org.apache.cxf.jaxrs.impl.UriBuilderImpl Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of cxf-bundle-jaxrs Show documentation
Show all versions of cxf-bundle-jaxrs Show documentation
Apache CXF JAX-RS Bundle Jar
/**
* 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.impl;
import java.lang.annotation.Annotation;
import java.lang.reflect.Method;
import java.net.URI;
import java.net.URISyntaxException;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
import javax.ws.rs.Path;
import javax.ws.rs.core.MultivaluedMap;
import javax.ws.rs.core.PathSegment;
import javax.ws.rs.core.UriBuilder;
import javax.ws.rs.core.UriBuilderException;
import org.apache.cxf.jaxrs.model.URITemplate;
import org.apache.cxf.jaxrs.utils.HttpUtils;
import org.apache.cxf.jaxrs.utils.JAXRSUtils;
public class UriBuilderImpl extends UriBuilder {
private boolean encode;
private String scheme;
private String userInfo;
private int port;
private String host;
private List paths;
private String fragment;
private MultivaluedMap query = new MetadataMap();
private MultivaluedMap matrix = new MetadataMap();
/**
* Creates builder with empty URI.
*/
public UriBuilderImpl() {
}
/**
* Creates builder initialized with given URI.
*
* @param uri initial value for builder
* @throws IllegalArgumentException when uri is null
*/
public UriBuilderImpl(URI uri) throws IllegalArgumentException {
setUriParts(uri);
}
@Override
public URI build() throws UriBuilderException {
return build(new Object[]{});
}
@Override
public URI build(Map map) throws IllegalArgumentException, UriBuilderException {
return buildFromMap(map);
}
@Override
public URI build(Object... values) throws IllegalArgumentException, UriBuilderException {
if (encode) {
for (int i = 0; i < values.length; i++) {
values[i] = HttpUtils.encodePartiallyEncoded(values[i].toString(), false);
}
}
return doBuild(encode, values);
}
private URI doBuild(boolean fromEncoded, Object... values) {
try {
String thePath = buildPath(fromEncoded);
thePath = substituteVarargs(thePath, values);
return buildURI(fromEncoded, thePath);
} catch (URISyntaxException ex) {
throw new UriBuilderException("URI can not be built", ex);
}
}
private URI buildURI(boolean fromEncoded, String thePath) throws URISyntaxException {
String theQuery = buildQuery(fromEncoded);
// TODO : do encodePartiallyEncoded only once here, do not do it inside buildPath()
// buildFromEncoded and buildFromEncodedMap - we'll need to be careful such that
// path '/' seperators are not encoded so probably we'll need to create PathSegments
// again if fromEncoded is set
if (fromEncoded) {
StringBuilder b = new StringBuilder();
b.append(scheme).append("://");
if (userInfo != null) {
b.append(userInfo).append('@');
}
b.append(host);
if (port != -1) {
b.append(':').append(port);
}
if (thePath != null && thePath.length() > 0) {
b.append(thePath.startsWith("/") ? thePath : '/' + thePath);
}
if (theQuery != null && theQuery.length() != 0) {
b.append('?').append(theQuery);
}
if (fragment != null) {
b.append('#').append(fragment);
}
return new URI(b.toString());
} else {
return new URI(scheme, userInfo, host, port, thePath, theQuery, fragment);
}
}
private String substituteVarargs(String path, Object... values) {
Map varValueMap = new HashMap();
URITemplate templ = new URITemplate(path);
// vars in set are properly ordered due to linking in hash set
Set uniqueVars = new LinkedHashSet(templ.getVariables());
if (values.length < uniqueVars.size()) {
throw new IllegalArgumentException("Unresolved variables; only " + values.length
+ " value(s) given for " + uniqueVars.size()
+ " unique variable(s)");
}
int idx = 0;
for (String var : uniqueVars) {
Object oval = values[idx++];
varValueMap.put(var, oval.toString());
}
return templ.substitute(varValueMap);
}
private URI buildFromMap(Map map) throws IllegalArgumentException,
UriBuilderException {
if (encode) {
Map decodedMap = new HashMap(map.size());
for (Map.Entry entry : map.entrySet()) {
decodedMap.put(entry.getKey(),
HttpUtils.encodePartiallyEncoded(entry.getValue().toString(), false));
}
return doBuildFromMap(decodedMap, encode);
} else {
return doBuildFromMap(map, encode);
}
}
private URI doBuildFromMap(Map map, boolean fromEncoded)
throws IllegalArgumentException, UriBuilderException {
try {
String thePath = buildPath(fromEncoded);
thePath = substituteMapped(thePath, map);
return buildURI(fromEncoded, thePath);
} catch (URISyntaxException ex) {
throw new UriBuilderException("URI can not be built", ex);
}
}
private String substituteMapped(String path, Map varValueMap) {
URITemplate templ = new URITemplate(path);
Set uniqueVars = new HashSet(templ.getVariables());
if (varValueMap.size() < uniqueVars.size()) {
throw new IllegalArgumentException("Unresolved variables; only " + varValueMap.size()
+ " value(s) given for " + uniqueVars.size()
+ " unique variable(s)");
}
return templ.substitute(varValueMap);
}
// CHECKSTYLE:OFF
@Override
public UriBuilder clone() {
UriBuilderImpl builder = new UriBuilderImpl();
builder.scheme = scheme;
builder.userInfo = userInfo;
builder.port = port;
builder.host = host;
builder.paths = new ArrayList(paths);
builder.fragment = fragment;
builder.query = new MetadataMap(query);
builder.matrix = new MetadataMap(matrix);
builder.encode = encode;
return builder;
}
// CHECKSTYLE:ON
@Override
public UriBuilder encode(boolean enable) {
this.encode = enable;
return this;
}
@Override
public UriBuilder fragment(String theFragment) throws IllegalArgumentException {
this.fragment = theFragment;
return this;
}
@Override
public UriBuilder host(String theHost) throws IllegalArgumentException {
this.host = theHost;
return this;
}
@Override
public UriBuilder matrixParam(String name, String value) throws IllegalArgumentException {
if (name == null) {
throw new IllegalArgumentException("name is null");
}
if (value != null) {
matrix.add(name, value);
} else {
matrix.remove(name);
}
return this;
}
@Override
public UriBuilder path(String... segments) throws IllegalArgumentException {
if (paths == null) {
paths = new ArrayList();
}
for (String seg : segments) {
paths.addAll(JAXRSUtils.getPathSegments(seg, false));
}
matrix.clear();
if (!paths.isEmpty()) {
matrix = paths.get(paths.size() - 1).getMatrixParameters();
}
return this;
}
@SuppressWarnings("unchecked")
@Override
public UriBuilder path(Class resource) throws IllegalArgumentException {
if (resource == null) {
throw new IllegalArgumentException("resource is null");
}
Annotation ann = resource.getAnnotation(Path.class);
if (ann == null) {
throw new IllegalArgumentException("Class '" + resource.getCanonicalName()
+ "' is not annotated with Path");
}
// path(String) decomposes multi-segment path when necessary
return path(((Path)ann).value());
}
@Override
public UriBuilder path(Method... methods) throws IllegalArgumentException {
if (methods == null) {
throw new IllegalArgumentException("methods is null");
}
for (Method method : methods) {
if (method == null) {
throw new IllegalArgumentException("method is null");
}
Annotation ann = method.getAnnotation(Path.class);
if (ann == null) {
throw new IllegalArgumentException("Method '" + method.getClass().getCanonicalName() + "."
+ method.getName() + "' is not annotated with Path");
}
// path(String) decomposes multi-segment path when necessary
path(((Path)ann).value());
}
return this;
}
@Override
public UriBuilder path(Class resource, String method) throws IllegalArgumentException {
if (resource == null) {
throw new IllegalArgumentException("resource is null");
}
if (method == null) {
throw new IllegalArgumentException("method is null");
}
Annotation foundAnn = null;
for (Method meth : resource.getMethods()) {
if (meth.getName().equals(method)) {
Annotation ann = meth.getAnnotation(Path.class);
if (foundAnn != null && ann != null) {
throw new IllegalArgumentException("Multiple Path annotations for '" + method
+ "' overloaded method");
}
foundAnn = ann;
}
}
if (foundAnn == null) {
throw new IllegalArgumentException("No Path annotation for '" + method + "' method");
}
// path(String) decomposes multi-segment path when necessary
return path(((Path)foundAnn).value());
}
@Override
public UriBuilder port(int thePort) throws IllegalArgumentException {
this.port = thePort;
return this;
}
@Override
public UriBuilder queryParam(String name, String value) throws IllegalArgumentException {
if (name == null) {
throw new IllegalArgumentException("name is null");
}
if (value != null) {
query.add(name, value);
} else {
query.remove(name);
}
return this;
}
public UriBuilder replaceMatrixParams(String m) throws IllegalArgumentException {
if (m == null) {
throw new IllegalArgumentException("name is null");
}
MultivaluedMap values = JAXRSUtils.getStructuredParams(m, ";", true);
for (Map.Entry> entry : values.entrySet()) {
matrix.put(entry.getKey(), entry.getValue());
}
return this;
}
@Override
public UriBuilder replaceQueryParams(String q) throws IllegalArgumentException {
if (q == null) {
throw new IllegalArgumentException("name is null");
}
MultivaluedMap values = JAXRSUtils.getStructuredParams(q, "&", true);
for (Map.Entry> entry : values.entrySet()) {
query.put(entry.getKey(), entry.getValue());
}
return this;
}
@Override
public UriBuilder scheme(String s) throws IllegalArgumentException {
scheme = s;
return this;
}
@Override
public UriBuilder schemeSpecificPart(String ssp) throws IllegalArgumentException {
// scheme-specific part is whatever after ":" of URI
// see: http://en.wikipedia.org/wiki/URI_scheme
try {
if (scheme == null) {
scheme = "http";
}
URI uri = new URI(scheme + "://" + ssp);
setUriParts(uri);
} catch (URISyntaxException e) {
throw new IllegalArgumentException("Wrong syntax of scheme-specific part", e);
}
return this;
}
@Override
public UriBuilder uri(URI uri) throws IllegalArgumentException {
setUriParts(uri);
return this;
}
@Override
public UriBuilder userInfo(String ui) throws IllegalArgumentException {
this.userInfo = ui;
return this;
}
@Override
public boolean isEncode() {
throw new UnsupportedOperationException("Not implemented :/");
}
@Override
public UriBuilder extension(String arg0) {
throw new UnsupportedOperationException("Not implemented :/");
}
@Override
public UriBuilder replacePath(String... path) throws IllegalArgumentException {
for (String p : path) {
paths = JAXRSUtils.getPathSegments(p, false);
}
if (!paths.isEmpty()) {
matrix = paths.get(paths.size() - 1).getMatrixParameters();
} else {
matrix.clear();
}
return this;
}
private void setUriParts(URI uri) {
if (uri == null) {
throw new IllegalArgumentException("uri is null");
}
scheme = uri.getScheme();
port = uri.getPort();
host = uri.getHost();
setPathAndMatrix(uri.getRawPath());
fragment = uri.getFragment();
query = JAXRSUtils.getStructuredParams(uri.getRawQuery(), "&", false);
userInfo = uri.getUserInfo();
}
private void setPathAndMatrix(String path) {
paths = JAXRSUtils.getPathSegments(path, false, false);
if (!paths.isEmpty()) {
matrix = paths.get(paths.size() - 1).getMatrixParameters();
} else {
matrix.clear();
}
}
private String buildPath(boolean fromEncoded) {
StringBuilder sb = new StringBuilder();
Iterator iter = paths.iterator();
while (iter.hasNext()) {
PathSegment ps = iter.next();
String p = ps.getPath();
if (p.length() != 0 || !iter.hasNext()) {
p = fromEncoded ? new URITemplate(p).encodeLiteralCharacters() : p;
if (!p.startsWith("/")) {
sb.append('/');
}
sb.append(p);
if (iter.hasNext()) {
buildMatrix(sb, ps.getMatrixParameters(), fromEncoded);
}
}
}
buildMatrix(sb, matrix, fromEncoded);
return sb.toString();
}
private String buildQuery(boolean fromEncoded) {
return buildParams(query, '&', fromEncoded);
}
/**
* Builds param string for query part or matrix part of URI.
*
* @param map query or matrix multivalued map
* @param separator params separator, '&' for query ';' for matrix
* @param fromEncoded if true then values will be decoded
* @return stringified params.
*/
private String buildParams(MultivaluedMap map, char separator,
boolean fromEncoded) {
boolean isQuery = separator == '&';
StringBuilder b = new StringBuilder();
for (Iterator>> it = map.entrySet().iterator(); it.hasNext();) {
Map.Entry> entry = it.next();
for (Iterator sit = entry.getValue().iterator(); sit.hasNext();) {
String val = sit.next();
if (fromEncoded) {
val = HttpUtils.encodePartiallyEncoded(val, isQuery);
}
b.append(entry.getKey()).append('=').append(val);
if (sit.hasNext() || it.hasNext()) {
b.append(separator);
}
}
}
return b.length() > 0 ? b.toString() : null;
}
/**
* Builds param string for matrix part of URI.
*
* @param sb buffer to add the matrix part to, will get ';' added if map is not empty
* @param map matrix multivalued map
*/
private void buildMatrix(StringBuilder sb, MultivaluedMap map,
boolean fromEncoded) {
if (!map.isEmpty()) {
sb.append(';');
sb.append(buildParams(map, ';', fromEncoded));
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy