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

com.google.api.client.http.GenericUrl Maven / Gradle / Ivy

Go to download

Google API Client Library for Java. Supports Java 5 (or higher) desktop (SE) and web (EE), Android, and Google App Engine.

There is a newer version: 1.4.1-beta
Show newest version
/*
 * Copyright (c) 2010 Google Inc.
 *
 * 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.google.api.client.http;

import com.google.api.client.escape.CharEscapers;
import com.google.api.client.escape.Escaper;
import com.google.api.client.escape.PercentEscaper;
import com.google.api.client.util.GenericData;
import com.google.api.client.util.Key;

import java.net.URI;
import java.net.URISyntaxException;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.Iterator;
import java.util.List;
import java.util.Map;

/**
 * URL builder in which the query parameters are specified as generic data
 * key/value pairs, based on the specification RFC 3986: Uniform Resource
 * Identifier (URI).
 * 

* The query parameters are specified with the data key name as the parameter * name, and the data value as the parameter value. Subclasses can declare * fields for known query parameters using the {@link Key} annotation. {@code * null} parameter names are not allowed, but {@code null} query values are * allowed. *

*

* Query parameter values are parsed using * {@link UrlEncodedParser#parse(String, Object)}. *

* * @since 1.0 * @author Yaniv Inbar */ public class GenericUrl extends GenericData { private static final Escaper URI_FRAGMENT_ESCAPER = new PercentEscaper("=&-_.!~*'()@:$,;/?:", false); /** Scheme (lowercase), for example {@code "https"}. */ public String scheme; /** Host, for example {@code "www.google.com"}. */ public String host; /** Port number or {@code -1} if undefined, for example {@code 443}. */ public int port = -1; /** * Decoded path component by parts with each part separated by a {@code '/'} * or {@code null} for none, for example {@code * "/m8/feeds/contacts/default/full"} is represented by {@code "", "m8", * "feeds", "contacts", "default", "full"}. *

* Use {@link #appendRawPath(String)} to append to the path, which ensures * that no extra slash is added. */ public List pathParts; /** Fragment component or {@code null} for none. */ public String fragment; public GenericUrl() { } /** * Constructs from an encoded URL. *

* Any known query parameters with pre-defined fields as data keys will be * parsed based on their data type. Any unrecognized query parameter will * always be parsed as a string. * * @param encodedUrl encoded URL, including any existing query parameters that * should be parsed * @throws IllegalArgumentException if URL has a syntax error */ public GenericUrl(String encodedUrl) { URI uri; try { uri = new URI(encodedUrl); } catch (URISyntaxException e) { throw new IllegalArgumentException(e); } this.scheme = uri.getScheme().toLowerCase(); this.host = uri.getHost(); this.port = uri.getPort(); this.pathParts = toPathParts(uri.getRawPath()); this.fragment = uri.getFragment(); String query = uri.getRawQuery(); if (query != null) { UrlEncodedParser.parse(query, this); } } @Override public int hashCode() { // TODO: optimize? return build().hashCode(); } @Override public boolean equals(Object obj) { if (this == obj) { return true; } if (!super.equals(obj) || !(obj instanceof GenericUrl)) { return false; } GenericUrl other = (GenericUrl) obj; // TODO: optimize? return build().equals(other.toString()); } @Override public String toString() { return build(); } @Override public GenericUrl clone() { GenericUrl result = (GenericUrl) super.clone(); result.pathParts = new ArrayList(this.pathParts); return result; } /** * Constructs the string representation of the URL, including the path * specified by {@link #pathParts} and the query parameters specified by this * generic URL. */ public final String build() { // scheme, host, port, and path StringBuilder buf = new StringBuilder(); buf.append(this.scheme).append("://").append(this.host); int port = this.port; if (port != -1) { buf.append(':').append(port); } List pathParts = this.pathParts; if (pathParts != null) { appendRawPathFromParts(buf); } // query parameters (similar to UrlEncodedContent) boolean first = true; for (Map.Entry nameValueEntry : entrySet()) { Object value = nameValueEntry.getValue(); if (value != null) { String name = CharEscapers.escapeUriQuery(nameValueEntry.getKey()); if (value instanceof Collection) { Collection collectionValue = (Collection) value; for (Object repeatedValue : collectionValue) { first = appendParam(first, buf, name, repeatedValue); } } else { first = appendParam(first, buf, name, value); } } } // URL fragment String fragment = this.fragment; if (fragment != null) { buf.append('#').append(URI_FRAGMENT_ESCAPER.escape(fragment)); } return buf.toString(); } /** * Returns the first query parameter value for the given query parameter name. * * @param name query parameter name * @return first query parameter value */ public Object getFirst(String name) { Object value = get(name); if (value instanceof Collection) { @SuppressWarnings("unchecked") Collection collectionValue = (Collection) value; Iterator iterator = collectionValue.iterator(); return iterator.hasNext() ? iterator.next() : null; } return value; } /** * Returns all query parameter values for the given query parameter name. * * @param name query parameter name * @return unmodifiable collection of query parameter values (possibly empty) */ public Collection getAll(String name) { Object value = get(name); if (value == null) { return Collections.emptySet(); } if (value instanceof Collection) { @SuppressWarnings("unchecked") Collection collectionValue = (Collection) value; return Collections.unmodifiableCollection(collectionValue); } return Collections.singleton(value); } /** * Returns the raw encoded path computed from the {@link #pathParts}. * * @return raw encoded path computed from the {@link #pathParts} or {@code * null} if {@link #pathParts} is {@code null} */ public String getRawPath() { List pathParts = this.pathParts; if (pathParts == null) { return null; } StringBuilder buf = new StringBuilder(); appendRawPathFromParts(buf); return buf.toString(); } /** * Sets the {@link #pathParts} from the given raw encoded path. * * @param encodedPath raw encoded path or {@code null} to set * {@link #pathParts} to {@code null} */ public void setRawPath(String encodedPath) { this.pathParts = toPathParts(encodedPath); } /** * Appends the given raw encoded path to the current {@link #pathParts}, * setting field only if it is {@code null} or empty. *

* The last part of the {@link #pathParts} is merged with the first part of * the path parts computed from the given encoded path. Thus, if the current * raw encoded path is {@code "a"}, and the given encoded path is {@code "b"}, * then the resulting raw encoded path is {@code "ab"}. * * @param encodedPath raw encoded path or {@code null} to ignore */ public void appendRawPath(String encodedPath) { if (encodedPath != null && encodedPath.length() != 0) { List pathParts = this.pathParts; List appendedPathParts = toPathParts(encodedPath); if (pathParts == null || pathParts.isEmpty()) { this.pathParts = appendedPathParts; } else { int size = pathParts.size(); pathParts.set( size - 1, pathParts.get(size - 1) + appendedPathParts.get(0)); pathParts.addAll( appendedPathParts.subList(1, appendedPathParts.size())); } } } /** * Returns the decoded path parts for the given encoded path. * * @param encodedPath slash-prefixed encoded path, for example {@code * "/m8/feeds/contacts/default/full"} * @return decoded path parts, with each part assumed to be preceded by a * {@code '/'}, for example {@code "", "m8", "feeds", "contacts", * "default", "full"}, or {@code null} for {@code null} or {@code ""} * input */ public static List toPathParts(String encodedPath) { if (encodedPath == null || encodedPath.length() == 0) { return null; } List result = new ArrayList(); int cur = 0; boolean notDone = true; while (notDone) { int slash = encodedPath.indexOf('/', cur); notDone = slash != -1; String sub; if (notDone) { sub = encodedPath.substring(cur, slash); } else { sub = encodedPath.substring(cur); } result.add(CharEscapers.decodeUri(sub)); cur = slash + 1; } return result; } private void appendRawPathFromParts(StringBuilder buf) { List pathParts = this.pathParts; int size = pathParts.size(); for (int i = 0; i < size; i++) { String pathPart = pathParts.get(i); if (i != 0) { buf.append('/'); } if (pathPart.length() != 0) { buf.append(CharEscapers.escapeUriPath(pathPart)); } } } private static boolean appendParam( boolean first, StringBuilder buf, String name, Object value) { if (first) { first = false; buf.append('?'); } else { buf.append('&'); } buf.append(name); String stringValue = CharEscapers.escapeUriQuery(value.toString()); if (stringValue.length() != 0) { buf.append('=').append(stringValue); } return first; } }