forked from GNUsocial/gnu-social
		
	
		
			
	
	
		
			158 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
		
		
			
		
	
	
			158 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
|   | <?php | ||
|  | /** | ||
|  |  * SOCKS5 proxy connection class | ||
|  |  * | ||
|  |  * PHP version 5 | ||
|  |  * | ||
|  |  * LICENSE: | ||
|  |  * | ||
|  |  * Copyright (c) 2008-2012, Alexey Borzov <avb@php.net> | ||
|  |  * All rights reserved. | ||
|  |  * | ||
|  |  * Redistribution and use in source and binary forms, with or without | ||
|  |  * modification, are permitted provided that the following conditions | ||
|  |  * are met: | ||
|  |  * | ||
|  |  *    * Redistributions of source code must retain the above copyright | ||
|  |  *      notice, this list of conditions and the following disclaimer. | ||
|  |  *    * Redistributions in binary form must reproduce the above copyright | ||
|  |  *      notice, this list of conditions and the following disclaimer in the | ||
|  |  *      documentation and/or other materials provided with the distribution. | ||
|  |  *    * The names of the authors may not be used to endorse or promote products | ||
|  |  *      derived from this software without specific prior written permission. | ||
|  |  * | ||
|  |  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS
 | ||
|  |  * IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
 | ||
|  |  * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR | ||
|  |  * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR | ||
|  |  * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, | ||
|  |  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, | ||
|  |  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR | ||
|  |  * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY | ||
|  |  * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING | ||
|  |  * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS | ||
|  |  * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||
|  |  * | ||
|  |  * @category HTTP | ||
|  |  * @package  HTTP_Request2 | ||
|  |  * @author   Alexey Borzov <avb@php.net> | ||
|  |  * @license  http://opensource.org/licenses/bsd-license.php New BSD License | ||
|  |  * @version  SVN: $Id: SOCKS5.php 324953 2012-04-08 07:24:12Z avb $ | ||
|  |  * @link     http://pear.php.net/package/HTTP_Request2 | ||
|  |  */ | ||
|  | 
 | ||
|  | /** Socket wrapper class used by Socket Adapter */ | ||
|  | require_once 'HTTP/Request2/SocketWrapper.php'; | ||
|  | 
 | ||
