package main import ( "flag" "fmt" "io/ioutil" "net/http" "os" "runtime" "sort" "strconv" "sync" "time" "github.com/cheggaaa/pb/v3" ) var version, ipFile, outputFile, versionNew string var disableDownload, ipv6Mode, allip bool var tcpPort, printResultNum, downloadSecond int var timeLimit, speedLimit float64 func init() { var printVersion bool var help = ` CloudflareSpeedTest ` + version + ` 测试 Cloudflare CDN 所有 IP 的延迟和速度,获取最快 IP (IPv4+IPv6)! https://github.com/XIU2/CloudflareSpeedTest 参数: -n 500 测速线程数量;越多测速越快,性能弱的设备 (如路由器) 请适当调低;(默认 500 最多 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.00 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", 500, "测速线程数量") 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(&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 = 500 } 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 } 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 var mu sync.Mutex var data = make([]CloudflareIPData, 0) var data2 = make([]CloudflareIPData, 0) downloadTestTime = time.Duration(downloadSecond) * time.Second // 开始延迟测速 fmt.Println("# XIU2/CloudflareSpeedTest " + version + "\n") if ipv6Mode { // IPv6 模式判断 fmt.Println("开始延迟测速(模式:TCP IPv6,端口:" + strconv.Itoa(tcpPort) + ",平均延迟上限:" + fmt.Sprintf("%.2f", timeLimit) + " ms):") } else { fmt.Println("开始延迟测速(模式:TCP IPv4,端口:" + strconv.Itoa(tcpPort) + ",平均延迟上限:" + fmt.Sprintf("%.2f", timeLimit) + " ms):") } 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 && timeLimit > 0 { for i := 0; i < len(data); i++ { if float64(data[i].pingTime) <= timeLimit { data2 = append(data2, data[i]) // 延迟满足条件时,添加到新数组中 } else { break } } data = data2 data2 = []CloudflareIPData{} } // 开始下载测速 if !disableDownload { // 如果禁用下载测速就跳过 if len(data) > 0 { // IP数组长度(IP数量) 大于 0 时才会继续下载测速 if len(data) < downloadTestCount { // 如果IP数组长度(IP数量) 小于下载测速数量(-dn),则次数修正为IP数 downloadTestCount = len(data) } var downloadTestCount2 int // 临时的下载测速次数,即实际的下载测速数量 if speedLimit > 0 { downloadTestCount2 = len(data) // 如果指定了 [下载速度下限] 条件,则临时变量改为总数量(即一直测速下去,直到凑够下载测速数量 -dn) } else { downloadTestCount2 = downloadTestCount // 如果没有指定 [下载速度下限] 条件,则临时变量为下载测速数量(-dn) } fmt.Println("开始下载测速(下载速度下限:" + fmt.Sprintf("%.2f", speedLimit) + " MB/s,下载测速数量:" + strconv.Itoa(downloadTestCount) + ",下载测速队列:" + strconv.Itoa(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)/1024/1024 >= speedLimit { data2 = append(data2, data[i]) // 高于下载速度下限时,添加到新数组中 bar.Add(1) if len(data2) == downloadTestCount { // 凑够满足条件的 IP 时(下载测速数量 -dn),就跳出循环 break } } } bar.Finish() } else { fmt.Println("\n[信息] 延迟测速结果 IP 数量为 0,跳过下载测速。") } } if len(data2) > 0 { // 如果该数组有内容,说明指定了 [下载测速下限] 条件,且最少有 1 个满足条件的 IP data = data2 } sort.Sort(CloudflareIPDataSetD(data)) // 排序(按下载速度,从高到低) if outputFile != "" { ExportCsv(outputFile, data) // 输出结果到文件 } printResult(data) // 显示最快结果 } // 显示最快结果 func printResult(data []CloudflareIPData) { sysType := runtime.GOOS if printResultNum > 0 { // 如果禁止直接输出结果就跳过 dateString := convertToString(data) // 转为多维数组 [][]String if len(dateString) > 0 { // IP数组长度(IP数量) 大于 0 时继续 if len(dateString) < printResultNum { // 如果IP数组长度(IP数量) 小于 打印次数,则次数改为IP数量 printResultNum = len(dateString) } if ipv6Mode { // IPv6 太长了,所以需要调整一下间隔 fmt.Printf("%-40s%-5s%-5s%-5s%-6s%-11s\n", "IP 地址", "已发送", "已接收", "丢包率", "平均延迟", "下载速度 (MB/s)") 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", "IP 地址", "已发送", "已接收", "丢包率", "平均延迟", "下载速度 (MB/s)") for i := 0; i < printResultNum; i++ { 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]) } } if versionNew != "" { fmt.Println("\n发现新版本 [" + versionNew + "]!请前往 [https://github.com/XIU2/CloudflareSpeedTest] 更新!") } if sysType == "windows" { // 如果是 Windows 系统,则需要按下 回车键 或 Ctrl+C 退出(避免通过双击运行时,测速完毕后直接关闭) if outputFile != "" { fmt.Printf("\n完整测速结果已写入 %v 文件,请使用记事本/表格软件查看。\n按下 回车键 或 Ctrl+C 退出。", outputFile) } else { fmt.Printf("\n按下 回车键 或 Ctrl+C 退出。") } var pause int fmt.Scanln(&pause) } else { // 其它系统直接退出 if outputFile != "" { fmt.Println("\n完整测速结果已写入 " + outputFile + " 文件,请使用记事本/表格软件查看。") } } } else { fmt.Println("\n[信息] 完整测速结果 IP 数量为 0,跳过输出结果。") } } else { fmt.Println("\n完整测速结果已写入 " + outputFile + " 文件,请使用记事本/表格软件查看。") } } // 检查更新 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 { // 读取资源数据 body: []byte body, err := ioutil.ReadAll(res.Body) // 关闭资源流 res.Body.Close() if err == nil { if string(body) != version { versionNew = string(body) } } } }