com.englishtown.vertx.jersey.impl.DefaultJerseyHandler Maven / Gradle / Ivy
/*
* The MIT License (MIT)
* Copyright © 2013 Englishtown
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the “Software”), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED “AS IS”, WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
* THE SOFTWARE.
*/
package com.englishtown.vertx.jersey.impl;
import com.englishtown.vertx.jersey.ApplicationHandlerDelegate;
import com.englishtown.vertx.jersey.JerseyHandler;
import com.englishtown.vertx.jersey.VertxContainer;
import com.englishtown.vertx.jersey.inject.ContainerResponseWriterProvider;
import com.englishtown.vertx.jersey.inject.VertxRequestProcessor;
import com.englishtown.vertx.jersey.security.DefaultSecurityContext;
import io.netty.handler.codec.http.HttpResponseStatus;
import io.netty.handler.codec.http.QueryStringDecoder;
import io.vertx.core.Handler;
import io.vertx.core.buffer.Buffer;
import io.vertx.core.http.HttpMethod;
import io.vertx.core.http.HttpServerRequest;
import io.vertx.core.http.HttpServerResponse;
import io.vertx.core.logging.Logger;
import io.vertx.core.logging.LoggerFactory;
import org.glassfish.hk2.api.TypeLiteral;
import org.glassfish.jersey.internal.MapPropertiesDelegate;
import org.glassfish.jersey.internal.util.collection.Ref;
import org.glassfish.jersey.server.ContainerRequest;
import javax.inject.Inject;
import javax.inject.Provider;
import javax.ws.rs.core.HttpHeaders;
import javax.ws.rs.core.MediaType;
import javax.ws.rs.core.UriBuilder;
import java.io.ByteArrayInputStream;
import java.io.InputStream;
import java.net.URI;
import java.net.URLEncoder;
import java.util.List;
import java.util.Map;
/**
* The Vertx {@link HttpServerRequest} handler that delegates handling to Jersey
*/
public class DefaultJerseyHandler implements JerseyHandler {
private Provider containerProvider;
private VertxContainer container;
private URI baseUri;
private Integer maxBodySize;
private final ContainerResponseWriterProvider responseWriterProvider;
private final List requestProcessors;
private static final Logger logger = LoggerFactory.getLogger(DefaultJerseyHandler.class);
@Inject
public DefaultJerseyHandler(
Provider containerProvider,
ContainerResponseWriterProvider responseWriterProvider,
List requestProcessors) {
this.containerProvider = containerProvider;
this.responseWriterProvider = responseWriterProvider;
this.requestProcessors = requestProcessors;
}
@Override
public URI getBaseUri() {
if (baseUri == null) {
baseUri = getContainer().getOptions().getBaseUri();
if (baseUri == null) {
throw new IllegalStateException("baseUri is null, have you called init() first?");
}
}
return baseUri;
}
public int getMaxBodySize() {
if (maxBodySize == null) {
maxBodySize = getContainer().getOptions().getMaxBodySize();
if (maxBodySize <= 0) {
maxBodySize = DefaultJerseyOptions.DEFAULT_MAX_BODY_SIZE;
}
}
return maxBodySize;
}
@Override
public ApplicationHandlerDelegate getDelegate() {
return getContainer().getApplicationHandlerDelegate();
}
@Override
public VertxContainer getContainer() {
if (container == null) {
container = containerProvider.get();
}
return container;
}
@Override
public JerseyHandler setContainer(VertxContainer container) {
this.container = container;
return this;
}
/**
* {@inheritDoc}
*/
@Override
public void handle(final HttpServerRequest vertxRequest) {
// Wait for the body for jersey to handle form/json/xml params
if (shouldReadData(vertxRequest)) {
if (logger.isDebugEnabled()) {
logger.debug("DefaultJerseyHandler - handle request and read body: " + vertxRequest.method() + " " + vertxRequest.uri());
}
final Buffer body = Buffer.buffer();
vertxRequest.handler(buffer -> {
body.appendBuffer(buffer);
if (body.length() > getMaxBodySize()) {
throw new RuntimeException("The input stream has exceeded the max allowed body size "
+ getMaxBodySize() + ".");
}
});
vertxRequest.endHandler(aVoid -> {
InputStream inputStream = new ByteArrayInputStream(body.getBytes());
DefaultJerseyHandler.this.handle(vertxRequest, inputStream);
});
} else {
if (logger.isDebugEnabled()) {
logger.debug("DefaultJerseyHandler - handle request: " + vertxRequest.method() + " " + vertxRequest.uri());
}
DefaultJerseyHandler.this.handle(vertxRequest, null);
}
}
protected void handle(
final HttpServerRequest vertxRequest,
final InputStream inputStream
) {
URI uri = getAbsoluteURI(vertxRequest);
boolean isSecure = "https".equalsIgnoreCase(uri.getScheme());
UriBuilder baseUriBuilder = UriBuilder.fromUri(uri)
.replacePath(getBaseUri().getPath())
.replaceQuery(null);
// Create the jersey request
final ContainerRequest jerseyRequest = new ContainerRequest(
baseUriBuilder.build(),
uri,
vertxRequest.method().name(),
new DefaultSecurityContext(isSecure),
new MapPropertiesDelegate());
handle(vertxRequest, inputStream, jerseyRequest);
}
protected URI getAbsoluteURI(HttpServerRequest vertxRequest) {
URI absoluteUri;
String hostAndPort = vertxRequest.headers().get(HttpHeaders.HOST);
try {
absoluteUri = URI.create(vertxRequest.absoluteURI());
if (hostAndPort != null && !hostAndPort.isEmpty()) {
String[] parts = hostAndPort.split(":");
String host = parts[0];
int port = (parts.length > 1 ? Integer.valueOf(parts[1]) : -1);
if (!host.equalsIgnoreCase(absoluteUri.getHost()) || port != absoluteUri.getPort()) {
absoluteUri = UriBuilder.fromUri(absoluteUri)
.host(host)
.port(port)
.build();
}
}
return absoluteUri;
} catch (IllegalArgumentException e) {
String uri = vertxRequest.uri();
if (!uri.contains("?")) {
throw e;
}
try {
logger.warn("Invalid URI: " + uri + ". Attempting to parse query string.", e);
QueryStringDecoder decoder = new QueryStringDecoder(uri);
StringBuilder sb = new StringBuilder(decoder.path() + "?");
for (Map.Entry> p : decoder.parameters().entrySet()) {
for (String value : p.getValue()) {
sb.append(p.getKey()).append("=").append(URLEncoder.encode(value, "UTF-8"));
}
}
return URI.create(sb.toString());
} catch (Exception e1) {
throw new RuntimeException(e1);
}
}
}
protected void handle(final HttpServerRequest vertxRequest,
final InputStream inputStream,
final ContainerRequest jerseyRequest) {
// Provide the vertx response writer
jerseyRequest.setWriter(responseWriterProvider.get(vertxRequest, jerseyRequest));
// Set entity stream if provided (form posts)
if (inputStream != null) {
jerseyRequest.setEntityStream(inputStream);
}
// Copy headers
for (Map.Entry header : vertxRequest.headers().entries()) {
jerseyRequest.getHeaders().add(header.getKey(), header.getValue());
}
// Set request scoped instances
jerseyRequest.setRequestScopedInitializer(locator -> {
locator.>getService((new TypeLiteral>() {
}).getType()).set(vertxRequest);
locator.>getService((new TypeLiteral>() {
}).getType()).set(vertxRequest.response());
});
// Call vertx before request processors
if (!requestProcessors.isEmpty()) {
// Pause the vert.x request if we haven't already read the body
if (inputStream == null) {
vertxRequest.pause();
}
callVertxRequestProcessor(0, vertxRequest, jerseyRequest, aVoid -> {
// Resume the vert.x request if we haven't already read the body
if (inputStream == null) {
vertxRequest.resume();
}
getDelegate().handle(jerseyRequest);
});
} else {
getDelegate().handle(jerseyRequest);
}
}
protected void callVertxRequestProcessor(
int index,
final HttpServerRequest vertxRequest,
final ContainerRequest jerseyRequest,
final Handler done
) {
if (index >= requestProcessors.size()) {
done.handle(null);
}
VertxRequestProcessor processor = requestProcessors.get(index);
final int next = index + 1;
try {
processor.process(vertxRequest, jerseyRequest, aVoid -> {
if (next >= requestProcessors.size()) {
done.handle(null);
} else {
callVertxRequestProcessor(next, vertxRequest, jerseyRequest, done);
}
});
} catch (Throwable t) {
logger.error("VertxRequestProcessor " + processor.getClass().getSimpleName() + " threw exception: " + t.getMessage(), t);
vertxRequest.response()
.setStatusCode(HttpResponseStatus.INTERNAL_SERVER_ERROR.code())
.end();
}
}
protected boolean shouldReadData(HttpServerRequest vertxRequest) {
HttpMethod method = vertxRequest.method();
// Only read input stream data for post/put methods
if (!(HttpMethod.POST == method || HttpMethod.PUT == method)) {
return false;
}
String contentType = vertxRequest.headers().get(HttpHeaders.CONTENT_TYPE);
if (contentType == null || contentType.isEmpty()) {
// Special handling for IE8 XDomainRequest where content-type is missing
// http://blogs.msdn.com/b/ieinternals/archive/2010/05/13/xdomainrequest-restrictions-limitations-and-workarounds.aspx
return true;
}
MediaType mediaType = MediaType.valueOf(contentType);
// Allow text/plain
if (MediaType.TEXT_PLAIN_TYPE.getType().equals(mediaType.getType())
&& MediaType.TEXT_PLAIN_TYPE.getSubtype().equals(mediaType.getSubtype())) {
return true;
}
// Only other media types accepted are application (will check subtypes next)
String applicationType = MediaType.APPLICATION_FORM_URLENCODED_TYPE.getType();
if (!applicationType.equalsIgnoreCase(mediaType.getType())) {
return false;
}
// Need to do some special handling for forms:
// Jersey doesn't properly handle when charset is included
if (mediaType.getSubtype().equalsIgnoreCase(MediaType.APPLICATION_FORM_URLENCODED_TYPE.getSubtype())) {
if (!mediaType.getParameters().isEmpty()) {
vertxRequest.headers().remove(HttpHeaders.CONTENT_TYPE);
vertxRequest.headers().add(HttpHeaders.CONTENT_TYPE, MediaType.APPLICATION_FORM_URLENCODED);
}
return true;
}
// Also accept json/xml sub types
return MediaType.APPLICATION_JSON_TYPE.getSubtype().equalsIgnoreCase(mediaType.getSubtype())
|| MediaType.APPLICATION_XML_TYPE.getSubtype().equalsIgnoreCase(mediaType.getSubtype());
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy