88 Commits

Author SHA1 Message Date
xiu2
3b957cb1a4 优化 [延迟测速上限/下载速度下限] 支持小数 #51 2021-02-04 16:16:29 +08:00
XIU2
faee85edb8 Merge pull request #53 from CrazyBoyFeng/master
优化 [延迟测速上限/下载速度下限] 支持小数  #51
2021-02-04 15:37:41 +08:00
CrazyBoyFeng
b901003dd1 优化 XIU2/CloudflareSpeedTest#51 2021-02-04 14:46:24 +08:00
xiu2
873200de80 补充 README.md 2021-02-03 13:28:28 +08:00
xiu2
ac29c9666e 补充 README.md 2021-02-02 18:14:56 +08:00
xiu2
a540789180 补充 README.md 2021-02-02 18:12:19 +08:00
xiu2
012865f02d 补充 README.md 2021-01-28 12:19:49 +08:00
xiu2
74c1744ca6 补充 README.md 2021-01-27 13:48:50 +08:00
xiu2
e66d44882e 补充 README.md 2021-01-27 13:12:43 +08:00
xiu2
0d3af2d5f8 补充 README.md 2021-01-27 13:10:05 +08:00
xiu2
70d11e3bc1 补充 README.md 2021-01-27 09:54:32 +08:00
xiu2
ec650d3084 补充 README.md 2021-01-26 19:24:31 +08:00
xiu2
4038a5b0aa 优化 README.md 2021-01-16 16:31:50 +08:00
xiu2
1ff4c24c59 优化 README.md 2021-01-16 16:14:24 +08:00
xiu2
ffbb161f79 优化 README.md 2021-01-11 10:25:32 +08:00
xiu2
3efabb5661 优化 README.md 2021-01-08 08:28:49 +08:00
xiu2
e2f23aeb48 优化 README.md 2021-01-08 08:11:37 +08:00
xiu2
97bbb8b5e9 优化 README.md 2021-01-07 07:37:26 +08:00
xiu2
5d75cb861e 优化 README.md 2021-01-07 07:36:30 +08:00
xiu2
eed4f4fa0b 优化 README.md 2021-01-06 09:35:03 +08:00
xiu2
e8537fb0ae 优化 指定测速条件时的进度条显示内容 2021-01-05 19:24:23 +08:00
xiu2
f1147c5cbf 添加 单独的 1.1.1.1 和 1.0.0.1(字位掩码 /32) 2021-01-05 16:03:30 +08:00
xiu2
8a7b0f12f5 优化 使用说明 2021-01-05 12:00:54 +08:00
xiu2
6a3504e98c 优化 帮助中的参数说明 2021-01-05 11:26:07 +08:00
xiu2
78a8f9c6c4 update 2021-01-05 11:24:58 +08:00
xiu2
aa11024026 update 2021-01-02 22:14:03 +08:00
xiu2
cf1a01b614 update 2021-01-02 21:10:42 +08:00
xiu2
b9159db975 update 2021-01-01 19:30:42 +08:00
xiu2
157b89d88e update 2021-01-01 19:27:38 +08:00
xiu2
d1c304ff59 update 2021-01-01 02:02:41 +08:00
xiu2
ceff5971d2 补充 README.md 2020-12-31 09:26:43 +08:00
xiu2
32d544184b update 2020-12-31 09:21:09 +08:00
xiu2
f82471671c update 2020-12-31 09:16:11 +08:00
xiu2
638273b7e7 修复 下载测速时间不准确、卡住的问题 2020-12-24 23:09:07 +08:00
xiu2
dc68529244 update 2020-12-23 11:01:37 +08:00
xiu2
29c927d3cd update 2020-12-22 23:57:46 +08:00
xiu2
fb190c661d update 2020-12-21 11:30:10 +08:00
xiu2
9e39be140a update 2020-12-21 11:15:11 +08:00
xiu2
09a578decf 优化 参数说明 2020-12-19 14:36:25 +08:00
xiu2
8ef6b3b7c2 优化 参数说明 2020-12-19 14:14:16 +08:00
xiu2
cc6b5dd7a6 update 2020-12-19 10:11:43 +08:00
xiu2
6c1166fc5e 新增 下载速度排序;修复 下载测速时间 -dt 参数自定义值无效的问题;回调 下载测速时间默认值为 10 秒 2020-12-19 09:54:18 +08:00
xiu2
f9ac05a072 update 2020-12-14 10:36:40 +08:00
xiu2
976dd79913 优化 IP 段子网掩码解析 2020-12-11 12:12:09 +08:00
xiu2
c8ef175207 优化 IP 段子网掩码解析 2020-12-11 03:12:28 +08:00
xiu2
31dc7aed3c update 2020-12-10 10:13:26 +08:00
xiu2
e3a6f80a14 update 2020-12-10 10:05:05 +08:00
xiu2
38e1d26341 update 2020-12-10 10:02:35 +08:00
xiu2
166d9abe7c 修复 上个版本更新导致的 IPv6 测速报错的问题。 2020-12-09 16:42:50 +08:00
xiu2
f9c310bfb4 update 2020-12-06 08:20:57 +08:00
xiu2
0d54b65f33 新增 测速全部 IP、检查版本更新 2020-12-05 16:03:21 +08:00
xiu2
9f2e5b5b5e add ipv6.txt 2020-11-30 18:10:41 +08:00
XIU2
7b4f6944be 优化 打印帮助时末尾换行(避免在命令行和下一行命令混在一起)
Merge pull request #13 from zhangsean/master
2020-11-30 17:10:07 +08:00
zhangsean
00b569d649 打印帮助换行 2020-11-30 17:05:07 +08:00
xiu2
4b4426c195 新增 IPv6 支持 2020-11-30 16:41:01 +08:00
xiu2
ff3a6d1d56 update 2020-11-25 12:31:57 +08:00
xiu2
a78c6e6270 update 2020-11-15 18:30:39 +08:00
xiu2
6b52fbf5ea update 2020-11-13 14:42:06 +08:00
xiu2
25fa4b65d8 优化 仅 Windows 系统才需要按下 回车键 或 Ctrl+C 退出 2020-11-12 08:11:38 +08:00
xiu2
dca761ec72 update 2020-11-11 22:56:53 +08:00
xiu2
0f5b18b305 update 2020-11-11 22:48:03 +08:00
xiu2
4e1678edc3 update 2020-11-11 19:35:47 +08:00
xiu2
65b451ec4d update 2020-11-11 19:28:14 +08:00
xiu2
72ecee9e26 update 2020-11-11 19:26:31 +08:00
xiu2
1d9f64a4a2 update 2020-11-11 18:19:12 +08:00
xiu2
40b22f660a 新增 指定延迟时间上限、下载速度下限条件 2020-11-11 18:10:53 +08:00
xiu2
12039f4850 修复 -p 0 时没有直接退出程序的问题;优化 代码 2020-11-10 21:03:09 +08:00
xiu2
129deeaf71 update 2020-11-10 19:50:43 +08:00
xiu2
3de6b38e00 优化 IP最后一段完全随机 2020-11-10 19:22:55 +08:00
xiu2
8c0e8732cc 新增 版本号标识 2020-11-10 15:55:52 +08:00
xiu2
8820c5f982 update 2020-11-10 15:35:24 +08:00
xiu2
d50c4806a6 调整 默认下载测速地址为自建地址 2020-11-08 16:46:27 +08:00
xiu2
306ce709c9 update 2020-11-08 10:48:24 +08:00
xiu2
956a35cab0 update 2020-11-08 10:43:26 +08:00
xiu2
3d49bb13ed update 2020-11-08 10:33:46 +08:00
xiu2
3ddd66b3c1 update 2020-11-07 10:38:36 +08:00
xiu2
9aa64db555 新增 自定义下载测速地址功能(-url https://xxx) 2020-11-07 10:07:00 +08:00
xiu2
9654cb8ea6 优化 下载测速文件大小 2020-11-06 12:32:23 +08:00
xiu2
13bae9c6f8 update 2020-11-06 10:01:35 +08:00
xiu2
f0fa3e4d0a update 2020-11-05 23:30:27 +08:00
xiu2
b83734b426 update 2020-11-05 23:26:18 +08:00
xiu2
c1348df16e update 2020-11-05 08:56:36 +08:00
xiu2
c52750ad9c update 2020-11-05 08:52:09 +08:00
xiu2
0e9461f3b7 update 2020-10-22 13:29:03 +08:00
xiu2
07e20028cc 修复 下载测速失效的问题 2020-10-07 02:27:56 +08:00
xiu2
4c92eae311 不输出结果文件 -o "" 改为 -o " " 2020-09-05 17:47:03 +08:00
xiu2
efdbc8f08e 优化直接输出结果排版;成功比率改为丢包率 2020-09-04 15:43:16 +08:00
xiu2
e85a03c651 新增 自定义TCP端口 功能等 2020-09-03 20:07:15 +08:00
7 changed files with 1090 additions and 589 deletions

View File

@@ -1,39 +1,167 @@
package main package main
import ( import (
"bufio" "bufio"
"log" "log"
"net" "net"
"os" "os"
) "strconv"
"strings"
func loadFirstIPOfRangeFromFile(ipFile string) []net.IPAddr { )
file, err := os.Open(ipFile)
if err != nil { // 根据子网掩码获取主机数量
log.Fatal(err) func getCidrHostNum(maskLen int) int {
} cidrIPNum := int(0)
firstIPs := make([]net.IPAddr, 0) if maskLen < 32 {
scanner := bufio.NewScanner(file) var i int = int(32 - maskLen - 1)
scanner.Split(bufio.ScanLines) for ; i >= 1; i-- {
for scanner.Scan() { cidrIPNum += 1 << i
IPString := scanner.Text() }
firstIP, IPRange, err := net.ParseCIDR(IPString) cidrIPNum += 2
if err != nil { } else {
log.Fatal(err) cidrIPNum = 1
} }
firstIP[15] = ipEndWith if cidrIPNum > 255 {
for IPRange.Contains(firstIP) { cidrIPNum = 255
firstIPCopy := make([]byte, len(firstIP)) }
copy(firstIPCopy, firstIP) return cidrIPNum
firstIPs = append(firstIPs, net.IPAddr{IP: firstIPCopy}) }
firstIP[14]++
if firstIP[14] == 0 { // 获取 IP 最后一段最小值和最大值
firstIP[13]++ func getCidrIPRange(cidr string) (uint8, uint8) {
if firstIP[13] == 0 { ip := strings.Split(cidr, "/")[0]
firstIP[12]++ ipSegs := strings.Split(ip, ".")
} maskLen, _ := strconv.Atoi(strings.Split(cidr, "/")[1])
} seg4MinIP, seg4MaxIP := getIPSeg4Range(ipSegs, maskLen)
} //ipPrefix := ipSegs[0] + "." + ipSegs[1] + "." + ipSegs[2] + "."
}
return firstIPs return seg4MinIP,
} seg4MaxIP
}
// 获取 IP 最后一段的区间
func getIPSeg4Range(ipSegs []string, maskLen int) (uint8, uint8) {
ipSeg, _ := strconv.Atoi(ipSegs[3])
return getIPSegRange(uint8(ipSeg), uint8(32-maskLen))
}
// 根据输入的基础IP地址和CIDR掩码计算一个IP片段的区间
func getIPSegRange(userSegIP, offset uint8) (uint8, uint8) {
var ipSegMax uint8 = 255
netSegIP := ipSegMax << offset
segMinIP := netSegIP & userSegIP
segMaxIP := userSegIP&(255<<offset) | ^(255 << offset)
return uint8(segMinIP), uint8(segMaxIP)
}
func loadFirstIPOfRangeFromFile(ipFile string) []net.IPAddr {
file, err := os.Open(ipFile)
if err != nil {
log.Fatal(err)
}
firstIPs := make([]net.IPAddr, 0)
scanner := bufio.NewScanner(file)
scanner.Split(bufio.ScanLines)
for scanner.Scan() {
IPString := scanner.Text()
firstIP, IPRange, err := net.ParseCIDR(IPString)
//fmt.Println(firstIP)
//fmt.Println(IPRange)
if err != nil {
log.Fatal(err)
}
if !ipv6Mode { // IPv4
minIP, maxIP := getCidrIPRange(scanner.Text()) // 获取 IP 最后一段最小值和最大值
Mask, _ := strconv.Atoi(strings.Split(scanner.Text(), "/")[1]) // 获取子网掩码
MaxIPNum := getCidrHostNum(Mask) // 根据子网掩码获取主机数量
for IPRange.Contains(firstIP) {
if allip { // 如果是测速全部 IP
for i := int(minIP); i <= int(maxIP); i++ { // 遍历 IP 最后一段最小值到最大值
firstIP[15] = uint8(i)
firstIPCopy := make([]byte, len(firstIP))
copy(firstIPCopy, firstIP)
firstIPs = append(firstIPs, net.IPAddr{IP: firstIPCopy})
}
} else { // 随机 IP 的最后一段 0.0.0.X
firstIP[15] = minIP + randipEndWith(MaxIPNum)
firstIPCopy := make([]byte, len(firstIP))
copy(firstIPCopy, firstIP)
firstIPs = append(firstIPs, net.IPAddr{IP: firstIPCopy})
}
firstIP[14]++ // 0.0.(X+1).X
if firstIP[14] == 0 {
firstIP[13]++ // 0.(X+1).X.X
if firstIP[13] == 0 {
firstIP[12]++ // (X+1).X.X.X
}
}
}
} else { //IPv6
var tempIP uint8
for IPRange.Contains(firstIP) {
//fmt.Println(firstIP)
//fmt.Println(firstIP[0], firstIP[1], firstIP[2], firstIP[3], firstIP[4], firstIP[5], firstIP[6], firstIP[7], firstIP[8], firstIP[9], firstIP[10], firstIP[11], firstIP[12], firstIP[13], firstIP[14], firstIP[15])
firstIP[15] = randipEndWith(255) // 随机 IP 的最后一段
firstIP[14] = randipEndWith(255) // 随机 IP 的最后一段
firstIPCopy := make([]byte, len(firstIP))
copy(firstIPCopy, firstIP)
firstIPs = append(firstIPs, net.IPAddr{IP: firstIPCopy})
tempIP = firstIP[13]
firstIP[13] += randipEndWith(255)
if firstIP[13] < tempIP {
tempIP = firstIP[12]
firstIP[12] += randipEndWith(255)
if firstIP[12] < tempIP {
tempIP = firstIP[11]
firstIP[11] += randipEndWith(255)
if firstIP[11] < tempIP {
tempIP = firstIP[10]
firstIP[10] += randipEndWith(255)
if firstIP[10] < tempIP {
tempIP = firstIP[9]
firstIP[9] += randipEndWith(255)
if firstIP[9] < tempIP {
tempIP = firstIP[8]
firstIP[8] += randipEndWith(255)
if firstIP[8] < tempIP {
tempIP = firstIP[7]
firstIP[7] += randipEndWith(255)
if firstIP[7] < tempIP {
tempIP = firstIP[6]
firstIP[6] += randipEndWith(255)
if firstIP[6] < tempIP {
tempIP = firstIP[5]
firstIP[5] += randipEndWith(255)
if firstIP[5] < tempIP {
tempIP = firstIP[4]
firstIP[4] += randipEndWith(255)
if firstIP[4] < tempIP {
tempIP = firstIP[3]
firstIP[3] += randipEndWith(255)
if firstIP[3] < tempIP {
tempIP = firstIP[2]
firstIP[2] += randipEndWith(255)
if firstIP[2] < tempIP {
tempIP = firstIP[1]
firstIP[1] += randipEndWith(255)
if firstIP[1] < tempIP {
tempIP = firstIP[0]
firstIP[0] += randipEndWith(255)
}
}
}
}
}
}
}
}
}
}
}
}
}
}
}
}
return firstIPs
}

369
README.md
View File

@@ -1,99 +1,270 @@
# XIU2/CloudflareSpeedTest # XIU2/CloudflareSpeedTest
[![Go Version](https://img.shields.io/github/go-mod/go-version/XIU2/CloudflareSpeedTest.svg?style=flat-square&label=Go&color=00ADD8)](https://github.com/XIU2/CloudflareSpeedTest/blob/master/go.mod) [![Go Version](https://img.shields.io/github/go-mod/go-version/XIU2/CloudflareSpeedTest.svg?style=flat-square&label=Go&color=00ADD8)](https://github.com/XIU2/CloudflareSpeedTest/blob/master/go.mod)
[![Release Version](https://img.shields.io/github/v/release/XIU2/CloudflareSpeedTest.svg?style=flat-square&label=Release&color=1784ff)](https://github.com/XIU2/CloudflareSpeedTest/releases/latest) [![Release Version](https://img.shields.io/github/v/release/XIU2/CloudflareSpeedTest.svg?style=flat-square&label=Release&color=1784ff)](https://github.com/XIU2/CloudflareSpeedTest/releases/latest)
[![GitHub license](https://img.shields.io/github/license/XIU2/CloudflareSpeedTest.svg?style=flat-square&label=License&color=f38020)](https://github.com/XIU2/CloudflareSpeedTest/blob/master/LICENSE) [![GitHub license](https://img.shields.io/github/license/XIU2/CloudflareSpeedTest.svg?style=flat-square&label=License&color=f38020)](https://github.com/XIU2/CloudflareSpeedTest/blob/master/LICENSE)
[![GitHub Star](https://img.shields.io/github/stars/XIU2/CloudflareSpeedTest.svg?style=flat-square&label=Star&color=f38020)](https://github.com/XIU2/CloudflareSpeedTest/stargazers) [![GitHub Star](https://img.shields.io/github/stars/XIU2/CloudflareSpeedTest.svg?style=flat-square&label=Star&color=f38020)](https://github.com/XIU2/CloudflareSpeedTest/stargazers)
[![GitHub Fork](https://img.shields.io/github/forks/XIU2/CloudflareSpeedTest.svg?style=flat-square&label=Fork&color=f38020)](https://github.com/XIU2/CloudflareSpeedTest/network/members) [![GitHub Fork](https://img.shields.io/github/forks/XIU2/CloudflareSpeedTest.svg?style=flat-square&label=Fork&color=f38020)](https://github.com/XIU2/CloudflareSpeedTest/network/members)
国外很多网站都在使用 Cloudflare CDN但分配给中国访客的 IP 并不友好。 国外很多网站都在使用 Cloudflare CDN但分配给中国访客的 IP 并不友好(高延迟/高丢包/速度慢等)
虽然 Cloudflare 公开了所有 [IP 段](https://www.cloudflare.com/ips/) ,但想要在这么多 IP 中找到适合自己的,怕是要累死,所以就有了这个软件。 虽然 Cloudflare 公开了所有 [IP 段](https://www.cloudflare.com/ips/) ,但想要在这么多 IP 中找到适合自己的,怕是要累死,所以就有了这个软件。
该软件可以**测试 Cloudflare CDN 所有 IP 的延迟和速度,获最快 IP** 该软件可以**测试 Cloudflare CDN 延迟和速度,获最快 IP (IPv4+IPv6)**!觉得好用请**点个⭐鼓励一下下~**
你可以将 IP 添加到 `Hosts` 文件中,以提高访问使用 Cloudflare CDN 服务的国外网站速度!
> _本项目也**适用于其他 CDN**,但是需要自行寻找 **CDN IP 段及下载测速地址**否则只能延迟测速。_
****
### 快速使用 > _我另一个开源项目 **[一个 \[油猴脚本\] 轻松解决「Github」文件下载速度慢的问题](https://github.com/XIU2/UserScript)**_
1. 下载编译好的可执行文件 [蓝奏云](https://www.lanzoux.com/b0742hkxe) / [Github](https://github.com/XIU2/CloudflareSpeedTest/releases) 并解压。 ****
2. 双击运行 `CloudflareST.exe`文件Windows系统等待测速... ## 快速使用
测速完毕后,会显示最快的 20 个 IP完整结果则保存在当前目录下的 `result.csv` 文件中,用记事本打开,排序为**延迟由低到高**,每一列用逗号分隔,分别是: ### 下载运行
```
IP 地址, 测试次数, 成功次数, 成功比率, 平均延迟, 下载速度 (MB/s) 1. 下载编译好的可执行文件 [蓝奏云](https://xiu.lanzoux.com/b0742hkxe) / [Github](https://github.com/XIU2/CloudflareSpeedTest/releases) 并解压。
104.27.70.18, 4, 4, 1.00, 150.79, 12.89 2. 双击运行 `CloudflareST.exe`文件Windows等待测速完成...
```
选择一个平均延迟与下载速度都不错的 IP 放到 `Hosts` 文件中(指向域名)。 <details>
<summary>「 Linux 下载运行 」</summary>
****
### 进阶使用 ****
直接双击运行使用的是默认参数,如果想要测试速度更快、测试结果更全面,可以自定义参数。 以下命令仅为示例,版本号和文件名请前往 [**Releases**](https://github.com/XIU2/CloudflareSpeedTest/releases) 查看。
``` cmd
C:\>CloudflareST.exe -h ``` bash
# 如果是第一次使用,则建议创建新文件夹(后续更新请跳过该步骤)
CloudflareSpeedTest mkdir CloudflareST
测试 Cloudflare CDN 所有 IP 的延迟和速度,获取最快 IP
https://github.com/XIU2/CloudflareSpeedTest # 进入文件夹(后续更新,只需要从这里重复下面的下载、解压命令即可)
cd CloudflareST
参数:
-n 500 # 下载 CloudflareST 压缩包(自行根据需求替换 URL 中版本号和文件名)
测速线程数量;数值越大速度越快,请勿超过 1000(结果误差大)(默认 500) wget -N https://github.com/XIU2/CloudflareSpeedTest/releases/download/v1.4.7/CloudflareST_linux_amd64.tar.gz
-t 4
延迟测速次数;单个 IP 测速次数,为 1 时将过滤丢包的IPTCP协议(默认 4) # 解压(不需要删除旧文件,会直接覆盖,自行根据需求替换 文件名)
-tp 443 tar -zxf CloudflareST_linux_amd64.tar.gz
延迟测速端口;延迟测速 TCP 协议的端口;(默认 443)
-dn 20 # 赋予执行权限
下载测速数量;延迟测速并排序后,从最低延迟起下载测速数量,请勿太多(速度慢)(默认 20) chmod +x CloudflareST
-dt 10
下载测速时间;单个 IP 测速最长时间,单位:秒;(默认 10) # 运行
-p 20 ./CloudflareST
直接显示结果;测速后直接显示指定数量的结果,为 -1 时不显示结果直接退出;(默认 20) ```
-f ip.txt
IP 数据文件;相对/绝对路径,如包含空格请加上引号;支持其他 CDN IP段记得禁用下载测速(默认 ip.txt) > **注意**:如果是在**路由器**上运行(如 OpenWrt请先**关闭路由器内的代理**,否则测速结果会不准确。
-o result.csv </details>
输出结果文件;相对/绝对路径,如包含空格请加上引号;允许 .txt 等后缀;(默认 result.csv)
-dd ****
禁用下载测速;如果带上该参数就是禁用下载测速;(默认 启用)
-v > _**注意建议测速时避开晚上高峰期20:00~24:00**,否则测速结果会与其他时间**差距很大...**_
打印程序版本
-h ### 结果示例
打印帮助说明
测速完毕后,默认会显示**最快的 20 个 IP**,示例(我联通白天测速结果):
示例:
CloudflareST.exe -n 500 -t 4 -dn 20 -dt 10 ```
CloudflareST.exe -n 500 -t 4 -dn 20 -dt 10 -f "ip.txt" -o "result.csv" -dd IP 地址 已发送 已接收 丢包率 平均延迟 下载速度 (MB/s)
CloudflareST.exe -n 500 -t 4 -dn 20 -dt 10 -f "C:\abc\ip.txt" -o "C:\abc\result.csv" -dd 104.27.200.69 4 4 0.00 146.23 28.64
``` 172.67.60.78 4 4 0.00 139.82 15.02
104.25.140.153 4 4 0.00 146.49 14.90
#### 使用示例 104.27.192.65 4 4 0.00 140.28 14.07
172.67.62.214 4 4 0.00 139.29 12.71
在 CMD 中运行,或者把启动参数添加到快捷方式中。 104.27.207.5 4 4 0.00 145.92 11.95
> **注意:** 不需要加上所有参数,按需选择,参数前后顺序随意。 172.67.54.193 4 4 0.00 146.71 11.55
104.22.66.8 4 4 0.00 147.42 11.11
``` cmd 104.27.197.63 4 4 0.00 131.29 10.26
# CMD 示例 172.67.58.91 4 4 0.00 140.19 9.14
CloudflareST.exe -n 500 -t 4 -dn 20 -dt 10 ...
```
# 指定 IP数据文件 及 输出结果文件(相对路径,即当前目录下)
CloudflareST.exe -n 500 -t 4 -dn 20 -dt 10 -f "ip.txt" -o "result.csv" -dd > _软件是先**延迟测速并按从低到高排序**后,再**从最低延迟的 IP 开始下载测速**的所以_
# 指定 IP数据文件 及 输出结果文件(绝对路径,即 C:\abc\ 目录下) 测速结果第一行就是**既下载速度最快、又平均延迟最低的最快 IP**!至于拿来干嘛?取决于你~
CloudflareST.exe -n 500 -t 4 -dn 20 -dt 10 -f "C:\abc\ip.txt" -o "C:\abc\result.csv" -dd
``` > _注意因为每次测速都是在 IP 段中随机 IP所以每次的测速结果都不可能相同这是正常的_
``` cmd 完整结果保存在当前目录下的 `result.csv` 文件中,用**记事本/表格软件**打开,格式如下:
# 快捷方式示例(右键快捷方式 - 目标)
## 如果有引号就放在引号外面,记得引号和 - 之间有空格。 ```
"D:\Program Files\CloudflareST\CloudflareST.exe" -n 500 -t 4 -dn 20 -dt 10 IP 地址, 已发送, 已接收, 丢包率, 平均延迟, 下载速度 (MB/s)
``` 104.27.200.69, 4, 4, 0.00, 146.23, 28.64
```
****
### 感谢项目 > _大家可以按自己需求对完整结果**进一步筛选处理**,或者去看一看进阶使用**指定过滤条件**_
* https://github.com/Spedoske/CloudflareScanner
****
意外发现了这个项目,看了之后发现正好解决了我的问题,但是我更喜欢用户命令行方式运行,这样会更方便、有更多使用姿势,于是我临时学了下 Golang 并 Fork 修改了一份命令行方式交互的版本,如果有什么问题可以告诉我,虽然我不一定会~ ## 进阶使用
**** 直接运行使用的是默认参数,如果想要测速结果更全面、更符合自己的要求,可以自定义参数。
### 许可证
The GPL-3.0 License. ``` cmd
C:\>CloudflareST.exe -h
CloudflareSpeedTest vX.X.X
测试 Cloudflare CDN 所有 IP 的延迟和速度,获取最快 IP (IPv4+IPv6)
https://github.com/XIU2/CloudflareSpeedTest
参数:
-n 500
测速线程数量;越多测速越快,性能弱的设备 (如路由器) 请适当调低;(默认 500 最多 1000 )
-t 4
延迟测速次数;单个 IP 延迟测速次数,为 1 时将过滤丢包的IPTCP协议(默认 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 )
-sl 5
下载速度下限;只输出高于指定下载速度的 IP凑够指定数量 [-dn] 才会停止测速;(默认 0 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
打印帮助说明
```
### 使用示例
Windows 是在 CMD 中运行,或者把相关参数添加到快捷方式目标中。
> **注意**:各参数均有**默认值**,使用默认值的参数是可以省略的(**按需选择**),参数**不分前后顺序**。
> **提示**Linux 系统只需要把下面命令中的 `CloudflareST.exe` 改为 `./CloudflareST` 即可。
#### \# IPv4/IPv6
``` bash
# 指定 IPv4 数据文件(-f 默认值就是 ip.txt所以该参数可以省略
CloudflareST.exe -f ip.txt
# 指定 IPv6 数据文件( ipv6.txt ),需要加上 -ipv6 参数
CloudflareST.exe -f ipv6.txt -ipv6
```
****
#### \# 文件相对/绝对路径
``` bash
# 指定 IPv4 数据文件,不显示结果直接退出,输出结果到文件(-p 值为 0
CloudflareST.exe -f ip.txt -p 0 -dd
# 指定 IPv4 数据文件,不输出结果到文件,直接显示结果(-p 值为 10 条,-o 值为空格)
CloudflareST.exe -f ip.txt -o " " -p 10 -dd
# 指定 IPv4 数据文件 及 输出结果到文件(相对路径,即当前目录下,如含空格请加上引号)
CloudflareST.exe -f ip.txt -o result.csv -dd
# 指定 IPv4 数据文件 及 输出结果到文件(绝对路径,即 C:\abc\ 目录下,如含空格请加上引号)
CloudflareST.exe -f C:\abc\ip.txt -o C:\abc\result.csv -dd
```
****
#### \# 自定义下载测速地址
``` bash
# 地址要求:可以直接下载、文件大小超过 200MB、用的是 Cloudflare CDN
CloudflareST.exe -url https://cf.xiu2.xyz/Github/CloudflareSpeedTest.png
```
****
#### \# 自定义测速条件
只有**同时满足三个条件**时才会停止测速。
``` bash
# 平均延迟上限9999 ms下载速度下限5 MB/s数量10 个
# 即需要找到 10 个平均延迟低于 9999 ms 且下载速度高于 5 MB/s 的 IP 才会停止测速。
CloudflareST.exe -sl 5 -dn 10
```
> 没有指定平均延迟上限时,如果一直**凑不够**满足条件的 IP 数量,就会**一直测速**下去。
> 所以建议**同时指定 下载速度下限 和 平均延迟上限**,这样测速到指定延迟上限还没凑够数量,就会终止测速。
``` bash
# 平均延迟上限200 ms下载速度下限5 MB/s数量10 个
# 即需要找到 10 个平均延迟低于 200 ms 且下载速度高于 5 MB/s 的 IP 才会停止测速。
CloudflareST.exe -tl 200 -sl 5 -dn 10
```
> 如果满足条件的 IP **一个都没找到**,那么就会**和不指定条件一样**输出完整结果。
> 所以建议先不指定条件测速一遍,看看平均延迟和下载速度大概在什么范围,避免指定条件**过低/过高**
> 因为 Cloudflare 公开的 IP 段是**回源 IP+任播 IP**,而**回源 IP**是无法用来指向网站的,所以下载测速是 0.00。
> 建议平时运行都加上 `-sl 1`(下载速度下限,最小值 1过滤掉**回源 IP**(下载测速小于 1MB/s 的结果)。
****
#### \# Windows 快捷方式
``` bash
## 右键快捷方式 - 目标
# 如果要不输出结果文件,那么请加上 -o " ",引号里的是空格(没有空格会导致该参数被省略)。
D:\ABC\CloudflareST\CloudflareST.exe -n 500 -t 4 -dn 20 -dt 5 -o " "
# 如果文件路径包含引号,则需要把启动参数放在引号外面,记得引号和 - 之间有空格。
"D:\Program Files\CloudflareST\CloudflareST.exe" -n 500 -t 4 -dn 20 -dt 5 -o " "
```
****
#### \# 单独测速 IP
如果要单独**对一个或多个 IP 进行测速**,只需要把这些 IP 按如下格式写入到任意文本文件中,例如:`1.txt`
```
1.1.1.1/32
2.2.2.2/32
```
然后运行 CloudflareST 时加上启动参数 `-f 1.txt` 即可。
``` bash
# 先进入 CloudflareST 所在目录,然后运行:
# Windows在 CMD 中运行)
CloudflareST.exe -f 1.txt
# Linux
./CloudflareST -f 1.txt
```
****
#### \# 自动更新 Hosts
考虑到很多人获得最快 Cloudflare CDN IP 后,需要替换 Hosts 文件中的 IP。
可以看这个 [**Issues**](https://github.com/XIU2/CloudflareSpeedTest/issues/42) 获取 **Windows/Linux 自动更新 Hosts 脚本**
****
#### \# 自动更新域名解析记录
如果你的域名托管在 Cloudflare则可以通过 Cloudflare 官方提供的 API 来自动更新域名解析记录。
可以看这个 [**Issues**](https://github.com/XIU2/CloudflareSpeedTest/issues/40) 获取**手动教程**或 **Windows/Linux 自动更新脚本**
****
## 问题反馈
如果你遇到什么问题,可以先去 [**Issues**](https://github.com/XIU2/CloudflareSpeedTest/issues) 里看看是否有别人问过了(记得去看下 [**Closed**](https://github.com/XIU2/CloudflareSpeedTest/issues?q=is%3Aissue+is%3Aclosed) 的)。
如果没找到类似问题,请新开个 [**Issues**](https://github.com/XIU2/CloudflareSpeedTest/issues/new) 来告诉我!
> _有问题请**大胆告诉我**,描述越详细越好(必要时可远程协助),如果不说那我怎么去完善功能或~~修复 BUG~~ 呢_
****
## 感谢项目
* https://github.com/Spedoske/CloudflareScanner
> _因为该项目已经很长时间没更新了而我又产生了很多功能需求所以我临时学了下 Go 语言就上手了(菜)..._
> _本软件基于该项目制作但**已添加大量功能及修复 BUG**,并根据大家的使用反馈积极添加、优化功能(闲)..._
****
## License
The GPL-3.0 License.

4
ip.txt
View File

@@ -1,3 +1,7 @@
1.1.1.0/24
1.0.0.0/24
1.1.1.1/32
1.0.0.1/32
173.245.48.0/20 173.245.48.0/20
103.21.244.0/22 103.21.244.0/22
103.22.200.0/22 103.22.200.0/22

42
ipv6.txt Normal file
View File

@@ -0,0 +1,42 @@
2606:4700:10::6814:0/112
2606:4700:10::ac43:0/112
2606:4700:3000::/48
2606:4700:3001::/48
2606:4700:3002::/48
2606:4700:3003::/48
2606:4700:3004::/48
2606:4700:3005::/48
2606:4700:3006::/48
2606:4700:3007::/48
2606:4700:3008::/48
2606:4700:3009::/48
2606:4700:3010::/48
2606:4700:3011::/48
2606:4700:3012::/48
2606:4700:3013::/48
2606:4700:3014::/48
2606:4700:3015::/48
2606:4700:3016::/48
2606:4700:3017::/48
2606:4700:3018::/48
2606:4700:3019::/48
2606:4700:3020::/48
2606:4700:3021::/48
2606:4700:3022::/48
2606:4700:3023::/48
2606:4700:3024::/48
2606:4700:3025::/48
2606:4700:3026::/48
2606:4700:3027::/48
2606:4700:3028::/48
2606:4700:3029::/48
2606:4700:3030::/48
2606:4700:3031::/48
2606:4700:3032::/48
2606:4700:3033::/48
2606:4700:3034::/48
2606:4700:3035::/48
2606:4700:3036::/48
2606:4700:3037::/48
2606:4700:3038::/48
2606:4700:3039::/48

440
main.go
View File

@@ -1,169 +1,271 @@
package main package main
import ( import (
"flag" "flag"
"fmt" "fmt"
"os" "io/ioutil"
"sort" "net/http"
"strconv" "os"
"sync" "runtime"
"time" "sort"
"strconv"
"github.com/cheggaaa/pb/v3" "sync"
) "time"
var version string "github.com/cheggaaa/pb/v3"
var disableDownload bool )
var tcpPort int
var ipFile string var version, ipFile, outputFile, versionNew string
var outputFile string var disableDownload, ipv6Mode, allip bool
var printResult int var tcpPort, printResultNum, downloadSecond int
var timeLimit, speedLimit float64
func init() {
var downloadSecond int64 func init() {
var printVersion bool var printVersion bool
const help = ` var help = `
CloudflareSpeedTest CloudflareSpeedTest ` + version + `
测试 Cloudflare CDN 所有 IP 的延迟和速度,获取最快 IP 测试 Cloudflare CDN 所有 IP 的延迟和速度,获取最快 IP (IPv4+IPv6)
https://github.com/XIU2/CloudflareSpeedTest https://github.com/XIU2/CloudflareSpeedTest
参数: 参数:
-n 500 -n 500
测速线程数量;数值越大速度越快,请勿超过 1000(结果误差大)(默认 500) 测速线程数量;越多测速越快,性能弱的设备 (如路由器) 请适当调低;(默认 500 最多 1000)
-t 4 -t 4
延迟测速次数;单个 IP 测速次数,为 1 时将过滤丢包的IPTCP协议(默认 4) 延迟测速次数;单个 IP 延迟测速次数,为 1 时将过滤丢包的IPTCP协议(默认 4)
-tp 443 -tp 443
延迟测速端口;延迟测速 TCP 协议的端口;(默认 443) 延迟测速端口;延迟测速 TCP 协议的端口;(默认 443)
-dn 20 -dn 20
下载测速数量;延迟测速并排序后,从最低延迟起下载测速数量,请勿太多(速度慢)(默认 20) 下载测速数量;延迟测速并排序后,从最低延迟起下载测速数量;(默认 20)
-dt 10 -dt 10
下载测速时间;单个 IP 测速最长时间,单位:秒;(默认 10) 下载测速时间;单个 IP 下载测速最长时间,单位:秒;(默认 10)
-p 20 -url https://cf.xiu2.xyz/Github/CloudflareSpeedTest.png
直接显示结果;测速后直接显示指定数量的结果,为 -1 时不显示结果直接退出;(默认 20) 下载测速地址;用来下载测速的 Cloudflare CDN 文件地址,如地址含有空格请加上引号;
-f ip.txt -tl 200
IP 数据文件;相对/绝对路径,如包含空格请加上引号;支持其他 CDN IP段记得禁用下载测速(默认 ip.txt) 平均延迟上限;只输出低于指定平均延迟的 IP与下载速度下限搭配使用(默认 9999.00 ms)
-o result.csv -sl 5
输出结果文件;相对/绝对路径,如包含空格请加上引号;允许 .txt 等后缀;(默认 result.csv) 下载速度下限;只输出高于指定下载速度的 IP凑够指定数量 [-dn] 才会停止测速;(默认 0.00 MB/s)
-dd -p 20
禁用下载测速;如果带上该参数就是禁用下载测速(默认 启用) 显示结果数量;测速后直接显示指定数量的结果,为 0 时不显示结果直接退出(默认 20)
-v -f ip.txt
打印程序版本 IP段数据文件如路径含有空格请加上引号支持其他 CDN IP段(默认 ip.txt)
-h -o result.csv
打印帮助说明 输出结果文件;如路径含有空格请加上引号;值为空格时不输出 [-o " "](默认 result.csv)
-dd
示例: 禁用下载测速;禁用后测速结果会按延迟排序 (默认按下载速度排序)(默认 启用)
CloudflareST.exe -n 500 -t 4 -dn 20 -dt 10 -ipv6
CloudflareST.exe -n 500 -t 4 -dn 20 -dt 10 -f "ip.txt" -o "result.csv" -dd IPv6测速模式确保 IP 段数据文件内只包含 IPv6 IP段软件不支持同时测速 IPv4+IPv6(默认 IPv4)
CloudflareST.exe -n 500 -t 4 -dn 20 -dt 10 -f "C:\abc\ip.txt" -o "C:\abc\result.csv" -dd` -allip
测速全部的IP对 IP 段中的每个 IP (仅支持 IPv4) 进行测速;(默认 每个 IP 段随机测速一个 IP)
flag.IntVar(&pingRoutine, "n", 500, "测速线程数量") -v
flag.IntVar(&pingTime, "t", 4, "延迟测速次数") 打印程序版本+检查版本更新
flag.IntVar(&tcpPort, "tp", 443, "延迟测速端口") -h
flag.IntVar(&downloadTestCount, "dn", 20, "下载测速数量") 打印帮助说明
flag.Int64Var(&downloadSecond, "dt", 10, "下载测速时间") `
flag.IntVar(&printResult, "p", 20, "直接显示结果")
flag.BoolVar(&disableDownload, "dd", false, "禁用下载测速") flag.IntVar(&pingRoutine, "n", 500, "测速线程数量")
flag.StringVar(&ipFile, "f", "ip.txt", "IP 数据文件") flag.IntVar(&pingTime, "t", 4, "延迟测速次数")
flag.StringVar(&outputFile, "o", "result.csv", "输出结果文件") flag.IntVar(&tcpPort, "tp", 443, "延迟测速端口")
flag.BoolVar(&printVersion, "v", false, "打印程序版本") flag.IntVar(&downloadTestCount, "dn", 20, "下载测速数量")
flag.IntVar(&downloadSecond, "dt", 10, "下载测速时间")
downloadTestTime = time.Duration(downloadSecond) * time.Second flag.StringVar(&url, "url", "https://cf.xiu2.xyz/Github/CloudflareSpeedTest.png", "下载测速地址")
flag.Float64Var(&timeLimit, "tl", 9999, "延迟时间上限")
flag.Usage = func() { fmt.Print(help) } flag.Float64Var(&speedLimit, "sl", 0, "下载速度下限")
flag.Parse() flag.IntVar(&printResultNum, "p", 20, "显示结果数量")
if printVersion { flag.BoolVar(&disableDownload, "dd", false, "禁用下载测速")
println(version) flag.BoolVar(&ipv6Mode, "ipv6", false, "禁用下载测速")
os.Exit(0) flag.BoolVar(&allip, "allip", false, "测速全部 IP")
} flag.StringVar(&ipFile, "f", "ip.txt", "IP 数据文件")
if pingRoutine <= 0 { flag.StringVar(&outputFile, "o", "result.csv", "输出结果文件")
pingRoutine = 500 flag.BoolVar(&printVersion, "v", false, "打印程序版本")
}
if pingTime <= 0 { flag.Usage = func() { fmt.Print(help) }
pingTime = 4 flag.Parse()
} if printVersion {
if tcpPort < 1 || tcpPort > 65535 { println(version)
tcpPort = 443 fmt.Println("检查版本更新中...")
} checkUpdate()
if downloadTestCount <= 0 { if versionNew != "" {
downloadTestCount = 20 fmt.Println("发现新版本 [" + versionNew + "]!请前往 [https://github.com/XIU2/CloudflareSpeedTest] 更新!")
} } else {
if downloadSecond <= 0 { fmt.Println("当前为最新版本 [" + version + "]")
downloadSecond = 10 }
} os.Exit(0)
if printResult == 0 { }
printResult = 20 if pingRoutine <= 0 {
} pingRoutine = 500
if ipFile == "" { }
ipFile = "ip.txt" if pingTime <= 0 {
} pingTime = 4
if outputFile == "" { }
outputFile = "result.csv" if tcpPort < 1 || tcpPort > 65535 {
} tcpPort = 443
} }
if downloadTestCount <= 0 {
func main() { downloadTestCount = 20
initipEndWith() // 随机数 }
failTime = pingTime // 设置接收次数 if downloadSecond <= 0 {
ips := loadFirstIPOfRangeFromFile(ipFile) // 读入IP downloadSecond = 10
pingCount := len(ips) * pingTime // 计算进度条总数IP*测试次数) }
bar := pb.Full.Start(pingCount) // 进度条总数 if url == "" {
var wg sync.WaitGroup url = "https://cf.xiu2.xyz/Github/CloudflareSpeedTest.png"
var mu sync.Mutex }
var data = make([]CloudflareIPData, 0) if timeLimit <= 0 {
timeLimit = 9999
fmt.Println("开始延迟测速模式TCP端口" + strconv.Itoa(tcpPort) + "") }
control := make(chan bool, pingRoutine) if speedLimit < 0 {
for _, ip := range ips { speedLimit = 0
wg.Add(1) }
control <- false if printResultNum < 0 {
handleProgress := handleProgressGenerator(bar) // 多线程进度条 printResultNum = 20
go tcpingGoroutine(&wg, &mu, ip, tcpPort, pingTime, &data, control, handleProgress) }
} if ipFile == "" {
wg.Wait() ipFile = "ip.txt"
bar.Finish() }
if outputFile == " " {
sort.Sort(CloudflareIPDataSet(data)) // 排序 outputFile = ""
}
// 下载测速 }
if !disableDownload { // 如果禁用下载测速就跳过
if len(data) > 0 { // IP数组长度(IP数量) 大于 0 时继续 func main() {
if len(data) < downloadTestCount { // 如果IP数组长度(IP数量) 小于 下载测速次数则次数改为IP数 go checkUpdate() // 检查版本更新
downloadTestCount = len(data) initRandSeed() // 置随机数种子
fmt.Println("\n[信息] IP数量小于下载测速次数下载测速次数改为IP数。\n") failTime = pingTime // 设置接收次数
} ips := loadFirstIPOfRangeFromFile(ipFile) // 读入IP
bar = pb.Simple.Start(downloadTestCount) pingCount := len(ips) * pingTime // 计算进度条总数IP*测试次数)
fmt.Println("开始下载测速:") bar := pb.Simple.Start(pingCount) // 进度条总数
for i := 0; i < downloadTestCount; i++ { var wg sync.WaitGroup
_, speed := DownloadSpeedHandler(data[i].ip) var mu sync.Mutex
data[i].downloadSpeed = speed var data = make([]CloudflareIPData, 0)
bar.Add(1) var data_2 = make([]CloudflareIPData, 0)
} downloadTestTime = time.Duration(downloadSecond) * time.Second
bar.Finish()
} else { fmt.Println("# XIU2/CloudflareSpeedTest " + version + "\n")
fmt.Println("\n[信息] IP数量为 0跳过下载测速。") if ipv6Mode {
} fmt.Println("开始延迟测速模式TCP IPv6端口" + strconv.Itoa(tcpPort) + "")
} } else {
fmt.Println("开始延迟测速模式TCP IPv4端口" + strconv.Itoa(tcpPort) + "")
// 直接输出结果 }
if printResult > 0 { // 如果禁用下载测速就跳过 control := make(chan bool, pingRoutine)
dateString := convertToString(data) // 转为多维数组 [][]String for _, ip := range ips {
if len(dateString) > 0 { // IP数组长度(IP数量) 大于 0 时继续 wg.Add(1)
if len(dateString) < printResult { // 如果IP数组长度(IP数量) 小于 打印次数则次数改为IP数量 control <- false
printResult = len(dateString) handleProgress := handleProgressGenerator(bar) // 多线程进度条
fmt.Println("\n[信息] IP数量小于显示结果数量显示结果数量改为IP数量。\n") go tcpingGoroutine(&wg, &mu, ip, tcpPort, pingTime, &data, control, handleProgress)
} }
fmt.Println("\nIP 地址 \t", "测试次数\t", "成功次数\t", "成功比率\t", "平均延迟\t", "下载速度 (MB/s)") wg.Wait()
for i := 0; i < printResult; i++ { bar.Finish()
fmt.Println(dateString[i][0], "\t", dateString[i][1], "\t\t", dateString[i][2], "\t\t", dateString[i][3], "\t\t", dateString[i][4], "\t", dateString[i][5])
} sort.Sort(CloudflareIPDataSet(data)) // 排序
fmt.Printf("\n完整内容请查看 %v 文件。请按 回车键 或 Ctrl+C 退出。", outputFile)
var pause int // 下载测速
fmt.Scanln(&pause) if !disableDownload { // 如果禁用下载测速就跳过
} else { if len(data) > 0 { // IP数组长度(IP数量) 大于 0 时继续
fmt.Println("\n[信息] IP数量为 0跳过输出结果。") if len(data) < downloadTestCount { // 如果IP数组长度(IP数量) 小于 下载测速次数则次数改为IP数
} //fmt.Println("\n[信息] IP 数量小于下载测速次数(" + strconv.Itoa(downloadTestCount) + " < " + strconv.Itoa(len(data)) + "下载测速次数改为IP数。\n")
} downloadTestCount = len(data)
}
// 输出结果到文件 var downloadTestCount_2 int // 临时的下载测速次数
ExportCsv(outputFile, data) if timeLimit == 9999 && speedLimit == 0 {
} downloadTestCount_2 = downloadTestCount // 如果没有指定条件,则临时变量为下载测速次数
} else if timeLimit > 0 || speedLimit >= 0 {
downloadTestCount_2 = len(data) // 如果指定了任意一个条件,则临时变量改为总数量
}
fmt.Println("开始下载测速(延迟时间上限:" + fmt.Sprintf("%.2f", timeLimit) + " ms下载速度下限" + fmt.Sprintf("%.2f", speedLimit) + " MB/s")
bar = pb.Simple.Start(downloadTestCount)
for i := 0; i < downloadTestCount_2; i++ {
_, speed := DownloadSpeedHandler(data[i].ip)
data[i].downloadSpeed = speed
if float64(data[i].pingTime) <= timeLimit && float64(speed)/1024/1024 >= speedLimit {
data_2 = append(data_2, data[i]) // 延迟和速度均满足条件时,添加到新数组中
bar.Add(1)
if len(data_2) == downloadTestCount { // 满足条件的 IP =下载测速次数,则跳出循环
break
}
} else if float64(data[i].pingTime) > timeLimit {
break
}
}
bar.Finish()
} else {
fmt.Println("\n[信息] IP数量为 0跳过下载测速。")
}
}
if len(data_2) > 0 { // 如果该数组有内容,说明进行过指定条件的下载测速
sort.Sort(CloudflareIPDataSetD(data_2)) // 排序
if outputFile != "" {
ExportCsv(outputFile, data_2) // 输出结果到文件(指定延迟时间或下载速度的)
}
printResult(data_2) // 显示最快结果(指定延迟时间或下载速度的)
} else {
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数量
//fmt.Println("\n[信息] IP 数量小于显示结果数量(" + strconv.Itoa(printResultNum) + " < " + strconv.Itoa(len(dateString)) + "显示结果数量改为IP数量。\n")
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)
}
}
}
}

332
tcping.go
View File

@@ -1,160 +1,172 @@
package main package main
import ( import (
"context" "context"
"io" "io"
"net" "net"
"net/http" "net/http"
"strconv" "strconv"
"sync" "sync"
"time" "time"
"github.com/VividCortex/ewma" "github.com/VividCortex/ewma"
) )
//bool connectionSucceed float32 time //bool connectionSucceed float32 time
func tcping(ip net.IPAddr, tcpPort int) (bool, float32) { func tcping(ip net.IPAddr, tcpPort int) (bool, float32) {
startTime := time.Now() startTime := time.Now()
conn, err := net.DialTimeout("tcp", ip.String()+":"+strconv.Itoa(tcpPort), tcpConnectTimeout) var fullAddress string
if err != nil { //fmt.Println(ip.String())
return false, 0 if ipv6Mode { // IPv6 需要加上 []
} else { fullAddress = "[" + ip.String() + "]:" + strconv.Itoa(tcpPort)
var endTime = time.Since(startTime) } else {
var duration = float32(endTime.Microseconds()) / 1000.0 fullAddress = ip.String() + ":" + strconv.Itoa(tcpPort)
_ = conn.Close() }
return true, duration conn, err := net.DialTimeout("tcp", fullAddress, tcpConnectTimeout)
} if err != nil {
} return false, 0
} else {
//pingReceived pingTotalTime var endTime = time.Since(startTime)
func checkConnection(ip net.IPAddr, tcpPort int) (int, float32) { var duration = float32(endTime.Microseconds()) / 1000.0
pingRecv := 0 _ = conn.Close()
var pingTime float32 = 0.0 return true, duration
for i := 1; i <= failTime; i++ { }
pingSucceed, pingTimeCurrent := tcping(ip, tcpPort) }
if pingSucceed {
pingRecv++ //pingReceived pingTotalTime
pingTime += pingTimeCurrent func checkConnection(ip net.IPAddr, tcpPort int) (int, float32) {
} pingRecv := 0
} var pingTime float32 = 0.0
return pingRecv, pingTime for i := 1; i <= failTime; i++ {
} pingSucceed, pingTimeCurrent := tcping(ip, tcpPort)
if pingSucceed {
//return Success packetRecv averagePingTime specificIPAddr pingRecv++
func tcpingHandler(ip net.IPAddr, tcpPort int, pingCount int, progressHandler func(e progressEvent)) (bool, int, float32, net.IPAddr) { pingTime += pingTimeCurrent
ipCanConnect := false }
pingRecv := 0 }
var pingTime float32 = 0.0 return pingRecv, pingTime
for !ipCanConnect { }
pingRecvCurrent, pingTimeCurrent := checkConnection(ip, tcpPort)
if pingRecvCurrent != 0 { //return Success packetRecv averagePingTime specificIPAddr
ipCanConnect = true func tcpingHandler(ip net.IPAddr, tcpPort int, pingCount int, progressHandler func(e progressEvent)) (bool, int, float32, net.IPAddr) {
pingRecv = pingRecvCurrent ipCanConnect := false
pingTime = pingTimeCurrent pingRecv := 0
} else { var pingTime float32 = 0.0
ip.IP[15]++ for !ipCanConnect {
if ip.IP[15] == 0 { pingRecvCurrent, pingTimeCurrent := checkConnection(ip, tcpPort)
break if pingRecvCurrent != 0 {
} ipCanConnect = true
break pingRecv = pingRecvCurrent
} pingTime = pingTimeCurrent
} } else {
if ipCanConnect { ip.IP[15]++
progressHandler(AvailableIPFound) if ip.IP[15] == 0 {
for i := failTime; i < pingCount; i++ { break
pingSuccess, pingTimeCurrent := tcping(ip, tcpPort) }
progressHandler(NormalPing) break
if pingSuccess { }
pingRecv++ }
pingTime += pingTimeCurrent if ipCanConnect {
} progressHandler(AvailableIPFound)
} for i := failTime; i < pingCount; i++ {
return true, pingRecv, pingTime / float32(pingRecv), ip pingSuccess, pingTimeCurrent := tcping(ip, tcpPort)
} else { progressHandler(NormalPing)
progressHandler(NoAvailableIPFound) if pingSuccess {
return false, 0, 0, net.IPAddr{} pingRecv++
} pingTime += pingTimeCurrent
} }
}
func tcpingGoroutine(wg *sync.WaitGroup, mutex *sync.Mutex, ip net.IPAddr, tcpPort int, pingCount int, csv *[]CloudflareIPData, control chan bool, progressHandler func(e progressEvent)) { return true, pingRecv, pingTime / float32(pingRecv), ip
defer wg.Done() } else {
success, pingRecv, pingTimeAvg, currentIP := tcpingHandler(ip, tcpPort, pingCount, progressHandler) progressHandler(NoAvailableIPFound)
if success { return false, 0, 0, net.IPAddr{}
mutex.Lock() }
var cfdata CloudflareIPData }
cfdata.ip = currentIP
cfdata.pingReceived = pingRecv func tcpingGoroutine(wg *sync.WaitGroup, mutex *sync.Mutex, ip net.IPAddr, tcpPort int, pingCount int, csv *[]CloudflareIPData, control chan bool, progressHandler func(e progressEvent)) {
cfdata.pingTime = pingTimeAvg defer wg.Done()
cfdata.pingCount = pingCount success, pingRecv, pingTimeAvg, currentIP := tcpingHandler(ip, tcpPort, pingCount, progressHandler)
*csv = append(*csv, cfdata) if success {
mutex.Unlock() mutex.Lock()
} var cfdata CloudflareIPData
<-control cfdata.ip = currentIP
} cfdata.pingReceived = pingRecv
cfdata.pingTime = pingTimeAvg
func GetDialContextByAddr(fakeSourceAddr string) func(ctx context.Context, network, address string) (net.Conn, error) { cfdata.pingCount = pingCount
return func(ctx context.Context, network, address string) (net.Conn, error) { *csv = append(*csv, cfdata)
c, e := (&net.Dialer{}).DialContext(ctx, network, fakeSourceAddr) mutex.Unlock()
return c, e }
} <-control
} }
//bool : can download,float32 downloadSpeed func GetDialContextByAddr(fakeSourceAddr string) func(ctx context.Context, network, address string) (net.Conn, error) {
func DownloadSpeedHandler(ip net.IPAddr) (bool, float32) { return func(ctx context.Context, network, address string) (net.Conn, error) {
var client = http.Client{ c, e := (&net.Dialer{}).DialContext(ctx, network, fakeSourceAddr)
Transport: nil, return c, e
CheckRedirect: nil, }
Jar: nil, }
Timeout: 0,
} //bool : can download,float32 downloadSpeed
client.Transport = &http.Transport{ func DownloadSpeedHandler(ip net.IPAddr) (bool, float32) {
DialContext: GetDialContextByAddr(ip.String() + ":443"), var client = http.Client{
} Transport: nil,
response, err := client.Get(url) CheckRedirect: nil,
Jar: nil,
if err != nil { Timeout: downloadTestTime,
return false, 0 }
} else { var fullAddress string
defer func() { _ = response.Body.Close() }() if ipv6Mode { // IPv6 需要加上 []
if response.StatusCode == 200 { fullAddress = "[" + ip.String() + "]:443"
timeStart := time.Now() } else {
timeEnd := timeStart.Add(downloadTestTime) fullAddress = ip.String() + ":443"
}
contentLength := response.ContentLength client.Transport = &http.Transport{
buffer := make([]byte, downloadBufferSize) DialContext: GetDialContextByAddr(fullAddress),
}
var contentRead int64 = 0 response, err := client.Get(url)
var timeSlice = downloadTestTime / 100 if err != nil {
var timeCounter = 1 return false, 0
var lastContentRead int64 = 0 } else {
defer func() { _ = response.Body.Close() }()
var nextTime = timeStart.Add(timeSlice * time.Duration(timeCounter)) if response.StatusCode == 200 {
e := ewma.NewMovingAverage() timeStart := time.Now()
timeEnd := timeStart.Add(downloadTestTime)
for contentLength != contentRead {
var currentTime = time.Now() contentLength := response.ContentLength
if currentTime.After(nextTime) { buffer := make([]byte, downloadBufferSize)
timeCounter += 1
nextTime = timeStart.Add(timeSlice * time.Duration(timeCounter)) var contentRead int64 = 0
e.Add(float64(contentRead - lastContentRead)) var timeSlice = downloadTestTime / 100
lastContentRead = contentRead var timeCounter = 1
} var lastContentRead int64 = 0
if currentTime.After(timeEnd) {
break var nextTime = timeStart.Add(timeSlice * time.Duration(timeCounter))
} e := ewma.NewMovingAverage()
bufferRead, err := response.Body.Read(buffer)
contentRead += int64(bufferRead) for contentLength != contentRead {
if err != nil { var currentTime = time.Now()
if err != io.EOF { if currentTime.After(nextTime) {
break timeCounter += 1
} else { nextTime = timeStart.Add(timeSlice * time.Duration(timeCounter))
e.Add(float64(contentRead-lastContentRead) / (float64(nextTime.Sub(currentTime)) / float64(timeSlice))) e.Add(float64(contentRead - lastContentRead))
} lastContentRead = contentRead
} }
} if currentTime.After(timeEnd) {
return true, float32(e.Value()) / (float32(downloadTestTime.Seconds()) / 100) break
} else { }
return false, 0 bufferRead, err := response.Body.Read(buffer)
} contentRead += int64(bufferRead)
} if err != nil {
} if err != io.EOF {
break
} else {
e.Add(float64(contentRead-lastContentRead) / (float64(nextTime.Sub(currentTime)) / float64(timeSlice)))
}
}
}
return true, float32(e.Value()) / (float32(downloadTestTime.Seconds()) / 120)
} else {
return false, 0
}
}
}

286
util.go
View File

@@ -1,122 +1,164 @@
package main package main
import ( import (
"encoding/csv" "encoding/csv"
"log" "log"
"math/rand" "math/rand"
"net" "net"
"os" "os"
"strconv" "strconv"
"time" "time"
"github.com/cheggaaa/pb/v3" "github.com/cheggaaa/pb/v3"
) )
type CloudflareIPData struct { type CloudflareIPData struct {
ip net.IPAddr ip net.IPAddr
pingCount int pingCount int
pingReceived int pingReceived int
recvRate float32 recvRate float32
downloadSpeed float32 downloadSpeed float32
pingTime float32 pingTime float32
} }
func (cf *CloudflareIPData) getRecvRate() float32 { func (cf *CloudflareIPData) getRecvRate() float32 {
if cf.recvRate == 0 { if cf.recvRate == 0 {
cf.recvRate = float32(cf.pingReceived) / float32(cf.pingCount) pingLost := cf.pingCount - cf.pingReceived
} cf.recvRate = float32(pingLost) / float32(cf.pingCount)
return cf.recvRate }
} return cf.recvRate
}
func ExportCsv(filePath string, data []CloudflareIPData) {
fp, err := os.Create(filePath) func ExportCsv(filePath string, data []CloudflareIPData) {
if err != nil { fp, err := os.Create(filePath)
log.Fatalf("创建文件["+filePath+"]句柄失败,%v", err) if err != nil {
return log.Fatalf("创建文件["+filePath+"]句柄失败,%v", err)
} return
defer fp.Close() }
w := csv.NewWriter(fp) //创建一个新的写入文件流 defer fp.Close()
w.Write([]string{"IP 地址", "测试次数", "成功次数", "成功比率", "平均延迟", "下载速度 (MB/s)"}) w := csv.NewWriter(fp) //创建一个新的写入文件流
w.WriteAll(convertToString(data)) w.Write([]string{"IP 地址", "已发送", "已接收", "丢包率", "平均延迟", "下载速度 (MB/s)"})
w.Flush() w.WriteAll(convertToString(data))
} w.Flush()
}
func (cf *CloudflareIPData) toString() []string {
result := make([]string, 6) func (cf *CloudflareIPData) toString() []string {
result[0] = cf.ip.String() result := make([]string, 6)
result[1] = strconv.Itoa(cf.pingCount) result[0] = cf.ip.String()
result[2] = strconv.Itoa(cf.pingReceived) result[1] = strconv.Itoa(cf.pingCount)
result[3] = strconv.FormatFloat(float64(cf.getRecvRate()), 'f', 2, 32) result[2] = strconv.Itoa(cf.pingReceived)
result[4] = strconv.FormatFloat(float64(cf.pingTime), 'f', 2, 32) result[3] = strconv.FormatFloat(float64(cf.getRecvRate()), 'f', 2, 32)
result[5] = strconv.FormatFloat(float64(cf.downloadSpeed)/1024/1024, 'f', 2, 32) result[4] = strconv.FormatFloat(float64(cf.pingTime), 'f', 2, 32)
return result result[5] = strconv.FormatFloat(float64(cf.downloadSpeed)/1024/1024, 'f', 2, 32)
} return result
}
func convertToString(data []CloudflareIPData) [][]string {
result := make([][]string, 0) func convertToString(data []CloudflareIPData) [][]string {
for _, v := range data { result := make([][]string, 0)
result = append(result, v.toString()) for _, v := range data {
} result = append(result, v.toString())
return result }
} return result
}
var pingTime int
var pingRoutine int var pingTime int
var pingRoutine int
var ipEndWith uint8 = 0
type progressEvent int
type progressEvent int
const (
const ( NoAvailableIPFound progressEvent = iota
NoAvailableIPFound progressEvent = iota AvailableIPFound
AvailableIPFound NormalPing
NormalPing )
)
var url string
const url string = "https://apple.freecdn.workers.dev/105/media/us/iphone-11-pro/2019/3bd902e4-0752-4ac1-95f8-6225c32aec6d/films/product/iphone-11-pro-product-tpl-cc-us-2019_1280x720h.mp4"
var downloadTestTime time.Duration
var downloadTestTime time.Duration
const downloadBufferSize = 1024
const downloadBufferSize = 1024
var downloadTestCount int
var downloadTestCount int
//const defaultTcpPort = 443
//const defaultTcpPort = 443 const tcpConnectTimeout = time.Second * 1
const tcpConnectTimeout = time.Second * 1
var failTime int
var failTime int
// 平均延迟排序(丢包另算)
type CloudflareIPDataSet []CloudflareIPData type CloudflareIPDataSet []CloudflareIPData
func initipEndWith() { // 下载速度排序
rand.Seed(time.Now().UnixNano()) type CloudflareIPDataSetD []CloudflareIPData
ipEndWith = uint8(rand.Intn(254) + 1)
} func initRandSeed() {
rand.Seed(time.Now().UnixNano())
func handleProgressGenerator(pb *pb.ProgressBar) func(e progressEvent) { }
return func(e progressEvent) {
switch e { func randipEndWith(num int) uint8 {
case NoAvailableIPFound: return uint8(rand.Intn(num))
pb.Add(pingTime) }
case AvailableIPFound:
pb.Add(failTime) func GetRandomString() string {
case NormalPing: str := "0123456789abcdef"
pb.Increment() bytes := []byte(str)
} result := []byte{}
} r := rand.New(rand.NewSource(time.Now().UnixNano()))
} for i := 0; i < 4; i++ {
result = append(result, bytes[r.Intn(len(bytes))])
func (cfs CloudflareIPDataSet) Len() int { }
return len(cfs) return string(result)
} }
func (cfs CloudflareIPDataSet) Less(i, j int) bool { func ipPadding(ip string) string {
if (cfs)[i].getRecvRate() != cfs[j].getRecvRate() { var ipLength int
return cfs[i].getRecvRate() > cfs[j].getRecvRate() var ipPrint string
} ipPrint = ip
return cfs[i].pingTime < cfs[j].pingTime ipLength = len(ipPrint)
} if ipLength < 15 {
for i := 0; i <= 15-ipLength; i++ {
func (cfs CloudflareIPDataSet) Swap(i, j int) { ipPrint += " "
cfs[i], cfs[j] = cfs[j], cfs[i] }
} }
return ipPrint
}
func handleProgressGenerator(pb *pb.ProgressBar) func(e progressEvent) {
return func(e progressEvent) {
switch e {
case NoAvailableIPFound:
pb.Add(pingTime)
case AvailableIPFound:
pb.Add(failTime)
case NormalPing:
pb.Increment()
}
}
}
func (cfs CloudflareIPDataSet) Len() int {
return len(cfs)
}
func (cfs CloudflareIPDataSet) Less(i, j int) bool {
if (cfs)[i].getRecvRate() != cfs[j].getRecvRate() {
return cfs[i].getRecvRate() < cfs[j].getRecvRate()
}
return cfs[i].pingTime < cfs[j].pingTime
}
func (cfs CloudflareIPDataSet) Swap(i, j int) {
cfs[i], cfs[j] = cfs[j], cfs[i]
}
func (cfs CloudflareIPDataSetD) Len() int {
return len(cfs)
}
func (cfs CloudflareIPDataSetD) Less(i, j int) bool {
return cfs[i].downloadSpeed > cfs[j].downloadSpeed
}
func (cfs CloudflareIPDataSetD) Swap(i, j int) {
cfs[i], cfs[j] = cfs[j], cfs[i]
}