com.gwtplatform.dispatch.rest.client.DefaultRestRequestBuilderFactory Maven / Gradle / Ivy
/**
* Copyright 2013 ArcBees Inc.
*
* 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 com.gwtplatform.dispatch.rest.client;
import java.util.Collection;
import java.util.List;
import java.util.Map;
import javax.inject.Inject;
import javax.ws.rs.core.HttpHeaders;
import javax.ws.rs.core.MediaType;
import com.github.nmorel.gwtjackson.client.exception.JsonMappingException;
import com.google.common.base.Strings;
import com.google.common.collect.Lists;
import com.google.common.collect.Maps;
import com.google.common.collect.Multimap;
import com.google.gwt.http.client.RequestBuilder;
import com.google.gwt.http.client.RequestBuilder.Method;
import com.gwtplatform.common.shared.UrlUtils;
import com.gwtplatform.dispatch.rest.client.serialization.Serialization;
import com.gwtplatform.dispatch.rest.shared.HttpMethod;
import com.gwtplatform.dispatch.rest.shared.MetadataType;
import com.gwtplatform.dispatch.rest.shared.RestAction;
import com.gwtplatform.dispatch.rest.shared.RestParameter;
import com.gwtplatform.dispatch.shared.ActionException;
import static com.google.gwt.user.client.rpc.RpcRequestBuilder.MODULE_BASE_HEADER;
/**
* Default implementation for {@link RestRequestBuilderFactory}.
*/
public class DefaultRestRequestBuilderFactory implements RestRequestBuilderFactory {
private static final Map HTTP_METHODS = Maps.newEnumMap(HttpMethod.class);
private static final String JSON_UTF8 = "application/json; charset=utf-8";
static {
HTTP_METHODS.put(HttpMethod.GET, RequestBuilder.GET);
HTTP_METHODS.put(HttpMethod.POST, RequestBuilder.POST);
HTTP_METHODS.put(HttpMethod.PUT, RequestBuilder.PUT);
HTTP_METHODS.put(HttpMethod.DELETE, RequestBuilder.DELETE);
HTTP_METHODS.put(HttpMethod.HEAD, RequestBuilder.HEAD);
}
private final ActionMetadataProvider metadataProvider;
private final Serialization serialization;
private final HttpRequestBuilderFactory httpRequestBuilderFactory;
private final UrlUtils urlUtils;
private final Multimap globalHeaderParams;
private final Multimap globalQueryParams;
private final String baseUrl;
private final String securityHeaderName;
private final Integer requestTimeoutMs;
@Inject
DefaultRestRequestBuilderFactory(ActionMetadataProvider metadataProvider,
Serialization serialization,
HttpRequestBuilderFactory httpRequestBuilderFactory,
UrlUtils urlUtils,
@GlobalHeaderParams Multimap globalHeaderParams,
@GlobalQueryParams Multimap globalQueryParams,
@RestApplicationPath String baseUrl,
@XsrfHeaderName String securityHeaderName,
@RequestTimeout Integer requestTimeoutMs) {
this.metadataProvider = metadataProvider;
this.serialization = serialization;
this.httpRequestBuilderFactory = httpRequestBuilderFactory;
this.urlUtils = urlUtils;
this.globalHeaderParams = globalHeaderParams;
this.globalQueryParams = globalQueryParams;
this.baseUrl = baseUrl;
this.securityHeaderName = securityHeaderName;
this.requestTimeoutMs = requestTimeoutMs;
}
@Override
public > RequestBuilder build(A action, String securityToken) throws ActionException {
Method httpMethod = HTTP_METHODS.get(action.getHttpMethod());
String url = buildUrl(action);
String xsrfToken = action.isSecured() ? securityToken : "";
RequestBuilder requestBuilder = httpRequestBuilderFactory.create(httpMethod, url);
requestBuilder.setTimeoutMillis(requestTimeoutMs);
buildHeaders(requestBuilder, xsrfToken, action);
buildBody(requestBuilder, action);
return requestBuilder;
}
/**
* Encodes the given {@link RestParameter} as a path parameter. The default implementation delegates to
* {@link UrlUtils#encodePathSegment(String)}.
*
* @param value the value to encode.
* @return the encoded path parameter.
* @throws ActionException if an exception occurred while encoding the path parameter.
* @see #encode(com.gwtplatform.dispatch.rest.shared.RestParameter)
*/
protected String encodePathParam(String value) throws ActionException {
return urlUtils.encodePathSegment(value);
}
/**
* Encodes the given {@link RestParameter} as a query parameter. The default implementation delegates to
* {@link UrlUtils#encodeQueryString(String)}.
*
* @param value the value to encode.
* @return the encoded query parameter.
* @throws ActionException if an exception occurred while encoding the query parameter.
* @see #encode(com.gwtplatform.dispatch.rest.shared.RestParameter)
*/
protected String encodeQueryParam(String value) throws ActionException {
return urlUtils.encodeQueryString(value);
}
/**
* Verify if the provided bodyType
can be serialized.
*
* @param bodyType the parameterized type to verify if it can be serialized.
* @return true
if bodyType
can be serialized, otherwise false
.
*/
protected boolean canSerialize(String bodyType) {
return serialization.canSerialize(bodyType);
}
/**
* Serialize the given object. We assume {@link #canSerialize(String)} returns true
or a runtime
* exception may be thrown.
*
* @param object the object to serialize.
* @param bodyType The parameterized type of the object to serialize.
* @return The serialized string.
*/
protected String serialize(Object object, String bodyType) {
return serialization.serialize(object, bodyType);
}
private void buildHeaders(RequestBuilder requestBuilder, String xsrfToken, RestAction> action)
throws ActionException {
String path = action.getPath();
List actionParams = action.getHeaderParams();
Collection applicableGlobalParams = globalHeaderParams.get(action.getHttpMethod());
// By setting the most generic headers first, we make sure they can be overridden by more specific ones
requestBuilder.setHeader(HttpHeaders.ACCEPT, MediaType.APPLICATION_JSON);
requestBuilder.setHeader(HttpHeaders.CONTENT_TYPE, JSON_UTF8);
if (!isAbsoluteUrl(path)) {
requestBuilder.setHeader(MODULE_BASE_HEADER, baseUrl);
}
if (!Strings.isNullOrEmpty(xsrfToken)) {
requestBuilder.setHeader(securityHeaderName, xsrfToken);
}
for (RestParameter parameter : applicableGlobalParams) {
String value = parameter.getStringValue();
if (value != null) {
requestBuilder.setHeader(parameter.getName(), value);
}
}
for (RestParameter param : actionParams) {
requestBuilder.setHeader(param.getName(), param.getStringValue());
}
}
private > void buildBody(RequestBuilder requestBuilder, A action) throws ActionException {
if (action.hasFormParams()) {
requestBuilder.setRequestData(buildQueryString(action.getFormParams()));
} else if (action.hasBodyParam()) {
requestBuilder.setRequestData(getSerializedValue(action, action.getBodyParam()));
}
}
private String buildUrl(RestAction> action) throws ActionException {
List queryParams = getGlobalQueryParamsForAction(action);
queryParams.addAll(action.getQueryParams());
String queryString = buildQueryString(queryParams);
if (!queryString.isEmpty()) {
queryString = "?" + queryString;
}
String path = buildPath(action.getPath(), action.getPathParams());
String prefix = "";
if (!isAbsoluteUrl(path)) {
prefix = baseUrl;
}
return prefix + path + queryString;
}
private List getGlobalQueryParamsForAction(RestAction> action) {
List queryParams = Lists.newArrayList();
for (RestParameter parameter : globalQueryParams.get(action.getHttpMethod())) {
String value = parameter.getStringValue();
if (value != null) {
queryParams.add(new RestParameter(parameter.getName(), value));
}
}
return queryParams;
}
private boolean isAbsoluteUrl(String path) {
return path.startsWith("http://") || path.startsWith("https://");
}
private String buildPath(String rawPath, List params) throws ActionException {
String path = rawPath;
for (RestParameter param : params) {
String encodedParam = encodePathParam(param.getStringValue());
path = path.replace("{" + param.getName() + "}", encodedParam);
}
return path;
}
private String buildQueryString(List params) throws ActionException {
StringBuilder queryString = new StringBuilder();
for (RestParameter param : params) {
queryString.append("&")
.append(param.getName())
.append("=")
.append(encodeQueryParam(param.getStringValue()));
}
if (queryString.length() != 0) {
queryString.deleteCharAt(0);
}
return queryString.toString();
}
private String getSerializedValue(RestAction> action, Object object) throws ActionException {
String bodyType = (String) metadataProvider.getValue(action, MetadataType.BODY_TYPE);
if (bodyType != null && canSerialize(bodyType)) {
try {
return serialize(object, bodyType);
} catch (JsonMappingException e) {
throw new ActionException("Unable to serialize request body. An unexpected error occurred.", e);
}
}
throw new ActionException("Unable to serialize request body. No serializer found.");
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy