C# HttpWebRequest 如何实现SOCKET5代理
Posted
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了C# HttpWebRequest 如何实现SOCKET5代理相关的知识,希望对你有一定的参考价值。
如题 好像只找到了HTTP 代理 我想链接SOCKET5然后去实现网页浏览怎么 实现 难道只能从底层构造SOCKET??
C# 实现Socket5代理协议通讯 这里主要讲的是用.NET实现基于Socket5下面的代理协议进行客户端的通讯,Socket4的实现是类似的,注意的事,这里不是讲用C#实现一个代理服务器,因为实现一个代理服务器需要实现很多协议,头大,而且现在市面上有很多现成的代理服务器用,性能又好,直接用不好吗?而是用C#来实现客户端通过一个代理服务器进行Socket5的通讯,这个简单点,一般来说用Socket5就好了,Socket4现在也逐渐淘汰,基本上代理服务器都支持Socket5。首先我们要了解下Socket5的代理协议是基于TCP协议之上的,从Socket4扩展上来的,用于提供给其他协议例如HTTP、FTP所作用的一套防火墙协议。(这里做点小解释,实际上应该是SOCKS V5协议,但是目前网上都用Socket5这样称呼,目前本文也暂时这样称) 然后下面简单的说下该协议的内容,具体分为两部分:
一部分是基于TCP协议的客户
当一个基于TCP协议的客户端希望与一个只能通过防火墙可以到达的目标(这是由实现所决定的)建立连接,它必须先建立一个与SOCKS服务器上SOCKS 端口的TCP连接。通常这个TCP端口是1080。当连接建立后,客户端进入协议的“握手(negotiation)”过程:认证方式的选择,根据选中的方式进行认证,然后发送转发的要求。SOCKS服务器检查这个要求,根据结果,或建立合适的连接,或拒绝。是不是有点像FTP?差不多。
握手的过程:
客户端连到服务器后,然后就发送请求来协商版本和认证方法: VERNMETHODSMETHODS111 to 255 这个版本的SOCKS协议中,VER字段被设置成X'05'。NMETHODS字段包含了在METHODS字段中出现的方法标示的数目(以字节为单位)。简化就是像服务器发送05 01 00
服务器从这些给定的方法中选择一个并发送一个方法选中的消息回客户端: VERMETHOD11如果选中的消息是X’FF’,这表示客户端所列出的方法列表中没有一个方法被选中,客户端必须关闭连接。
这里我们关注的是 X’00’ 不需要认证和X’02’ 用户名/密码,简化是服务器会返回的前两个字节
会是05 00或者05 02,02的时候进行验证。
验证过程结束后,客户端就发送详细的请求信息。如果协商的方法中有以完整性检查和/或安全性为目的的封装,这些请求必须按照该方法所定义的方式进行封装。
握手完成之后,要进行一个请求连接,这个就是对远程服务器的连接,我们知道当前既然连接的是代理服务器,我们实际上发送数据的对象并不是它,也就是我们要告诉代理服务器去连接真实的某某,现在就是在做这一步了。
SOCKS请求的格式如下: VERCMDRSVATYPDST.ADDRDST.PROT11X’00’1Variable2其中
· VER 协议版本: X’05’
· CMD
· CONNECT:X’01’
· BIND:X’02’
· UDP ASSOCIATE:X’03’
· RSV 保留
· ATYP 后面的地址类型
· IPV4:X’01’
· 域名:X’03’
· IPV6:X’04’'
· DST.ADDR 目的地址
· DST.PORT 以网络字节顺序出现的端口号
SOCKS服务器会根据源地址和目的地址来分析请求,然后根据请求类型返回一个或多个应答。
ATYP字段中描述了地址字段(DST.ADDR,BND.ADDR)所包含的地址类型:
· X'01'
基于IPV4的IP地址,4个字节长
· X'03'
基于域名的地址,地址字段中的第一字节是以字节为单位的该域名的长度,没有结尾的NUL字节。
· X'04'
基于IPV6的IP地址,16个字节长。
Variable表示该域的长度是可变的。
以最常用的IP表示法为例,加入我们的代理服务器地址是10.10.1.254,现在我们要通过代理服务器访问192.168.1.2这个IP地址,这里不要看做局域网地址,因为是通过代理进行访问的,
前期传递了IP对象IPEndPoint destIP;
byte [] data = new byte[10];
data[0]=5;data[1]=1;data[2]=0;data[3]=1;//前4个字节
Array.Copy(destIP.Address.GetAddressBytes(), 0, data, 4, 4); //IP地址
Array.Copy(BitConverter.GetBytes(
IPAddress.HostToNetworkOrder(destIP.Port)), 2, data, 8, 2); //端口号
这个data就是要发送的请求了.
代理服务器这边会根据请求,以如下格式返回: VERREPRSVATYPBND.ADDRBND.PORT11X’00’1Variable2其中:
· VER 协议版本: X’05’
· REP 应答字段:
· X’00’ 成功
· X’01’ 普通的SOCKS服务器请求失败
· X’02’ 现有的规则不允许的连接
· X’03’ 网络不可达
· X’04’ 主机不可达
· X’05’ 连接被拒
· X’06’ TTL超时
· X’07’ 不支持的命令
· X’08’ 不支持的地址类型
· X’09’ – X’FF’ 未定义
· RSV 保留
· ATYP 后面的地址类型
· IPV4:X’01’
· 域名:X’03’
· IPV6:X’04’
· BND.ADDR 服务器绑定的地址
· BND.PORT 以网络字节顺序表示的服务器绑定的段口
标识为RSV的字段必须设为X’00’。返回的信息里面都看到,最重要是第二个域,如果为0,就表示成功,这第二个域对于接收到bytes来说是第1个字节,及data[1].
如果选中的方法中有以完整性检查和/或安全性为目的的封装,这些应答必须按照该方法所定义的方式进行封装。完成这步请求就完成了Socket5整个TCP客户端的连接了,剩下的工作,你就如正常的连接一下向代理服务器发送数据,简单来说这时候你就把代理服务器看成远程你要连接的对象就行了,代理服务器此时就是个透明的网络连接。至于整个C#示例,可以到我的空间下载,这个是博客园文件下载链接。 对于UDP的客户端连接,这里给出协议规范,可以自行进行实验。
在UDP ASSOCIATE应答中由BND.PORT指明了服务器所使用的UDP端口,一个基于UDP协议的客户必须发送数据报至UDP转发服务器的该端口上。如果协商的认证方法中有以完整性、认证和/或安全性为目的的封装,这些数据报必须按照该方法所定义的方式进行封装。每个UDP数据报都有一个UDP请求头在其首部: RSVFRAGATYPDST.ADDRDST.PORTDATA211Variable2Variable在UDP请求头中的字段是:· RSV 保留 X’0000’
· FRAG 当前的分段号
· ATYP 后面的地址类型
· IPV4:X’01’
· 域名:X’03’
· IPV6:X’04’
· DST.ADDR 目的地址
· DST.PORT 以网络字节顺序出现的端口号
· DATA 用户数据
当一个UDP转发服务器转发一个UDP数据报时,不会发送任何通知给客户端;同样,它也将丢弃任何它不能发至远端主机的数据报。当UDP转发服务器从远端服务器收到一个应答的数据报时,必须加上上述UDP请求头,并对数据报进行封装。
UDP转发服务器必须从SOCKS服务器得到期望的客户端IP地址,并将数据报发送到UDP ASSOCIATE应答中给定的端口号。如果数据报从任何IP地址到来,而该IP地址与该特定连接中指定的IP地址不同,那么该数据报会被丢弃。
FRAG字段指明数据报是否是一些分片中的一片。如果SOCKS服务器要实现这个功能,X’00’指明数据报是独立的;其他则越大越是数据报的尾端。介于 1到127之间的值说明了该分片在分片序列里的位置。每个接收者都为这些分片提供一个重组队列和一个重组的计时器。这个重组队列必须在重组计时器超时后重新初始化,并丢弃相应的数据报。或者当一个新到达的数据报有一个比当前在处理的数据报序列中最大的FRAG值要小时,也必须重新初始化从组队列。重组计时器必须小于5秒。只要有可能,应用程序最好不要使用分片。
分片的实现是可选的;如果某实现不支持分片,所有FRAG字段不为0的数据报都必须被丢弃。
一个SOCKS的UDP编程界面(The programming interface for a SOCKS-aware UDP)必须报告当前可用UDP数据报缓存空间小于操作系统提供的实际空间。
· 如果 ATYP是 X’01’ - 10+method_dependent octets smaller
· 如果 ATYP是X’03’ - 262+method_dependent octets smaller
· 如果 ATYP是X’04’ - 20+method_dependent octets smaller
参考网址: http://www.cnblogs.com/zhujiechang/archive/2008/10/21/1316308.html 参考技术A 具体的代码请参考如下内容
http://wenda.io/questions/293920/use-webclient-with-socks-proxy.html
请注意,不支持 WebRequest/WebResponse 参考技术B 你都说了是http 你怎么用tcp的东西?
c# httpwebrequest 模拟登陆
登录的时候有验证码,如果拿到这个验证码文件(php文件)现在可以用程序单独识别这个验证码了,但是在post提交登录的时候 提交用户名和密码的时候怎么将这个验证码提交上去呢,验证码是刷新一次就会变的,
要实现的功能:
1、用户名加密码加验证码实现登录,并得到cookie。cookie登录是需要刷新一次的吧,不然登录不成功。 (验证码识别这块已经解决,重点是如何结合httpwebrequest来提交这个验证码)
2、然后用cookie去访问某个页面返回其源码
您如果是做开发最好是下个firefox 或者Google的浏览器
firefox里面自带了个“开发人员工具"或者下个firebug更好,它可以更全面的为您记录/浏览,html元素/脚本/css/http请求等
在http请求中您很方便的看到目标网页是如何将这些参数组织并传递给后台页面的。
至于Cookie,这个基本会在后台服务器端生成,如果在客户端生成的话基本上这个网站也就是个幼儿园级别的了,所以我们无法调用某接口为自己创建Cookie
但是,如果我们分析清楚,目标网站是如何利用参数登录的,包括都调用了什么js脚本/在哪个框架中提出http请求的,那么咱们完全模拟的话,就可以让它的服务器端程序为咱们创建合法的Cookie
再就是现在很多网站利用类似检查盗链的方式检查该http请求是否是从我理想的位置提出的,如果不是就拒绝提供服务。
如果真的比较棘手的话我们还可以分析生成Cookie,通过Cookie欺骗,完成利用Cookie进行访问的过程。 参考技术A 先获取验证码保存cookie然后模拟登录的时候带上已经保存的cookie就可以了追问
那要用到get方法哈
追答用的 Get方法 获取验证码然后保存cookie 提交的时候发送cookie就行了
追问能给个例子吗 最好用dedecms的登录和文章发布
追答public class Login
CookieContainer cookies = new CookieContainer();
HttpWebRequest wrq;
byte[] onedata = ;
//
/// 获取Cookie
///
public void LoadCookie()
try
string src = "网页地址";
wrq = (HttpWebRequest)WebRequest.Create(src);
wrq.CookieContainer = cookies;
wrq.Method = "POST";
wrq.Timeout = 10000;
wrq.ContentType = "application/x-www-form-urlencoded";
wrq.ContentLength = onedata.Length;
Stream myseream = wrq.GetRequestStream();
myseream.Write(onedata, 0, onedata.Length);
catch
///
/// 获取验证码方法
///
public void LoadImg()
string img = "验证码地址";
try
wrq = (HttpWebRequest)WebRequest.Create(img);//请求的URL
wrq.CookieContainer = cookies;//*发送COOKIE
wrq.Method = "GET";
wrq.ContentType = "application/x-www-form-urlencoded";
wrq.Timeout = 10000;
//获取返回资源
HttpWebResponse response = (HttpWebResponse)wrq.GetResponse();
//获取流
Image bitmapImage = Bitmap.FromStream(response.GetResponseStream()) as Bitmap;
im = bitmapImage;
this.pbimg.Image = bitmapImage;
catch
贴不上去了 post提交的时候 带上wrq.CookieContainer = cookies;//*发送COOKIE
麻烦把这个源码发一份给我吧 243652695的qq邮箱
本回答被提问者采纳 参考技术B 已发送,请检查追问发送到哪里啦
以上是关于C# HttpWebRequest 如何实现SOCKET5代理的主要内容,如果未能解决你的问题,请参考以下文章
用c#编写webservice时,如何获取调用它的httpwebrequest中?后面的参数 非常感谢。
C#通过WebClient/HttpWebRequest实现http的post/get方法
如何在 C# 中设置 HttpWebRequest 的内容?
C# HttpWebRequest - 如何确定是不是发生了 HTTP 301?