forked from akkuman/rotateproxy
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcheck.go
128 lines (119 loc) · 3.02 KB
/
check.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
package rotateproxy
import (
"crypto/tls"
"fmt"
"io/ioutil"
"net/http"
"net/url"
"strings"
"time"
)
type IPInfo struct {
Status string `json:"status"`
Country string `json:"country"`
CountryCode string `json:"countryCode"`
Region string `json:"region"`
RegionName string `json:"regionName"`
City string `json:"city"`
Zip string `json:"zip"`
Lat float64 `json:"lat"`
Lon float64 `json:"lon"`
Timezone string `json:"timezone"`
Isp string `json:"isp"`
Org string `json:"org"`
As string `json:"as"`
Query string `json:"query"`
}
func CheckProxyAlive(proxyURL string) (respBody string, timeout int64, avail bool) {
proxy, _ := url.Parse(proxyURL)
httpclient := &http.Client{
Transport: &http.Transport{
Proxy: http.ProxyURL(proxy),
TLSClientConfig: &tls.Config{InsecureSkipVerify: true},
DisableKeepAlives: true,
},
Timeout: 20 * time.Second,
}
startTime := time.Now()
resp, err := httpclient.Get("http://cip.cc/")
if err != nil {
return "", 0, false
}
defer resp.Body.Close()
timeout = int64(time.Since(startTime))
body, err := ioutil.ReadAll(resp.Body)
if err != nil {
return "", 0, false
}
if !strings.Contains(string(body), "地址") {
return "", 0, false
}
return string(body), timeout, true
}
func CheckProxyWithCheckURL(proxyURL string, checkURL string) (timeout int64, avail bool) {
fmt.Printf("check %s: %s\n", proxyURL, checkURL)
proxy, _ := url.Parse(proxyURL)
httpclient := &http.Client{
Transport: &http.Transport{
Proxy: http.ProxyURL(proxy),
TLSClientConfig: &tls.Config{InsecureSkipVerify: true},
DisableKeepAlives: true,
},
Timeout: 20 * time.Second,
}
startTime := time.Now()
resp, err := httpclient.Get(checkURL)
if err != nil {
return 0, false
}
defer resp.Body.Close()
timeout = int64(time.Since(startTime))
// TODO: support regex
if resp.StatusCode != 200 {
return 0, false
}
return timeout, true
}
func StartCheckProxyAlive(checkURL string) {
go func() {
ticker := time.NewTicker(120 * time.Second)
for {
select {
case <-crawlDone:
fmt.Println("Checking")
checkAlive(checkURL)
fmt.Println("Check done")
case <-ticker.C:
checkAlive(checkURL)
}
}
}()
}
func checkAlive(checkURL string) {
proxies, err := QueryProxyURL()
if err != nil {
fmt.Printf("[!] query db error: %v\n", err)
}
for i := range proxies {
proxy := proxies[i]
go func() {
respBody, timeout, avail := CheckProxyAlive(proxy.URL)
if avail {
if checkURL != "" {
timeout, avail = CheckProxyWithCheckURL(proxy.URL, checkURL)
}
if avail {
fmt.Printf("%v 可用\n", proxy.URL)
SetProxyURLAvail(proxy.URL, timeout, CanBypassGFW(respBody))
return
}
}
AddProxyURLRetry(proxy.URL)
}()
}
}
func CanBypassGFW(respBody string) bool {
return strings.Contains(respBody, "香港") ||
strings.Contains(respBody, "台湾") ||
strings.Contains(respBody, "澳门") || !strings.Contains(respBody, "中国")
}