123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193 |
- //go:build windows
- package yu_hpsocket
- import (
- yu_base64 "gogs.qqck.cn/s/tools/base64"
- yu_fast "gogs.qqck.cn/s/tools/fast"
- yu_proxy "gogs.qqck.cn/s/tools/proxy"
- yu_strconv "gogs.qqck.cn/s/tools/strconv"
- yu_sys "gogs.qqck.cn/s/tools/sys"
- "net"
- "net/netip"
- )
- // SetProxy
- //
- // @Description: 设置代理
- func (t *Client[T]) SetProxy(info *yu_proxy.Info) {
- t.proxy_info = info
- return
- }
- // DelProxy
- //
- // @Description: 清除代理
- func (t *Client[T]) DelProxy() {
- t.proxy_info.Host = ""
- t.proxy_info.User = ""
- t.proxy_info.Pass = ""
- t.proxy_info.Type = 0
- t.proxy_info.Timeout = 0
- return
- }
- func (t *Client[T]) connect_proxy() bool {
- var j_addr netip.Addr
- var j_err error
- var j_doamin bool
- if j_addr, j_err = netip.ParseAddr(t._RemoteAddress); j_err != nil {
- j_addrs, j_err := net.LookupHost(t._RemoteAddress)
- if j_err != nil || len(j_addrs) == 0 {
- t.Stop()
- return false
- }
- if j_addr, j_err = netip.ParseAddr(j_addrs[0]); j_err != nil {
- t.Stop()
- return false
- }
- j_doamin = true
- }
- if t.proxy_info.IsHttp() {
- j_buf := make([]byte, 0, 256)
- j_buf = append(j_buf, "CONNECT "...)
- if t.proxy_info.Domain {
- j_buf = append(j_buf, t._RemoteAddress...)
- } else if j_addr.IsValid() {
- if j_addr.Is4() {
- j_buf = append(j_buf, j_addr.String()...)
- } else if j_addr.Is6() {
- j_buf = append(j_buf, '[')
- j_buf = append(j_buf, j_addr.String()...)
- j_buf = append(j_buf, ']')
- } else {
- t.Stop()
- return false
- }
- } else {
- t.Stop()
- return false
- }
- j_buf = append(j_buf, ':')
- j_buf = append(j_buf, yu_strconv.FormatInt(t._Port)...)
- j_buf = append(j_buf, " HTTP/1.1\r\n"...)
- j_buf = append(j_buf, "Proxy-Connection: Keep-Alive"...)
- j_buf = append(j_buf, 13, 10)
- if t.proxy_info.IsAuthorization() {
- j_buf = append(j_buf, "Proxy-Authorization: Basic "...)
- j_buf = append(j_buf, yu_base64.Std.EncodeS2S(t.proxy_info.User+":"+t.proxy_info.Pass)...)
- j_buf = append(j_buf, 13, 10)
- }
- j_buf = append(j_buf, 13, 10)
- if !yu_sys.Socket_write_tcp(t.socket, j_buf, 30, 0) {
- t.Stop()
- return false
- }
- if j_buf = yu_sys.Socket_read_tcp(t.socket, 1024, 30, 0); len(j_buf) == 0 {
- t.Stop()
- return false
- }
- // "HTTP/1.1 200 Connection Established"、"HTTP/1.1 200 Connection established"
- // HTTP/1.0 200 OK
- if len(j_buf) < 36 || yu_fast.B2S(j_buf[9:12]) != "200" {
- t.Stop()
- return false
- }
- switch yu_fast.B2S(j_buf[13:35]) {
- case "Connection Established", "Connection established", "connection Established", "connection established":
- default:
- t.Stop()
- return false
- }
- return true
- }
- if t.proxy_info.IsSocks5() {
- // https://zhuanlan.zhihu.com/p/458173597?utm_id=0
- // https://www.jianshu.com/p/97873541510f
- j_buf := make([]byte, 0, 256)
- j_buf = append(j_buf, 5) // VER 协议版本号
- j_buf = append(j_buf, 1) // NMETHODS 客户端支持的认证方法数量
- // X'00' 不需要身份验证(NO AUTHENTICATION REQUIRED)
- // X'01' GSSAPI
- // X'02' 用户密码认证(USERNAME/PASSWORD)
- // X'03' to X'7F' IANA ASSIGNED
- // X'80' to X'FE' RESERVED FOR PRIVATE METHODS
- if t.proxy_info.IsAuthorization() {
- j_buf = append(j_buf, 0x02) // METHODS:认证方法
- } else {
- j_buf = append(j_buf, 0x00) // METHODS:认证方法
- }
- if !yu_sys.Socket_write_tcp(t.socket, j_buf, 30, 0) {
- t.Stop()
- return false
- }
- if j_buf = yu_sys.Socket_read_tcp(t.socket, 1024, 30, 0); len(j_buf) == 0 {
- t.Stop()
- return false
- }
- if len(j_buf) != 2 || j_buf[0] != 5 {
- t.Stop()
- return false
- }
- if j_buf[1] == 0x02 {
- j_buf = append(j_buf[0:0], 1) // VERSION 认证子协商版本(与SOCKS协议版本的0x05无关系)
- j_buf = append(j_buf, byte(len(t.proxy_info.User)))
- j_buf = append(j_buf, t.proxy_info.User...)
- j_buf = append(j_buf, byte(len(t.proxy_info.Pass)))
- j_buf = append(j_buf, t.proxy_info.Pass...)
- if !yu_sys.Socket_write_tcp(t.socket, j_buf, 30, 0) {
- t.Stop()
- return false
- }
- if j_buf = yu_sys.Socket_read_tcp(t.socket, 1024, 30, 0); len(j_buf) == 0 {
- t.Stop()
- return false
- }
- if len(j_buf) != 2 || (j_buf[0] != 1 && j_buf[0] != 5) || j_buf[1] != 0 {
- t.Stop()
- return false
- }
- } else if j_buf[1] != 0x00 {
- t.Stop()
- return false
- }
- j_buf = append(j_buf[0:0], 5) // VER 协议版本号
- j_buf = append(j_buf, 0x01) // 0x01 CONNECT 连接目标服务器
- j_buf = append(j_buf, 0) // RSV 保留字段
- if t.proxy_info.Domain && j_doamin {
- j_buf = append(j_buf, 0x03) // 0x03 域名地址(没有打错,就是没有0x02),域名地址的第1个字节为域名长度,剩下字节为域名名称字节数组
- j_buf = append(j_buf, byte(len(t._RemoteAddress)))
- j_buf = append(j_buf, t._RemoteAddress...)
- } else if j_addr.IsValid() {
- if j_addr.Is4() {
- j_buf = append(j_buf, 0x01) // 0x01 IP V4地址
- } else if j_addr.Is6() {
- j_buf = append(j_buf, 0x04) // 0x04 IP V6地址
- } else {
- t.Stop()
- return false
- }
- j_buf = append(j_buf, j_addr.AsSlice()...)
- } else {
- t.Stop()
- return false
- }
- j_buf = append(j_buf, byte(t._Port>>8), byte(t._Port))
- if !yu_sys.Socket_write_tcp(t.socket, j_buf, 30, 0) {
- t.Stop()
- return false
- }
- if j_buf = yu_sys.Socket_read_tcp(t.socket, 1024, 30, 0); len(j_buf) == 0 {
- t.Stop()
- return false
- }
- if len(j_buf) < 10 || j_buf[0] != 5 || j_buf[1] != 0 || j_buf[2] != 0 {
- t.Stop()
- return false
- }
- return true
- }
- t.Stop()
- return false
- }
|