/* * Copyright (C) 2010 The Android Open Source Project * * 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 android.net; import android.annotation.NonNull; import android.annotation.Nullable; import android.compat.annotation.UnsupportedAppUsage; import android.os.Parcel; import android.os.Parcelable; import android.text.TextUtils; import java.net.InetSocketAddress; import java.net.URLConnection; import java.util.List; import java.util.Locale; /** * Describes a proxy configuration. * * Proxy configurations are already integrated within the {@code java.net} and * Apache HTTP stack. So {@link URLConnection} and Apache's {@code HttpClient} will use * them automatically. * * Other HTTP stacks will need to obtain the proxy info from * {@link Proxy#PROXY_CHANGE_ACTION} broadcast as the extra {@link Proxy#EXTRA_PROXY_INFO}. */ public class ProxyInfo implements Parcelable { private final String mHost; private final int mPort; private final String mExclusionList; private final String[] mParsedExclusionList; private final Uri mPacFileUrl; /** *@hide */ public static final String LOCAL_EXCL_LIST = ""; /** *@hide */ public static final int LOCAL_PORT = -1; /** *@hide */ public static final String LOCAL_HOST = "localhost"; /** * Constructs a {@link ProxyInfo} object that points at a Direct proxy * on the specified host and port. */ public static ProxyInfo buildDirectProxy(String host, int port) { return new ProxyInfo(host, port, null); } /** * Constructs a {@link ProxyInfo} object that points at a Direct proxy * on the specified host and port. * * The proxy will not be used to access any host in exclusion list, exclList. * * @param exclList Hosts to exclude using the proxy on connections for. These * hosts can use wildcards such as *.example.com. */ public static ProxyInfo buildDirectProxy(String host, int port, List exclList) { String[] array = exclList.toArray(new String[exclList.size()]); return new ProxyInfo(host, port, TextUtils.join(",", array), array); } /** * Construct a {@link ProxyInfo} that will download and run the PAC script * at the specified URL. */ public static ProxyInfo buildPacProxy(Uri pacUri) { return new ProxyInfo(pacUri); } /** * Construct a {@link ProxyInfo} object that will download and run the PAC script at the * specified URL and port. */ @NonNull public static ProxyInfo buildPacProxy(@NonNull Uri pacUrl, int port) { return new ProxyInfo(pacUrl, port); } /** * Create a ProxyProperties that points at a HTTP Proxy. * @hide */ @UnsupportedAppUsage public ProxyInfo(String host, int port, String exclList) { mHost = host; mPort = port; mExclusionList = exclList; mParsedExclusionList = parseExclusionList(mExclusionList); mPacFileUrl = Uri.EMPTY; } /** * Create a ProxyProperties that points at a PAC URL. * @hide */ public ProxyInfo(@NonNull Uri pacFileUrl) { mHost = LOCAL_HOST; mPort = LOCAL_PORT; mExclusionList = LOCAL_EXCL_LIST; mParsedExclusionList = parseExclusionList(mExclusionList); if (pacFileUrl == null) { throw new NullPointerException(); } mPacFileUrl = pacFileUrl; } /** * Create a ProxyProperties that points at a PAC URL. * @hide */ public ProxyInfo(String pacFileUrl) { mHost = LOCAL_HOST; mPort = LOCAL_PORT; mExclusionList = LOCAL_EXCL_LIST; mParsedExclusionList = parseExclusionList(mExclusionList); mPacFileUrl = Uri.parse(pacFileUrl); } /** * Only used in PacManager after Local Proxy is bound. * @hide */ public ProxyInfo(@NonNull Uri pacFileUrl, int localProxyPort) { mHost = LOCAL_HOST; mPort = localProxyPort; mExclusionList = LOCAL_EXCL_LIST; mParsedExclusionList = parseExclusionList(mExclusionList); if (pacFileUrl == null) { throw new NullPointerException(); } mPacFileUrl = pacFileUrl; } private static String[] parseExclusionList(String exclusionList) { if (exclusionList == null) { return new String[0]; } else { return exclusionList.toLowerCase(Locale.ROOT).split(","); } } private ProxyInfo(String host, int port, String exclList, String[] parsedExclList) { mHost = host; mPort = port; mExclusionList = exclList; mParsedExclusionList = parsedExclList; mPacFileUrl = Uri.EMPTY; } /** * A copy constructor to hold proxy properties. */ public ProxyInfo(@Nullable ProxyInfo source) { if (source != null) { mHost = source.getHost(); mPort = source.getPort(); mPacFileUrl = source.mPacFileUrl; mExclusionList = source.getExclusionListAsString(); mParsedExclusionList = source.mParsedExclusionList; } else { mHost = null; mPort = 0; mExclusionList = null; mParsedExclusionList = null; mPacFileUrl = Uri.EMPTY; } } /** * @hide */ public InetSocketAddress getSocketAddress() { InetSocketAddress inetSocketAddress = null; try { inetSocketAddress = new InetSocketAddress(mHost, mPort); } catch (IllegalArgumentException e) { } return inetSocketAddress; } /** * Returns the URL of the current PAC script or null if there is * no PAC script. */ public Uri getPacFileUrl() { return mPacFileUrl; } /** * When configured to use a Direct Proxy this returns the host * of the proxy. */ public String getHost() { return mHost; } /** * When configured to use a Direct Proxy this returns the port * of the proxy */ public int getPort() { return mPort; } /** * When configured to use a Direct Proxy this returns the list * of hosts for which the proxy is ignored. */ public String[] getExclusionList() { return mParsedExclusionList; } /** * comma separated * @hide */ @Nullable public String getExclusionListAsString() { return mExclusionList; } /** * Return true if the pattern of proxy is valid, otherwise return false. */ public boolean isValid() { if (!Uri.EMPTY.equals(mPacFileUrl)) return true; return Proxy.PROXY_VALID == Proxy.validate(mHost == null ? "" : mHost, mPort == 0 ? "" : Integer.toString(mPort), mExclusionList == null ? "" : mExclusionList); } /** * @hide */ public java.net.Proxy makeProxy() { java.net.Proxy proxy = java.net.Proxy.NO_PROXY; if (mHost != null) { try { InetSocketAddress inetSocketAddress = new InetSocketAddress(mHost, mPort); proxy = new java.net.Proxy(java.net.Proxy.Type.HTTP, inetSocketAddress); } catch (IllegalArgumentException e) { } } return proxy; } @Override public String toString() { StringBuilder sb = new StringBuilder(); if (!Uri.EMPTY.equals(mPacFileUrl)) { sb.append("PAC Script: "); sb.append(mPacFileUrl); } if (mHost != null) { sb.append("["); sb.append(mHost); sb.append("] "); sb.append(Integer.toString(mPort)); if (mExclusionList != null) { sb.append(" xl=").append(mExclusionList); } } else { sb.append("[ProxyProperties.mHost == null]"); } return sb.toString(); } @Override public boolean equals(Object o) { if (!(o instanceof ProxyInfo)) return false; ProxyInfo p = (ProxyInfo)o; // If PAC URL is present in either then they must be equal. // Other parameters will only be for fall back. if (!Uri.EMPTY.equals(mPacFileUrl)) { return mPacFileUrl.equals(p.getPacFileUrl()) && mPort == p.mPort; } if (!Uri.EMPTY.equals(p.mPacFileUrl)) { return false; } if (mExclusionList != null && !mExclusionList.equals(p.getExclusionListAsString())) { return false; } if (mHost != null && p.getHost() != null && mHost.equals(p.getHost()) == false) { return false; } if (mHost != null && p.mHost == null) return false; if (mHost == null && p.mHost != null) return false; if (mPort != p.mPort) return false; return true; } /** * Implement the Parcelable interface * @hide */ public int describeContents() { return 0; } @Override /* * generate hashcode based on significant fields */ public int hashCode() { return ((null == mHost) ? 0 : mHost.hashCode()) + ((null == mExclusionList) ? 0 : mExclusionList.hashCode()) + mPort; } /** * Implement the Parcelable interface. * @hide */ public void writeToParcel(Parcel dest, int flags) { if (!Uri.EMPTY.equals(mPacFileUrl)) { dest.writeByte((byte)1); mPacFileUrl.writeToParcel(dest, 0); dest.writeInt(mPort); return; } else { dest.writeByte((byte)0); } if (mHost != null) { dest.writeByte((byte)1); dest.writeString(mHost); dest.writeInt(mPort); } else { dest.writeByte((byte)0); } dest.writeString(mExclusionList); dest.writeStringArray(mParsedExclusionList); } public static final @android.annotation.NonNull Creator CREATOR = new Creator() { public ProxyInfo createFromParcel(Parcel in) { String host = null; int port = 0; if (in.readByte() != 0) { Uri url = Uri.CREATOR.createFromParcel(in); int localPort = in.readInt(); return new ProxyInfo(url, localPort); } if (in.readByte() != 0) { host = in.readString(); port = in.readInt(); } String exclList = in.readString(); String[] parsedExclList = in.readStringArray(); ProxyInfo proxyProperties = new ProxyInfo(host, port, exclList, parsedExclList); return proxyProperties; } public ProxyInfo[] newArray(int size) { return new ProxyInfo[size]; } }; }