Home | History | Annotate | Download | only in net
      1 /*
      2  * Copyright (C) 2010 The Android Open Source Project
      3  *
      4  * Licensed under the Apache License, Version 2.0 (the "License");
      5  * you may not use this file except in compliance with the License.
      6  * You may obtain a copy of the License at
      7  *
      8  *      http://www.apache.org/licenses/LICENSE-2.0
      9  *
     10  * Unless required by applicable law or agreed to in writing, software
     11  * distributed under the License is distributed on an "AS IS" BASIS,
     12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
     13  * See the License for the specific language governing permissions and
     14  * limitations under the License.
     15  */
     16 
     17 package android.net;
     18 
     19 
     20 import android.annotation.UnsupportedAppUsage;
     21 import android.os.Parcel;
     22 import android.os.Parcelable;
     23 import android.text.TextUtils;
     24 
     25 import java.net.InetSocketAddress;
     26 import java.net.URLConnection;
     27 import java.util.List;
     28 import java.util.Locale;
     29 
     30 /**
     31  * Describes a proxy configuration.
     32  *
     33  * Proxy configurations are already integrated within the {@code java.net} and
     34  * Apache HTTP stack. So {@link URLConnection} and Apache's {@code HttpClient} will use
     35  * them automatically.
     36  *
     37  * Other HTTP stacks will need to obtain the proxy info from
     38  * {@link Proxy#PROXY_CHANGE_ACTION} broadcast as the extra {@link Proxy#EXTRA_PROXY_INFO}.
     39  */
     40 public class ProxyInfo implements Parcelable {
     41 
     42     private final String mHost;
     43     private final int mPort;
     44     private final String mExclusionList;
     45     private final String[] mParsedExclusionList;
     46     private final Uri mPacFileUrl;
     47 
     48     /**
     49      *@hide
     50      */
     51     public static final String LOCAL_EXCL_LIST = "";
     52     /**
     53      *@hide
     54      */
     55     public static final int LOCAL_PORT = -1;
     56     /**
     57      *@hide
     58      */
     59     public static final String LOCAL_HOST = "localhost";
     60 
     61     /**
     62      * Constructs a {@link ProxyInfo} object that points at a Direct proxy
     63      * on the specified host and port.
     64      */
     65     public static ProxyInfo buildDirectProxy(String host, int port) {
     66         return new ProxyInfo(host, port, null);
     67     }
     68 
     69     /**
     70      * Constructs a {@link ProxyInfo} object that points at a Direct proxy
     71      * on the specified host and port.
     72      *
     73      * The proxy will not be used to access any host in exclusion list, exclList.
     74      *
     75      * @param exclList Hosts to exclude using the proxy on connections for.  These
     76      *                 hosts can use wildcards such as *.example.com.
     77      */
     78     public static ProxyInfo buildDirectProxy(String host, int port, List<String> exclList) {
     79         String[] array = exclList.toArray(new String[exclList.size()]);
     80         return new ProxyInfo(host, port, TextUtils.join(",", array), array);
     81     }
     82 
     83     /**
     84      * Construct a {@link ProxyInfo} that will download and run the PAC script
     85      * at the specified URL.
     86      */
     87     public static ProxyInfo buildPacProxy(Uri pacUri) {
     88         return new ProxyInfo(pacUri);
     89     }
     90 
     91     /**
     92      * Create a ProxyProperties that points at a HTTP Proxy.
     93      * @hide
     94      */
     95     @UnsupportedAppUsage
     96     public ProxyInfo(String host, int port, String exclList) {
     97         mHost = host;
     98         mPort = port;
     99         mExclusionList = exclList;
    100         mParsedExclusionList = parseExclusionList(mExclusionList);
    101         mPacFileUrl = Uri.EMPTY;
    102     }
    103 
    104     /**
    105      * Create a ProxyProperties that points at a PAC URL.
    106      * @hide
    107      */
    108     public ProxyInfo(Uri pacFileUrl) {
    109         mHost = LOCAL_HOST;
    110         mPort = LOCAL_PORT;
    111         mExclusionList = LOCAL_EXCL_LIST;
    112         mParsedExclusionList = parseExclusionList(mExclusionList);
    113         if (pacFileUrl == null) {
    114             throw new NullPointerException();
    115         }
    116         mPacFileUrl = pacFileUrl;
    117     }
    118 
    119     /**
    120      * Create a ProxyProperties that points at a PAC URL.
    121      * @hide
    122      */
    123     public ProxyInfo(String pacFileUrl) {
    124         mHost = LOCAL_HOST;
    125         mPort = LOCAL_PORT;
    126         mExclusionList = LOCAL_EXCL_LIST;
    127         mParsedExclusionList = parseExclusionList(mExclusionList);
    128         mPacFileUrl = Uri.parse(pacFileUrl);
    129     }
    130 
    131     /**
    132      * Only used in PacManager after Local Proxy is bound.
    133      * @hide
    134      */
    135     public ProxyInfo(Uri pacFileUrl, int localProxyPort) {
    136         mHost = LOCAL_HOST;
    137         mPort = localProxyPort;
    138         mExclusionList = LOCAL_EXCL_LIST;
    139         mParsedExclusionList = parseExclusionList(mExclusionList);
    140         if (pacFileUrl == null) {
    141             throw new NullPointerException();
    142         }
    143         mPacFileUrl = pacFileUrl;
    144     }
    145 
    146     private static String[] parseExclusionList(String exclusionList) {
    147         if (exclusionList == null) {
    148             return new String[0];
    149         } else {
    150             return exclusionList.toLowerCase(Locale.ROOT).split(",");
    151         }
    152     }
    153 
    154     private ProxyInfo(String host, int port, String exclList, String[] parsedExclList) {
    155         mHost = host;
    156         mPort = port;
    157         mExclusionList = exclList;
    158         mParsedExclusionList = parsedExclList;
    159         mPacFileUrl = Uri.EMPTY;
    160     }
    161 
    162     // copy constructor instead of clone
    163     /**
    164      * @hide
    165      */
    166     public ProxyInfo(ProxyInfo source) {
    167         if (source != null) {
    168             mHost = source.getHost();
    169             mPort = source.getPort();
    170             mPacFileUrl = source.mPacFileUrl;
    171             mExclusionList = source.getExclusionListAsString();
    172             mParsedExclusionList = source.mParsedExclusionList;
    173         } else {
    174             mHost = null;
    175             mPort = 0;
    176             mExclusionList = null;
    177             mParsedExclusionList = null;
    178             mPacFileUrl = Uri.EMPTY;
    179         }
    180     }
    181 
    182     /**
    183      * @hide
    184      */
    185     public InetSocketAddress getSocketAddress() {
    186         InetSocketAddress inetSocketAddress = null;
    187         try {
    188             inetSocketAddress = new InetSocketAddress(mHost, mPort);
    189         } catch (IllegalArgumentException e) { }
    190         return inetSocketAddress;
    191     }
    192 
    193     /**
    194      * Returns the URL of the current PAC script or null if there is
    195      * no PAC script.
    196      */
    197     public Uri getPacFileUrl() {
    198         return mPacFileUrl;
    199     }
    200 
    201     /**
    202      * When configured to use a Direct Proxy this returns the host
    203      * of the proxy.
    204      */
    205     public String getHost() {
    206         return mHost;
    207     }
    208 
    209     /**
    210      * When configured to use a Direct Proxy this returns the port
    211      * of the proxy
    212      */
    213     public int getPort() {
    214         return mPort;
    215     }
    216 
    217     /**
    218      * When configured to use a Direct Proxy this returns the list
    219      * of hosts for which the proxy is ignored.
    220      */
    221     public String[] getExclusionList() {
    222         return mParsedExclusionList;
    223     }
    224 
    225     /**
    226      * comma separated
    227      * @hide
    228      */
    229     public String getExclusionListAsString() {
    230         return mExclusionList;
    231     }
    232 
    233     /**
    234      * @hide
    235      */
    236     public boolean isValid() {
    237         if (!Uri.EMPTY.equals(mPacFileUrl)) return true;
    238         return Proxy.PROXY_VALID == Proxy.validate(mHost == null ? "" : mHost,
    239                 mPort == 0 ? "" : Integer.toString(mPort),
    240                 mExclusionList == null ? "" : mExclusionList);
    241     }
    242 
    243     /**
    244      * @hide
    245      */
    246     public java.net.Proxy makeProxy() {
    247         java.net.Proxy proxy = java.net.Proxy.NO_PROXY;
    248         if (mHost != null) {
    249             try {
    250                 InetSocketAddress inetSocketAddress = new InetSocketAddress(mHost, mPort);
    251                 proxy = new java.net.Proxy(java.net.Proxy.Type.HTTP, inetSocketAddress);
    252             } catch (IllegalArgumentException e) {
    253             }
    254         }
    255         return proxy;
    256     }
    257 
    258     @Override
    259     public String toString() {
    260         StringBuilder sb = new StringBuilder();
    261         if (!Uri.EMPTY.equals(mPacFileUrl)) {
    262             sb.append("PAC Script: ");
    263             sb.append(mPacFileUrl);
    264         }
    265         if (mHost != null) {
    266             sb.append("[");
    267             sb.append(mHost);
    268             sb.append("] ");
    269             sb.append(Integer.toString(mPort));
    270             if (mExclusionList != null) {
    271                 sb.append(" xl=").append(mExclusionList);
    272             }
    273         } else {
    274             sb.append("[ProxyProperties.mHost == null]");
    275         }
    276         return sb.toString();
    277     }
    278 
    279     @Override
    280     public boolean equals(Object o) {
    281         if (!(o instanceof ProxyInfo)) return false;
    282         ProxyInfo p = (ProxyInfo)o;
    283         // If PAC URL is present in either then they must be equal.
    284         // Other parameters will only be for fall back.
    285         if (!Uri.EMPTY.equals(mPacFileUrl)) {
    286             return mPacFileUrl.equals(p.getPacFileUrl()) && mPort == p.mPort;
    287         }
    288         if (!Uri.EMPTY.equals(p.mPacFileUrl)) {
    289             return false;
    290         }
    291         if (mExclusionList != null && !mExclusionList.equals(p.getExclusionListAsString())) {
    292             return false;
    293         }
    294         if (mHost != null && p.getHost() != null && mHost.equals(p.getHost()) == false) {
    295             return false;
    296         }
    297         if (mHost != null && p.mHost == null) return false;
    298         if (mHost == null && p.mHost != null) return false;
    299         if (mPort != p.mPort) return false;
    300         return true;
    301     }
    302 
    303     /**
    304      * Implement the Parcelable interface
    305      * @hide
    306      */
    307     public int describeContents() {
    308         return 0;
    309     }
    310 
    311     @Override
    312     /*
    313      * generate hashcode based on significant fields
    314      */
    315     public int hashCode() {
    316         return ((null == mHost) ? 0 : mHost.hashCode())
    317                 + ((null == mExclusionList) ? 0 : mExclusionList.hashCode())
    318                 + mPort;
    319     }
    320 
    321     /**
    322      * Implement the Parcelable interface.
    323      * @hide
    324      */
    325     public void writeToParcel(Parcel dest, int flags) {
    326         if (!Uri.EMPTY.equals(mPacFileUrl)) {
    327             dest.writeByte((byte)1);
    328             mPacFileUrl.writeToParcel(dest, 0);
    329             dest.writeInt(mPort);
    330             return;
    331         } else {
    332             dest.writeByte((byte)0);
    333         }
    334         if (mHost != null) {
    335             dest.writeByte((byte)1);
    336             dest.writeString(mHost);
    337             dest.writeInt(mPort);
    338         } else {
    339             dest.writeByte((byte)0);
    340         }
    341         dest.writeString(mExclusionList);
    342         dest.writeStringArray(mParsedExclusionList);
    343     }
    344 
    345     public static final @android.annotation.NonNull Creator<ProxyInfo> CREATOR =
    346         new Creator<ProxyInfo>() {
    347             public ProxyInfo createFromParcel(Parcel in) {
    348                 String host = null;
    349                 int port = 0;
    350                 if (in.readByte() != 0) {
    351                     Uri url = Uri.CREATOR.createFromParcel(in);
    352                     int localPort = in.readInt();
    353                     return new ProxyInfo(url, localPort);
    354                 }
    355                 if (in.readByte() != 0) {
    356                     host = in.readString();
    357                     port = in.readInt();
    358                 }
    359                 String exclList = in.readString();
    360                 String[] parsedExclList = in.readStringArray();
    361                 ProxyInfo proxyProperties = new ProxyInfo(host, port, exclList, parsedExclList);
    362                 return proxyProperties;
    363             }
    364 
    365             public ProxyInfo[] newArray(int size) {
    366                 return new ProxyInfo[size];
    367             }
    368         };
    369 }
    370