|  | /** | ||
|  |  * SOCKS5 proxy connection class (used by Socket Adapter) | ||
|  |  * | ||
|  |  * @category HTTP | ||
|  |  * @package  HTTP_Request2 | ||
|  |  * @author   Alexey Borzov <avb@php.net> | ||
|  |  * @license  http://opensource.org/licenses/bsd-license.php New BSD License | ||
|  |  * @version  Release: 2.1.1 | ||
|  |  * @link     http://pear.php.net/package/HTTP_Request2 | ||
|  |  * @link     http://pear.php.net/bugs/bug.php?id=19332 | ||
|  |  * @link     http://tools.ietf.org/html/rfc1928 | ||
|  |  */ | ||
|  | class HTTP_Request2_SOCKS5 extends HTTP_Request2_SocketWrapper | ||
|  | { | ||
|  |     /** | ||
|  |      * Constructor, tries to connect and authenticate to a SOCKS5 proxy | ||
|  |      * | ||
|  |      * @param string $address    Proxy address, e.g. 'tcp://localhost:1080' | ||
|  |      * @param int    $timeout    Connection timeout (seconds) | ||
|  |      * @param array  $sslOptions SSL context options | ||
|  |      * @param string $username   Proxy user name | ||
|  |      * @param string $password   Proxy password | ||
|  |      * | ||
|  |      * @throws HTTP_Request2_LogicException | ||
|  |      * @throws HTTP_Request2_ConnectionException | ||
|  |      * @throws HTTP_Request2_MessageException | ||
|  |      */ | ||
|  |     public function __construct( | ||
|  |         $address, $timeout = 10, array $sslOptions = array(), | ||
|  |         $username = null, $password = null | ||
|  |     ) { | ||
|  |         parent::__construct($address, $timeout, $sslOptions); | ||
|  | 
 | ||
|  |         if (strlen($username)) { | ||
|  |             $request = pack('C4', 5, 2, 0, 2); | ||
|  |         } else { | ||
|  |             $request = pack('C3', 5, 1, 0); | ||
|  |         } | ||
|  |         $this->write($request); | ||
|  |         $response = unpack('Cversion/Cmethod', $this->read(3)); | ||
|  |         if (5 != $response['version']) { | ||
|  |             throw new HTTP_Request2_MessageException( | ||
|  |                 'Invalid version received from SOCKS5 proxy: ' . $response['version'], | ||
|  |                 HTTP_Request2_Exception::MALFORMED_RESPONSE | ||
|  |             ); | ||
|  |         } | ||
|  |         switch ($response['method']) { | ||
|  |         case 2: | ||
|  |             $this->performAuthentication($username, $password); | ||
|  |         case 0: | ||
|  |             break; | ||
|  |         default: | ||
|  |             throw new HTTP_Request2_ConnectionException( | ||
|  |                 "Connection rejected by proxy due to unsupported auth method" | ||
|  |             ); | ||
|  |         } | ||
|  |     } | ||
|  | 
 | ||
|  |     /** | ||
|  |      * Performs username/password authentication for SOCKS5 | ||
|  |      * | ||
|  |      * @param string $username Proxy user name | ||
|  |      * @param string $password Proxy password | ||
|  |      * | ||
|  |      * @throws HTTP_Request2_ConnectionException | ||
|  |      * @throws HTTP_Request2_MessageException | ||
|  |      * @link http://tools.ietf.org/html/rfc1929 | ||
|  |      */ | ||
|  |     protected function performAuthentication($username, $password) | ||
|  |     { | ||
|  |         $request  = pack('C2', 1, strlen($username)) . $username | ||
|  |                     . pack('C', strlen($password)) . $password; | ||
|  | 
 | ||
|  |         $this->write($request); | ||
|  |         $response = unpack('Cvn/Cstatus', $this->read(3)); | ||
|  |         if (1 != $response['vn'] || 0 != $response['status']) { | ||
|  |             throw new HTTP_Request2_ConnectionException( | ||
|  |                 'Connection rejected by proxy due to invalid username and/or password' | ||
|  |             ); | ||
|  |         } | ||
|  |     } | ||
|  | 
 | ||
|  |     /** | ||
|  |      * Connects to a remote host via proxy | ||
|  |      * | ||
|  |      * @param string $remoteHost Remote host | ||
|  |      * @param int    $remotePort Remote port | ||
|  |      * | ||
|  |      * @throws HTTP_Request2_ConnectionException | ||
|  |      * @throws HTTP_Request2_MessageException | ||
|  |      */ | ||
|  |     public function connect($remoteHost, $remotePort) | ||
|  |     { | ||
|  |         $request = pack('C5', 0x05, 0x01, 0x00, 0x03, strlen($remoteHost)) | ||
|  |                    . $remoteHost . pack('n', $remotePort); | ||
|  | 
 | ||
|  |         $this->write($request); | ||
|  |         $response = unpack('Cversion/Creply/Creserved', $this->read(1024)); | ||
|  |         if (5 != $response['version'] || 0 != $response['reserved']) { | ||
|  |             throw new HTTP_Request2_MessageException( | ||
|  |                 'Invalid response received from SOCKS5 proxy', | ||
|  |                 HTTP_Request2_Exception::MALFORMED_RESPONSE | ||
|  |             ); | ||
|  |         } elseif (0 != $response['reply']) { | ||
|  |             throw new HTTP_Request2_ConnectionException( | ||
|  |                 "Unable to connect to {$remoteHost}:{$remotePort} through SOCKS5 proxy", | ||
|  |                 0, $response['reply'] | ||
|  |             ); | ||
|  |         } | ||
|  |     } | ||
|  | } | ||
|  | ?>
 |