检查 PHP 中是 SOCKS4 还是 5?

Posted

技术标签:

【中文标题】检查 PHP 中是 SOCKS4 还是 5?【英文标题】:Check if SOCKS4 or 5 in PHP? 【发布时间】:2011-12-12 07:21:20 【问题描述】:

基本上,我有一个代理列表。我想将它们分成 SOCKS4 和 SOCKS5。我想编写一个小的 php 脚本来为我做这件事。我将如何检测它在 PHP 中的类型?

【问题讨论】:

你为什么只通过换词问同样的问题两次:***.com/questions/6997113/…。你忘了那个问题吗? @SomnathMuluk 错误。这很奇怪。 我们可以看看你的一些代码吗? @Starx,我还没有这个。我根本不知道该怎么做。 【参考方案1】:

您需要自己编写一些小代码来尝试连接您的任何代理并检查 socks 版本。不同版本的连接协议和错误代码记录在wikipedia page about SOCKS。

考虑到这一点,剩下的或多或少是与 PHP 的标准套接字连接。

例子:

$proxies = array( '66.135.131.74:1681', '172.52.61.244:48943',
                  '75.101.237.217:1080', '76.68.128.165:39879',);

foreach ($proxies as $index => $proxy)

    $type = SOCKSVersion::getType($proxy);
    $typeName = SOCKSVersion::getTypeName($type);
    printf("Proxy #%d: %s\n", $index, $typeName);

输出:

Proxy #0: SOCKS4
Proxy #1: SOCKS4
Proxy #2: Unknown
Proxy #3: SOCKS4

此示例性实现仅检查 SOCKS4,但可以通过添加类似于 isSocks4() 的方法轻松扩展以测试 SOCK4a 和 SOCKS5:

/**
 * SOCKS server identifiation class.
 */
class SOCKSVersion

    const TYPE_UNKNOWN = 0;
    const TYPE_SOCKS4 = 1;
    const TYPE_SOCKS4a = 2;
    const TYPE_SOCKS5 = 3;

    /**
     * @var string[]
     */
    private static $typeNames = array(
        self::TYPE_UNKNOWN => 'Unknown',
        self::TYPE_SOCKS4 => 'SOCKS4',
        self::TYPE_SOCKS4a => 'SOCKS4a',
        self::TYPE_SOCKS5 => 'SOCKS5',
    );

    /**
     * @var int
     */
    private $timeout = 30;

    /**
     * @var int
     */
    private $host, $port;

    /**
     * @var string[]
     */
    private $errors;

    /**
     * @var string[]
     */
    private $socks4Errors = array(
        91 => "Request rejected or failed",
        92 => "Request failed because client is not running identd (or not reachable from the server)",
        93 => "Request failed because client's identd could not confirm the user ID string in the request",
    );

    public function __construct($endpoint)
    
        $this->setEndpoint($endpoint);
    

    /**
     * @static
     * @param string $proxy
     * @return int any of the TYPE_* constants
     */
    public static function getType($proxy)
    
        $socks = new self($proxy);
        return $socks->getSocksVersion();
    

    /**
     * @static
     * @param int $type
     * @return string
     */
    public static function getTypeName($type)
    
        $typeNames = self::$typeNames;
        if (isset($typeNames[$type])) 
            return $typeNames[$type];
        
        return $typeNames[self::TYPE_UNKNOWN];
    

    public function setEndpoint($endpoint)
    
        if (!$parts = parse_url('http://' . $endpoint)) 
            throw new InvalidArgumentException(sprintf('Unable to parse endpoint "%s".', $endpoint));
        
        if (empty($parts['host'])) 
            throw new InvalidArgumentException('No host given.');
        
        if (empty($parts['port'])) 
            throw new InvalidArgumentException('No port given.');
        
        $this->host = $parts['host'];
        $this->port = $parts['port'];
    

    /**
     * @return int any of the TYPE_* constants
     */
    public function getSocksVersion()
    
        try 
            if ($this->isSocks4()) 
                return self::TYPE_SOCKS4;
            
         catch (BadFunctionCallException $e) 
            $this->errors[] = sprintf("SOCKS4 Test: ", $this->host, $e->getMessage());
        
        return self::TYPE_UNKNOWN;
    

    public function isSocks4()
    
        $socket = stream_socket_client("tcp://" . $this->host . ":" . $this->port, $errno, $errstr, $this->timeout, STREAM_CLIENT_CONNECT);
        if (!$socket) 
            throw new BadFunctionCallException(sprintf('Socket-Error #%d: %s', $errno, $errstr));
        

        // SOCKS4; @link <http://en.wikipedia.org/wiki/SOCKS#Protocol>
        $userId = "";
        $packet = "\x04\x01" . pack("n", $this->port) . pack("H*", dechex(ip2long($this->host))) . $userId . "\0";
        fwrite($socket, $packet, strlen($packet));
        $response = fread($socket, 9);
        if (strlen($response) == 8 && (ord($response[0]) == 0 || ord($response[0]) == 4)) 
            $status = ord($response[1]);
            if ($status != 90) 
                throw new BadFunctionCallException(sprintf("Error from SOCKS4 server: %s.", $this->socks4Errors[$status]));
            
         else 
            throw new BadFunctionCallException("The SOCKS server returned an invalid response");
        
        fclose($socket);

        return TRUE;
    

希望这会有所帮助。如果您引入多个版本,则应改进错误处理,并且如果在之前的测试中连接失败,则不要多次连接到同一主机。

【讨论】:

【参考方案2】:

我认为你能做的最好的就是首先尝试通过尝试最高版本 - 5 来建立 CURL 连接。

  curl_setopt($curl, CURLOPT_PROXYTYPE, CURLPROXY_SOCKS5);

无论哪种方式,这都会给您答案。执行后检查curl_error。如果没有错误,您使用的是 SOCKS5,否则您使用的是 SOCKS4。

【讨论】:

【参考方案3】:

根据 RFC1928,要建立 SOCKS 连接,首先要将这些字节发送到服务器:

1 byte      SOCKS version
1 byte      Number of authentication methods (n)
n bytes     List of method identifiers

服务器响应

1 byte      SOCKS version
1 byte      Accepted method

这在第 4 版和第 5 版 SOCKS 中很常见。因此,您可以从一个版本(例如 5)开始,如果服务器没有相应响应,则回退到另一个版本。

【讨论】:

你能提供一个简单的例子吗?我对代理不是很熟悉。 @Mr.Pallazzo:那个例子没有展示如何获取类型,你必须将它作为参数传递.. 这在 SOCKS 的第 4 和第 5 版本之间并不常见。 Socks4 不支持 auth 方法,不响应请求

以上是关于检查 PHP 中是 SOCKS4 还是 5?的主要内容,如果未能解决你的问题,请参考以下文章

检查 Integer 在 groovy 中是 Null 还是数字?

如何检查PHP中的字母是大写还是小写?

如何在PHP中检查它是xml文件还是json文件

file_exists 函数的 PHP 说明,它是检查实际文件还是仅检查路径

PHP 如何检查数字是奇数还是偶数

PHP检查变量是真还是假[重复]