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

com.nimbusds.oauth2.sdk.util.URLUtils Maven / Gradle / Ivy

/*
 * oauth2-oidc-sdk
 *
 * Copyright 2012-2016, Connect2id Ltd and contributors.
 *
 * 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.nimbusds.oauth2.sdk.util;


import java.io.UnsupportedEncodingException;
import java.net.*;
import java.util.*;


/**
 * URL operations.
 */
public final class URLUtils {

	
	/**
	 * The default UTF-8 character set.
	 */
	public static final String CHARSET = "utf-8";
	
	
	/**
	 * Gets the base part (protocol, host, port and path) of the specified
	 * URL.
	 *
	 * @param url The URL. May be {@code null}.
	 *
	 * @return The base part of the URL, {@code null} if the original URL 
	 *         is {@code null} or doesn't specify a protocol.
	 */
	public static URL getBaseURL(final URL url) {
	
		if (url == null)
			return null;
		
		try {
			return new URL(url.getProtocol(), url.getHost(), url.getPort(), url.getPath());
			
		} catch (MalformedURLException e) {
		
			return null;
		}
	}


	/**
	 * Sets the encoded query of the specified URL.
	 *
	 * @param url   The URL. May be {@code null}.
	 * @param query The encoded query, {@code null} if not specified.
	 *
	 * @return The new URL.
	 */
	public static URL setEncodedQuery(final URL url, final String query) {

		if (url == null) {
			return null;
		}

		try {
			URI uri = url.toURI();
			StringBuilder sb = new StringBuilder(URIUtils.getBaseURI(uri).toString());
			if (query != null && ! query.isEmpty()) {
				sb.append('?');
				sb.append(query);
			}
			if (uri.getRawFragment() != null) {
				sb.append('#');
				sb.append(uri.getRawFragment());

			}
			return new URL(sb.toString());
		} catch (MalformedURLException | URISyntaxException e) {
			throw new IllegalArgumentException(e);
		}
        }


	/**
	 * Sets the encoded fragment of the specified URL.
	 *
	 * @param url      The URL. May be {@code null}.
	 * @param fragment The encoded fragment, {@code null} if not specified.
	 *
	 * @return The new URL.
	 */
	public static URL setEncodedFragment(final URL url, final String fragment) {

		if (url == null) {
			return null;
		}

		try {
			URI uri = url.toURI();
			StringBuilder sb = new StringBuilder(URIUtils.getBaseURI(uri).toString());
			if (uri.getRawQuery() != null) {
				sb.append('?');
				sb.append(uri.getRawQuery());
			}
			if (fragment != null && ! fragment.isEmpty()) {
				sb.append('#');
				sb.append(fragment);

			}
			return new URL(sb.toString());
		} catch (MalformedURLException | URISyntaxException e) {
			throw new IllegalArgumentException(e);
		}
	}


