nbbrd.io.http.URLQueryBuilder Maven / Gradle / Ivy
/*
* Copyright 2017 National Bank of Belgium
*
* Licensed under the EUPL, Version 1.1 or - as soon they will be approved
* by the European Commission - subsequent versions of the EUPL (the "Licence");
* You may not use this work except in compliance with the Licence.
* You may obtain a copy of the Licence at:
*
* http://ec.europa.eu/idabc/eupl
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the Licence is distributed on an "AS IS" basis,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the Licence for the specific language governing permissions and
* limitations under the Licence.
*/
package nbbrd.io.http;
import lombok.AccessLevel;
import lombok.NonNull;
import nbbrd.design.StaticFactoryMethod;
import java.io.UncheckedIOException;
import java.io.UnsupportedEncodingException;
import java.net.MalformedURLException;
import java.net.URL;
import java.net.URLEncoder;
import java.nio.charset.StandardCharsets;
import java.util.*;
/**
* @author Philippe Charles
*/
@lombok.RequiredArgsConstructor(access = AccessLevel.PRIVATE)
public final class URLQueryBuilder {
@StaticFactoryMethod
public static @NonNull URLQueryBuilder of(@NonNull URL endPoint) {
return new URLQueryBuilder(endPoint);
}
@lombok.NonNull
private final URL endPoint;
private boolean trailingSlash = false;
private final List paths = new ArrayList<>();
private final Map params = new LinkedHashMap<>();
/**
* Appends a trailing slash to the final URL.
*
* @param trailingSlash specify if a trailing slash is required
* @return this builder
* @see https://en.wikipedia.org/wiki/URI_normalization
*/
@NonNull
public URLQueryBuilder trailingSlash(boolean trailingSlash) {
this.trailingSlash = trailingSlash;
return this;
}
/**
* Appends the specified path to the current URL.
*
* @param path a non-null path
* @return this builder
* @throws NullPointerException if path is null
*/
@NonNull
public URLQueryBuilder path(@NonNull String path) {
paths.add(path);
return this;
}
/**
* Appends the specified path to the current URL.
*
* @param path a non-null path
* @return this builder
* @throws NullPointerException if path is null
*/
@NonNull
public URLQueryBuilder path(@NonNull List path) {
paths.addAll(path);
return this;
}
/**
* Appends the specified parameter to the current URL.
*
* @param key a non-null key
* @param value a non-null value
* @return this builder
* @throws NullPointerException if key or value is null
*/
@NonNull
public URLQueryBuilder param(@NonNull String key, @NonNull String value) {
params.put(key, value);
return this;
}
/**
* Appends the specified parameter to the current URL.
*
* @param key a non-null key
* @return this builder
* @throws NullPointerException if key or value is null
*/
@NonNull
public URLQueryBuilder param(@NonNull String key) {
params.put(key, null);
return this;
}
@Override
public String toString() {
StringBuilder result = new StringBuilder();
String endPointAsString = endPoint.toString();
if (!paths.isEmpty() && endPointAsString.charAt(endPointAsString.length() - 1) == '/') {
result.append(endPointAsString, 0, endPointAsString.length() - 1);
} else {
result.append(endPointAsString);
}
for (String path : paths) {
result.append('/').append(encode(path));
}
if (trailingSlash) {
result.append('/');
}
Iterator> paramsIterator = params.entrySet().iterator();
if (paramsIterator.hasNext()) {
result.append('?');
appendParam(result, paramsIterator.next());
while (paramsIterator.hasNext()) {
result.append('&');
appendParam(result, paramsIterator.next());
}
}
return result.toString();
}
private void appendParam(StringBuilder result, Map.Entry o) {
result.append(encode(o.getKey()));
String value = o.getValue();
if (value != null) {
result.append('=').append(encode(value));
}
}
/**
* Creates a new URL using the specified path and parameters.
*
* @return a new URL
* @throws MalformedURLException if no protocol is specified, or an unknown protocol is found, or spec is null,
* or the parsed URL fails to comply with the specific syntax of the associated protocol.
*/
@NonNull
public URL build() throws MalformedURLException {
return new URL(toString());
}
private static String encode(String s) {
try {
return URLEncoder.encode(s, StandardCharsets.UTF_8.name()).replace("+", "%20");
} catch (UnsupportedEncodingException ex) {
throw new UncheckedIOException(ex);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy