httping.go 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151
  1. package task
  2. import (
  3. //"crypto/tls"
  4. //"fmt"
  5. "io"
  6. "log"
  7. "net"
  8. "net/http"
  9. "regexp"
  10. "strings"
  11. "sync"
  12. "time"
  13. )
  14. var (
  15. Httping bool
  16. HttpingStatusCode int
  17. HttpingCFColo string
  18. HttpingCFColomap *sync.Map
  19. ColoRegexp = regexp.MustCompile(`[A-Z]{3}`)
  20. )
  21. // pingReceived pingTotalTime
  22. func (p *Ping) httping(ip *net.IPAddr) (int, time.Duration, string) {
  23. hc := http.Client{
  24. Timeout: time.Second * 2,
  25. Transport: &http.Transport{
  26. DialContext: getDialContext(ip),
  27. //TLSClientConfig: &tls.Config{InsecureSkipVerify: true}, // 跳过证书验证
  28. },
  29. CheckRedirect: func(req *http.Request, via []*http.Request) error {
  30. return http.ErrUseLastResponse // 阻止重定向
  31. },
  32. }
  33. // 先访问一次获得 HTTP 状态码 及 Cloudflare Colo
  34. var colo string
  35. {
  36. request, err := http.NewRequest(http.MethodHead, URL, nil)
  37. if err != nil {
  38. return 0, 0, ""
  39. }
  40. 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")
  41. response, err := hc.Do(request)
  42. if err != nil {
  43. return 0, 0, ""
  44. }
  45. defer response.Body.Close()
  46. //fmt.Println("IP:", ip, "StatusCode:", response.StatusCode, response.Request.URL)
  47. // 如果未指定的 HTTP 状态码,或指定的状态码不合规,则默认只认为 200、301、302 才算 HTTPing 通过
  48. if HttpingStatusCode == 0 || HttpingStatusCode < 100 && HttpingStatusCode > 599 {
  49. if response.StatusCode != 200 && response.StatusCode != 301 && response.StatusCode != 302 {
  50. return 0, 0, ""
  51. }
  52. } else {
  53. if response.StatusCode != HttpingStatusCode {
  54. return 0, 0, ""
  55. }
  56. }
  57. io.Copy(io.Discard, response.Body)
  58. // 通过头部 Server 值判断是 Cloudflare 还是 AWS CloudFront 并设置 cfRay 为各自的机场地区码完整内容
  59. colo = getHeaderColo(response.Header)
  60. // 只有指定了地区才匹配机场地区码
  61. if HttpingCFColo != "" {
  62. // 判断是否匹配指定的地区码
  63. colo = p.filterColo(colo)
  64. if colo == "" { // 没有匹配到地区码或不符合指定地区则直接结束该 IP 测试
  65. return 0, 0, ""
  66. }
  67. }
  68. }
  69. // 循环测速计算延迟
  70. success := 0
  71. var delay time.Duration
  72. for i := 0; i < PingTimes; i++ {
  73. request, err := http.NewRequest(http.MethodHead, URL, nil)
  74. if err != nil {
  75. log.Fatal("意外的错误,情报告:", err)
  76. return 0, 0, ""
  77. }
  78. 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")
  79. if i == PingTimes-1 {
  80. request.Header.Set("Connection", "close")
  81. }
  82. startTime := time.Now()
  83. response, err := hc.Do(request)
  84. if err != nil {
  85. continue
  86. }
  87. success++
  88. io.Copy(io.Discard, response.Body)
  89. _ = response.Body.Close()
  90. duration := time.Since(startTime)
  91. delay += duration
  92. }
  93. return success, delay, colo
  94. }
  95. func MapColoMap() *sync.Map {
  96. if HttpingCFColo == "" {
  97. return nil
  98. }
  99. // 将 -cfcolo 参数指定的地区地区码转为大写并格式化
  100. colos := strings.Split(strings.ToUpper(HttpingCFColo), ",")
  101. colomap := &sync.Map{}
  102. for _, colo := range colos {
  103. colomap.Store(colo, colo)
  104. }
  105. return colomap
  106. }
  107. // 从响应头中获取 地区码 值
  108. func getHeaderColo(header http.Header) (colo string) {
  109. // 如果是 Cloudflare 的服务器,则获取 CF-RAY 头部
  110. if header.Get("Server") == "cloudflare" {
  111. colo = header.Get("CF-RAY") // 示例 cf-ray: 7bd32409eda7b020-SJC
  112. } else { // 如果是 AWS CloudFront 的服务器,则获取 X-Amz-Cf-Pop 头部
  113. colo = header.Get("x-amz-cf-pop") // 示例 X-Amz-Cf-Pop: SIN52-P1
  114. }
  115. // 如果没有获取到头部信息,说明不是 Cloudflare 和 AWS CloudFront,则直接返回空字符串
  116. if colo == "" {
  117. return ""
  118. }
  119. // 正则匹配并返回 机场地区码
  120. return ColoRegexp.FindString(colo)
  121. }
  122. // 处理地区码
  123. func (p *Ping) filterColo(colo string) string {
  124. if colo == "" {
  125. return ""
  126. }
  127. // 如果没有指定 -cfcolo 参数,则直接返回
  128. if HttpingCFColomap == nil {
  129. return colo
  130. }
  131. // 匹配 机场地区码 是否为指定的地区
  132. _, ok := HttpingCFColomap.Load(colo)
  133. if ok {
  134. return colo
  135. }
  136. return ""
  137. }