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