/**
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
package org.apache.cxf.jaxrs.provider;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.lang.annotation.Annotation;
import java.lang.reflect.Type;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import javax.ws.rs.Consumes;
import javax.ws.rs.Encoded;
import javax.ws.rs.Produces;
import javax.ws.rs.WebApplicationException;
import javax.ws.rs.core.Context;
import javax.ws.rs.core.MediaType;
import javax.ws.rs.core.MultivaluedMap;
import javax.ws.rs.ext.MessageBodyReader;
import javax.ws.rs.ext.MessageBodyWriter;
import javax.ws.rs.ext.Provider;
import org.apache.cxf.jaxrs.ext.MessageContext;
import org.apache.cxf.jaxrs.ext.form.Form;
import org.apache.cxf.jaxrs.ext.multipart.Attachment;
import org.apache.cxf.jaxrs.ext.multipart.MultipartBody;
import org.apache.cxf.jaxrs.impl.MetadataMap;
import org.apache.cxf.jaxrs.utils.AnnotationUtils;
import org.apache.cxf.jaxrs.utils.FormUtils;
import org.apache.cxf.jaxrs.utils.HttpUtils;
import org.apache.cxf.jaxrs.utils.multipart.AttachmentUtils;
import org.apache.cxf.message.Message;
import org.apache.cxf.phase.PhaseInterceptorChain;
@Produces({"application/x-www-form-urlencoded", "multipart/form-data" })
@Consumes({"application/x-www-form-urlencoded", "multipart/form-data" })
@Provider
public class FormEncodingProvider implements
MessageBodyReader<Object>, MessageBodyWriter<Object> {
private FormValidator validator;
@Context private MessageContext mc;
private String attachmentDir;
private String attachmentThreshold;
private String attachmentMaxSize;
private boolean expectEncoded;
public void setExpectedEncoded(boolean expect) {
this.expectEncoded = expect;
}
public void setAttachmentDirectory(String dir) {
attachmentDir = dir;
}
public void setAttachmentThreshold(String threshold) {
attachmentThreshold = threshold;
}
public void setAttachmentMaxSize(String maxSize) {
attachmentMaxSize = maxSize;
}
public void setValidator(FormValidator formValidator) {
validator = formValidator;
}
public boolean isReadable(Class<?> type, Type genericType,
Annotation[] annotations, MediaType mt) {
return isSupported(type, genericType, annotations, mt);
}
public Object readFrom(
Class<Object> clazz, Type genericType, Annotation[] annotations, MediaType mt,
MultivaluedMap<String, String> headers, InputStream is)
throws IOException {
if (is == null) {
return null;
}
try {
if (mt.isCompatible(MediaType.MULTIPART_FORM_DATA_TYPE)) {
MultipartBody body = AttachmentUtils.getMultipartBody(mc);
if (MultipartBody.class.isAssignableFrom(clazz)) {
return body;
} else if (Attachment.class.isAssignableFrom(clazz)) {
return body.getRootAttachment();
}
}
MultivaluedMap<String, String> params = createMap(clazz);
populateMap(params, is, mt, !keepEncoded(annotations));
validateMap(params);
persistParamsOnMessage(params);
return getFormObject(clazz, params);
} catch (WebApplicationException e) {
throw e;
} catch (Exception e) {
throw new WebApplicationException(e);
}
}
protected boolean keepEncoded(Annotation[] anns) {
return AnnotationUtils.getAnnotation(anns, Encoded.class) != null
|| expectEncoded;
}
protected void persistParamsOnMessage(MultivaluedMap<String, String> params) {
Message message = PhaseInterceptorChain.getCurrentMessage();
if (message != null) {
message.put(FormUtils.FORM_PARAM_MAP, params);
}
}
@SuppressWarnings("unchecked")
protected MultivaluedMap<String, String> createMap(Class<?> clazz) throws Exception {
if (clazz == MultivaluedMap.class || clazz == Form.class) {
return new MetadataMap<String, String>();
}
return (MultivaluedMap<String, String>)clazz.newInstance();
}
@SuppressWarnings("unchecked")
private Object getFormObject(Class<?> clazz, MultivaluedMap<String, String> params) {
return Form.class.isAssignableFrom(clazz) ? new Form((MultivaluedMap)params) : params;
}
/**
* Retrieve map of parameters from the passed in message
*
* @param message
* @return a Map of parameters.
*/
protected void populateMap(MultivaluedMap<String, String> params,
InputStream is, MediaType mt, boolean decode) {
if (mt.isCompatible(MediaType.MULTIPART_FORM_DATA_TYPE)) {
MultipartBody body =
AttachmentUtils.getMultipartBody(mc, attachmentDir, attachmentThreshold, attachmentMaxSize);
FormUtils.populateMapFromMultipart(params, body, decode);
} else {
String enc = HttpUtils.getEncoding(mt, "UTF-8");
FormUtils.populateMapFromString(params,
FormUtils.readBody(is, enc),
enc,
decode,
mc != null ? mc.getHttpServletRequest() : null);
}
}
protected void validateMap(MultivaluedMap<String, String> params) {
if (validator != null) {
validator.validate(params);
}
}
public long getSize(Object t, Class<?> type,
Type genericType, Annotation[] annotations,
MediaType mediaType) {
return -1;
}
public boolean isWriteable(Class<?> type, Type genericType, Annotation[] annotations,
MediaType mt) {
return isSupported(type, genericType, annotations, mt);
}
private boolean isSupported(Class<?> type, Type genericType, Annotation[] annotations,
MediaType mt) {
return (MultivaluedMap.class.isAssignableFrom(type) || Form.class.isAssignableFrom(type))
|| (mt.getType().equalsIgnoreCase("multipart")
&& mt.isCompatible(MediaType.MULTIPART_FORM_DATA_TYPE)
&& (MultivaluedMap.class.isAssignableFrom(type) || Form.class.isAssignableFrom(type)));
}
@SuppressWarnings("unchecked")
public void writeTo(Object obj, Class<?> c, Type t, Annotation[] anns,
MediaType mt, MultivaluedMap<String, Object> headers, OutputStream os)
throws IOException, WebApplicationException {
MultivaluedMap<String, String> map =
(MultivaluedMap<String, String>)(obj instanceof Form ? ((Form)obj).getData() : obj);
boolean encoded = keepEncoded(anns);
String encoding = HttpUtils.getSetEncoding(mt, headers, "UTF-8");
for (Iterator<Map.Entry<String, List<String>>> it = map.entrySet().iterator(); it.hasNext();) {
Map.Entry<String, List<String>> entry = it.next();
for (Iterator<String> entryIterator = entry.getValue().iterator(); entryIterator.hasNext();) {
String value = entryIterator.next();
os.write(entry.getKey().getBytes(encoding));
os.write('=');
String data = encoded ? value : HttpUtils.urlEncode(value);
os.write(data.getBytes(encoding));
if (entryIterator.hasNext() || it.hasNext()) {
os.write('&');
}
}
}
}
}