main.go 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278
  1. package main
  2. import (
  3. "flag"
  4. "fmt"
  5. "io/ioutil"
  6. "net/http"
  7. "os"
  8. "runtime"
  9. "sort"
  10. "strconv"
  11. "sync"
  12. "time"
  13. "github.com/cheggaaa/pb/v3"
  14. )
  15. var version, ipFile, outputFile, versionNew string
  16. var disableDownload, ipv6Mode, allip bool
  17. var tcpPort, printResultNum, downloadSecond int
  18. var timeLimit, speedLimit float64
  19. func init() {
  20. var printVersion bool
  21. var help = `
  22. CloudflareSpeedTest ` + version + `
  23. 测试 Cloudflare CDN 所有 IP 的延迟和速度,获取最快 IP (IPv4+IPv6)!
  24. https://github.com/XIU2/CloudflareSpeedTest
  25. 参数:
  26. -n 500
  27. 测速线程数量;越多测速越快,性能弱的设备 (如路由器) 请适当调低;(默认 500 最多 1000)
  28. -t 4
  29. 延迟测速次数;单个 IP 延迟测速次数,为 1 时将过滤丢包的IP,TCP协议;(默认 4)
  30. -tp 443
  31. 延迟测速端口;延迟测速 TCP 协议的端口;(默认 443)
  32. -dn 20
  33. 下载测速数量;延迟测速并排序后,从最低延迟起下载测速的数量;(默认 20)
  34. -dt 10
  35. 下载测速时间;单个 IP 下载测速最长时间,单位:秒;(默认 10)
  36. -url https://cf.xiu2.xyz/Github/CloudflareSpeedTest.png
  37. 下载测速地址;用来下载测速的 Cloudflare CDN 文件地址,如地址含有空格请加上引号;
  38. -tl 200
  39. 平均延迟上限;只输出低于指定平均延迟的 IP,可单独使用也可搭配下载速度下限;(默认 9999.00 ms)
  40. -sl 5
  41. 下载速度下限;只输出高于指定下载速度的 IP,凑够指定数量 [-dn] 才会停止测速;(默认 0.00 MB/s)
  42. -p 20
  43. 显示结果数量;测速后直接显示指定数量的结果,为 0 时不显示结果直接退出;(默认 20)
  44. -f ip.txt
  45. IP段数据文件;如路径含有空格请加上引号;支持其他 CDN IP段;(默认 ip.txt)
  46. -o result.csv
  47. 输出结果文件;如路径含有空格请加上引号;值为空格时不输出 [-o " "];(默认 result.csv)
  48. -dd
  49. 禁用下载测速;禁用后测速结果会按延迟排序 (默认按下载速度排序);(默认 启用)
  50. -ipv6
  51. IPv6测速模式;确保 IP 段数据文件内只包含 IPv6 IP段,软件不支持同时测速 IPv4+IPv6;(默认 IPv4)
  52. -allip
  53. 测速全部的IP;对 IP 段中的每个 IP (仅支持 IPv4) 进行测速;(默认 每个 IP 段随机测速一个 IP)
  54. -v
  55. 打印程序版本+检查版本更新
  56. -h
  57. 打印帮助说明
  58. `
  59. flag.IntVar(&pingRoutine, "n", 500, "测速线程数量")
  60. flag.IntVar(&pingTime, "t", 4, "延迟测速次数")
  61. flag.IntVar(&tcpPort, "tp", 443, "延迟测速端口")
  62. flag.IntVar(&downloadTestCount, "dn", 20, "下载测速数量")
  63. flag.IntVar(&downloadSecond, "dt", 10, "下载测速时间")
  64. flag.StringVar(&url, "url", "https://cf.xiu2.xyz/Github/CloudflareSpeedTest.png", "下载测速地址")
  65. flag.Float64Var(&timeLimit, "tl", 9999, "平均延迟上限")
  66. flag.Float64Var(&speedLimit, "sl", 0, "下载速度下限")
  67. flag.IntVar(&printResultNum, "p", 20, "显示结果数量")
  68. flag.BoolVar(&disableDownload, "dd", false, "禁用下载测速")
  69. flag.BoolVar(&ipv6Mode, "ipv6", false, "禁用下载测速")
  70. flag.BoolVar(&allip, "allip", false, "测速全部 IP")
  71. flag.StringVar(&ipFile, "f", "ip.txt", "IP 数据文件")
  72. flag.StringVar(&outputFile, "o", "result.csv", "输出结果文件")
  73. flag.BoolVar(&printVersion, "v", false, "打印程序版本")
  74. flag.Usage = func() { fmt.Print(help) }
  75. flag.Parse()
  76. if printVersion {
  77. println(version)
  78. fmt.Println("检查版本更新中...")
  79. checkUpdate()
  80. if versionNew != "" {
  81. fmt.Println("发现新版本 [" + versionNew + "]!请前往 [https://github.com/XIU2/CloudflareSpeedTest] 更新!")
  82. } else {
  83. fmt.Println("当前为最新版本 [" + version + "]!")
  84. }
  85. os.Exit(0)
  86. }
  87. if pingRoutine <= 0 {
  88. pingRoutine = 500
  89. }
  90. if pingTime <= 0 {
  91. pingTime = 4
  92. }
  93. if tcpPort < 1 || tcpPort > 65535 {
  94. tcpPort = 443
  95. }
  96. if downloadTestCount <= 0 {
  97. downloadTestCount = 20
  98. }
  99. if downloadSecond <= 0 {
  100. downloadSecond = 10
  101. }
  102. if url == "" {
  103. url = "https://cf.xiu2.xyz/Github/CloudflareSpeedTest.png"
  104. }
  105. if timeLimit <= 0 {
  106. timeLimit = 9999
  107. }
  108. if speedLimit < 0 {
  109. speedLimit = 0
  110. }
  111. if printResultNum < 0 {
  112. printResultNum = 20
  113. }
  114. if ipFile == "" {
  115. ipFile = "ip.txt"
  116. }
  117. if outputFile == " " {
  118. outputFile = ""
  119. }
  120. }
  121. func main() {
  122. go checkUpdate() // 检查版本更新
  123. initRandSeed() // 置随机数种子
  124. failTime = pingTime // 设置接收次数
  125. ips := loadFirstIPOfRangeFromFile(ipFile) // 读入IP
  126. pingCount := len(ips) * pingTime // 计算进度条总数(IP*测试次数)
  127. bar := pb.Simple.Start(pingCount) // 进度条总数
  128. var wg sync.WaitGroup
  129. var mu sync.Mutex
  130. var data = make([]CloudflareIPData, 0)
  131. var data2 = make([]CloudflareIPData, 0)
  132. downloadTestTime = time.Duration(downloadSecond) * time.Second
  133. // 开始延迟测速
  134. fmt.Println("# XIU2/CloudflareSpeedTest " + version + "\n")
  135. if ipv6Mode { // IPv6 模式判断
  136. fmt.Println("开始延迟测速(模式:TCP IPv6,端口:" + strconv.Itoa(tcpPort) + ",平均延迟上限:" + fmt.Sprintf("%.2f", timeLimit) + " ms):")
  137. } else {
  138. fmt.Println("开始延迟测速(模式:TCP IPv4,端口:" + strconv.Itoa(tcpPort) + ",平均延迟上限:" + fmt.Sprintf("%.2f", timeLimit) + " ms):")
  139. }
  140. control := make(chan bool, pingRoutine)
  141. for _, ip := range ips {
  142. wg.Add(1)
  143. control <- false
  144. handleProgress := handleProgressGenerator(bar) // 多线程进度条
  145. go tcpingGoroutine(&wg, &mu, ip, tcpPort, pingTime, &data, control, handleProgress)
  146. }
  147. wg.Wait()
  148. bar.Finish()
  149. sort.Sort(CloudflareIPDataSet(data)) // 排序(按延迟,从低到高,不同丢包率会分开单独按延迟和丢包率排序)
  150. // 延迟测速完毕后,以 [平均延迟上限] 条件过滤结果
  151. if timeLimit < 9999 && timeLimit > 0 {
  152. for i := 0; i < len(data); i++ {
  153. if float64(data[i].pingTime) <= timeLimit {
  154. data2 = append(data2, data[i]) // 延迟满足条件时,添加到新数组中
  155. } else {
  156. break
  157. }
  158. }
  159. data = data2
  160. data2 = []CloudflareIPData{}
  161. }
  162. // 开始下载测速
  163. if !disableDownload { // 如果禁用下载测速就跳过
  164. if len(data) > 0 { // IP数组长度(IP数量) 大于 0 时才会继续下载测速
  165. if len(data) < downloadTestCount { // 如果IP数组长度(IP数量) 小于下载测速数量(-dn),则次数修正为IP数
  166. downloadTestCount = len(data)
  167. }
  168. var downloadTestCount2 int // 临时的下载测速次数,即实际的下载测速数量
  169. if speedLimit > 0 {
  170. downloadTestCount2 = len(data) // 如果指定了 [下载速度下限] 条件,则临时变量改为总数量(即一直测速下去,直到凑够下载测速数量 -dn)
  171. } else {
  172. downloadTestCount2 = downloadTestCount // 如果没有指定 [下载速度下限] 条件,则临时变量为下载测速数量(-dn)
  173. }
  174. fmt.Println("开始下载测速(下载速度下限:" + fmt.Sprintf("%.2f", speedLimit) + " MB/s,下载测速数量:" + strconv.Itoa(downloadTestCount) + ",下载测速队列:" + strconv.Itoa(downloadTestCount2) + "):")
  175. bar = pb.Simple.Start(downloadTestCount)
  176. for i := 0; i < downloadTestCount2; i++ {
  177. _, speed := DownloadSpeedHandler(data[i].ip)
  178. data[i].downloadSpeed = speed
  179. // 在每个 IP 下载测速后,以 [下载速度下限] 条件过滤结果
  180. if float64(speed)/1024/1024 >= speedLimit {
  181. data2 = append(data2, data[i]) // 高于下载速度下限时,添加到新数组中
  182. bar.Add(1)
  183. if len(data2) == downloadTestCount { // 凑够满足条件的 IP 时(下载测速数量 -dn),就跳出循环
  184. break
  185. }
  186. }
  187. }
  188. bar.Finish()
  189. } else {
  190. fmt.Println("\n[信息] 延迟测速结果 IP 数量为 0,跳过下载测速。")
  191. }
  192. }
  193. if len(data2) > 0 { // 如果该数组有内容,说明指定了 [下载测速下限] 条件,且最少有 1 个满足条件的 IP
  194. data = data2
  195. }
  196. sort.Sort(CloudflareIPDataSetD(data)) // 排序(按下载速度,从高到低)
  197. if outputFile != "" {
  198. ExportCsv(outputFile, data) // 输出结果到文件
  199. }
  200. printResult(data) // 显示最快结果
  201. }
  202. // 显示最快结果
  203. func printResult(data []CloudflareIPData) {
  204. sysType := runtime.GOOS
  205. if printResultNum > 0 { // 如果禁止直接输出结果就跳过
  206. dateString := convertToString(data) // 转为多维数组 [][]String
  207. if len(dateString) > 0 { // IP数组长度(IP数量) 大于 0 时继续
  208. if len(dateString) < printResultNum { // 如果IP数组长度(IP数量) 小于 打印次数,则次数改为IP数量
  209. printResultNum = len(dateString)
  210. }
  211. if ipv6Mode { // IPv6 太长了,所以需要调整一下间隔
  212. fmt.Printf("%-40s%-5s%-5s%-5s%-6s%-11s\n", "IP 地址", "已发送", "已接收", "丢包率", "平均延迟", "下载速度 (MB/s)")
  213. for i := 0; i < printResultNum; i++ {
  214. fmt.Printf("%-42s%-8s%-8s%-8s%-10s%-15s\n", ipPadding(dateString[i][0]), dateString[i][1], dateString[i][2], dateString[i][3], dateString[i][4], dateString[i][5])
  215. }
  216. } else {
  217. fmt.Printf("%-16s%-5s%-5s%-5s%-6s%-11s\n", "IP 地址", "已发送", "已接收", "丢包率", "平均延迟", "下载速度 (MB/s)")
  218. for i := 0; i < printResultNum; i++ {
  219. fmt.Printf("%-18s%-8s%-8s%-8s%-10s%-15s\n", ipPadding(dateString[i][0]), dateString[i][1], dateString[i][2], dateString[i][3], dateString[i][4], dateString[i][5])
  220. }
  221. }
  222. if versionNew != "" {
  223. fmt.Println("\n发现新版本 [" + versionNew + "]!请前往 [https://github.com/XIU2/CloudflareSpeedTest] 更新!")
  224. }
  225. if sysType == "windows" { // 如果是 Windows 系统,则需要按下 回车键 或 Ctrl+C 退出(避免通过双击运行时,测速完毕后直接关闭)
  226. if outputFile != "" {
  227. fmt.Printf("\n完整测速结果已写入 %v 文件,请使用记事本/表格软件查看。\n按下 回车键 或 Ctrl+C 退出。", outputFile)
  228. } else {
  229. fmt.Printf("\n按下 回车键 或 Ctrl+C 退出。")
  230. }
  231. var pause int
  232. fmt.Scanln(&pause)
  233. } else { // 其它系统直接退出
  234. if outputFile != "" {
  235. fmt.Println("\n完整测速结果已写入 " + outputFile + " 文件,请使用记事本/表格软件查看。")
  236. }
  237. }
  238. } else {
  239. fmt.Println("\n[信息] 完整测速结果 IP 数量为 0,跳过输出结果。")
  240. }
  241. } else {
  242. fmt.Println("\n完整测速结果已写入 " + outputFile + " 文件,请使用记事本/表格软件查看。")
  243. }
  244. }
  245. // 检查更新
  246. func checkUpdate() {
  247. timeout := time.Duration(10 * time.Second)
  248. client := http.Client{Timeout: timeout}
  249. res, err := client.Get("https://api.xiuer.pw/ver/cloudflarespeedtest.txt")
  250. if err == nil {
  251. // 读取资源数据 body: []byte
  252. body, err := ioutil.ReadAll(res.Body)
  253. // 关闭资源流
  254. res.Body.Close()
  255. if err == nil {
  256. if string(body) != version {
  257. versionNew = string(body)
  258. }
  259. }
  260. }
  261. }