httpclient.java

来自「This is a resource based on j2me embedde」· Java 代码 · 共 983 行 · 第 1/2 页

JAVA
983
字号
/* * @(#)HttpClient.java	1.124 06/10/10 * * Copyright  1990-2008 Sun Microsystems, Inc. All Rights Reserved.   * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER   *    * This program is free software; you can redistribute it and/or   * modify it under the terms of the GNU General Public License version   * 2 only, as published by the Free Software Foundation.    *    * This program is distributed in the hope that it will be useful, but   * WITHOUT ANY WARRANTY; without even the implied warranty of   * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU   * General Public License version 2 for more details (a copy is   * included at /legal/license.txt).    *    * You should have received a copy of the GNU General Public License   * version 2 along with this work; if not, write to the Free Software   * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA   * 02110-1301 USA    *    * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa   * Clara, CA 95054 or visit www.sun.com if you need additional   * information or have any questions.  * */package sun.net.www.http;import java.io.*;import java.net.*;import java.util.*;import sun.net.NetworkClient;import sun.net.ProgressEntry;import sun.net.ProgressData;import sun.net.www.MessageHeader;import sun.net.www.HeaderParser;import sun.net.www.MeteredStream;import sun.misc.RegexpPool;import java.security.*;/** * @version 1.115, 08/30/01 * @author Herb Jellinek * @author Dave Brown */public class HttpClient extends NetworkClient {    // whether this httpclient comes from the cache    protected boolean cachedHttpClient = false;    private boolean inCache;    // Http requests we send    MessageHeader requests;    // Http data we send with the headers    PosterOutputStream poster = null;    // if we've had one io error    boolean failedOnce = false;    /** regexp pool of hosts for which we should connect directly, not Proxy     *  these are intialized from a property.     */    private static RegexpPool nonProxyHostsPool = null;    /** The string source of nonProxyHostsPool     */    private static String nonProxyHostsSource = null;    /** Response code for CONTINUE */    private static final int	HTTP_CONTINUE = 100;    /** Default port number for http daemons. TODO: make these private */    static final int	httpPortNumber = 80;    /** return default port number (subclasses may override) */    protected int getDefaultPort () { return httpPortNumber; }    /* The following three data members are left in for binary */    /* backwards-compatibility.  Unfortunately, HotJava sets them directly */    /* when it wants to change the settings.  The new design has us not */    /* cache these, so this is unnecessary, but eliminating the data members */    /* would break HJB 1.1 under JDK 1.2. */    /* */    /* These data members are not used, and their values are meaningless. */    /* TODO:  Take them out for JDK 2.0! */    /**     * @deprecated     */    public static String proxyHost = null;    /**     * @deprecated     */    public static int proxyPort = 80;    /* instance-specific proxy fields override the static fields if set.     * Used by FTP.  These are set to the true proxy host/port if     * usingProxy is true.     */    private String instProxy = null;    private int instProxyPort = -1;    /* All proxying (generic as well as instance-specific) may be     * disabled through use of this flag     */    protected boolean proxyDisabled;    // are we using proxy in this instance?    public boolean usingProxy = false;    // target host, port for the URL    protected String host;    protected int port;    /* where we cache currently open, persistent connections */    protected static KeepAliveCache kac = new KeepAliveCache();    private static boolean keepAliveProp = true;    volatile boolean keepingAlive = false;     /* this is a keep-alive connection */    int keepAliveConnections = -1;    /* number of keep-alives left */    /**Idle timeout value, in milliseconds. Zero means infinity,     * iff keepingAlive=true.     * Unfortunately, we can't always believe this one.  If I'm connected     * through a Netscape proxy to a server that sent me a keep-alive     * time of 15 sec, the proxy unilaterally terminates my connection     * after 5 sec.  So we have to hard code our effective timeout to     * 4 sec for the case where we're using a proxy. *SIGH*     */    int keepAliveTimeout = 0;    /** Url being fetched. */    protected URL	url;        /* if set, the client will be reused and must not be put in cache */    public boolean reuse = false;     /**     * A NOP method kept for backwards binary compatibility     * @deprecated -- system properties are no longer cached.     */    public static synchronized void resetProperties() {    }    int getKeepAliveTimeout() {	return keepAliveTimeout;    }        /**     * @return the proxy host to use, as defined by system properties.     */    private String getProxyHost() {	String host = (String) java.security.AccessController.doPrivileged(                new sun.security.action.GetPropertyAction("http.proxyHost"));	if (host == null) {	    /* maintain compatibility with 1.0.2, before	     * the properties namespace was so polluted	     * and these properties were called, respectively,	     * "proxyHost" "proxyPort".  Hopefully there won't	     * be any conflicts.	     */	    host = (String) java.security.AccessController.doPrivileged(                    new sun.security.action.GetPropertyAction("proxyHost"));	}	if (host != null && host.length() == 0) {	    /* System.getProp() will give us an empty String, ""	     * for a defined but "empty" property.	     */	    host = null;	}	return host;    }    /**     * @return the proxy port to use, as defined by system properties     */    private int getProxyPort() {	final int port[] = {0};	java.security.AccessController.doPrivileged(	    new java.security.PrivilegedAction() {	    public Object run() {		if (System.getProperty("http.proxyHost") != null) {		    port[0] =			Integer.getInteger("http.proxyPort", 80).intValue();		} else {		    /* maintain compatibility with 1.0.2, before		     * the properties namespace was so polluted		     * and these properties were called, respectively,		     * "proxyHost" "proxyPort".  Hopefully there won't		     * be any conflicts.		     */		    port[0] = Integer.getInteger("proxyPort", 80).intValue();		}		return null;	    }	});	return port[0];    }    static {	String keepAlive = 	    (String) java.security.AccessController.doPrivileged(			 new java.security.PrivilegedAction() {		         public Object run() {			     return System.getProperty("http.keepAlive");			 }	    });	if (keepAlive != null) {	    keepAliveProp = Boolean.valueOf(keepAlive).booleanValue(); 	} else {	    keepAliveProp = true; 	}    }    /**     * @return true iff http keep alive is set (i.e. enabled).  Defaults     *		to true if the system property http.keepAlive isn't set.     */    public boolean getHttpKeepAliveSet() {	return keepAliveProp;    }    /**      * @return true if host matches a host specified via     * the http.nonProxyHosts property     */    private boolean matchNonProxyHosts(String host) {	synchronized (getClass()) {	    String rawList = (String) java.security.AccessController.doPrivileged(                    new sun.security.action.GetPropertyAction("http.nonProxyHosts"));	    if (rawList == null) {	        nonProxyHostsPool = null;	    } else {		if (!rawList.equals(nonProxyHostsSource)) {	            RegexpPool pool = new RegexpPool();	            StringTokenizer st = new StringTokenizer(rawList, "|", false);                    try {                        while (st.hasMoreTokens()) {                            pool.add(st.nextToken().toLowerCase(), Boolean.TRUE);                        }                    } catch (sun.misc.REException ex) {                        System.err.println("Error in http.nonProxyHosts system property:  " + ex);		    }		    nonProxyHostsPool = pool;                }	    }	    nonProxyHostsSource = rawList;        }	/*	 * Match against non-proxy hosts	 */	if (nonProxyHostsPool == null) {	    return false;	}	if (nonProxyHostsPool.match(host) != null) {	    return true;	} else {	    return false;	}    }    protected HttpClient() {    }    private HttpClient(URL url)    throws IOException {	this(url, (String)null, -1, false);    }    protected HttpClient(URL url,			 boolean proxyDisabled) throws IOException {	this(url, null, -1, proxyDisabled);    }    /* This package-only CTOR should only be used for FTP piggy-backed on HTTP     * HTTP URL's that use this won't take advantage of keep-alive.     * Additionally, this constructor may be used as a last resort when the     * first HttpClient gotten through New() failed (probably b/c of a     * Keep-Alive mismatch).     *     * NOTE: That documentation is wrong ... it's not package-private any more     */    public HttpClient(URL url, String proxy, int proxyPort)    throws IOException {	this(url, proxy, proxyPort, false);    }    /*     * This constructor gives "ultimate" flexibility, including the ability     * to bypass implicit proxying.  Sometimes we need to be using tunneling     * (transport or network level) instead of proxying (application level),     * for example when we don't want the application level data to become     * visible to third parties.     *     * @param url		the URL to which we're connecting     * @param proxy		proxy to use for this URL (e.g. forwarding)     * @param proxyPort		proxy port to use for this URL     * @param proxyDisabled	true to disable default proxying     */    private HttpClient(URL url, String proxy, int proxyPort,		       boolean proxyDisabled)	throws IOException {	this.proxyDisabled = proxyDisabled;	if (!proxyDisabled) {	    this.instProxy = proxy;	    this.instProxyPort = (proxyPort < 0)		? getDefaultPort()		: proxyPort;	}	/* try to set host to "%d.%d.%d.%d" string if	 * visible - Sprint bug - brown */	try {	    InetAddress addr = InetAddress.getByName(url.getHost());	    this.host = addr.getHostAddress();	} catch (UnknownHostException ignored) {	    this.host = url.getHost();	}	this.url = url;	port = url.getPort();	if (port == -1) {	    port = getDefaultPort();	}	openServer();    }    /* This class has no public constructor for HTTP.  This method is used to     * get an HttpClient to the specifed URL.  If there's currently an     * active HttpClient to that server/port, you'll get that one.     */    public static HttpClient New(URL url)    throws IOException {	return HttpClient.New(url, true);    }    public static HttpClient New(URL url, boolean useCache)	throws IOException {	return HttpClient.New(url, (String)null, -1, useCache);    }    public static HttpClient New(URL url, String proxy, int proxyPort,				 boolean useCache)	throws IOException {	HttpClient ret = null;	if (useCache) {	    /* see if one's already around */	    ret = (HttpClient) kac.get(url, null);	    if (ret != null) {		synchronized (ret) {		    ret.cachedHttpClient = true;		    // Uncomment this line if assertions are always		    //    turned on for libraries		    //assert ret.inCache;	    	    ret.inCache = false;		}	    }	}	if (ret == null) {	    ret = new HttpClient(url, proxy, proxyPort);	} else {	    SecurityManager security = System.getSecurityManager();	    if (security != null) {		security.checkConnect(url.getHost(), url.getPort());	    }	    ret.url = url;	}		return ret;    }    /* return it to the cache as still usable, if:     * 1) It's keeping alive, AND     * 2) It still has some connections left, AND     * 3) It hasn't had a error (PrintStream.checkError())     * 4) It hasn't timed out     *     * If this client is not keepingAlive, it should have been     * removed from the cache in the parseHeaders() method.     */    public void finished() {	if (reuse) /* will be reused */	    return;	keepAliveConnections--;	if (keepAliveConnections > 0 && isKeepingAlive() &&	       !(serverOutput.checkError())) {	    /* This connection is keepingAlive && still valid.	     * Return it to the cache.	     */	    putInKeepAliveCache();	} else {	    closeServer();	}    }    protected synchronized void putInKeepAliveCache() {	if (inCache) {	    // Uncomment this line if assertions are always	    //    turned on for libraries	    //assert false : "Duplicate put to keep alive cache";	    return;	}	inCache = true;	kac.put(url, null, this);    }    /*     * Close an idle connection to this URL (if it exists in the     * cache).     */    public void closeIdleConnection() {	HttpClient http = (HttpClient) kac.get(url, null);	if (http != null) {	    http.closeServer();	}    }    /* We're very particular here about what our InputStream to the server     * looks like for reasons that are apparent if you can decipher the     * method parseHTTP().  That's why this method is overidden from the     * superclass.     */    public void openServer(String server, int port) throws IOException {	serverSocket = doConnect(server, port);	try { 	    serverOutput = new PrintStream( 	        new BufferedOutputStream(serverSocket.getOutputStream()),  					 false, encoding); 	} catch (UnsupportedEncodingException e) { 	    throw new InternalError(encoding+" encoding not found"); 	}	serverSocket.setTcpNoDelay(true);    }    /*     * Returns true if the http request should be tunneled through proxy.     * An example where this is the case is Https.     */    public boolean needsTunneling() {	return false;    }    /*     * Returns true if this httpclient is from cache     */    public boolean isCachedConnection() {	return cachedHttpClient;    }    /*     * Finish any work left after the socket connection is     * established.  In the normal http case, it's a NO-OP. Subclass     * may need to override this. An example is Https, where for     * direct connection to the origin server, ssl handshake needs to     * be done; for proxy tunneling, the socket needs to be converted     * into an SSL socket before ssl handshake can take place.     */    public void afterConnect() throws IOException, UnknownHostException {	// NO-OP. Needs to be overwritten by HttpsClient    }    /*     * call openServer in a privileged block     */    private synchronized void privilegedOpenServer(final String proxyHost,						   final int proxyPort)	 throws IOException    {	try {	    java.security.AccessController.doPrivileged(		new java.security.PrivilegedExceptionAction() {		public Object run() throws IOException {		    openServer(proxyHost, proxyPort);		    return null;		}	    });	} catch (java.security.PrivilegedActionException pae) {	    throw (IOException) pae.getException();	}    }    /*     * call super.openServer     */    private void superOpenServer(final String proxyHost,				 final int proxyPort)	throws IOException, UnknownHostException    {	super.openServer(proxyHost, proxyPort);    } 

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?