1 /* 2 * Copyright (C) 2007 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.SdkConstant; 20 import android.annotation.SdkConstant.SdkConstantType; 21 import android.content.ContentResolver; 22 import android.content.Context; 23 import android.database.ContentObserver; 24 import android.net.ProxyProperties; 25 import android.os.Handler; 26 import android.os.SystemProperties; 27 import android.text.TextUtils; 28 import android.provider.Settings; 29 import android.util.Log; 30 31 import java.net.InetAddress; 32 import java.net.InetSocketAddress; 33 import java.net.ProxySelector; 34 import java.net.SocketAddress; 35 import java.net.URI; 36 import java.net.UnknownHostException; 37 import java.util.concurrent.locks.ReadWriteLock; 38 import java.util.concurrent.locks.ReentrantReadWriteLock; 39 import java.util.List; 40 import java.util.regex.Matcher; 41 import java.util.regex.Pattern; 42 43 import junit.framework.Assert; 44 45 import org.apache.http.conn.routing.HttpRoute; 46 import org.apache.http.conn.routing.HttpRoutePlanner; 47 import org.apache.http.conn.scheme.SchemeRegistry; 48 import org.apache.http.HttpHost; 49 import org.apache.http.HttpRequest; 50 import org.apache.http.impl.conn.ProxySelectorRoutePlanner; 51 import org.apache.http.protocol.HttpContext; 52 53 /** 54 * A convenience class for accessing the user and default proxy 55 * settings. 56 */ 57 public final class Proxy { 58 59 // Set to true to enable extra debugging. 60 private static final boolean DEBUG = false; 61 private static final String TAG = "Proxy"; 62 63 /** 64 * Used to notify an app that's caching the default connection proxy 65 * that either the default connection or its proxy has changed. 66 * The intent will have the following extra value:</p> 67 * <ul> 68 * <li><em>EXTRA_PROXY_INFO</em> - The ProxyProperties for the proxy. Non-null, 69 * though if the proxy is undefined the host string 70 * will be empty. 71 * </ul> 72 * 73 * <p class="note">This is a protected intent that can only be sent by the system 74 */ 75 @SdkConstant(SdkConstantType.BROADCAST_INTENT_ACTION) 76 public static final String PROXY_CHANGE_ACTION = "android.intent.action.PROXY_CHANGE"; 77 /** {@hide} **/ 78 public static final String EXTRA_PROXY_INFO = "proxy"; 79 80 private static ConnectivityManager sConnectivityManager = null; 81 82 // Hostname / IP REGEX validation 83 // Matches blank input, ips, and domain names 84 private static final String NAME_IP_REGEX = 85 "[a-zA-Z0-9]+(\\-[a-zA-Z0-9]+)*(\\.[a-zA-Z0-9]+(\\-[a-zA-Z0-9]+)*)*"; 86 87 private static final String HOSTNAME_REGEXP = "^$|^" + NAME_IP_REGEX + "$"; 88 89 private static final Pattern HOSTNAME_PATTERN; 90 91 private static final String EXCLLIST_REGEXP = "$|^(.?" + NAME_IP_REGEX 92 + ")+(,(.?" + NAME_IP_REGEX + "))*$"; 93 94 private static final Pattern EXCLLIST_PATTERN; 95 96 static { 97 HOSTNAME_PATTERN = Pattern.compile(HOSTNAME_REGEXP); 98 EXCLLIST_PATTERN = Pattern.compile(EXCLLIST_REGEXP); 99 } 100 101 /** 102 * Return the proxy object to be used for the URL given as parameter. 103 * @param ctx A Context used to get the settings for the proxy host. 104 * @param url A URL to be accessed. Used to evaluate exclusion list. 105 * @return Proxy (java.net) object containing the host name. If the 106 * user did not set a hostname it returns the default host. 107 * A null value means that no host is to be used. 108 * {@hide} 109 */ getProxy(Context ctx, String url)110 public static final java.net.Proxy getProxy(Context ctx, String url) { 111 String host = ""; 112 if (url != null) { 113 URI uri = URI.create(url); 114 host = uri.getHost(); 115 } 116 117 if (!isLocalHost(host)) { 118 if (sConnectivityManager == null) { 119 sConnectivityManager = (ConnectivityManager)ctx.getSystemService( 120 Context.CONNECTIVITY_SERVICE); 121 } 122 if (sConnectivityManager == null) return java.net.Proxy.NO_PROXY; 123 124 ProxyProperties proxyProperties = sConnectivityManager.getProxy(); 125 126 if (proxyProperties != null) { 127 if (!proxyProperties.isExcluded(host)) { 128 return proxyProperties.makeProxy(); 129 } 130 } 131 } 132 return java.net.Proxy.NO_PROXY; 133 } 134 135 136 /** 137 * Return the proxy host set by the user. 138 * @param ctx A Context used to get the settings for the proxy host. 139 * @return String containing the host name. If the user did not set a host 140 * name it returns the default host. A null value means that no 141 * host is to be used. 142 * @deprecated Use standard java vm proxy values to find the host, port 143 * and exclusion list. This call ignores the exclusion list. 144 */ getHost(Context ctx)145 public static final String getHost(Context ctx) { 146 java.net.Proxy proxy = getProxy(ctx, null); 147 if (proxy == java.net.Proxy.NO_PROXY) return null; 148 try { 149 return ((InetSocketAddress)(proxy.address())).getHostName(); 150 } catch (Exception e) { 151 return null; 152 } 153 } 154 155 /** 156 * Return the proxy port set by the user. 157 * @param ctx A Context used to get the settings for the proxy port. 158 * @return The port number to use or -1 if no proxy is to be used. 159 * @deprecated Use standard java vm proxy values to find the host, port 160 * and exclusion list. This call ignores the exclusion list. 161 */ getPort(Context ctx)162 public static final int getPort(Context ctx) { 163 java.net.Proxy proxy = getProxy(ctx, null); 164 if (proxy == java.net.Proxy.NO_PROXY) return -1; 165 try { 166 return ((InetSocketAddress)(proxy.address())).getPort(); 167 } catch (Exception e) { 168 return -1; 169 } 170 } 171 172 /** 173 * Return the default proxy host specified by the carrier. 174 * @return String containing the host name or null if there is no proxy for 175 * this carrier. 176 * @deprecated Use standard java vm proxy values to find the host, port and 177 * exclusion list. This call ignores the exclusion list and no 178 * longer reports only mobile-data apn-based proxy values. 179 */ getDefaultHost()180 public static final String getDefaultHost() { 181 String host = System.getProperty("http.proxyHost"); 182 if (TextUtils.isEmpty(host)) return null; 183 return host; 184 } 185 186 /** 187 * Return the default proxy port specified by the carrier. 188 * @return The port number to be used with the proxy host or -1 if there is 189 * no proxy for this carrier. 190 * @deprecated Use standard java vm proxy values to find the host, port and 191 * exclusion list. This call ignores the exclusion list and no 192 * longer reports only mobile-data apn-based proxy values. 193 */ getDefaultPort()194 public static final int getDefaultPort() { 195 if (getDefaultHost() == null) return -1; 196 try { 197 return Integer.parseInt(System.getProperty("http.proxyPort")); 198 } catch (NumberFormatException e) { 199 return -1; 200 } 201 } 202 203 /** 204 * Returns the preferred proxy to be used by clients. This is a wrapper 205 * around {@link android.net.Proxy#getHost()}. 206 * 207 * @param context the context which will be passed to 208 * {@link android.net.Proxy#getHost()} 209 * @param url the target URL for the request 210 * @note Calling this method requires permission 211 * android.permission.ACCESS_NETWORK_STATE 212 * @return The preferred proxy to be used by clients, or null if there 213 * is no proxy. 214 * {@hide} 215 */ getPreferredHttpHost(Context context, String url)216 public static final HttpHost getPreferredHttpHost(Context context, 217 String url) { 218 java.net.Proxy prefProxy = getProxy(context, url); 219 if (prefProxy.equals(java.net.Proxy.NO_PROXY)) { 220 return null; 221 } else { 222 InetSocketAddress sa = (InetSocketAddress)prefProxy.address(); 223 return new HttpHost(sa.getHostName(), sa.getPort(), "http"); 224 } 225 } 226 isLocalHost(String host)227 private static final boolean isLocalHost(String host) { 228 if (host == null) { 229 return false; 230 } 231 try { 232 if (host != null) { 233 if (host.equalsIgnoreCase("localhost")) { 234 return true; 235 } 236 if (NetworkUtils.numericToInetAddress(host).isLoopbackAddress()) { 237 return true; 238 } 239 } 240 } catch (IllegalArgumentException iex) { 241 } 242 return false; 243 } 244 245 /** 246 * Validate syntax of hostname, port and exclusion list entries 247 * {@hide} 248 */ validate(String hostname, String port, String exclList)249 public static void validate(String hostname, String port, String exclList) { 250 Matcher match = HOSTNAME_PATTERN.matcher(hostname); 251 Matcher listMatch = EXCLLIST_PATTERN.matcher(exclList); 252 253 if (!match.matches()) { 254 throw new IllegalArgumentException(); 255 } 256 257 if (!listMatch.matches()) { 258 throw new IllegalArgumentException(); 259 } 260 261 if (hostname.length() > 0 && port.length() == 0) { 262 throw new IllegalArgumentException(); 263 } 264 265 if (port.length() > 0) { 266 if (hostname.length() == 0) { 267 throw new IllegalArgumentException(); 268 } 269 int portVal = -1; 270 try { 271 portVal = Integer.parseInt(port); 272 } catch (NumberFormatException ex) { 273 throw new IllegalArgumentException(); 274 } 275 if (portVal <= 0 || portVal > 0xFFFF) { 276 throw new IllegalArgumentException(); 277 } 278 } 279 } 280 281 static class AndroidProxySelectorRoutePlanner 282 extends org.apache.http.impl.conn.ProxySelectorRoutePlanner { 283 284 private Context mContext; 285 AndroidProxySelectorRoutePlanner(SchemeRegistry schreg, ProxySelector prosel, Context context)286 public AndroidProxySelectorRoutePlanner(SchemeRegistry schreg, ProxySelector prosel, 287 Context context) { 288 super(schreg, prosel); 289 mContext = context; 290 } 291 292 @Override chooseProxy(List<java.net.Proxy> proxies, HttpHost target, HttpRequest request, HttpContext context)293 protected java.net.Proxy chooseProxy(List<java.net.Proxy> proxies, HttpHost target, 294 HttpRequest request, HttpContext context) { 295 return getProxy(mContext, target.getHostName()); 296 } 297 298 @Override determineProxy(HttpHost target, HttpRequest request, HttpContext context)299 protected HttpHost determineProxy(HttpHost target, HttpRequest request, 300 HttpContext context) { 301 return getPreferredHttpHost(mContext, target.getHostName()); 302 } 303 304 @Override determineRoute(HttpHost target, HttpRequest request, HttpContext context)305 public HttpRoute determineRoute(HttpHost target, HttpRequest request, 306 HttpContext context) { 307 HttpHost proxy = getPreferredHttpHost(mContext, target.getHostName()); 308 if (proxy == null) { 309 return new HttpRoute(target); 310 } else { 311 return new HttpRoute(target, null, proxy, false); 312 } 313 } 314 } 315 316 /** @hide */ getAndroidProxySelectorRoutePlanner(Context context)317 public static final HttpRoutePlanner getAndroidProxySelectorRoutePlanner(Context context) { 318 AndroidProxySelectorRoutePlanner ret = new AndroidProxySelectorRoutePlanner( 319 new SchemeRegistry(), ProxySelector.getDefault(), context); 320 return ret; 321 } 322 323 /** @hide */ setHttpProxySystemProperty(ProxyProperties p)324 public static final void setHttpProxySystemProperty(ProxyProperties p) { 325 String host = null; 326 String port = null; 327 String exclList = null; 328 if (p != null) { 329 host = p.getHost(); 330 port = Integer.toString(p.getPort()); 331 exclList = p.getExclusionList(); 332 } 333 setHttpProxySystemProperty(host, port, exclList); 334 } 335 336 /** @hide */ setHttpProxySystemProperty(String host, String port, String exclList)337 public static final void setHttpProxySystemProperty(String host, String port, String exclList) { 338 if (exclList != null) exclList = exclList.replace(",", "|"); 339 if (false) Log.d(TAG, "setHttpProxySystemProperty :"+host+":"+port+" - "+exclList); 340 if (host != null) { 341 System.setProperty("http.proxyHost", host); 342 System.setProperty("https.proxyHost", host); 343 } else { 344 System.clearProperty("http.proxyHost"); 345 System.clearProperty("https.proxyHost"); 346 } 347 if (port != null) { 348 System.setProperty("http.proxyPort", port); 349 System.setProperty("https.proxyPort", port); 350 } else { 351 System.clearProperty("http.proxyPort"); 352 System.clearProperty("https.proxyPort"); 353 } 354 if (exclList != null) { 355 System.setProperty("http.nonProxyHosts", exclList); 356 System.setProperty("https.nonProxyHosts", exclList); 357 } else { 358 System.clearProperty("http.nonProxyHosts"); 359 System.clearProperty("https.nonProxyHosts"); 360 } 361 } 362 } 363