br.com.caelum.vraptor.observer.DeserializingObserver Maven / Gradle / Ivy
/***
* Copyright (c) 2009 Caelum - www.caelum.com.br/opensource All rights reserved.
*
* Licensed 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 br.com.caelum.vraptor.observer;
import static java.util.Arrays.asList;
import static org.slf4j.LoggerFactory.getLogger;
import java.io.IOException;
import java.util.List;
import javax.enterprise.context.ApplicationScoped;
import javax.enterprise.event.Observes;
import javax.inject.Inject;
import javax.servlet.http.HttpServletRequest;
import org.slf4j.Logger;
import br.com.caelum.vraptor.Consumes;
import br.com.caelum.vraptor.controller.ControllerMethod;
import br.com.caelum.vraptor.core.MethodInfo;
import br.com.caelum.vraptor.events.InterceptorsReady;
import br.com.caelum.vraptor.events.MethodReady;
import br.com.caelum.vraptor.ioc.Container;
import br.com.caelum.vraptor.serialization.Deserializer;
import br.com.caelum.vraptor.serialization.Deserializers;
import br.com.caelum.vraptor.view.Status;
/**
* Important: this class must observe {@link MethodReady}
* because it is fired just before {@link ExecuteMethod} execution
*
* @author Lucas Cavalcanti
* @author Rafael Ferreira
* @author Rodrigo Turini
*/
@ApplicationScoped
public class DeserializingObserver {
private final Deserializers deserializers;
private final Container container;
private static final Logger logger = getLogger(DeserializingObserver.class);
/**
* @deprecated CDI eyes only
*/
protected DeserializingObserver() {
this(null, null);
}
@Inject
public DeserializingObserver(Deserializers deserializers, Container container) {
this.deserializers = deserializers;
this.container = container;
}
public void deserializes(@Observes InterceptorsReady event, HttpServletRequest request,
MethodInfo methodInfo, Status status) throws IOException {
ControllerMethod method = event.getControllerMethod();
if (!method.containsAnnotation(Consumes.class)) return;
List supported = asList(method.getMethod().getAnnotation(Consumes.class).value());
if(request.getContentType() == null) {
logger.warn("Request does not have Content-Type and parameters will be not deserialized");
return;
}
String contentType = mime(request.getContentType());
if (!supported.isEmpty() && !supported.contains(contentType)) {
unsupported("Request with media type [%s]. Expecting one of %s.", status, contentType, supported);
return;
}
Deserializer deserializer = deserializers.deserializerFor(contentType, container);
if (deserializer == null) {
unsupported("Unable to handle media type [%s]: no deserializer found.", status, contentType);
return;
}
Object[] deserialized = deserializer.deserialize(request.getInputStream(), method);
logger.debug("Deserialized parameters for {} are {} ", method, deserialized);
for (int i = 0; i < deserialized.length; i++) {
if (deserialized[i] != null) {
methodInfo.setParameter(i, deserialized[i]);
}
}
}
private static String mime(String contentType) {
if (contentType.contains(";")) {
return contentType.split(";")[0];
}
return contentType;
}
private void unsupported(String message, Status status, Object... params) {
status.unsupportedMediaType(String.format(message, params));
}
}