![白帽子安全开发实战](https://wfqqreader-1252317822.image.myqcloud.com/cover/529/37323529/b_37323529.jpg)
2.3 代理服务扫描器
![](https://epubservercos.yuewen.com/BCD655/19773741601353506/epubprivate/OEBPS/Images/70_01.jpg?sign=1739372859-aHj85aWTwFDd3Vn03muyFeg5OGJ2gefg-0-3e17799281d10c32b0e88dbc4ff0249b)
代理服务扫描器的作用是判断一个端口上是否监听了代理服务,它的使用场景有以下几个。
● 定期扫描自己公司的服务器,排查是否有外网服务器开启了代理服务。如果运维或研发人员安全意识薄弱,为了方便在服务器中部署了代理软件,攻击者发现后就可以通过代理绕过网络边界设置的种种防御措施直接进入内网,渗透防御措施相对薄弱的内网服务器。
● 扫描并搜集公网上的代理,用来完善代理池。
● 批量检测一些代理网站上提供的免费代理是否有效。
2.3.1 HTTP/HTTPS代理检测模块
Go语言标准库的net/http包中有HTTP客户端和服务器的具体实现,利用此包可以很方便地编写HTTP客户端和服务器端的程序。例如,以下代码可以发起HTTP的GET请求:
![](https://epubservercos.yuewen.com/BCD655/19773741601353506/epubprivate/OEBPS/Images/70_02.jpg?sign=1739372859-fLy7vTZ3Lsd4c8b7AkM3aqwFcpFXxMF1-0-a84ddb536d528806b3a635cc2e9e65d8)
在Sublime Text中执行后会返回笔者blog的内容,如图2-12所示。
![](https://epubservercos.yuewen.com/BCD655/19773741601353506/epubprivate/OEBPS/Images/71_01.jpg?sign=1739372859-ubyjhI8jbBpB2GJzDWpvlE40nWXrspH0-0-91e9ac334cc9aeb882aafe5c41d06487)
●图2-12 Go的HTTP客户端测试
HTTP/HTTPS代理服务扫描的原理是利用代理发起HTTP请求,如果利用代理能访问目标网站,则该代理是有效的。
为上面HTTP请求增加使用代理的功能,自定义一个HTTP Client对象即可,详细的代码如下所示:
![](https://epubservercos.yuewen.com/BCD655/19773741601353506/epubprivate/OEBPS/Images/71_02.jpg?sign=1739372859-LRozFedtCLhgQZdVQyicVPeEnwsZWN4p-0-49e398e6e621bb0c6d5a10547d51b18a)
![](https://epubservercos.yuewen.com/BCD655/19773741601353506/epubprivate/OEBPS/Images/72_01.jpg?sign=1739372859-CPFi92T6MYKNdx39EIE4SGRwocqCtAnm-0-c9d7ed299ed12db5ce78c5e316610af6)
给HTTP Client增加使用代理的功能后再请求目标网站,返回的情况有以下两种。
● 代理的IP或端口无法访问,此时会返回connect:connection refused错误,说明代理是无效的。
● 如果测试的代理是Web服务,会返回此Web服务的内容,如果只判断是否有数据返回,会造成误报,如图2-13所示。
![](https://epubservercos.yuewen.com/BCD655/19773741601353506/epubprivate/OEBPS/Images/72_02.jpg?sign=1739372859-vyL8dHnR0HqOiAOKmfDPc1UrMcboEbBC-0-fc751bf80cafffaf5a2d6a8616bc338c)
●图2-13 代理检测误报效果
解决误报的方式是给HTTP Client增加代理后,访问一个特定的网站,判断返回的内容中是否包含特定的关键字。例如,访问http://email.163.com/,如果返回的内容中包含“<title>网易免费邮箱”这几个关键字,说明测试的代理是有效的。
除了使用Go标准库中的net/http包外,也可以使用第三方包来实现HTTP代理测试的功能,如github.com/parnurzeal/gorequest、github.com/levigross/grequests包等。grequests包类似于Python语言中的requests库。
使用github.com/parnurzeal/gorequest包通过代理提交HTTP请求的示例代码如下所示:
![](https://epubservercos.yuewen.com/BCD655/19773741601353506/epubprivate/OEBPS/Images/73_01.jpg?sign=1739372859-Rpolxgbu9txFQDKfV53n0rCfwJIOkFsP-0-e894112e585e3c9e3682ef90192da508)
可以看出,利用此包给HTTP请求增加使用代理的功能非常方便,只要调用gorequest对象的Proxy方法即可。运行结果如图2-14所示。
![](https://epubservercos.yuewen.com/BCD655/19773741601353506/epubprivate/OEBPS/Images/73_02.jpg?sign=1739372859-vIOgDltYcM6J0rEeZdKDe42zva2yyO1I-0-59dfa9a47a78840997bbeba0fedd9ba9)
●图2-14 gorequest测试
使用github.com/levigross/grequests包通过代理提交HTTP请求的代码如下所示:
![](https://epubservercos.yuewen.com/BCD655/19773741601353506/epubprivate/OEBPS/Images/74_01.jpg?sign=1739372859-pitQ5R8dUcLTEdCAC7cn0C85ljljcKEG-0-cf356720547035f97dcd66bbba44846d)
从以上示例可以得知,grequests库是通过grequests.RequestOptions来设置代理的。运行结果如图2-15所示。
![](https://epubservercos.yuewen.com/BCD655/19773741601353506/epubprivate/OEBPS/Images/74_02.jpg?sign=1739372859-pSBjVY2gOX4n2TbneQzt4Xv9H2HulIlE-0-1d27419f2c5e6b0974e235d5ca76f24b)
●图2-15 grequests测试
前面介绍使用HTTP代表的3种不同的方式,为了减少对外部的依赖,笔者采用Go语言标准库的net/http包,并封装了一个HTTP代理检测函数,详细的代码如下所示:
![](https://epubservercos.yuewen.com/BCD655/19773741601353506/epubprivate/OEBPS/Images/75_01.jpg?sign=1739372859-6FPGrCipKdvC3cwI7cCXhdGx9g9ut0q1-0-c3da4912611668e33882675d24646cb9)
![](https://epubservercos.yuewen.com/BCD655/19773741601353506/epubprivate/OEBPS/Images/76_01.jpg?sign=1739372859-wAM8AVy6I8oxuW2Q5MX2HlnC36hBTGp5-0-dc98717e21912f3f0239d27e43d16608)
以上代码中,变量HttpProxyProtocol与WebUrl的含义分别如下所述。
● HttpProxyProtocol中定义了检测的代理协议的种类,如HTTP代表与HTTPS代理。
● WebUrl表示用代理访问的Web,用于通过返回的内容判断代理是否有效。
2.3.2 SOCKS代理检测模块
Go语言的"golang.org/x/net/proxy"包支持给HTTP客户端增加SOCKS5代理。在使用这个包之前,需要通过go get-u golang.org/x/net/proxy命令进行安装。
如果无法访问golang.org这个域名,可以使用https://goproxy.io/或https://goproxy.cn/等Go模块代理。
golang.org/x/net/proxy包的使用方法如下。
1)创建一个dialer,它包含了SOCK5代理服务器的地址、用户名和密码。下面是创建dialer的代码,这里假设代理服务器的IP为sec.lu,端口为1080,用户名为xsec,密码为xsec,代码片断如下所示:
![](https://epubservercos.yuewen.com/BCD655/19773741601353506/epubprivate/OEBPS/Images/76_02.jpg?sign=1739372859-sRjJBUu90gQPR63XA0Fwf16ynAKltz7c-0-9c58fa829813e62d1ef98f5531d668d1)
如果代理服务器并不需要用户名和密码,可以将proxy.SOCKS5函数的第三个参数设置为nil。
2)创建一个transport,它会利用刚才创建的dialer进行TCP连接,创建transport的代码如下所示:
![](https://epubservercos.yuewen.com/BCD655/19773741601353506/epubprivate/OEBPS/Images/76_03.jpg?sign=1739372859-yUOp7y9k43j1CzpcO0MdlNKz0wHunjcZ-0-d037f8579a3dd7127c137d8a19de77b9)
3)创建一个HTTP Client就可使用SOCKS5代理了,代码如下所示:
![](https://epubservercos.yuewen.com/BCD655/19773741601353506/epubprivate/OEBPS/Images/77_01.jpg?sign=1739372859-XcrNvAj0Ivus89YVHbkCP61FuPQMVBUc-0-640ee5a6f64fe86b97352b55b0b0d0d1)
完整的示例代码如下所示:
![](https://epubservercos.yuewen.com/BCD655/19773741601353506/epubprivate/OEBPS/Images/77_02.jpg?sign=1739372859-0hw0t3Y2sohXBhLVDMm5o7QejcTCeAPb-0-6a88783b793a47b0d5acd7117799820f)
这个示例的作用是检测sec.lu:1080这个SOCKS5代理是否有效,可以在Sublime Text中通过GoSublime直接运行以上代码,结果如图2-16所示。
![](https://epubservercos.yuewen.com/BCD655/19773741601353506/epubprivate/OEBPS/Images/78_01.jpg?sign=1739372859-ELTKzE960DtTQzjpjDR6srMRnv5mdfgT-0-50df2dd09536e04122287f0c6f715a9e)
●图2-16 SOCKS5代理测试
另外还有一个第三方库h12.io/socks也可以实现使用SOCKS5代理的功能,它不仅支持SOCKS5,还支持SOCKS4, SOCKS4A。使用方法如下所示:
![](https://epubservercos.yuewen.com/BCD655/19773741601353506/epubprivate/OEBPS/Images/78_02.jpg?sign=1739372859-UymmDYSyqE8cc9QWRaLNOeeaqiM3hKqR-0-fc5161a454d826062ffb61a4e9b509aa)
![](https://epubservercos.yuewen.com/BCD655/19773741601353506/epubprivate/OEBPS/Images/79_01.jpg?sign=1739372859-XxzICVuxY7KNjr87ZgSpRZkNPif6z2iK-0-6058f1c478ac44940dbd3226413991d2)
这里使用h12.io/socks包实现SOCKS代理检测模块。按照与HTTP代理检测模块相同的函数原型封装一个SOCKS代理检测函数,详细的代码如下所示:
![](https://epubservercos.yuewen.com/BCD655/19773741601353506/epubprivate/OEBPS/Images/79_02.jpg?sign=1739372859-0YxWoa5L5tzh5908WWsCG9A5WckuJtcU-0-8f4767f44fa471e69efab9fce7f9b9b2)
![](https://epubservercos.yuewen.com/BCD655/19773741601353506/epubprivate/OEBPS/Images/80_01.jpg?sign=1739372859-jIztj9LmO1iQl2RMb25hjei9xaKN3Lhy-0-1a9cd5f472988e28a179c884264afbc4)
SockProxyProtocol是需要检测协议的map,检测程序会测试这个map中的协议是否有效。
2.3.3 代理服务扫描模块任务执行功能的实现
前面已经实现了HTTP与SOCKS代理检测的功能,接下来实现扫描任务的调度功能。程序的执行流程如下。
1)从文件列表中读取待检测的代理服务器的IP、端口列表。
2)将待扫描的ipList分割为组,按组进行扫描。
1.读取待扫描的ipList的功能的实现
待扫描的ipList的文件格式为每行一个ip:port,如下所示:
![](https://epubservercos.yuewen.com/BCD655/19773741601353506/epubprivate/OEBPS/Images/80_02.jpg?sign=1739372859-NnSY3CmCyywObANzqeI2i9xOFy3LmexG-0-da0a28ce347e8a7e6303ed9a6ff5db3b)
定义一个ProxyAddr结构用来保存每条待扫描的记录,如下所示:
![](https://epubservercos.yuewen.com/BCD655/19773741601353506/epubprivate/OEBPS/Images/80_03.jpg?sign=1739372859-35e43sYXo1TK4q1x0UeBcAzLtsk2ZZ3x-0-04ec625de168dcad1e2bd651fd557aa4)
使用标准库中的bufio包逐行读取文件,然后用strings包进行分割,最后返回一个[]ProxyAddr即可,详细的代码如下所示:
![](https://epubservercos.yuewen.com/BCD655/19773741601353506/epubprivate/OEBPS/Images/80_04.jpg?sign=1739372859-5QaPRwsej9bAxV3K4lW6uwJpNxSD6gaT-0-ab009fc28d7f319392423c39fa91a2e8)
![](https://epubservercos.yuewen.com/BCD655/19773741601353506/epubprivate/OEBPS/Images/81_01.jpg?sign=1739372859-LdgtrzCWqN82plNr45gKi66krBvf7EPJ-0-9b9fc2c2f9329b608f8145a4eea0e771)
2.扫描任务分割与调度的功能实现
扫描任务的组数是根据待扫描的ipList的总数与扫描的并发数计算出来的,每次最多扫描ScanNum个ipList,代码如下所示:
![](https://epubservercos.yuewen.com/BCD655/19773741601353506/epubprivate/OEBPS/Images/81_02.jpg?sign=1739372859-o7Z3qZK61vu6YPR1YZ9Y119b0FEXvPaj-0-5f6c9ff5ebf56aad929ff6024a1b6ead)
![](https://epubservercos.yuewen.com/BCD655/19773741601353506/epubprivate/OEBPS/Images/82_01.jpg?sign=1739372859-VBu4TjTvDHIlO2WynmpemhBowQLyqTZP-0-89ded4eaa98d081b817f3bc33f90eb91)
CheckProxy的作用是检测一组ipList,其中包括检测HTTP代表与SOCKS代理。HTTP代表有HTTP/HTTPS两种,SOCKS代理有SOCKS4、SOCKS4A与SOCKS5三种,详细的代码如下所示:
![](https://epubservercos.yuewen.com/BCD655/19773741601353506/epubprivate/OEBPS/Images/82_02.jpg?sign=1739372859-STrjcS2DOYmNDiuP0amVgr1vJTMVbJqv-0-bc54b4711d916d14aebc5ec3d677e1c0)
2.3.4 代理扫描器命令行的实现
最终实现的代理扫描器的命令行参数为./main scan-debug=true-scan_num=30-timeout=10-filename=iplist.txt,每个参数的含义如下。
● scan表示用默认参数启动代理服务器的扫描。
● --debug表示是否启用Debug模式,值为true和false。
● --scan_num表示扫描的并发数,默认值为100。
● --timeout表示每个扫描连接的超时时间,默认值为5s。
● --filename表示等待扫描的代理服务器的列表,默认值为iplist.txt。
前面多次介绍过github.com/urfave/cli包的用法,实现以上命令行参数只需要定义一个cli.Command即可,完整的代码如下所示:
![](https://epubservercos.yuewen.com/BCD655/19773741601353506/epubprivate/OEBPS/Images/83_01.jpg?sign=1739372859-mQPrjcVaDFykPr5jlrzryH3nKPPNzrOV-0-15ba95fdc5d59526d5a45aa6c46f9f69)
实际的参数处理与任务调度入口在proxy.Scan的Action中,代码可以参考scanner/proxy-scanner/proxy/proxy.go文件。
2.3.5 代理扫描器测试
代理扫描器已经开发完成,其命令行参数如图2-17所示。
![](https://epubservercos.yuewen.com/BCD655/19773741601353506/epubprivate/OEBPS/Images/83_02.jpg?sign=1739372859-2f9C2nLUNokk7WigS8iv4RN6djl2d9oV-0-b002bc4dbc9a2e7a335fa828a542c60d)
●图2-17 代理扫描器命令行参数
有一些网站提供了免费的代理服务器列表,如http://free-proxy.cz/en/网站,可以选择代理的类型、国家,这里选择China,单击Filter proxies可以过滤出国家为中国的代理服务器,最后单击Export IP:Port按钮即可将IP:Port复制出来,如图2-18所示。
![](https://epubservercos.yuewen.com/BCD655/19773741601353506/epubprivate/OEBPS/Images/84_01.jpg?sign=1739372859-ceEMrVPJwyFEephieOupkb1CDcEnLUKi-0-4a159c72244036773b04a7e241a09219)
●图2-18 免费代理网站
将IP:Port复制到代理扫描器当前目录中的iplist.txt中,然后通过以后命令启动扫描:
![](https://epubservercos.yuewen.com/BCD655/19773741601353506/epubprivate/OEBPS/Images/84_02.jpg?sign=1739372859-PdOR5dNHRhxeZdNG2Vd9a5eU00kxzxEB-0-d50bba7f7ba0291029073837989bcb89)
等待几秒钟后,扫描器输出了结果,用时大约5s,输出结果中有两个是SOCKS5代理,其他的全是SOCKS4或SOCKS4A代理,如图2-19所示。
![](https://epubservercos.yuewen.com/BCD655/19773741601353506/epubprivate/OEBPS/Images/84_03.jpg?sign=1739372859-Pk1YMvEBayYDYwtUJAuQEocKAAmkVgEh-0-2035d6b9f6447c725fd2d03559888e52)
●图2-19 代理验证效果
接下来用QQ自带的代理测试功能来测试扫描结果中SOCKS5代理的有效性,最终得出的结论是代理扫描器的检测结果是准确的,如图2-20所示。
![](https://epubservercos.yuewen.com/BCD655/19773741601353506/epubprivate/OEBPS/Images/85_01.jpg?sign=1739372859-ZA1EhNSBDE17UOabvytB95AtyoCzijXD-0-76dee2dede189c798eb1c3276951c0a0)
●图2-20 QQ的代理测试效果