8 Commits

Author SHA1 Message Date
xiu2
7e5804b7ba 新增 支持仅指定 [平均延迟上限] 条件 2021-02-18 23:31:01 +08:00
xiu2
cee772547b README.md 2021-02-17 11:38:01 +08:00
xiu2
1a939f752b README.md 2021-02-10 19:48:01 +08:00
xiu2
5b45f400a2 补充 README.md 2021-02-06 20:24:35 +08:00
xiu2
3b43b21b83 补充 README.md 2021-02-05 22:08:34 +08:00
xiu2
2f284efddd 补充 README.md 2021-02-04 22:59:43 +08:00
xiu2
f960ce4a4b 补充 README.md 2021-02-04 18:45:56 +08:00
xiu2
8ae1d495af 补充 README.md 2021-02-04 17:32:14 +08:00
2 changed files with 97 additions and 59 deletions

View File

@@ -6,7 +6,7 @@
[![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)
国外很多网站都在使用 Cloudflare CDN但分配给中国访客的 IP 并不友好(高延迟/高丢包/速度慢等)。
[国外很多网站](https://github.com/XIU2/CloudflareSpeedTest/discussions/62)都在使用 Cloudflare CDN但分配给中国访客的 IP 并不友好(高延迟/高丢包/速度慢等)。
虽然 Cloudflare 公开了所有 [IP 段](https://www.cloudflare.com/ips/) ,但想要在这么多 IP 中找到适合自己的,怕是要累死,所以就有了这个软件。
该软件可以**测试 Cloudflare CDN 延迟和速度,获取最快 IP (IPv4+IPv6)**!觉得好用请**点个⭐鼓励一下下~**
@@ -24,7 +24,7 @@
2. 双击运行 `CloudflareST.exe`文件Windows等待测速完成...
<details>
<summary>「 Linux 下载运行 」</summary>
<summary><code><strong>「 点击查看 Linux 下载运行命令示例 」</strong></code></summary>
****
@@ -50,18 +50,21 @@ chmod +x CloudflareST
./CloudflareST
```
> **注意**:如果是在**路由器**上运行(如 OpenWrt请先**关闭路由器内的代理**,否则测速结果会不准确。
> 如果平**均延迟非常低**(如 0.xx则说明 CloudflareST **测速时走了代理**,请先关闭代理软件后再测速。
> 如果在**路由器**上运行(如 OpenWrt请先关闭路由器内的代理否则测速结果会**不准确且无法使用**。
</details>
****
> _**注意建议测速时避开晚上高峰期20:00~24:00**,否则测速结果会与其他时间**差距很大...**_
> [_**在 Android 手机上运行 CloudflareST 测速的简单教程 ...**_](https://github.com/XIU2/CloudflareSpeedTest/discussions/61)
> _**建议测速时避开晚上高峰期20:00~24:00**,否则测速结果会与其他时间**相差很大...**_
### 结果示例
测速完毕后,默认会显示**最快的 20 个 IP**,示例(我联通白天测速结果):
```
``` bash
IP 地址 已发送 已接收 丢包率 平均延迟 下载速度 (MB/s)
104.27.200.69 4 4 0.00 146.23 28.64
172.67.60.78 4 4 0.00 139.82 15.02
@@ -74,14 +77,15 @@ IP 地址 已发送 已接收 丢包率 平均延迟 下载速度
104.27.197.63 4 4 0.00 131.29 10.26
172.67.58.91 4 4 0.00 140.19 9.14
...
# 如果平均延迟非常低(如 0.xx则说明 CloudflareST 测速时走了代理,请先关闭代理软件后再测速。
# 如果在路由器上运行(如 OpenWrt请先关闭路由器内的代理否则测速结果会不准确且无法使用。
# 因为每次测速都是在每个 IP 段中随机 IP所以每次的测速结果都不可能相同这是正常的
# 软件是先 延迟测速并按从低到高排序后,再从 最低延迟的 IP 开始下载测速的,所以:
```
> _软件是先**延迟测速并按从低到高排序**后,再**从最低延迟的 IP 开始下载测速**的所以_
测速结果第一行就是**既下载速度最快、又平均延迟最低的最快 IP**!至于拿来干嘛?取决于你~
> _注意因为每次测速都是在 IP 段中随机 IP所以每次的测速结果都不可能相同这是正常的_
完整结果保存在当前目录下的 `result.csv` 文件中,用**记事本/表格软件**打开,格式如下:
```
@@ -117,9 +121,9 @@ https://github.com/XIU2/CloudflareSpeedTest
-url https://cf.xiu2.xyz/Github/CloudflareSpeedTest.png
下载测速地址;用来下载测速的 Cloudflare CDN 文件地址,如地址含有空格请加上引号;
-tl 200
平均延迟上限;只输出低于指定平均延迟的 IP下载速度下限搭配使用(默认 9999 ms )
平均延迟上限;只输出低于指定平均延迟的 IP可单独使用也可搭配下载速度下限;(默认 9999.00 ms)
-sl 5
下载速度下限;只输出高于指定下载速度的 IP凑够指定数量 [-dn] 才会停止测速;(默认 0 MB/s )
下载速度下限;只输出高于指定下载速度的 IP凑够指定数量 [-dn] 才会停止测速;(默认 0.00 MB/s )
-p 20
显示结果数量;测速后直接显示指定数量的结果,为 0 时不显示结果直接退出;(默认 20 )
-f ip.txt
@@ -180,28 +184,50 @@ CloudflareST.exe -url https://cf.xiu2.xyz/Github/CloudflareSpeedTest.png
****
#### \# 自定义测速条件
只有**同时满足三个条件**时才会停止测速。
- 仅指定 **[平均延迟上限]** 条件
``` bash
# 平均延迟上限:9999 ms下载速度下限5 MB/s数量10 个
# 即需要找到 10 个平均延迟低于 9999 ms 且下载速度高于 5 MB/s 的 IP 才会停止测速
# 平均延迟上限:200 ms下载速度下限0 MB/s数量10 个
# 即找到平均延迟低于 200 ms 的 IP然后再按延迟从低到高进行 10 次下载测速
CloudflareST.exe -tl 200 -dn 10
```
> 如果没有一个 IP **平均延迟低于 200ms**,那么不会输出任何内容。
- 仅指定 **[平均延迟上限]** 条件,且**只延迟测速,不下载测速**
``` bash
# 平均延迟上限200 ms下载速度下限0 MB/s数量无限 个
# 即只输出低于 200ms 的 IP且不再下载测速因为不再下载测速所以 -dn 参数就无效了)
CloudflareST.exe -tl 200 -dd
```
- 仅指定 **[下载速度下限]** 条件
``` 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
# 平均延迟上限、下载速度下限均支持小数(如 -sl 0.5
# 平均延迟上限:200 ms下载速度下限5.6 MB/s数量10 个( 可选)
# 即需要找到 10 个平均延迟低于 200 ms 且下载速度高于 5 .6MB/s 的 IP 才会停止测速
CloudflareST.exe -tl 200 -sl 5.6 -dn 10
```
> 如果满足条件的 IP **一个都没找到**,那么就会**和不指定条件一样**输出完整结果
> 如果没有一个 IP **平均延迟低于 200ms**,那么不会输出任何内容
> 如果没有一个 IP **下载速度高于 5.6 MB/s**,那么就会**和不指定 [下载速度下限] 条件一样**输出结果。
> 所以建议先不指定条件测速一遍,看看平均延迟和下载速度大概在什么范围,避免指定条件**过低/过高**
> 因为 Cloudflare 公开的 IP 段是**回源 IP+任播 IP**,而**回源 IP**是无法用来指向网站的,所以下载测速是 0.00。
> 建议平时运行都加上 `-sl 1`(下载速度下限,最小值 1过滤掉**回源 IP**(下载测速小于 1MB/s 的结果)。
> 运行时可以加上 `-sl 0.01`(下载速度下限,最小值 0.01过滤掉**回源 IP**(下载测速小于 0.01MB/s 的结果)。
****
#### \# Windows 快捷方式
@@ -219,9 +245,12 @@ D:\ABC\CloudflareST\CloudflareST.exe -n 500 -t 4 -dn 20 -dt 5 -o " "
如果要单独**对一个或多个 IP 进行测速**,只需要把这些 IP 按如下格式写入到任意文本文件中,例如:`1.txt`
```
``` json
1.1.1.1/32
2.2.2.2/32
1.0.0.1/24
// 子网掩码 /32 指的是这个 IP 本身,即 1.1.1.1
// 子网掩码 /24 指的是这个 IP 最后一段,即 1.0.0.1~1.0.0.255
```
然后运行 CloudflareST 时加上启动参数 `-f 1.txt` 即可。
@@ -233,6 +262,8 @@ CloudflareST.exe -f 1.txt
# Linux
./CloudflareST -f 1.txt
# 对于 IP 段 1.0.0.1/24 软件只会随机最后一段1.0.0.1~255如果要测速该 IP 段中的所有 IP需要加上 -allip 参数。
```
****

83
main.go
View File

@@ -41,7 +41,7 @@ https://github.com/XIU2/CloudflareSpeedTest
-url https://cf.xiu2.xyz/Github/CloudflareSpeedTest.png
下载测速地址;用来下载测速的 Cloudflare CDN 文件地址,如地址含有空格请加上引号;
-tl 200
平均延迟上限;只输出低于指定平均延迟的 IP下载速度下限搭配使用(默认 9999.00 ms)
平均延迟上限;只输出低于指定平均延迟的 IP可单独使用也可搭配下载速度下限;(默认 9999.00 ms)
-sl 5
下载速度下限;只输出高于指定下载速度的 IP凑够指定数量 [-dn] 才会停止测速;(默认 0.00 MB/s)
-p 20
@@ -68,7 +68,7 @@ https://github.com/XIU2/CloudflareSpeedTest
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(&timeLimit, "tl", 9999, "平均延迟上限")
flag.Float64Var(&speedLimit, "sl", 0, "下载速度下限")
flag.IntVar(&printResultNum, "p", 20, "显示结果数量")
flag.BoolVar(&disableDownload, "dd", false, "禁用下载测速")
@@ -136,14 +136,15 @@ func main() {
var wg sync.WaitGroup
var mu sync.Mutex
var data = make([]CloudflareIPData, 0)
var data_2 = make([]CloudflareIPData, 0)
var data2 = make([]CloudflareIPData, 0)
downloadTestTime = time.Duration(downloadSecond) * time.Second
// 开始延迟测速
fmt.Println("# XIU2/CloudflareSpeedTest " + version + "\n")
if ipv6Mode {
fmt.Println("开始延迟测速模式TCP IPv6端口" + strconv.Itoa(tcpPort) + "")
if ipv6Mode { // IPv6 模式判断
fmt.Println("开始延迟测速模式TCP IPv6端口" + strconv.Itoa(tcpPort) + ",平均延迟上限:" + fmt.Sprintf("%.2f", timeLimit) + " ms")
} else {
fmt.Println("开始延迟测速模式TCP IPv4端口" + strconv.Itoa(tcpPort) + "")
fmt.Println("开始延迟测速模式TCP IPv4端口" + strconv.Itoa(tcpPort) + ",平均延迟上限:" + fmt.Sprintf("%.2f", timeLimit) + " ms")
}
control := make(chan bool, pingRoutine)
for _, ip := range ips {
@@ -155,54 +156,61 @@ func main() {
wg.Wait()
bar.Finish()
sort.Sort(CloudflareIPDataSet(data)) // 排序
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数量) 小于 下载测速数,则次数为IP数
//fmt.Println("\n[信息] IP 数量小于下载测速次数(" + strconv.Itoa(downloadTestCount) + " < " + strconv.Itoa(len(data)) + "下载测速次数改为IP数。\n")
if len(data) > 0 { // IP数组长度(IP数量) 大于 0 时才会继续下载测速
if len(data) < downloadTestCount { // 如果IP数组长度(IP数量) 小于下载测速数量(-dn,则次数修正为IP数
downloadTestCount = len(data)
}
var downloadTestCount_2 int // 临时的下载测速次数
if timeLimit == 9999 && speedLimit == 0 {
downloadTestCount_2 = downloadTestCount // 如果没有指定条件,则临时变量为下载测速次数
} else if timeLimit > 0 || speedLimit >= 0 {
downloadTestCount_2 = len(data) // 如果指定了任意一个条件,则临时变量改为总数量
var downloadTestCount2 int // 临时的下载测速次数,即实际的下载测速数量
if speedLimit > 0 {
downloadTestCount2 = len(data) // 如果指定了 [下载速度下限] 条件,则临时变量改为总数量(即一直测速下去,直到凑够下载测速数量 -dn
} else {
downloadTestCount2 = downloadTestCount // 如果没有指定 [下载速度下限] 条件,则临时变量为下载测速数量(-dn
}
fmt.Println("开始下载测速(延迟时间上限:" + fmt.Sprintf("%.2f", timeLimit) + " ms下载速度下限:" + fmt.Sprintf("%.2f", speedLimit) + " MB/s")
fmt.Println("开始下载测速(下载速度下限:" + fmt.Sprintf("%.2f", speedLimit) + " MB/s,下载测速数量:" + strconv.Itoa(downloadTestCount) + ",下载测速队列:" + strconv.Itoa(downloadTestCount2) + "")
bar = pb.Simple.Start(downloadTestCount)
for i := 0; i < downloadTestCount_2; i++ {
for i := 0; i < downloadTestCount2; 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]) // 延迟和速度均满足条件时,添加到新数组中
// 在每个 IP 下载测速后,以 [下载速度下限] 条件过滤结果
if float64(speed)/1024/1024 >= speedLimit {
data2 = append(data2, data[i]) // 高于下载速度下限时,添加到新数组中
bar.Add(1)
if len(data_2) == downloadTestCount { // 满足条件的 IP =下载测速次数,则跳出循环
if len(data2) == downloadTestCount { // 凑够满足条件的 IP 时(下载测速数量 -dn跳出循环
break
}
} else if float64(data[i].pingTime) > timeLimit {
break
}
}
bar.Finish()
} else {
fmt.Println("\n[信息] IP数量为 0跳过下载测速。")
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) // 显示最快结果
if len(data2) > 0 { // 如果该数组有内容,说明指定了 [下载测速下限] 条件,且最少有 1 个满足条件的 IP
data = data2
}
sort.Sort(CloudflareIPDataSetD(data)) // 排序(按下载速度,从高到低)
if outputFile != "" {
ExportCsv(outputFile, data) // 输出结果到文件
}
printResult(data) // 显示最快结果
}
// 显示最快结果
@@ -212,7 +220,6 @@ func printResult(data []CloudflareIPData) {
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 太长了,所以需要调整一下间隔
@@ -231,7 +238,7 @@ func printResult(data []CloudflareIPData) {
fmt.Println("\n发现新版本 [" + versionNew + "]!请前往 [https://github.com/XIU2/CloudflareSpeedTest] 更新!")
}
if sysType == "windows" { // 如果是 Windows 系统,则需要按下 回车键 或 Ctrl+C 退出
if sysType == "windows" { // 如果是 Windows 系统,则需要按下 回车键 或 Ctrl+C 退出(避免通过双击运行时,测速完毕后直接关闭)
if outputFile != "" {
fmt.Printf("\n完整测速结果已写入 %v 文件,请使用记事本/表格软件查看。\n按下 回车键 或 Ctrl+C 退出。", outputFile)
} else {
@@ -245,7 +252,7 @@ func printResult(data []CloudflareIPData) {
}
}
} else {
fmt.Println("\n[信息] IP数量为 0跳过输出结果。")
fmt.Println("\n[信息] 完整测速结果 IP 数量为 0跳过输出结果。")
}
} else {
fmt.Println("\n完整测速结果已写入 " + outputFile + " 文件,请使用记事本/表格软件查看。")