All Downloads are FREE. Search and download functionalities are using the official Maven repository.

feign.RequestTemplate Maven / Gradle / Ivy

/**
 * Copyright 2012-2019 The Feign Authors
 *
 * 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 feign;

import feign.Request.HttpMethod;
import feign.template.HeaderTemplate;
import feign.template.QueryTemplate;
import feign.template.UriTemplate;
import feign.template.UriUtils;
import java.io.Serializable;
import java.net.URI;
import java.nio.charset.Charset;
import java.util.AbstractMap.SimpleImmutableEntry;
import java.util.*;
import java.util.Map.Entry;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import java.util.stream.Collectors;
import static feign.Util.*;

/**
 * Request Builder for an HTTP Target.
 * 

* This class is a variation on a UriTemplate, where, in addition to the uri, Headers and Query * information also support template expressions. *

*/ @SuppressWarnings({"WeakerAccess", "UnusedReturnValue"}) public final class RequestTemplate implements Serializable { private static final Pattern QUERY_STRING_PATTERN = Pattern.compile("(? queries = new LinkedHashMap<>(); private final Map headers = new TreeMap<>(String.CASE_INSENSITIVE_ORDER); private String target; private String fragment; private boolean resolved = false; private UriTemplate uriTemplate; private HttpMethod method; private transient Charset charset = Util.UTF_8; private Request.Body body = Request.Body.empty(); private boolean decodeSlash = true; private CollectionFormat collectionFormat = CollectionFormat.EXPLODED; /** * Create a new Request Template. */ public RequestTemplate() { super(); } /** * Create a new Request Template. * * @param target for the template. * @param uriTemplate for the template. * @param method of the request. * @param charset for the request. * @param body of the request, may be null * @param decodeSlash if the request uri should encode slash characters. * @param collectionFormat when expanding collection based variables. */ private RequestTemplate(String target, String fragment, UriTemplate uriTemplate, HttpMethod method, Charset charset, Request.Body body, boolean decodeSlash, CollectionFormat collectionFormat) { this.target = target; this.fragment = fragment; this.uriTemplate = uriTemplate; this.method = method; this.charset = charset; this.body = body; this.decodeSlash = decodeSlash; this.collectionFormat = (collectionFormat != null) ? collectionFormat : CollectionFormat.EXPLODED; } /** * Create a Request Template from an existing Request Template. * * @param requestTemplate to copy from. * @return a new Request Template. */ public static RequestTemplate from(RequestTemplate requestTemplate) { RequestTemplate template = new RequestTemplate(requestTemplate.target, requestTemplate.fragment, requestTemplate.uriTemplate, requestTemplate.method, requestTemplate.charset, requestTemplate.body, requestTemplate.decodeSlash, requestTemplate.collectionFormat); if (!requestTemplate.queries().isEmpty()) { template.queries.putAll(requestTemplate.queries); } if (!requestTemplate.headers().isEmpty()) { template.headers.putAll(requestTemplate.headers); } return template; } /** * Create a Request Template from an existing Request Template. * * @param toCopy template. * @deprecated replaced by {@link RequestTemplate#from(RequestTemplate)} */ @Deprecated public RequestTemplate(RequestTemplate toCopy) { checkNotNull(toCopy, "toCopy"); this.target = toCopy.target; this.fragment = toCopy.fragment; this.method = toCopy.method; this.queries.putAll(toCopy.queries); this.headers.putAll(toCopy.headers); this.charset = toCopy.charset; this.body = toCopy.body; this.decodeSlash = toCopy.decodeSlash; this.collectionFormat = (toCopy.collectionFormat != null) ? toCopy.collectionFormat : CollectionFormat.EXPLODED; this.uriTemplate = toCopy.uriTemplate; this.resolved = false; } /** * Resolve all expressions using the variable value substitutions provided. Variable values will * be pct-encoded, if they are not already. * * @param variables containing the variable values to use when resolving expressions. * @return a new Request Template with all of the variables resolved. */ public RequestTemplate resolve(Map variables) { StringBuilder uri = new StringBuilder(); /* create a new template form this one, but explicitly */ RequestTemplate resolved = RequestTemplate.from(this); if (this.uriTemplate == null) { /* create a new uri template using the default root */ this.uriTemplate = UriTemplate.create("", !this.decodeSlash, this.charset); } uri.append(this.uriTemplate.expand(variables)); /* * for simplicity, combine the queries into the uri and use the resulting uri to seed the * resolved template. */ if (!this.queries.isEmpty()) { /* * since we only want to keep resolved query values, reset any queries on the resolved copy */ resolved.queries(Collections.emptyMap()); StringBuilder query = new StringBuilder(); Iterator queryTemplates = this.queries.values().iterator(); while (queryTemplates.hasNext()) { QueryTemplate queryTemplate = queryTemplates.next(); String queryExpanded = queryTemplate.expand(variables); if (Util.isNotBlank(queryExpanded)) { query.append(queryExpanded); if (queryTemplates.hasNext()) { query.append("&"); } } } String queryString = query.toString(); if (!queryString.isEmpty()) { Matcher queryMatcher = QUERY_STRING_PATTERN.matcher(uri); if (queryMatcher.find()) { /* the uri already has a query, so any additional queries should be appended */ uri.append("&"); } else { uri.append("?"); } uri.append(queryString); } } /* add the uri to result */ resolved.uri(uri.toString()); /* headers */ if (!this.headers.isEmpty()) { /* * same as the query string, we only want to keep resolved values, so clear the header map on * the resolved instance */ resolved.headers(Collections.emptyMap()); for (HeaderTemplate headerTemplate : this.headers.values()) { /* resolve the header */ String header = headerTemplate.expand(variables); if (!header.isEmpty()) { /* split off the header values and add it to the resolved template */ String headerValues = header.substring(header.indexOf(" ") + 1); if (!headerValues.isEmpty()) { resolved.header(headerTemplate.getName(), headerValues); } } } } resolved.body(this.body.expand(variables)); /* mark the new template resolved */ resolved.resolved = true; return resolved; } /** * Resolves all expressions, using the variables provided. Values not present in the {@code * alreadyEncoded} map are pct-encoded. * * @param unencoded variable values to substitute. * @param alreadyEncoded variable names. * @return a resolved Request Template * @deprecated use {@link RequestTemplate#resolve(Map)}. Values already encoded are recognized as * such and skipped. */ @SuppressWarnings("unused") @Deprecated RequestTemplate resolve(Map unencoded, Map alreadyEncoded) { return this.resolve(unencoded); } /** * Creates a {@link Request} from this template. The template must be resolved before calling this * method, or an {@link IllegalStateException} will be thrown. * * @return a new Request instance. * @throws IllegalStateException if this template has not been resolved. */ public Request request() { if (!this.resolved) { throw new IllegalStateException("template has not been resolved."); } return Request.create(this.method, this.url(), this.headers(), this.requestBody()); } /** * Set the Http Method. * * @param method to use. * @return a RequestTemplate for chaining. * @deprecated see {@link RequestTemplate#method(HttpMethod)} */ @Deprecated public RequestTemplate method(String method) { checkNotNull(method, "method"); try { this.method = HttpMethod.valueOf(method); } catch (IllegalArgumentException iae) { throw new IllegalArgumentException("Invalid HTTP Method: " + method); } return this; } /** * Set the Http Method. * * @param method to use. * @return a RequestTemplate for chaining. */ public RequestTemplate method(HttpMethod method) { checkNotNull(method, "method"); this.method = method; return this; } /** * The Request Http Method. * * @return Http Method. */ public String method() { return (method != null) ? method.name() : null; } /** * Set whether do encode slash {@literal /} characters when resolving this template. * * @param decodeSlash if slash literals should not be encoded. * @return a RequestTemplate for chaining. */ public RequestTemplate decodeSlash(boolean decodeSlash) { this.decodeSlash = decodeSlash; this.uriTemplate = UriTemplate.create(this.uriTemplate.toString(), !this.decodeSlash, this.charset); return this; } /** * If slash {@literal /} characters are not encoded when resolving. * * @return true if slash literals are not encoded, false otherwise. */ public boolean decodeSlash() { return decodeSlash; } /** * The Collection Format to use when resolving variables that represent {@link Iterable}s or * {@link Collection}s * * @param collectionFormat to use. * @return a RequestTemplate for chaining. */ public RequestTemplate collectionFormat(CollectionFormat collectionFormat) { this.collectionFormat = collectionFormat; return this; } /** * The Collection Format that will be used when resolving {@link Iterable} and {@link Collection} * variables. * * @return the collection format set */ @SuppressWarnings("unused") public CollectionFormat collectionFormat() { return collectionFormat; } /** * Append the value to the template. *

* This method is poorly named and is used primarily to store the relative uri for the request. It * has been replaced by {@link RequestTemplate#uri(String)} and will be removed in a future * release. *

* * @param value to append. * @return a RequestTemplate for chaining. * @deprecated see {@link RequestTemplate#uri(String, boolean)} */ @Deprecated public RequestTemplate append(CharSequence value) { /* proxy to url */ if (this.uriTemplate != null) { return this.uri(value.toString(), true); } return this.uri(value.toString()); } /** * Insert the value at the specified point in the template uri. *

* This method is poorly named has undocumented behavior. When the value contains a fully * qualified http request url, the value is always inserted at the beginning of the uri. *

*

* Due to this, use of this method is not recommended and remains for backward compatibility. It * has been replaced by {@link RequestTemplate#target(String)} and will be removed in a future * release. *

* * @param pos in the uri to place the value. * @param value to insert. * @return a RequestTemplate for chaining. * @deprecated see {@link RequestTemplate#target(String)} */ @SuppressWarnings("unused") @Deprecated public RequestTemplate insert(int pos, CharSequence value) { return target(value.toString()); } /** * Set the Uri for the request, replacing the existing uri if set. * * @param uri to use, must be a relative uri. * @return a RequestTemplate for chaining. */ public RequestTemplate uri(String uri) { return this.uri(uri, false); } /** * Set the uri for the request. * * @param uri to use, must be a relative uri. * @param append if the uri should be appended, if the uri is already set. * @return a RequestTemplate for chaining. */ public RequestTemplate uri(String uri, boolean append) { /* validate and ensure that the url is always a relative one */ if (UriUtils.isAbsolute(uri)) { throw new IllegalArgumentException("url values must be not be absolute."); } if (uri == null) { uri = "/"; } else if ((!uri.isEmpty() && !uri.startsWith("/") && !uri.startsWith("{") && !uri.startsWith("?") && !uri.startsWith(";"))) { /* if the start of the url is a literal, it must begin with a slash. */ uri = "/" + uri; } /* * templates may provide query parameters. since we want to manage those explicity, we will need * to extract those out, leaving the uriTemplate with only the path to deal with. */ Matcher queryMatcher = QUERY_STRING_PATTERN.matcher(uri); if (queryMatcher.find()) { String queryString = uri.substring(queryMatcher.start() + 1); /* parse the query string */ this.extractQueryTemplates(queryString, append); /* reduce the uri to the path */ uri = uri.substring(0, queryMatcher.start()); } int fragmentIndex = uri.indexOf('#'); if (fragmentIndex > -1) { fragment = uri.substring(fragmentIndex); uri = uri.substring(0, fragmentIndex); } /* replace the uri template */ if (append && this.uriTemplate != null) { this.uriTemplate = UriTemplate.append(this.uriTemplate, uri); } else { this.uriTemplate = UriTemplate.create(uri, !this.decodeSlash, this.charset); } return this; } /** * Set the target host for this request. * * @param target host for this request. Must be an absolute target. * @return a RequestTemplate for chaining. */ public RequestTemplate target(String target) { /* target can be empty */ if (Util.isBlank(target)) { return this; } /* verify that the target contains the scheme, host and port */ if (!UriUtils.isAbsolute(target)) { throw new IllegalArgumentException("target values must be absolute."); } if (target.endsWith("/")) { target = target.substring(0, target.length() - 1); } try { /* parse the target */ URI targetUri = URI.create(target); if (Util.isNotBlank(targetUri.getRawQuery())) { /* * target has a query string, we need to make sure that they are recorded as queries */ this.extractQueryTemplates(targetUri.getRawQuery(), true); } /* strip the query string */ this.target = targetUri.getScheme() + "://" + targetUri.getAuthority() + targetUri.getPath(); if (targetUri.getFragment() != null) { this.fragment = "#" + targetUri.getFragment(); } } catch (IllegalArgumentException iae) { /* the uri provided is not a valid one, we can't continue */ throw new IllegalArgumentException("Target is not a valid URI.", iae); } return this; } /** * The URL for the request. If the template has not been resolved, the url will represent a uri * template. * * @return the url */ public String url() { /* build the fully qualified url with all query parameters */ StringBuilder url = new StringBuilder(this.path()); if (!this.queries.isEmpty()) { url.append(this.queryLine()); } if (fragment != null) { url.append(fragment); } return url.toString(); } /** * The Uri Path. * * @return the uri path. */ public String path() { /* build the fully qualified url with all query parameters */ StringBuilder path = new StringBuilder(); if (this.target != null) { path.append(this.target); } if (this.uriTemplate != null) { path.append(this.uriTemplate.toString()); } if (path.length() == 0) { /* no path indicates the root uri */ path.append("/"); } return path.toString(); } /** * List all of the template variable expressions for this template. * * @return a list of template variable names */ public List variables() { /* combine the variables from the uri, query, header, and body templates */ List variables = new ArrayList<>(this.uriTemplate.getVariables()); /* queries */ for (QueryTemplate queryTemplate : this.queries.values()) { variables.addAll(queryTemplate.getVariables()); } /* headers */ for (HeaderTemplate headerTemplate : this.headers.values()) { variables.addAll(headerTemplate.getVariables()); } /* body */ variables.addAll(this.body.getVariables()); return variables; } /** * @see RequestTemplate#query(String, Iterable) */ public RequestTemplate query(String name, String... values) { if (values == null) { return query(name, Collections.emptyList()); } return query(name, Arrays.asList(values)); } /** * Specify a Query String parameter, with the specified values. Values can be literals or template * expressions. * * @param name of the parameter. * @param values for this parameter. * @return a RequestTemplate for chaining. */ public RequestTemplate query(String name, Iterable values) { return appendQuery(name, values, this.collectionFormat); } /** * Specify a Query String parameter, with the specified values. Values can be literals or template * expressions. * * @param name of the parameter. * @param values for this parameter. * @param collectionFormat to use when resolving collection based expressions. * @return a Request Template for chaining. */ public RequestTemplate query(String name, Iterable values, CollectionFormat collectionFormat) { return appendQuery(name, values, collectionFormat); } /** * Appends the query name and values. * * @param name of the parameter. * @param values for the parameter, may be expressions. * @param collectionFormat to use when resolving collection based query variables. * @return a RequestTemplate for chaining. */ private RequestTemplate appendQuery(String name, Iterable values, CollectionFormat collectionFormat) { if (!values.iterator().hasNext()) { /* empty value, clear the existing values */ this.queries.remove(name); return this; } /* create a new query template out of the information here */ this.queries.compute(name, (key, queryTemplate) -> { if (queryTemplate == null) { return QueryTemplate.create(name, values, this.charset, collectionFormat); } else { return QueryTemplate.append(queryTemplate, values, collectionFormat); } }); return this; } /** * Sets the Query Parameters. * * @param queries to use for this request. * @return a RequestTemplate for chaining. */ @SuppressWarnings("unused") public RequestTemplate queries(Map> queries) { if (queries == null || queries.isEmpty()) { this.queries.clear(); } else { queries.forEach(this::query); } return this; } /** * Return an immutable Map of all Query Parameters and their values. * * @return registered Query Parameters. */ public Map> queries() { Map> queryMap = new LinkedHashMap<>(); this.queries.forEach((key, queryTemplate) -> { List values = new ArrayList<>(queryTemplate.getValues()); /* add the expanded collection, but lock it */ queryMap.put(key, Collections.unmodifiableList(values)); }); return Collections.unmodifiableMap(queryMap); } /** * @see RequestTemplate#header(String, Iterable) */ public RequestTemplate header(String name, String... values) { return header(name, Arrays.asList(values)); } /** * Specify a Header, with the specified values. Values can be literals or template expressions. * * @param name of the header. * @param values for this header. * @return a RequestTemplate for chaining. */ public RequestTemplate header(String name, Iterable values) { if (name == null || name.isEmpty()) { throw new IllegalArgumentException("name is required."); } if (values == null) { values = Collections.emptyList(); } return appendHeader(name, values); } /** * Clear on reader from {@link RequestTemplate} * * @param name of the header. * @return a RequestTemplate for chaining. */ public RequestTemplate removeHeader(String name) { if (name == null || name.isEmpty()) { throw new IllegalArgumentException("name is required."); } this.headers.remove(name); return this; } /** * Create a Header Template. * * @param name of the header * @param values for the header, may be expressions. * @return a RequestTemplate for chaining. */ private RequestTemplate appendHeader(String name, Iterable values) { if (!values.iterator().hasNext()) { /* empty value, clear the existing values */ this.headers.remove(name); return this; } this.headers.compute(name, (headerName, headerTemplate) -> { if (headerTemplate == null) { return HeaderTemplate.create(headerName, values); } else { return HeaderTemplate.append(headerTemplate, values); } }); return this; } /** * Headers for this Request. * * @param headers to use. * @return a RequestTemplate for chaining. */ public RequestTemplate headers(Map> headers) { if (headers != null && !headers.isEmpty()) { headers.forEach(this::header); } else { this.headers.clear(); } return this; } /** * Returns an immutable copy of the Headers for this request. * * @return the currently applied headers. */ public Map> headers() { Map> headerMap = new TreeMap<>(String.CASE_INSENSITIVE_ORDER); this.headers.forEach((key, headerTemplate) -> { List values = new ArrayList<>(headerTemplate.getValues()); /* add the expanded collection, but only if it has values */ if (!values.isEmpty()) { headerMap.put(key, Collections.unmodifiableList(values)); } }); return Collections.unmodifiableMap(headerMap); } /** * Sets the Body and Charset for this request. * * @param bodyData to send, can be null. * @param charset of the encoded data. * @return a RequestTemplate for chaining. * @deprecated use {@link RequestTemplate#body(feign.Request.Body)} instead */ @Deprecated public RequestTemplate body(byte[] bodyData, Charset charset) { this.body(Request.Body.encoded(bodyData, charset)); return this; } /** * Set the Body for this request. Charset is assumed to be UTF_8. Data must be encoded. * * @param bodyText to send. * @return a RequestTemplate for chaining. * @deprecated use {@link RequestTemplate#body(feign.Request.Body)} instead */ @Deprecated public RequestTemplate body(String bodyText) { byte[] bodyData = bodyText != null ? bodyText.getBytes(UTF_8) : null; return body(bodyData, UTF_8); } /** * Set the Body for this request. * * @param body to send. * @return a RequestTemplate for chaining. */ public RequestTemplate body(Request.Body body) { this.body = body; header(CONTENT_LENGTH); if (body.length() > 0) { header(CONTENT_LENGTH, String.valueOf(body.length())); } return this; } /** * Charset of the Request Body, if known. * * @return the currently applied Charset. */ public Charset requestCharset() { return charset; } /** * The Request Body. * * @return the request body. * @deprecated replaced by {@link RequestTemplate#requestBody()} */ @Deprecated public byte[] body() { return body.asBytes(); } /** * Specify the Body Template to use. Can contain literals and expressions. * * @param bodyTemplate to use. * @return a RequestTemplate for chaining. * @deprecated replaced by {@link RequestTemplate#body(feign.Request.Body)} */ @Deprecated public RequestTemplate bodyTemplate(String bodyTemplate) { this.body(Request.Body.bodyTemplate(bodyTemplate, Util.UTF_8)); return this; } /** * Body Template to resolve. * * @return the unresolved body template. */ public String bodyTemplate() { return body.bodyTemplate(); } @Override public String toString() { return request().toString(); } /** * Return if the variable exists on the uri, query, or headers, in this template. * * @param variable to look for. * @return true if the variable exists, false otherwise. */ public boolean hasRequestVariable(String variable) { return this.getRequestVariables().contains(variable); } /** * Retrieve all uri, header, and query template variables. * * @return a List of all the variable names. */ public Collection getRequestVariables() { final Collection variables = new LinkedHashSet<>(this.uriTemplate.getVariables()); this.queries.values().forEach(queryTemplate -> variables.addAll(queryTemplate.getVariables())); this.headers.values() .forEach(headerTemplate -> variables.addAll(headerTemplate.getVariables())); return variables; } /** * If this template has been resolved. * * @return true if the template has been resolved, false otherwise. */ @SuppressWarnings("unused") public boolean resolved() { return this.resolved; } /** * The Query String for the template. Expressions are not resolved. * * @return the Query String. */ public String queryLine() { StringBuilder queryString = new StringBuilder(); if (!this.queries.isEmpty()) { Iterator iterator = this.queries.values().iterator(); while (iterator.hasNext()) { QueryTemplate queryTemplate = iterator.next(); String query = queryTemplate.toString(); if (query != null && !query.isEmpty()) { queryString.append(query); if (iterator.hasNext()) { queryString.append("&"); } } } } /* remove any trailing ampersands */ String result = queryString.toString(); if (result.endsWith("&")) { result = result.substring(0, result.length() - 1); } if (!result.isEmpty()) { result = "?" + result; } return result; } private void extractQueryTemplates(String queryString, boolean append) { /* split the query string up into name value pairs */ Map> queryParameters = Arrays.stream(queryString.split("&")) .map(this::splitQueryParameter) .collect(Collectors.groupingBy( SimpleImmutableEntry::getKey, LinkedHashMap::new, Collectors.mapping(Entry::getValue, Collectors.toList()))); /* add them to this template */ if (!append) { /* clear the queries and use the new ones */ this.queries.clear(); } queryParameters.forEach(this::query); } private SimpleImmutableEntry splitQueryParameter(String pair) { int eq = pair.indexOf("="); final String name = (eq > 0) ? pair.substring(0, eq) : pair; final String value = (eq > 0 && eq < pair.length()) ? pair.substring(eq + 1) : null; return new SimpleImmutableEntry<>(name, value); } public Request.Body requestBody() { return this.body; } /** * Factory for creating RequestTemplate. */ interface Factory { /** * create a request template using args passed to a method invocation. */ RequestTemplate create(Object[] argv); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy