基于go手动写个转发代理服务的代码实现
由于公司经常需要异地办公,在调试的时候需要用到内网环境,因此手动写了个代理转发服务器給兄弟们用:socks5proxy。
选型上,语言上就选择了Go,简单清晰,转发协议选择了socks5。
SOCKS5协议介绍
SOCKS是一种网络传输协议,主要用于客户端与外网服务器之间通讯的中间传递,SOCKS是"SOCKetS"的缩写。SOCKS5是SOCKS4的升级版,其主要多了鉴定、IPv6、UDP支持。
SOCKS5协议可以分为三个部分:
(1)协议版本及认证方式
(2)根据认证方式执行对应的认证
(3)请求信息
(1)协议版本及认证方式
创建与SOCKS5服务器的TCP连接后客户端需要先发送请求来协议版本及认证方式,
VER | NMETHODS | METHODS |
---|---|---|
1 | 1 | 1-255 |
- VER是SOCKS版本,这里应该是0x05;
- NMETHODS是METHODS部分的长度;
- METHODS是客户端支持的认证方式列表,每个方法占1字节。当前的定义是:
- 0x00不需要认证
- 0x01GSSAPI
- 0x02用户名、密码认证
- 0x03-0x7F由IANA分配(保留)
- 0x80-0xFE为私人方法保留
- 0xFF无可接受的方法
服务器回复客户端可用方法:
VER | METHOD |
---|---|
1 | 1 |
- VER是SOCKS版本,这里应该是0x05;
- METHOD是服务端选中的方法。如果返回0xFF表示没有一个认证方法被选中,客户端需要关闭连接。
代码实现:
typeProtocolVersionstruct{ VERuint8 NMETHODSuint8 METHODS[]uint8 } func(s*ProtocolVersion)handshake(connnet.Conn)error{ b:=make([]byte,255) n,err:=conn.Read(b) iferr!=nil{ log.Println(err) returnerr } s.VER=b[0]//ReadBytereadsandreturnsasinglebyte,第一个参数为socks的版本号 s.NMETHODS=b[1]//nmethods是记录methods的长度的。nmethods的长度是1个字节 ifn!=int(2+s.NMETHODS){ returnerrors.New("协议错误,sNMETHODS不对") } s.METHODS=b[2:2+s.NMETHODS]//读取指定长度信息,读取正好len(buf)长度的字节。如果字节数不是指定长度,则返回错误信息和正确的字节数 ifs.VER!=5{ returnerrors.New("该协议不是socks5协议") } //服务器回应客户端消息: //第一个参数表示版本号为5,即socks5协议, //第二个参数表示服务端选中的认证方法,0即无需密码访问,2表示需要用户名和密码进行验证。 resp:=[]byte{5,0} conn.Write(resp) returnnil }
(2)根据认证方式执行对应的认证
SOCKS5协议提供5种认证方式:
- 0x00不需要认证
- 0x01GSSAPI
- 0x02用户名、密码认证
- 0x03-0x7F由IANA分配(保留)
- 0x80-0xFE为私人方法保留
这里就主要介绍用户名、密码认证。在客户端、服务端协商使用用户名密码认证后,客户端发出用户名密码:
鉴定协议版本 | 用户名长度 | 用户名 | 密码长度 | 密码 |
---|---|---|---|---|
1 | 1 | 动态 | 1 | 动态 |
服务器鉴定后发出如下回应:
其中鉴定状态0x00表示成功,0x01表示失败。
代码实现:
typeSocks5Authstruct{ VERuint8 ULENuint8 UNAMEstring PLENuint8 PASSWDstring } func(s*Socks5Auth)authenticate(connnet.Conn)error{ b:=make([]byte,128) n,err:=conn.Read(b) iferr!=nil{ log.Println(err) returnerr } s.VER=b[0] ifs.VER!=5{ returnerrors.New("该协议不是socks5协议") } s.ULEN=b[1] s.UNAME=string(b[2:2+s.ULEN]) s.PLEN=b[2+s.ULEN+1] s.PASSWD=string(b[n-int(s.PLEN):n]) log.Println(s.UNAME,s.PASSWD) ifusername!=s.UNAME||passwd!=s.PASSWD{ returnerrors.New("账号密码错误") } /** 回应客户端,响应客户端连接成功 TheserververifiesthesuppliedUNAMEandPASSWD,andsendsthe followingresponse: +----+--------+ |VER|STATUS| +----+--------+ |1|1| +----+--------+ ASTATUSfieldofX'00'indicatessuccess.Iftheserverreturnsa `failure'(STATUSvalueotherthanX'00')status,itMUSTclosethe connection. */ resp:=[]byte{0x05,0x00} conn.Write(resp) returnnil }
但其实,现在大家都习惯自己采用加密流的方式进行加密,很少采用用户名密码的形式进行加密,后面章节会介绍一种对SOCKS的混淆加密方式。
(3)请求信息
认证结束后客户端就可以发送请求信息。如果认证方法有特殊封装要求,请求必须按照方法所定义的方式进行封装解密,其原始格式如下:
VER | CMD | RSV | ATYP | DST.ADDR | DST.PORT |
---|---|---|---|---|---|
1 | 1 | 0x00 | 1 | 动态 | 2 |
- VER是SOCKS版本,这里应该是0x05;
- CMD是SOCK的命令码
- 0x01表示CONNECT请求
- 0x02表示BIND请求
- 0x03表示UDP转发
- RSV0x00,保留
- ATYPDST.ADDR类型
- DST.ADDR目的地址
- 0x01IPv4地址,DST.ADDR部分4字节长度
- 0x03域名,DST.ADDR部分第一个字节为域名长度,DST.ADDR剩余的内容为域名,没有\0结尾。
- 0x04IPv6地址,16个字节长度。
- DST.PORT网络字节序表示的目的端口
代码实现:
typeSocks5Resolutionstruct{ VERuint8 CMDuint8 RSVuint8 ATYPuint8 DSTADDR[]byte DSTPORTuint16 DSTDOMAINstring RAWADDR*net.TCPAddr } func(s*Socks5Resolution)lstRequest(connnet.Conn)error{ b:=make([]byte,128) n,err:=conn.Read(b) iferr!=nil||n<7{ log.Println(err) returnerrors.New("请求协议错误") } s.VER=b[0] ifs.VER!=5{ returnerrors.New("该协议不是socks5协议") } s.CMD=b[1] ifs.CMD!=1{ returnerrors.New("客户端请求类型不为代理连接,其他功能暂时不支持.") } s.RSV=b[2]//RSV保留字端,值长度为1个字节 s.ATYP=b[3] switchs.ATYP{ case1: //IPV4address:X'01' s.DSTADDR=b[4:4+net.IPv4len] case3: //DOMAINNAME:X'03' s.DSTDOMAIN=string(b[5:n-2]) ipAddr,err:=net.ResolveIPAddr("ip",s.DSTDOMAIN) iferr!=nil{ returnerr } s.DSTADDR=ipAddr.IP case4: //IPV6address:X'04' s.DSTADDR=b[4:4+net.IPv6len] default: returnerrors.New("IP地址错误") } s.DSTPORT=binary.BigEndian.Uint16(b[n-2:n]) //DSTADDR全部换成IP地址,可以防止DNS污染和封杀 s.RAWADDR=&net.TCPAddr{ IP:s.DSTADDR, Port:int(s.DSTPORT), } /** 回应客户端,响应客户端连接成功 +----+-----+-------+------+----------+----------+ |VER|REP|RSV|ATYP|BND.ADDR|BND.PORT| +----+-----+-------+------+----------+----------+ |1|1|X'00'|1|Variable|2| +----+-----+-------+------+----------+----------+ */ resp:=[]byte{0x05,0x00,0x00,0x01,0x00,0x00,0x00,0x00,0x00,0x00} conn.Write(resp) returnnil }
(4)最后将信息进行转发即可
代码实现:
wg:=new(sync.WaitGroup) wg.Add(2) gofunc(){ deferwg.Done() deferdstServer.Close() io.Copy(dstServer,client) }() gofunc(){ deferwg.Done() deferclient.Close() io.Copy(client,dstServer) }() wg.Wait()
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持毛票票。