123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151 |
- package task
- import (
- //"crypto/tls"
- //"fmt"
- "io"
- "log"
- "net"
- "net/http"
- "regexp"
- "strings"
- "sync"
- "time"
- )
- var (
- Httping bool
- HttpingStatusCode int
- HttpingCFColo string
- HttpingCFColomap *sync.Map
- ColoRegexp = regexp.MustCompile(`[A-Z]{3}`)
- )
- // pingReceived pingTotalTime
- func (p *Ping) httping(ip *net.IPAddr) (int, time.Duration, string) {
- hc := http.Client{
- Timeout: time.Second * 2,
- Transport: &http.Transport{
- DialContext: getDialContext(ip),
- //TLSClientConfig: &tls.Config{InsecureSkipVerify: true}, // 跳过证书验证
- },
- CheckRedirect: func(req *http.Request, via []*http.Request) error {
- return http.ErrUseLastResponse // 阻止重定向
- },
- }
- // 先访问一次获得 HTTP 状态码 及 Cloudflare Colo
- var colo string
- {
- request, err := http.NewRequest(http.MethodHead, URL, nil)
- if err != nil {
- return 0, 0, ""
- }
- request.Header.Set("User-Agent", "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_6) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/98.0.4758.80 Safari/537.36")
- response, err := hc.Do(request)
- if err != nil {
- return 0, 0, ""
- }
- defer response.Body.Close()
- //fmt.Println("IP:", ip, "StatusCode:", response.StatusCode, response.Request.URL)
- // 如果未指定的 HTTP 状态码,或指定的状态码不合规,则默认只认为 200、301、302 才算 HTTPing 通过
- if HttpingStatusCode == 0 || HttpingStatusCode < 100 && HttpingStatusCode > 599 {
- if response.StatusCode != 200 && response.StatusCode != 301 && response.StatusCode != 302 {
- return 0, 0, ""
- }
- } else {
- if response.StatusCode != HttpingStatusCode {
- return 0, 0, ""
- }
- }
- io.Copy(io.Discard, response.Body)
- // 通过头部 Server 值判断是 Cloudflare 还是 AWS CloudFront 并设置 cfRay 为各自的机场地区码完整内容
- colo = getHeaderColo(response.Header)
- // 只有指定了地区才匹配机场地区码
- if HttpingCFColo != "" {
- // 判断是否匹配指定的地区码
- colo = p.filterColo(colo)
- if colo == "" { // 没有匹配到地区码或不符合指定地区则直接结束该 IP 测试
- return 0, 0, ""
- }
- }
- }
- // 循环测速计算延迟
- success := 0
- var delay time.Duration
- for i := 0; i < PingTimes; i++ {
- request, err := http.NewRequest(http.MethodHead, URL, nil)
- if err != nil {
- log.Fatal("意外的错误,情报告:", err)
- return 0, 0, ""
- }
- request.Header.Set("User-Agent", "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_6) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/98.0.4758.80 Safari/537.36")
- if i == PingTimes-1 {
- request.Header.Set("Connection", "close")
- }
- startTime := time.Now()
- response, err := hc.Do(request)
- if err != nil {
- continue
- }
- success++
- io.Copy(io.Discard, response.Body)
- _ = response.Body.Close()
- duration := time.Since(startTime)
- delay += duration
- }
- return success, delay, colo
- }
- func MapColoMap() *sync.Map {
- if HttpingCFColo == "" {
- return nil
- }
- // 将 -cfcolo 参数指定的地区地区码转为大写并格式化
- colos := strings.Split(strings.ToUpper(HttpingCFColo), ",")
- colomap := &sync.Map{}
- for _, colo := range colos {
- colomap.Store(colo, colo)
- }
- return colomap
- }
- // 从响应头中获取 地区码 值
- func getHeaderColo(header http.Header) (colo string) {
- // 如果是 Cloudflare 的服务器,则获取 CF-RAY 头部
- if header.Get("Server") == "cloudflare" {
- colo = header.Get("CF-RAY") // 示例 cf-ray: 7bd32409eda7b020-SJC
- } else { // 如果是 AWS CloudFront 的服务器,则获取 X-Amz-Cf-Pop 头部
- colo = header.Get("x-amz-cf-pop") // 示例 X-Amz-Cf-Pop: SIN52-P1
- }
- // 如果没有获取到头部信息,说明不是 Cloudflare 和 AWS CloudFront,则直接返回空字符串
- if colo == "" {
- return ""
- }
- // 正则匹配并返回 机场地区码
- return ColoRegexp.FindString(colo)
- }
- // 处理地区码
- func (p *Ping) filterColo(colo string) string {
- if colo == "" {
- return ""
- }
- // 如果没有指定 -cfcolo 参数,则直接返回
- if HttpingCFColomap == nil {
- return colo
- }
- // 匹配 机场地区码 是否为指定的地区
- _, ok := HttpingCFColomap.Load(colo)
- if ok {
- return colo
- }
- return ""
- }
|