io.gravitee.am.common.web.UriBuilder Maven / Gradle / Ivy
/**
* Copyright (C) 2015 The Gravitee team (http://gravitee.io)
*
* 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 io.gravitee.am.common.web;
import io.gravitee.am.common.oauth2.Parameters;
import io.gravitee.am.common.utils.ConstantKeys;
import java.net.URI;
import java.net.URISyntaxException;
import java.net.URLDecoder;
import java.net.URLEncoder;
import java.nio.charset.StandardCharsets;
import java.util.Map;
import java.util.TreeMap;
import java.util.function.BiConsumer;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
/**
* @author Titouan COMPIEGNE (titouan.compiegne at graviteesource.com)
* @author Alexandre FARIA (contact at alexandrefaria.net)
* @author GraviteeSource Team
*/
public class UriBuilder {
private static final String SCHEME_REGEX = "([^:/?#]+):";
private static final String HTTP_REGEX = "(?i)(http|https):";
private static final String USERINFO_REGEX = "([^@\\[/?#]*)";
private static final String HOST_IPV4_REGEX = "[^\\[/?#:]*";
private static final String HOST_IPV6_REGEX = "\\[[\\p{XDigit}\\:\\.]*[%\\p{Alnum}]*\\]";
private static final String HOST_REGEX = "(" + HOST_IPV6_REGEX + "|" + HOST_IPV4_REGEX + ")";
private static final String PORT_REGEX = "(\\d*(?:\\{[^/]+?\\})?)";
private static final String PATH_REGEX = "([^?#]*)";
private static final String QUERY_REGEX = "([^#]*)";
private static final String LAST_REGEX = "(.*)";
// Regex patterns that matches URIs. See RFC 3986, appendix B
private static final Pattern URI_PATTERN = Pattern.compile(
"^(" + SCHEME_REGEX + ")?" + "(//(" + USERINFO_REGEX + "@)?" + HOST_REGEX + "(:" + PORT_REGEX +
")?" + ")?" + PATH_REGEX + "(\\?" + QUERY_REGEX + ")?" + "(#" + LAST_REGEX + ")?");
private static final Pattern HTTP_URL_PATTERN = Pattern.compile(
"^" + HTTP_REGEX + "(//(" + USERINFO_REGEX + "@)?" + HOST_REGEX + "(:" + PORT_REGEX +
")?" + ")?" + PATH_REGEX + "(\\?" + QUERY_REGEX + ")?" + "(#" + LAST_REGEX + ")?");
private static final Pattern HTTP_PATTERN = Pattern.compile(HTTP_REGEX.replace(":", ""));
private static final String LOCALHOST_HOST_REGEX = "^localhost$";
private static final String LOCALHOST_IPV4_REGEX = "^127(?:\\.[0-9]+){0,2}\\.[0-9]+$";
private static final String LOCALHOST_IPV6_REGEX = "^(?:0*\\:)*?:?0*1$";
private static final Pattern LOCALHOST_PATTERN = Pattern.compile(
LOCALHOST_HOST_REGEX + "|" + LOCALHOST_IPV4_REGEX + "|" + LOCALHOST_IPV6_REGEX);
private String scheme;
private String host;
private int port = -1;
private String userInfo;
private String path;
private String query;
private String fragment;
public static UriBuilder newInstance() {
return new UriBuilder();
}
public static UriBuilder fromURIString(String uri) {
Matcher matcher = URI_PATTERN.matcher(uri);
if (matcher.matches()) {
UriBuilder builder = new UriBuilder();
String scheme = matcher.group(2);
String userInfo = matcher.group(5);
String host = matcher.group(6);
String port = matcher.group(8);
String path = matcher.group(9);
String query = matcher.group(11);
String fragment = matcher.group(13);
builder.scheme(scheme);
builder.host(host);
if (port != null && !port.isEmpty()) {
builder.port(Integer.parseInt(port));
}
builder.userInfo(userInfo);
builder.path(path);
builder.query(query);
builder.fragment(fragment);
return builder;
} else {
throw new IllegalArgumentException("[" + uri + "] is not a valid URI");
}
}
public static UriBuilder fromHttpUrl(String httpUrl) {
Matcher matcher = HTTP_URL_PATTERN.matcher(httpUrl);
if (matcher.matches()) {
UriBuilder builder = new UriBuilder();
String scheme = matcher.group(1);
builder.scheme(scheme != null ? scheme.toLowerCase() : null);
builder.userInfo(matcher.group(4));
String host = matcher.group(5);
if ((scheme != null && !scheme.isEmpty()) && (host == null || host.isEmpty())) {
throw new IllegalArgumentException("[" + httpUrl + "] is not a valid HTTP URL");
}
builder.host(host);
String port = matcher.group(7);
if (port != null && !port.isEmpty()) {
builder.port(Integer.parseInt(port));
}
builder.path(matcher.group(8));
builder.query(matcher.group(10));
builder.fragment(matcher.group(12));
return builder;
} else {
throw new IllegalArgumentException("[" + httpUrl + "] is not a valid HTTP URL");
}
}
/**
* Convert a String to the application/x-www-form-urlencoded MIME format
*/
public static String encodeURIComponent(String s) {
if (s == null) {
return null;
}
return URLEncoder.encode(s, StandardCharsets.UTF_8);
}
public static String decodeURIComponent(String s) {
try {
return URLDecoder.decode(s, StandardCharsets.UTF_8);
} catch (Exception e) {
return s;
}
}
public UriBuilder scheme(String scheme) {
this.scheme = scheme;
return this;
}
public UriBuilder host(String host) {
this.host = host;
return this;
}
public UriBuilder port(int port) {
this.port = port;
return this;
}
public UriBuilder userInfo(String userInfo) {
this.userInfo = userInfo;
return this;
}
public UriBuilder path(String path) {
this.path = path;
return this;
}
public UriBuilder query(String query) {
this.query = query;
return this;
}
public UriBuilder fragment(String fragment) {
this.fragment = fragment;
return this;
}
public UriBuilder parameters(Map parameters) {
if (parameters != null) {
parameters.forEach(this::addParameter);
}
return this;
}
public UriBuilder parameters(Iterable> parameters) {
if (parameters != null) {
parameters.forEach(e -> addParameter(e.getKey(), e.getValue()));
}
return this;
}
public UriBuilder addParameter(String parameter, String value) {
if (query == null) {
query = "";
}
if (!query.isEmpty()) {
query += "&";
}
query += parameter + "=" + value;
return this;
}
/**
* Add a query param if the value isn't null, otherwise return this builder unmodified
*/
public UriBuilder addNotNullParameter(String parameter, String value) {
if (value == null) {
return this;
}
return addParameter(parameter, value);
}
/**
* Add a param to the fragment if the value isn't null, otherwise return this builder unmodified
*/
public UriBuilder addNotNullFragmentParameter(String parameter, String value) {
if (value == null) {
return this;
}
return addFragmentParameter(parameter, value);
}
/**
* Add a query param if the value isn't null, otherwise return this builder unmodified
*/
public UriBuilder addParameterIfHasValue(String paremeter, String value) {
if (value != null) {
return addParameter(paremeter, value);
} else {
return this;
}
}
public UriBuilder addFragmentParameter(String parameter, String value) {
if (fragment == null) {
fragment = "";
}
if (!fragment.isEmpty()) {
fragment += "&";
}
fragment += parameter + "=" + value;
return this;
}
public URI build() throws URISyntaxException {
return new URI(scheme, userInfo, host, port, path, query, fragment);
}
public String buildString() {
final StringBuilder sb = new StringBuilder();
if (this.scheme != null) {
sb.append(this.scheme).append(':');
}
if (this.host != null) {
sb.append("//");
if (this.userInfo != null) {
sb.append(this.userInfo).append("@");
}
sb.append(this.host);
if (this.port >= 0) {
sb.append(":").append(this.port);
}
}
if (this.path != null) {
sb.append(this.path);
}
if (this.query != null) {
sb.append("?").append(this.query);
}
if (this.fragment != null) {
sb.append("#").append(this.fragment);
}
return sb.toString();
}
public String buildRootUrl() {
final StringBuilder sb = new StringBuilder();
if (this.scheme != null) {
sb.append(this.scheme).append(':');
}
if (this.host != null) {
sb.append("//");
sb.append(this.host);
if (this.port >= 0) {
sb.append(":").append(this.port);
}
}
return sb.toString();
}
public static boolean isHttp(String scheme) {
return HTTP_PATTERN.matcher(scheme.toLowerCase()).matches();
}
public static boolean isLocalhost(String host) {
return LOCALHOST_PATTERN.matcher(host.toLowerCase()).matches();
}
public static String buildErrorRedirect(String baseRedirectUri, ErrorInfo error, boolean fragment, Map extraParams) throws URISyntaxException {
final URI redirectUri = UriBuilder.fromURIString(baseRedirectUri).build();
var parameters = new TreeMap();
if (error.error() != null) {
parameters.put(ConstantKeys.ERROR_PARAM_KEY, error.error());
}
if (error.code() != null) {
parameters.put(ConstantKeys.ERROR_CODE_PARAM_KEY, error.code());
}
if (error.description() != null) {
parameters.put(ConstantKeys.ERROR_DESCRIPTION_PARAM_KEY, error.description());
}
if (error.state() != null) {
parameters.put(Parameters.STATE, error.state());
}
parameters.putAll(extraParams);
// create final redirect uri
final UriBuilder finalRedirectUri = UriBuilder.fromURIString(redirectUri.toString());
BiConsumer addParameter = fragment
? (k, v) -> finalRedirectUri.addNotNullFragmentParameter(k, encodeURIComponent(v))
: (k, v) -> finalRedirectUri.addNotNullParameter(k, encodeURIComponent(v));
parameters.forEach(addParameter);
return finalRedirectUri.buildString();
}
public static String buildErrorRedirect(String baseRedirectUri, ErrorInfo error, boolean fragment) throws URISyntaxException {
return buildErrorRedirect(baseRedirectUri, error, fragment, Map.of());
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy