-
Notifications
You must be signed in to change notification settings - Fork 98
/
Copy pathcore.go
132 lines (119 loc) · 3.35 KB
/
core.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
package main
import (
"bytes"
"github.com/sirupsen/logrus"
"io"
"net"
"strings"
"sync"
"time"
)
func listen(rule *ruleStructure, wg *sync.WaitGroup) {
defer wg.Done()
//监听
listener, err := net.Listen("tcp", rule.Listen)
if err != nil {
logrus.Errorf("[%s] failed to listen at %s", rule.Name, rule.Listen)
return
}
logrus.Infof("[%s] listing at %s", rule.Name, rule.Listen)
for {
//处理客户端连接
conn, err := listener.Accept()
if err != nil {
logrus.Errorf("[%s] failed to accept at %s", rule.Name, rule.Listen)
time.Sleep(time.Second * 1)
continue
}
//判断黑名单
if len(rule.Blacklist) != 0 {
clientIP := conn.RemoteAddr().String()
clientIP = clientIP[0:strings.LastIndex(clientIP, ":")]
if rule.Blacklist[clientIP] {
logrus.Infof("[%s] disconnected ip in blacklist: %s", rule.Name, clientIP)
conn.Close()
continue
}
}
//判断是否是正则模式
if rule.EnableRegexp {
go handleRegexp(conn, rule)
} else {
go handleNormal(conn, rule)
}
}
}
func handleNormal(conn net.Conn, rule *ruleStructure) {
defer conn.Close()
var target net.Conn
//正常模式下挨个连接直到成功连接
for _, v := range rule.Targets {
c, err := net.Dial("tcp", v.Address)
if err != nil {
logrus.Errorf("[%s] try to handle connection (%s) failed because target (%s) connected failed, try next target.",
rule.Name, conn.RemoteAddr(), v.Address)
continue
}
target = c
break
}
if target == nil {
logrus.Errorf("[%s] unable to handle connection (%s) because all targets connected failed",
rule.Name, conn.RemoteAddr())
return
}
logrus.Debugf("[%s] handle connection (%s) to target (%s)", rule.Name, conn.RemoteAddr(), target.RemoteAddr())
defer target.Close()
//io桥
go io.Copy(conn, target)
io.Copy(target, conn)
}
func handleRegexp(conn net.Conn, rule *ruleStructure) {
defer conn.Close()
//正则模式下需要客户端的第一个数据包判断特征,所以需要设置一个超时
conn.SetReadDeadline(time.Now().Add(time.Millisecond * time.Duration(rule.FirstPacketTimeout)))
//获取第一个数据包
firstPacket, err := waitFirstPacket(conn)
if err != nil {
logrus.Errorf("[%s] unable to handle connection (%s) because failed to get first packet : %s",
rule.Name, conn.RemoteAddr(), err.Error())
return
}
var target net.Conn
//挨个匹配正则
for _, v := range rule.Targets {
if !v.regexp.Match(firstPacket) {
continue
}
c, err := net.Dial("tcp", v.Address)
if err != nil {
logrus.Errorf("[%s] try to handle connection (%s) failed because target (%s) connected failed, try next match target.",
rule.Name, conn.RemoteAddr(), v.Address)
continue
}
target = c
break
}
if target == nil {
logrus.Errorf("[%s] unable to handle connection (%s) because no match target",
rule.Name, conn.RemoteAddr())
return
}
logrus.Debugf("[%s] handle connection (%s) to target (%s)", rule.Name, conn.RemoteAddr(), target.RemoteAddr())
//匹配到了,去除掉刚才设定的超时
conn.SetReadDeadline(time.Time{})
//把第一个数据包发送给目标
io.Copy(target, bytes.NewReader(firstPacket))
defer target.Close()
//io桥
go io.Copy(conn, target)
io.Copy(target, conn)
}
func waitFirstPacket(conn net.Conn) ([]byte, error) {
buf := make([]byte, 2048)
n, err := conn.Read(buf)
if err != nil {
return nil, err
}
return buf[:n], nil
}