httping.go 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185
  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. "github.com/XIU2/CloudflareSpeedTest/utils"
  14. )
  15. var (
  16. Httping bool
  17. HttpingStatusCode int
  18. HttpingCFColo string
  19. HttpingCFColomap *sync.Map
  20. ColoRegexp = regexp.MustCompile(`[A-Z]{3}`)
  21. )
  22. // pingReceived pingTotalTime
  23. func (p *Ping) httping(ip *net.IPAddr) (int, time.Duration, string) {
  24. hc := http.Client{
  25. Timeout: time.Second * 2,
  26. Transport: &http.Transport{
  27. DialContext: getDialContext(ip),
  28. //TLSClientConfig: &tls.Config{InsecureSkipVerify: true}, // 跳过证书验证
  29. },
  30. CheckRedirect: func(req *http.Request, via []*http.Request) error {
  31. return http.ErrUseLastResponse // 阻止重定向
  32. },
  33. }
  34. // 先访问一次获得 HTTP 状态码 及 Cloudflare Colo
  35. var colo string
  36. {
  37. request, err := http.NewRequest(http.MethodHead, URL, nil)
  38. if err != nil {
  39. if utils.Debug { // 调试模式下,输出更多信息
  40. fmt.Printf("\033[31m[调试] IP: %s, 延迟测速请求创建失败,错误信息: %v, 测速地址: %s\033[0m\n", ip.String(), err, URL)
  41. }
  42. return 0, 0, ""
  43. }
  44. 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")
  45. response, err := hc.Do(request)
  46. if err != nil {
  47. return 0, 0, ""
  48. }
  49. defer response.Body.Close()
  50. //fmt.Println("IP:", ip, "StatusCode:", response.StatusCode, response.Request.URL)
  51. // 如果未指定的 HTTP 状态码,或指定的状态码不合规,则默认只认为 200、301、302 才算 HTTPing 通过
  52. if HttpingStatusCode == 0 || HttpingStatusCode < 100 && HttpingStatusCode > 599 {
  53. if response.StatusCode != 200 && response.StatusCode != 301 && response.StatusCode != 302 {
  54. if utils.Debug { // 调试模式下,输出更多信息
  55. fmt.Printf("\033[31m[调试] IP: %s, 延迟测速终止,HTTP 状态码: %d, 测速地址: %s\033[0m\n", ip.String(), response.StatusCode, URL)
  56. }
  57. return 0, 0, ""
  58. }
  59. } else {
  60. if response.StatusCode != HttpingStatusCode {
  61. if utils.Debug { // 调试模式下,输出更多信息
  62. fmt.Printf("\033[31m[调试] IP: %s, 延迟测速终止,HTTP 状态码: %d, 指定的 HTTP 状态码 %d, 测速地址: %s\033[0m\n", ip.String(), response.StatusCode, HttpingStatusCode, URL)
  63. }
  64. return 0, 0, ""
  65. }
  66. }
  67. io.Copy(io.Discard, response.Body)
  68. // 通过头部 Server 值判断是 Cloudflare 还是 AWS CloudFront 并设置 cfRay 为各自的机场地区码完整内容
  69. colo = getHeaderColo(response.Header)
  70. // 只有指定了地区才匹配机场地区码
  71. if HttpingCFColo != "" {
  72. // 判断是否匹配指定的地区码
  73. colo = p.filterColo(colo)
  74. if colo == "" { // 没有匹配到地区码或不符合指定地区则直接结束该 IP 测试
  75. if utils.Debug { // 调试模式下,输出更多信息
  76. fmt.Printf("\033[31m[调试] IP: %s, 地区码不匹配: %s\033[0m\n", ip.String(), colo)
  77. }
  78. return 0, 0, ""
  79. }
  80. }
  81. }
  82. // 循环测速计算延迟
  83. success := 0
  84. var delay time.Duration
  85. for i := 0; i < PingTimes; i++ {
  86. request, err := http.NewRequest(http.MethodHead, URL, nil)
  87. if err != nil {
  88. log.Fatal("意外的错误,情报告:", err)
  89. return 0, 0, ""
  90. }
  91. 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")
  92. if i == PingTimes-1 {
  93. request.Header.Set("Connection", "close")
  94. }
  95. startTime := time.Now()
  96. response, err := hc.Do(request)
  97. if err != nil {
  98. continue
  99. }
  100. success++
  101. io.Copy(io.Discard, response.Body)
  102. _ = response.Body.Close()
  103. duration := time.Since(startTime)
  104. delay += duration
  105. }
  106. return success, delay, colo
  107. }
  108. func MapColoMap() *sync.Map {
  109. if HttpingCFColo == "" {
  110. return nil
  111. }
  112. // 将 -cfcolo 参数指定的地区地区码转为大写并格式化
  113. colos := strings.Split(strings.ToUpper(HttpingCFColo), ",")
  114. colomap := &sync.Map{}
  115. for _, colo := range colos {
  116. colomap.Store(colo, colo)
  117. }
  118. return colomap
  119. }
  120. // 从响应头中获取 地区码 值
  121. func getHeaderColo(header http.Header) (colo string) {
  122. // 如果是 Cloudflare 的服务器,则获取 cf-ray 头部
  123. if header.Get("Server") == "cloudflare" {
  124. colo = header.Get("cf-ray") // 示例 cf-ray: 7bd32409eda7b020-SJC
  125. } else { // 反之则默认当成 AWS CloudFront 的服务器获取(如果不是则会获取到空,下面会判断并处理的)
  126. colo = header.Get("x-amz-cf-pop") // 示例 x-amz-cf-pop: SIN52-P1
  127. }
  128. // Fastly CDN 的头部信息,测试地址 https://fastly.jsdelivr.net/gh/XIU2/CloudflareSpeedTest@master/go.mod
  129. // x-served-by: cache-qpg1275-QPG
  130. // x-served-by: cache-fra-etou8220141-FRA, cache-hhr-khhr2060043-HHR(最后一个为实际位置)
  131. // Gcore CDN 的头部信息,测试地址 https://assets.gcore.pro/assets/icons/shield-lock.svg
  132. // x-id-fe: sg1-hw-edge-gc12
  133. // x-shard: sg1-shard0-default
  134. // x-id: sg1-hw-edge-gc2
  135. // CDN77 的头部信息,测试地址 https://www.cdn77.com
  136. // Server: CDN77-Turbo
  137. // X-77-Pop: losangelesUSCA
  138. // x-77-pop: frankfurtDE
  139. // x-77-pop: amsterdamNL
  140. // x-77-pop: singaporeSG
  141. // 如果没有获取到头部信息,说明不是 Cloudflare 和 AWS CloudFront,则直接返回空字符串
  142. if colo == "" {
  143. return ""
  144. }
  145. // 正则匹配并返回 机场地区码
  146. /*matches := ColoRegexp.FindAllString(colo, -1) // 适用于 Fastly 这种有多个地区码的
  147. if len(matches) == 0 {
  148. return ""
  149. }
  150. return matches[len(matches)-1]*/
  151. return ColoRegexp.FindString(colo)
  152. }
  153. // 处理地区码
  154. func (p *Ping) filterColo(colo string) string {
  155. if colo == "" {
  156. return ""
  157. }
  158. // 如果没有指定 -cfcolo 参数,则直接返回
  159. if HttpingCFColomap == nil {
  160. return colo
  161. }
  162. // 匹配 机场地区码 是否为指定的地区
  163. _, ok := HttpingCFColomap.Load(colo)
  164. if ok {
  165. return colo
  166. }
  167. return ""
  168. }