	/**
	 * Performs {@code application/x-www-form-urlencoded} encoding on the
	 * specified parameter keys and values.
	 *
	 * @param params A map of the parameters. May be empty or {@code null}.
	 *
	 * @return The encoded parameters, {@code null} if not specified.
	 */
	public static Map> urlEncodeParameters(final Map> params) {

		if (MapUtils.isEmpty(params)) {
			return params;
		}

		Map> out = new LinkedHashMap<>(); // preserve order

		for (Map.Entry> entry: params.entrySet()) {

			try {
				String newKey = entry.getKey() != null ? URLEncoder.encode(entry.getKey(), CHARSET) : null;

				List newValues;

				if (entry.getValue() != null) {

					newValues = new LinkedList<>();

					for (String value : entry.getValue()) {

						if (value != null) {
							newValues.add(URLEncoder.encode(value, CHARSET));
						} else {
							newValues.add(null); // preserve null values
						}
					}
				} else {
					newValues = null;
				}

				out.put(newKey, newValues);

			} catch (UnsupportedEncodingException e) {
				// UTF-8 must always be supported
				throw new RuntimeException(e);
			}
		}

		return out;
	}
	
	
	/**
	 * Serialises the specified map of parameters into a URL query string. 
	 * The parameter keys and values are 
	 * {@code application/x-www-form-urlencoded} encoded.
	 *
	 * 

Parameters with {@code null} keys or values are ignored and not * serialised. * *

Note that the '?' character preceding the query string in GET * requests is not included in the returned string. * *

Example query string: * *

	 * response_type=code
	 * &client_id=s6BhdRkqt3
	 * &state=xyz
	 * &redirect_uri=https%3A%2F%2Fclient%2Eexample%2Ecom%2Fcb
	 * 
* *

The opposite method is {@link #parseParameters}. * * @param params A map of the URL query parameters. May be empty or * {@code null}. * * @return The serialised URL query string, empty if no parameters. */ public static String serializeParameters(final Map> params) { if (params == null || params.isEmpty()) return ""; Map> encodedParams = urlEncodeParameters(params); StringBuilder sb = new StringBuilder(); for (Map.Entry> entry: encodedParams.entrySet()) { if (entry.getKey() == null || entry.getValue() == null) continue; for (String value: entry.getValue()) { if (value == null) { value = ""; } if (sb.length() > 0) sb.append('&'); sb.append(entry.getKey()); sb.append('='); sb.append(value); } } return sb.toString(); } /** * Serialises the specified map of parameters into a URL query string. * Supports multiple key / value pairs that have the same key. The * parameter keys and values are * {@code application/x-www-form-urlencoded} encoded. * *

Parameters with {@code null} keys or values are ignored and not * serialised. * *

Note that the '?' character preceding the query string in GET * requests is not included in the returned string. * *

Example query string: * *

	 * response_type=code
	 * &client_id=s6BhdRkqt3
	 * &state=xyz
	 * &redirect_uri=https%3A%2F%2Fclient%2Eexample%2Ecom%2Fcb
	 * 
* *

The opposite method is {@link #parseParameters}. * * @param params A map of the URL query parameters. May be empty or * {@code null}. * * @return The serialised URL query string, empty if no parameters. */ public static String serializeParametersAlt(final Map params) { if (params == null) { return serializeParameters(null); } Map> out = new HashMap<>(); for (Map.Entry entry: params.entrySet()) { if (entry.getValue() == null) { out.put(entry.getKey(), null); } else { out.put(entry.getKey(), Arrays.asList(entry.getValue())); } } return serializeParameters(out); } /** * Parses the specified URL query string into a parameter map. If a * parameter has multiple values only the first one will be saved. The * parameter keys and values are * {@code application/x-www-form-urlencoded} decoded. * *

Note that the '?' character preceding the query string in GET * requests must not be included. * *

Example query string: * *

	 * response_type=code
	 * &client_id=s6BhdRkqt3
	 * &state=xyz
	 * &redirect_uri=https%3A%2F%2Fclient%2Eexample%2Ecom%2Fcb
	 * 
* *

The opposite method {@link #serializeParameters}. * * @param query The URL query string to parse. May be {@code null}. * * @return A map of the URL query parameters, empty if none are found. */ public static Map> parseParameters(final String query) { Map> params = new LinkedHashMap<>(); if (StringUtils.isBlank(query)) { return params; // empty map } StringTokenizer st = new StringTokenizer(query.trim(), "&"); while(st.hasMoreTokens()) { String param = st.nextToken(); String[] pair = param.split("=", 2); // Split around the first '=', see issue #169 String key, value; try { key = URLDecoder.decode(pair[0], CHARSET); value = pair.length > 1 ? URLDecoder.decode(pair[1], CHARSET) : ""; } catch (UnsupportedEncodingException e) { // UTF-8 should always be supported continue; } catch (Exception e) { // Handle "IllegalArgumentException: URLDecoder: Incomplete trailing escape (%) pattern", etc continue; } if (params.containsKey(key)) { // Append value List updatedValueList = new LinkedList<>(params.get(key)); updatedValueList.add(value); params.put(key, Collections.unmodifiableList(updatedValueList)); } else { params.put(key, Collections.singletonList(value)); } } return params; } /** * Prevents public instantiation. */ private URLUtils() {} }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy