package main import ( "flag" "fmt" "io/ioutil" "net/http" "os" "runtime" "sort" "sync" "time" "github.com/cheggaaa/pb/v3" ) var ( version, ipFile, outputFile, versionNew string disableDownload, ipv6Mode, allip bool tcpPort, printResultNum, downloadSecond int timeLimit, timeLimitLow, speedLimit float64 ) func init() { var printVersion bool var help = ` CloudflareSpeedTest ` + version + ` 测试 Cloudflare CDN 所有 IP 的延迟和速度,获取最快 IP (IPv4+IPv6)! https://github.com/XIU2/CloudflareSpeedTest 参数: -n 200 测速线程数量;越多测速越快,性能弱的设备 (如路由器) 请勿太高;(默认 200 最多 1000) -t 4 延迟测速次数;单个 IP 延迟测速次数,为 1 时将过滤丢包的IP,TCP协议;(默认 4) -tp 443 延迟测速端口;延迟测速 TCP 协议的端口;(默认 443) -dn 20 下载测速数量;延迟测速并排序后,从最低延迟起下载测速的数量;(默认 20) -dt 10 下载测速时间;单个 IP 下载测速最长时间,单位:秒;(默认 10) -url https://cf.xiu2.xyz/Github/CloudflareSpeedTest.png 下载测速地址;用来下载测速的 Cloudflare CDN 文件地址,如地址含有空格请加上引号; -tl 200 平均延迟上限;只输出低于指定平均延迟的 IP,可与其他上限/下限搭配;(默认 9999 ms) -tll 40 平均延迟下限;只输出高于指定平均延迟的 IP,可与其他上限/下限搭配,过滤被假蔷的 IP;(默认 0 ms) -sl 5 下载速度下限;只输出高于指定下载速度的 IP,凑够指定数量 [-dn] 才会停止测速;(默认 0.00 MB/s) -p 20 显示结果数量;测速后直接显示指定数量的结果,为 0 时不显示结果直接退出;(默认 20) -f ip.txt IP段数据文件;如路径含有空格请加上引号;支持其他 CDN IP段;(默认 ip.txt) -o result.csv 写入结果文件;如路径含有空格请加上引号;值为空时不写入文件 [-o ""];(默认 result.csv) -dd 禁用下载测速;禁用后测速结果会按延迟排序 (默认按下载速度排序);(默认 启用) -ipv6 IPv6测速模式;确保 IP 段数据文件内只包含 IPv6 IP段,软件不支持同时测速 IPv4+IPv6;(默认 IPv4) -allip 测速全部的IP;对 IP 段中的每个 IP (仅支持 IPv4) 进行测速;(默认 每个 IP 段随机测速一个 IP) -v 打印程序版本+检查版本更新 -h 打印帮助说明 ` flag.IntVar(&pingRoutine, "n", 200, "测速线程数量") flag.IntVar(&pingTime, "t", 4, "延迟测速次数") flag.IntVar(&tcpPort, "tp", 443, "延迟测速端口") flag.IntVar(&downloadTestCount, "dn", 20, "下载测速数量") flag.IntVar(&downloadSecond, "dt", 10, "下载测速时间") flag.StringVar(&url, "url", "https://cf.xiu2.xyz/Github/CloudflareSpeedTest.png", "下载测速地址") flag.Float64Var(&timeLimit, "tl", 9999, "平均延迟上限") flag.Float64Var(&timeLimitLow, "tll", 0, "平均延迟下限") flag.Float64Var(&speedLimit, "sl", 0, "下载速度下限") flag.IntVar(&printResultNum, "p", 20, "显示结果数量") flag.BoolVar(&disableDownload, "dd", false, "禁用下载测速") flag.BoolVar(&ipv6Mode, "ipv6", false, "禁用下载测速") flag.BoolVar(&allip, "allip", false, "测速全部 IP") flag.StringVar(&ipFile, "f", "ip.txt", "IP 数据文件") flag.StringVar(&outputFile, "o", "result.csv", "输出结果文件") flag.BoolVar(&printVersion, "v", false, "打印程序版本") flag.Usage = func() { fmt.Print(help) } flag.Parse() if printVersion { println(version) fmt.Println("检查版本更新中...") checkUpdate() if versionNew != "" { fmt.Println("发现新版本 [" + versionNew + "]!请前往 [https://github.com/XIU2/CloudflareSpeedTest] 更新!") } else { fmt.Println("当前为最新版本 [" + version + "]!") } os.Exit(0) } if pingRoutine <= 0 { pingRoutine = 200 } if pingTime <= 0 { pingTime = 4 } if tcpPort < 1 || tcpPort > 65535 { tcpPort = 443 } if downloadTestCount <= 0 { downloadTestCount = 20 } if downloadSecond <= 0 { downloadSecond = 10 } if url == "" { url = "https://cf.xiu2.xyz/Github/CloudflareSpeedTest.png" } if timeLimit <= 0 || timeLimit > 9999 { timeLimit = 9999 } if timeLimitLow < 0 || timeLimitLow > 9999 { timeLimitLow = 0 } if speedLimit < 0 { speedLimit = 0 } if printResultNum < 0 { printResultNum = 20 } if ipFile == "" { ipFile = "ip.txt" } if outputFile == " " { outputFile = "" } } func main() { go checkUpdate() // 检查版本更新 initRandSeed() // 置随机数种子 failTime = pingTime // 设置接收次数 ips := loadFirstIPOfRangeFromFile(ipFile) // 读入IP pingCount := len(ips) * pingTime // 计算进度条总数(IP*测试次数) bar := pb.Simple.Start(pingCount) // 进度条总数 var ( wg sync.WaitGroup mu sync.Mutex data = make([]CloudflareIPData, 0) data2 = make([]CloudflareIPData, 0) ) downloadTestTime = time.Duration(downloadSecond) * time.Second // 开始延迟测速 fmt.Printf("# XIU2/CloudflareSpeedTest %s \n", version) ipVersion := "IPv4" if ipv6Mode { // IPv6 模式判断 ipVersion = "IPv6" } fmt.Printf("开始延迟测速(模式:TCP %s,端口:%d ,平均延迟上限:%.2f ms,平均延迟下限:%.2f ms):\n", ipVersion, tcpPort, timeLimit, timeLimitLow) control := make(chan bool, pingRoutine) for _, ip := range ips { wg.Add(1) // control <- false handleProgress := handleProgressGenerator(bar) // 多线程进度条 go tcpingGoroutine(&wg, &mu, ip, tcpPort, pingTime, &data, control, handleProgress) } wg.Wait() bar.Finish() sort.Sort(CloudflareIPDataSet(data)) // 排序(按延迟,从低到高,不同丢包率会分开单独按延迟和丢包率排序) // 延迟测速完毕后,以 [平均延迟上限] + [平均延迟下限] 条件过滤结果 if timeLimit != 9999 || timeLimitLow != 0 { for i := 0; i < len(data); i++ { if float64(data[i].pingTime) > timeLimit { // 平均延迟上限 break } if float64(data[i].pingTime) <= timeLimitLow { // 平均延迟下限 continue } data2 = append(data2, data[i]) // 延迟满足条件时,添加到新数组中 } data = data2 data2 = []CloudflareIPData{} } // 开始下载测速 if !disableDownload { // 如果禁用下载测速就跳过 testDownloadSpeed(data, data2, bar) } if len(data2) > 0 { // 如果该数组有内容,说明指定了 [下载测速下限] 条件,且最少有 1 个满足条件的 IP data = data2 } sort.Sort(CloudflareIPDataSetD(data)) // 排序(按下载速度,从高到低) if outputFile != "" { ExportCsv(outputFile, data) // 输出结果到文件 } printResult(data) // 显示最快结果 } func testDownloadSpeed(data, data2 []CloudflareIPData, bar *pb.ProgressBar) { if len(data) <= 0 { // IP数组长度(IP数量) 大于 0 时才会继续下载测速 fmt.Println("\n[信息] 延迟测速结果 IP 数量为 0,跳过下载测速。") return } if len(data) < downloadTestCount { // 如果IP数组长度(IP数量) 小于下载测速数量(-dn),则次数修正为IP数 downloadTestCount = len(data) } // 临时的下载测速次数,即实际的下载测速数量 downloadTestCount2 := downloadTestCount // 如果没有指定 [下载速度下限] 条件,则临时变量为下载测速数量(-dn) if speedLimit > 0 { // 如果指定了 [下载速度下限] 条件,则临时变量改为总数量(即一直测速下去,直到凑够下载测速数量 -dn) downloadTestCount2 = len(data) } fmt.Printf("开始下载测速(下载速度下限:%.2f MB/s,下载测速数量:%d,下载测速队列:%d):\n", speedLimit, downloadTestCount, downloadTestCount2) bar = pb.Simple.Start(downloadTestCount) for i := 0; i < downloadTestCount2; i++ { _, speed := DownloadSpeedHandler(data[i].ip) data[i].downloadSpeed = speed // 在每个 IP 下载测速后,以 [下载速度下限] 条件过滤结果 if float64(speed) >= speedLimit*1024*1024 { data2 = append(data2, data[i]) // 高于下载速度下限时,添加到新数组中 bar.Add(1) if len(data2) == downloadTestCount { // 凑够满足条件的 IP 时(下载测速数量 -dn),就跳出循环 break } } } bar.Finish() } // 显示最快结果 func printResult(data []CloudflareIPData) { sysType := runtime.GOOS if printResultNum <= 0 { // 如果禁止直接输出结果就跳过 fmt.Printf("完整测速结果已写入 %v 文件,请使用记事本/表格软件查看。\n", outputFile) return } dateString := convertToString(data) // 转为多维数组 [][]String if len(dateString) <= 0 { // IP数组长度(IP数量) 大于 0 时继续 fmt.Println("\n[信息] 完整测速结果 IP 数量为 0,跳过输出结果。") return } if len(dateString) < printResultNum { // 如果IP数组长度(IP数量) 小于 打印次数,则次数改为IP数量 printResultNum = len(dateString) } resHeader := []interface{}{"IP 地址", "已发送", "已接收", "丢包率", "平均延迟", "下载速度 (MB/s)"} if ipv6Mode { // IPv6 太长了,所以需要调整一下间隔 fmt.Printf("%-40s%-5s%-5s%-5s%-6s%-11s\n", resHeader...) for i := 0; i < printResultNum; i++ { 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]) } } else { fmt.Printf("%-16s%-5s%-5s%-5s%-6s%-11s\n", resHeader...) for i := 0; i < printResultNum; i++ { fmt.Printf("%-18s%-8s%-8s%-8s%-15s%-15s\n", ipPadding(dateString[i][0]), dateString[i][1], dateString[i][2], dateString[i][3], dateString[i][4], dateString[i][5]) } } if versionNew != "" { fmt.Printf("\n*** 发现新版本 [%s]!请前往 [https://github.com/XIU2/CloudflareSpeedTest] 更新! ***\n", versionNew) } if outputFile != "" { fmt.Printf("完整测速结果已写入 %v 文件,请使用记事本/表格软件查看。\n", outputFile) } if sysType == "windows" { // 如果是 Windows 系统,则需要按下 回车键 或 Ctrl+C 退出(避免通过双击运行时,测速完毕后直接关闭) fmt.Printf("按下 回车键 或 Ctrl+C 退出。\n") var pause int fmt.Scanln(&pause) } } // 检查更新 func checkUpdate() { timeout := time.Duration(10 * time.Second) client := http.Client{Timeout: timeout} res, err := client.Get("https://api.xiuer.pw/ver/cloudflarespeedtest.txt") if err != nil { return } // 读取资源数据 body: []byte body, err := ioutil.ReadAll(res.Body) if err != nil { return } // 关闭资源流 defer res.Body.Close() if string(body) != version { versionNew = string(body) } }