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