mirror of
				https://github.com/oneclickvirt/ecs.git
				synced 2025-10-25 00:10:24 +08:00 
			
		
		
		
	Compare commits
	
		
			122 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 0ce9d94c87 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 2cfd5af3c0 | ||
|   | ed66e2804a | ||
|   | 174bf303af | ||
|   | b75f42ffe5 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 56b71ac53f | ||
|   | 1045d3fab8 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 7bd2b59d58 | ||
|   | cc1da7ea7c | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 1a002a1681 | ||
|   | ee2b55e7eb | ||
|   | a55cebf94b | ||
|   | 0571a8df13 | ||
|   | f29a2829f3 | ||
|   | 33b8e0396f | ||
|   | c259073d1b | ||
|   | 07ebc8cab5 | ||
|   | 1824051e53 | ||
|   | 9f93a2e59d | ||
|   | 8cd09182da | ||
|   | 9bb776d411 | ||
|   | 12f2da9da2 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 7aa70ac1fd | ||
|   | 04ce926582 | ||
|   | 73eb38eed1 | ||
|   | 9bc8a934b1 | ||
|   | 7b729e073b | ||
|   | 111126ae90 | ||
|   | 9397f789be | ||
|   | 5a1dda6483 | ||
|   | e322c717c0 | ||
|   | 778b33142b | ||
|   | aa9f361380 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 3236c60359 | ||
|   | 73b0f30ddc | ||
|   | 825da78bd5 | ||
|   | 5d2f3c7f96 | ||
|   | 61247a206e | ||
|   | f0daad1360 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 2d23fb55a0 | ||
|   | a73dbf2d0b | ||
|   | b38dd713d9 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | e66ef1f106 | ||
|   | f6ee1e40ec | ||
|   | cb2bf0a7e5 | ||
|   | ad017db5a6 | ||
|   | a99f58518a | ||
|   | 2e59bac322 | ||
|   | 4132b1daff | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 53296b745a | ||
|   | 74630e9615 | ||
|   | 5ec7924214 | ||
|   | 7a7fdc26a0 | ||
|   | d4c855de92 | ||
|   | 7c22dee443 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 797496b640 | ||
|   | 5b686abdc8 | ||
|   | f99a37edbe | ||
|   | 4ff49c8b90 | ||
|   | 1d9257beb3 | ||
|   | fc6ccb9f92 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 88a2a7fdc9 | ||
|   | 5ff18ed7c7 | ||
|   | df897db244 | ||
|   | 9a8680491c | ||
|   | 33f81fd6aa | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 940703c3f9 | ||
|   | 1c2e9cdab9 | ||
|   | 3e6524fa0e | ||
|   | 026f40dc4c | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 110c58d401 | ||
|   | 06e76a9c33 | ||
|   | 6b88a81c02 | ||
|   | 5482506bab | ||
|   | b7130db8ce | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | dc5e3b7852 | ||
|   | 6937e69a0a | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | a68d33739c | ||
|   | 94e0441801 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 39be183fda | ||
|   | dbc1506518 | ||
|   | 149f5673d2 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | c1b7302485 | ||
|   | bf44ea9324 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 191ddfd668 | ||
|   | 89a99a7428 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | c474c71091 | ||
|   | 43b2c8aca3 | ||
|   | 96117a040e | ||
|   | c5aeda45bd | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 0b2ac51f09 | ||
|   | ffe1b65a2b | ||
|   | a4bfd4d143 | ||
|   | edbcf1c245 | ||
|   | 4c65417ea6 | ||
|   | 2cf7484881 | ||
|   | d5da2a59b6 | ||
|   | 372deb59eb | ||
|   | 8e4c6dfd3e | ||
|   | 8fc828d416 | ||
|   | 8a3fbd79e6 | ||
|   | 5628f1bb9c | ||
|   | cadbb2a45c | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 56d7560471 | ||
|   | 75e7eb1b25 | ||
|   | 24ba56cfa6 | ||
|   | ebefd64a3d | ||
|   | 4d83ffea02 | ||
|   | 01a4084462 | ||
|   | 6674093425 | ||
|   | 6d2e56b1ec | ||
|   | 2a736d3e70 | ||
|   | d02383b8cb | ||
|   | 0caba0ea60 | ||
|   | 6f92b8a994 | ||
|   | cc34baf9e1 | ||
|   | 84d8963684 | ||
|   | a94e9a6284 | ||
|   | 06007c191f | ||
|   | d013b8f90c | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 42002fdae1 | ||
|   | d7628a5a57 | 
							
								
								
									
										14
									
								
								.back/OldFunction.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								.back/OldFunction.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | ||||
| // clearScreen 清屏 | ||||
| func clearScreen() { | ||||
| 	var cmd *exec.Cmd | ||||
| 	switch runtime.GOOS { | ||||
| 	case "windows": | ||||
| 		cmd = exec.Command("cmd", "/c", "cls") | ||||
| 	case "darwin": | ||||
| 		cmd = exec.Command("clear") | ||||
| 	default: | ||||
| 		cmd = exec.Command("clear") | ||||
| 	} | ||||
| 	cmd.Stdout = os.Stdout | ||||
| 	_ = cmd.Run() | ||||
| } | ||||
| @@ -1,9 +0,0 @@ | ||||
| package backtrace | ||||
|  | ||||
| import ( | ||||
| 	"github.com/oneclickvirt/backtrace/bk" | ||||
| ) | ||||
|  | ||||
| func BackTrace(enableIpv6 bool) { | ||||
| 	backtrace.BackTrace(enableIpv6) | ||||
| } | ||||
| @@ -1,21 +0,0 @@ | ||||
| package backtrace | ||||
|  | ||||
| import ( | ||||
| 	"testing" | ||||
| ) | ||||
|  | ||||
| //func TestGeneratePrefixMap(t *testing.T) { | ||||
| //	prefix := "223.119.8.0/21" | ||||
| //	prefixList := GeneratePrefixList(prefix) | ||||
| //	if prefixList != nil { | ||||
| //		// 打印生成的IP地址前缀列表 | ||||
| //		for _, ip := range prefixList { | ||||
| //			fmt.Println(ip) | ||||
| //		} | ||||
| //	} | ||||
| //} | ||||
|  | ||||
| // 本包仅测试,无实际使用 | ||||
| func TestBackTrace(t *testing.T) { | ||||
| 	BackTrace(false) | ||||
| } | ||||
| @@ -12,5 +12,5 @@ func Basic(language string) { | ||||
| 	ipInfo, _, _ := network.NetworkCheck("both", false, language) | ||||
| 	systemInfo := system.CheckSystemInfo(language) | ||||
| 	basicInfo := strings.ReplaceAll(systemInfo+ipInfo, "\n\n", "\n") | ||||
| 	fmt.Printf(basicInfo) | ||||
| 	fmt.Print(basicInfo) | ||||
| } | ||||
|   | ||||
							
								
								
									
										467
									
								
								.back/build_binary_cgo.old
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										467
									
								
								.back/build_binary_cgo.old
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,467 @@ | ||||
| name: Build and Release | ||||
| on: | ||||
|   workflow_dispatch: | ||||
|     tags: | ||||
|       - "v*.*.*" | ||||
|  | ||||
| jobs: | ||||
|   build: | ||||
|     name: Release Check And Build | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Checkout code | ||||
|         uses: actions/checkout@v4 | ||||
|         with: | ||||
|           fetch-depth: 0 | ||||
|  | ||||
|       - name: Get latest tag | ||||
|         id: tag | ||||
|         run: | | ||||
|           TAG=$(git describe --tags --abbrev=0 2>/dev/null || echo "v0.1.0") | ||||
|           echo "tag=$TAG" >> $GITHUB_OUTPUT | ||||
|           echo "version=${TAG#v}" >> $GITHUB_OUTPUT | ||||
|  | ||||
|       - name: Generate changelog | ||||
|         id: changelog | ||||
|         run: | | ||||
|           TAG="${{ steps.tag.outputs.tag }}" | ||||
|           PREV_TAG=$(git describe --tags --abbrev=0 "$TAG^" 2>/dev/null || echo "") | ||||
|           if [ -z "$PREV_TAG" ]; then | ||||
|             CHANGELOG=$(git log --oneline --pretty=format:"* %H %s" "$TAG" | head -20) | ||||
|           else | ||||
|             CHANGELOG=$(git log --oneline --pretty=format:"* %H %s" "$PREV_TAG..$TAG") | ||||
|           fi | ||||
|           FULL_CHANGELOG="## Changelog"$'\n'"$CHANGELOG" | ||||
|           echo "$FULL_CHANGELOG" > changelog.txt | ||||
|           echo "changelog<<EOF" >> $GITHUB_OUTPUT | ||||
|           echo "$FULL_CHANGELOG" >> $GITHUB_OUTPUT | ||||
|           echo "EOF" >> $GITHUB_OUTPUT | ||||
|  | ||||
|       - name: Create or update release | ||||
|         run: | | ||||
|           TAG="${{ steps.tag.outputs.tag }}" | ||||
|           CHANGELOG_BODY=$(cat changelog.txt | jq -Rs .) | ||||
|           RELEASE_EXISTS=$(curl -s -H "Authorization: Bearer ${{ secrets.GHT }}" "https://api.github.com/repos/${{ github.repository }}/releases/tags/$TAG" | jq -r '.id // empty') | ||||
|           if [ -z "$RELEASE_EXISTS" ]; then | ||||
|             curl -s -X POST -H "Authorization: Bearer ${{ secrets.GHT }}" \ | ||||
|               -H "Content-Type: application/json" \ | ||||
|               -d "{\"tag_name\":\"$TAG\",\"name\":\"$TAG\",\"body\":$CHANGELOG_BODY,\"draft\":false,\"prerelease\":false}" \ | ||||
|               "https://api.github.com/repos/${{ github.repository }}/releases" | ||||
|           else | ||||
|             curl -s -X PATCH -H "Authorization: Bearer ${{ secrets.GHT }}" \ | ||||
|               -H "Content-Type: application/json" \ | ||||
|               -d "{\"body\":$CHANGELOG_BODY}" \ | ||||
|               "https://api.github.com/repos/${{ github.repository }}/releases/$RELEASE_EXISTS" | ||||
|           fi | ||||
|  | ||||
|       - name: Delete existing release assets | ||||
|         run: | | ||||
|           TAG="${{ steps.tag.outputs.tag }}" | ||||
|           RELEASE_ID=$(curl -s -H "Authorization: Bearer ${{ secrets.GHT }}" "https://api.github.com/repos/${{ github.repository }}/releases/tags/$TAG" | jq -r '.id') | ||||
|           if [ "$RELEASE_ID" != "null" ]; then | ||||
|             ASSETS=$(curl -s -H "Accept: application/vnd.github.v3+json" "https://api.github.com/repos/${{ github.repository }}/releases/$RELEASE_ID/assets" | jq -r '.[] | .id') | ||||
|             for asset in $ASSETS; do | ||||
|               curl -s -X DELETE -H "Authorization: Bearer ${{ secrets.GHT }}" "https://api.github.com/repos/${{ github.repository }}/releases/assets/$asset" | ||||
|             done | ||||
|             sleep 30 | ||||
|           fi | ||||
|  | ||||
|   release-binary: | ||||
|     name: Release Go Binary | ||||
|     needs: build | ||||
|     strategy: | ||||
|       fail-fast: false | ||||
|       matrix: | ||||
|         include: | ||||
|           - goos: linux | ||||
|             goarch: amd64 | ||||
|             cgo_enabled: "1" | ||||
|             cc: x86_64-linux-gnu-gcc | ||||
|             cflags: "-O2 -static -fno-stack-protector" | ||||
|             ldflags: "-extldflags=-static -s -w" | ||||
|             packages: "build-essential gcc" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: linux | ||||
|             goarch: 386 | ||||
|             cgo_enabled: "1" | ||||
|             cc: x86_64-linux-gnu-gcc | ||||
|             cflags: "-m32 -O1 -march=i686 -mtune=generic -fno-stack-protector" | ||||
|             ldflags: "-extldflags=-static -s -w" | ||||
|             packages: "build-essential gcc-multilib" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: linux | ||||
|             goarch: arm64 | ||||
|             cgo_enabled: "1" | ||||
|             cc: aarch64-linux-gnu-gcc | ||||
|             cflags: "-O1 -fno-stack-protector" | ||||
|             ldflags: "-extldflags=-static -s -w" | ||||
|             packages: "build-essential gcc-aarch64-linux-gnu" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: linux | ||||
|             goarch: riscv64 | ||||
|             cgo_enabled: "1" | ||||
|             cc: riscv64-linux-gnu-gcc | ||||
|             cflags: "-O1 -fno-stack-protector" | ||||
|             ldflags: "-extldflags=-static -s -w" | ||||
|             packages: "build-essential gcc-riscv64-linux-gnu" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: linux | ||||
|             goarch: mips64 | ||||
|             cgo_enabled: "1" | ||||
|             cc: mips64-linux-gnuabi64-gcc | ||||
|             cflags: "-O1 -fno-stack-protector" | ||||
|             ldflags: "-extldflags=-static -s -w" | ||||
|             packages: "build-essential gcc-mips64-linux-gnuabi64" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: linux | ||||
|             goarch: mips64le | ||||
|             cgo_enabled: "1" | ||||
|             cc: mips64el-linux-gnuabi64-gcc | ||||
|             cflags: "-O1 -fno-stack-protector" | ||||
|             ldflags: "-extldflags=-static -s -w" | ||||
|             packages: "build-essential gcc-mips64el-linux-gnuabi64" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: linux | ||||
|             goarch: ppc64le | ||||
|             cgo_enabled: "1" | ||||
|             cc: powerpc64le-linux-gnu-gcc | ||||
|             cflags: "-O1 -fno-stack-protector" | ||||
|             ldflags: "-extldflags=-static -s -w" | ||||
|             packages: "build-essential gcc-powerpc64le-linux-gnu" | ||||
|             runner: ubuntu-latest | ||||
|            | ||||
|           - goos: linux | ||||
|             goarch: arm | ||||
|             # goarm: 7 | ||||
|             cgo_enabled: "1" | ||||
|             cc: arm-linux-gnueabihf-gcc | ||||
|             cflags: "-O1 -fno-stack-protector" | ||||
|             ldflags: "-extldflags=-static -s -w" | ||||
|             packages: "build-essential gcc-arm-linux-gnueabihf" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: windows | ||||
|             goarch: amd64 | ||||
|             cgo_enabled: "1" | ||||
|             cc: x86_64-w64-mingw32-gcc | ||||
|             cflags: "-O2 -static-libgcc -static-libstdc++" | ||||
|             ldflags: "-extldflags=-static -s -w" | ||||
|             packages: "build-essential gcc-mingw-w64-x86-64" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: windows | ||||
|             goarch: 386 | ||||
|             cgo_enabled: "1" | ||||
|             cc: i686-w64-mingw32-gcc | ||||
|             cflags: "-O2 -static-libgcc -static-libstdc++" | ||||
|             ldflags: "-extldflags=-static -s -w" | ||||
|             packages: "build-essential gcc-mingw-w64-i686" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: windows | ||||
|             goarch: arm64 | ||||
|             cgo_enabled: "0" | ||||
|             ldflags: "-extldflags=-static -s -w" | ||||
|             packages: "build-essential" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: darwin | ||||
|             goarch: amd64 | ||||
|             cgo_enabled: "0" | ||||
|             ldflags: "-s -w" | ||||
|             runner: macos-latest | ||||
|  | ||||
|           - goos: darwin | ||||
|             goarch: arm64 | ||||
|             cgo_enabled: "0" | ||||
|             ldflags: "-s -w" | ||||
|             runner: macos-latest | ||||
|  | ||||
|           - goos: linux | ||||
|             goarch: s390x | ||||
|             cgo_enabled: "0" | ||||
|             ldflags: "-s -w" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: linux | ||||
|             goarch: mips | ||||
|             cgo_enabled: "0" | ||||
|             ldflags: "-s -w" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: linux | ||||
|             goarch: mipsle | ||||
|             cgo_enabled: "0" | ||||
|             ldflags: "-s -w" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: linux | ||||
|             goarch: ppc64 | ||||
|             cgo_enabled: "0" | ||||
|             ldflags: "-s -w" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: windows | ||||
|             goarch: arm64 | ||||
|             cgo_enabled: "0" | ||||
|             ldflags: "-s -w" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: freebsd | ||||
|             goarch: amd64 | ||||
|             cgo_enabled: "0" | ||||
|             ldflags: "-s -w" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: freebsd | ||||
|             goarch: arm64 | ||||
|             cgo_enabled: "0" | ||||
|             ldflags: "-s -w" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|     runs-on: ${{ matrix.runner }} | ||||
|     steps: | ||||
|       - name: Checkout code | ||||
|         uses: actions/checkout@v4 | ||||
|         with: | ||||
|           fetch-depth: 0 | ||||
|  | ||||
|       - name: Set up Go | ||||
|         uses: actions/setup-go@v5 | ||||
|         with: | ||||
|           go-version: 1.24.5 | ||||
|  | ||||
|       - name: Configure Git for Private Modules | ||||
|         run: | | ||||
|           git config --global url."https://${{ secrets.GHT }}@github.com/".insteadOf "https://github.com/" | ||||
|           git config --global url."git@github.com:".insteadOf "https://github.com/" | ||||
|         env: | ||||
|           GITHUB_TOKEN: ${{ secrets.GHT }} | ||||
|  | ||||
|       - name: Install cross-compilation tools | ||||
|         if: matrix.runner != 'macos-latest' | ||||
|         run: | | ||||
|           sudo apt-get update -qq | ||||
|           case "${{ matrix.goos }}-${{ matrix.goarch }}" in | ||||
|             linux-386) | ||||
|               sudo apt-get install -y build-essential gcc-multilib g++-multilib ;; | ||||
|             linux-arm64) | ||||
|               sudo apt-get install -y build-essential gcc-aarch64-linux-gnu ;; | ||||
|             linux-riscv64) | ||||
|               sudo apt-get install -y build-essential gcc-riscv64-linux-gnu ;; | ||||
|             linux-mips64) | ||||
|               sudo apt-get install -y build-essential gcc-mips64-linux-gnuabi64 ;; | ||||
|             linux-mips64le) | ||||
|               sudo apt-get install -y build-essential gcc-mips64el-linux-gnuabi64 ;; | ||||
|             linux-ppc64le) | ||||
|               sudo apt-get install -y build-essential gcc-powerpc64le-linux-gnu ;; | ||||
|             linux-arm) | ||||
|               sudo apt-get install -y build-essential gcc-arm-linux-gnueabihf ;; | ||||
|             windows-amd64|windows-386) | ||||
|               sudo apt-get install -y build-essential gcc-mingw-w64-x86-64 gcc-mingw-w64-i686 ;; | ||||
|             *) | ||||
|               sudo apt-get install -y build-essential ;; | ||||
|           esac | ||||
|  | ||||
|       - name: Get latest tag | ||||
|         id: tag | ||||
|         run: | | ||||
|           TAG=$(git describe --tags --abbrev=0 2>/dev/null || echo "v0.1.0") | ||||
|           echo "tag=$TAG" >> $GITHUB_OUTPUT | ||||
|           echo "version=${TAG#v}" >> $GITHUB_OUTPUT | ||||
|  | ||||
|       - name: Build Binary | ||||
|         env: | ||||
|           CGO_ENABLED: ${{ matrix.cgo_enabled }} | ||||
|           GOOS: ${{ matrix.goos }} | ||||
|           GOARCH: ${{ matrix.goarch }} | ||||
|           CC: ${{ matrix.cc }} | ||||
|           CGO_CFLAGS: ${{ matrix.cflags }} | ||||
|           CGO_LDFLAGS: ${{ matrix.ldflags }} | ||||
|         run: | | ||||
|           go clean -cache -modcache -testcache | ||||
|            | ||||
|           # 设置额外的环境变量 | ||||
|           if [[ -n "${{ matrix.goarm }}" ]]; then | ||||
|             export GOARM=${{ matrix.goarm }} | ||||
|           fi | ||||
|           if [[ -n "${{ matrix.gomips }}" ]]; then | ||||
|             export GOMIPS=${{ matrix.gomips }} | ||||
|           fi | ||||
|            | ||||
|           # 针对 Darwin 的特殊处理 | ||||
|           if [[ "${{ matrix.cgo_enabled }}" == "1" && "${{ matrix.goos }}" == "darwin" ]]; then | ||||
|             if [[ "${{ matrix.goarch }}" == "amd64" ]]; then | ||||
|               export CC="x86_64-apple-darwin21.4-clang" | ||||
|               export CXX="x86_64-apple-darwin21.4-clang++" | ||||
|             elif [[ "${{ matrix.goarch }}" == "arm64" ]]; then | ||||
|               export CC="aarch64-apple-darwin21.4-clang" | ||||
|               export CXX="aarch64-apple-darwin21.4-clang++" | ||||
|             fi | ||||
|             export OSXCROSS_ROOT="${OSXCROSS_ROOT}" | ||||
|           elif [[ "${{ matrix.cgo_enabled }}" == "1" && "${{ matrix.runner }}" != "macos-latest" ]]; then | ||||
|             # 对于 Windows 的特殊处理 | ||||
|             if [[ "${{ matrix.goos }}" == "windows" ]]; then | ||||
|               export CGO_LDFLAGS="-static-libgcc -static-libstdc++" | ||||
|             fi | ||||
|           fi | ||||
|            | ||||
|           # 测试编译器(仅在启用 CGO 时) | ||||
|           if [[ "${{ matrix.cgo_enabled }}" == "1" && -n "$CC" ]]; then | ||||
|             echo 'int main() { return 0; }' > test.c | ||||
|             $CC $CGO_CFLAGS test.c -o test || exit 1 | ||||
|             rm -f test.c test | ||||
|           fi | ||||
|            | ||||
|           # 清理和准备 | ||||
|           rm -rf vendor/ | ||||
|           go mod download | ||||
|           go mod tidy | ||||
|           mkdir -p bin | ||||
|            | ||||
|           # 设置二进制文件名 | ||||
|           BINARY_NAME="goecs" | ||||
|           if [[ "${{ matrix.goos }}" == "windows" ]]; then | ||||
|             BINARY_NAME="${BINARY_NAME}.exe" | ||||
|           fi | ||||
|            | ||||
|           # 构建 LDFLAGS | ||||
|           LDFLAGS="-s -w -X main.version=${{ steps.tag.outputs.version }} -X main.arch=${{ matrix.goarch }}" | ||||
|           if [[ "${{ matrix.cgo_enabled }}" == "1" ]]; then | ||||
|             LDFLAGS="${LDFLAGS} -checklinkname=0 ${{ matrix.ldflags }}" | ||||
|           else | ||||
|             LDFLAGS="${LDFLAGS} -checklinkname=0 ${{ matrix.ldflags }}" | ||||
|           fi | ||||
|            | ||||
|           # 执行构建 | ||||
|           echo "Building for GOOS=$GOOS GOARCH=$GOARCH CGO_ENABLED=$CGO_ENABLED" | ||||
|           go build -a -o bin/$BINARY_NAME -ldflags="$LDFLAGS" -trimpath ./ | ||||
|            | ||||
|           # 验证文件是否存在 | ||||
|           [[ -f "bin/$BINARY_NAME" ]] || exit 1 | ||||
|            | ||||
|           # 显示构建信息 | ||||
|           echo "Built binary: bin/$BINARY_NAME" | ||||
|           ls -la bin/ | ||||
|           if command -v file >/dev/null 2>&1; then | ||||
|             file bin/$BINARY_NAME | ||||
|           fi | ||||
|  | ||||
|       - name: Create ZIP archive | ||||
|         run: | | ||||
|           cd bin | ||||
|           BINARY_NAME="goecs" | ||||
|           if [[ "${{ matrix.goos }}" == "windows" ]]; then | ||||
|             BINARY_NAME="${BINARY_NAME}.exe" | ||||
|           fi | ||||
|           ZIP_NAME="goecs_${{ matrix.goos }}_${{ matrix.goarch }}" | ||||
|           if [[ -n "${{ matrix.goarm }}" ]]; then | ||||
|             ZIP_NAME="${ZIP_NAME}v${{ matrix.goarm }}" | ||||
|           fi | ||||
|           if [[ -n "${{ matrix.gomips }}" ]]; then | ||||
|             ZIP_NAME="${ZIP_NAME}_${{ matrix.gomips }}" | ||||
|           fi | ||||
|           ZIP_NAME="${ZIP_NAME}.zip" | ||||
|           zip "$ZIP_NAME" "$BINARY_NAME" | ||||
|  | ||||
|       - name: Upload to Release | ||||
|         run: | | ||||
|           TAG="${{ steps.tag.outputs.tag }}" | ||||
|           RELEASE_ID=$(curl -s -H "Authorization: Bearer ${{ secrets.GHT }}" "https://api.github.com/repos/${{ github.repository }}/releases/tags/$TAG" | jq -r '.id') | ||||
|           cd bin | ||||
|           for file in *.zip; do | ||||
|             if [[ -f "$file" ]]; then | ||||
|               curl -s -H "Authorization: Bearer ${{ secrets.GHT }}" \ | ||||
|                 -H "Content-Type: application/zip" \ | ||||
|                 --data-binary @"$file" \ | ||||
|                 "https://uploads.github.com/repos/${{ github.repository }}/releases/$RELEASE_ID/assets?name=$file" | ||||
|             fi | ||||
|           done | ||||
|  | ||||
|   checksums: | ||||
|     name: Generate Checksums | ||||
|     runs-on: ubuntu-latest | ||||
|     needs: release-binary | ||||
|     steps: | ||||
|       - name: Checkout code | ||||
|         uses: actions/checkout@v4 | ||||
|         with: | ||||
|           fetch-depth: 0 | ||||
|  | ||||
|       - name: Get latest tag | ||||
|         id: tag | ||||
|         run: | | ||||
|           TAG=$(git describe --tags --abbrev=0 2>/dev/null || echo "v0.1.0") | ||||
|           echo "tag=$TAG" >> $GITHUB_OUTPUT | ||||
|  | ||||
|       - name: Download release assets | ||||
|         run: | | ||||
|           TAG="${{ steps.tag.outputs.tag }}" | ||||
|           RELEASE_ID=$(curl -s -H "Authorization: Bearer ${{ secrets.GHT }}" "https://api.github.com/repos/${{ github.repository }}/releases/tags/$TAG" | jq -r '.id') | ||||
|           mkdir -p assets | ||||
|           ASSETS=$(curl -s -H "Accept: application/vnd.github.v3+json" "https://api.github.com/repos/${{ github.repository }}/releases/$RELEASE_ID/assets") | ||||
|           echo "$ASSETS" | jq -r '.[] | select(.name | endswith(".zip")) | .browser_download_url' | while read url; do | ||||
|             filename=$(basename "$url") | ||||
|             curl -L -H "Authorization: Bearer ${{ secrets.GHT }}" "$url" -o "assets/$filename" | ||||
|           done | ||||
|  | ||||
|       - name: Generate checksums | ||||
|         run: | | ||||
|           cd assets | ||||
|           sha256sum *.zip > checksums.txt | ||||
|           if [[ -f "../goecs.sh" ]]; then | ||||
|             sha256sum ../goecs.sh >> checksums.txt | ||||
|           fi | ||||
|  | ||||
|       - name: Upload checksums | ||||
|         run: | | ||||
|           TAG="${{ steps.tag.outputs.tag }}" | ||||
|           RELEASE_ID=$(curl -s -H "Authorization: Bearer ${{ secrets.GHT }}" "https://api.github.com/repos/${{ github.repository }}/releases/tags/$TAG" | jq -r '.id') | ||||
|           curl -s -H "Authorization: Bearer ${{ secrets.GHT }}" \ | ||||
|             -H "Content-Type: text/plain" \ | ||||
|             --data-binary @assets/checksums.txt \ | ||||
|             "https://uploads.github.com/repos/${{ github.repository }}/releases/$RELEASE_ID/assets?name=checksums.txt" | ||||
|  | ||||
|   update-script: | ||||
|     name: Update Script Version | ||||
|     runs-on: ubuntu-latest | ||||
|     needs: checksums | ||||
|     steps: | ||||
|       - name: Checkout code | ||||
|         uses: actions/checkout@v4 | ||||
|         with: | ||||
|           fetch-depth: 0 | ||||
|  | ||||
|       - name: Get latest tag | ||||
|         id: tag | ||||
|         run: | | ||||
|           TAG=$(git describe --tags --abbrev=0 2>/dev/null || echo "v0.1.0") | ||||
|           echo "tag=$TAG" >> $GITHUB_OUTPUT | ||||
|           echo "version=${TAG#v}" >> $GITHUB_OUTPUT | ||||
|  | ||||
|       - name: Update goecs.sh version | ||||
|         run: | | ||||
|           VERSION="${{ steps.tag.outputs.version }}" | ||||
|           BRANCH="master" | ||||
|           git config --global user.name "github-actions[bot]" | ||||
|           git config --global user.email "github-actions[bot]@users.noreply.github.com" | ||||
|           git config --global --unset url."git@github.com:".insteadOf || true | ||||
|           git fetch origin $BRANCH | ||||
|           git checkout $BRANCH | ||||
|           if [ -f "goecs.sh" ]; then | ||||
|             sed -i "s/\(_yellow \"Unable to get version info, using default version \).*\(\".*\)/\1$VERSION\2/" "goecs.sh" | ||||
|             sed -i "s/\(ECS_VERSION=\"\).*\(\"\)/\1$VERSION\2/" "goecs.sh" | ||||
|             if ! git diff --quiet "goecs.sh"; then | ||||
|               git add "goecs.sh" | ||||
|               git commit -m "chore: update ECS_VERSION to $VERSION in goecs.sh" | ||||
|               git push origin $BRANCH | ||||
|             fi | ||||
|           fi | ||||
|         env: | ||||
|           GITHUB_TOKEN: ${{ secrets.GHT }} | ||||
							
								
								
									
										527
									
								
								.back/build_binary_musl.old
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										527
									
								
								.back/build_binary_musl.old
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,527 @@ | ||||
| name: Build and Release | ||||
| on: | ||||
|   workflow_dispatch: | ||||
|     tags: | ||||
|       - "v*.*.*" | ||||
|  | ||||
| jobs: | ||||
|   build: | ||||
|     name: Release Check And Build | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Checkout code | ||||
|         uses: actions/checkout@v4 | ||||
|         with: | ||||
|           fetch-depth: 0 | ||||
|  | ||||
|       - name: Get latest tag | ||||
|         id: tag | ||||
|         run: | | ||||
|           TAG=$(git describe --tags --abbrev=0 2>/dev/null || echo "v0.1.0") | ||||
|           echo "tag=$TAG" >> $GITHUB_OUTPUT | ||||
|           echo "version=${TAG#v}" >> $GITHUB_OUTPUT | ||||
|  | ||||
|       - name: Generate changelog | ||||
|         id: changelog | ||||
|         run: | | ||||
|           TAG="${{ steps.tag.outputs.tag }}" | ||||
|           PREV_TAG=$(git describe --tags --abbrev=0 "$TAG^" 2>/dev/null || echo "") | ||||
|           if [ -z "$PREV_TAG" ]; then | ||||
|             CHANGELOG=$(git log --oneline --pretty=format:"* %H %s" "$TAG" | head -20) | ||||
|           else | ||||
|             CHANGELOG=$(git log --oneline --pretty=format:"* %H %s" "$PREV_TAG..$TAG") | ||||
|           fi | ||||
|           FULL_CHANGELOG="## Changelog"$'\n'"$CHANGELOG" | ||||
|           echo "$FULL_CHANGELOG" > changelog.txt | ||||
|           echo "changelog<<EOF" >> $GITHUB_OUTPUT | ||||
|           echo "$FULL_CHANGELOG" >> $GITHUB_OUTPUT | ||||
|           echo "EOF" >> $GITHUB_OUTPUT | ||||
|  | ||||
|       - name: Create or update release | ||||
|         run: | | ||||
|           TAG="${{ steps.tag.outputs.tag }}" | ||||
|           CHANGELOG_BODY=$(cat changelog.txt | jq -Rs .) | ||||
|           RELEASE_EXISTS=$(curl -s -H "Authorization: Bearer ${{ secrets.GHT }}" "https://api.github.com/repos/${{ github.repository }}/releases/tags/$TAG" | jq -r '.id // empty') | ||||
|           if [ -z "$RELEASE_EXISTS" ]; then | ||||
|             curl -s -X POST -H "Authorization: Bearer ${{ secrets.GHT }}" \ | ||||
|               -H "Content-Type: application/json" \ | ||||
|               -d "{\"tag_name\":\"$TAG\",\"name\":\"$TAG\",\"body\":$CHANGELOG_BODY,\"draft\":false,\"prerelease\":false}" \ | ||||
|               "https://api.github.com/repos/${{ github.repository }}/releases" | ||||
|           else | ||||
|             curl -s -X PATCH -H "Authorization: Bearer ${{ secrets.GHT }}" \ | ||||
|               -H "Content-Type: application/json" \ | ||||
|               -d "{\"body\":$CHANGELOG_BODY}" \ | ||||
|               "https://api.github.com/repos/${{ github.repository }}/releases/$RELEASE_EXISTS" | ||||
|           fi | ||||
|  | ||||
|       - name: Delete existing release assets | ||||
|         run: | | ||||
|           TAG="${{ steps.tag.outputs.tag }}" | ||||
|           RELEASE_ID=$(curl -s -H "Authorization: Bearer ${{ secrets.GHT }}" "https://api.github.com/repos/${{ github.repository }}/releases/tags/$TAG" | jq -r '.id') | ||||
|           if [ "$RELEASE_ID" != "null" ]; then | ||||
|             ASSETS=$(curl -s -H "Accept: application/vnd.github.v3+json" "https://api.github.com/repos/${{ github.repository }}/releases/$RELEASE_ID/assets" | jq -r '.[] | .id') | ||||
|             for asset in $ASSETS; do | ||||
|               curl -s -X DELETE -H "Authorization: Bearer ${{ secrets.GHT }}" "https://api.github.com/repos/${{ github.repository }}/releases/assets/$asset" | ||||
|             done | ||||
|             sleep 30 | ||||
|           fi | ||||
|  | ||||
|   build-musl-toolchain: | ||||
|     name: Build musl Cross-Compiler Toolchain | ||||
|     runs-on: ubuntu-latest | ||||
|     strategy: | ||||
|       matrix: | ||||
|         target: | ||||
|           - x86_64-linux-musl | ||||
|           - i686-linux-musl | ||||
|           - aarch64-linux-musl | ||||
|           - riscv64-linux-musl | ||||
|           - mips64-linux-musl | ||||
|           - mips64el-linux-musl | ||||
|           - powerpc64le-linux-musl | ||||
|           - arm-linux-musleabihf | ||||
|     steps: | ||||
|       - name: Install build dependencies | ||||
|         run: | | ||||
|           sudo apt-get update -qq | ||||
|           sudo apt-get install -y build-essential curl | ||||
|  | ||||
|       - name: Cache musl toolchain | ||||
|         id: cache-musl | ||||
|         uses: actions/cache@v3 | ||||
|         with: | ||||
|           path: /opt/musl-${{ matrix.target }} | ||||
|           key: musl-toolchain-${{ matrix.target }}-v2 | ||||
|  | ||||
|       - name: Build musl cross-compiler | ||||
|         if: steps.cache-musl.outputs.cache-hit != 'true' | ||||
|         run: | | ||||
|           # Clone musl-cross-make | ||||
|           git clone https://github.com/richfelker/musl-cross-make.git | ||||
|           cd musl-cross-make | ||||
|            | ||||
|           # Create config for target | ||||
|           cat > config.mak << EOF | ||||
|           TARGET = ${{ matrix.target }} | ||||
|           OUTPUT = /opt/musl-${{ matrix.target }} | ||||
|           COMMON_CONFIG += --disable-nls | ||||
|           GCC_CONFIG += --enable-languages=c,c++ | ||||
|           GCC_CONFIG += --disable-libquadmath --disable-decimal-float | ||||
|           GCC_CONFIG += --disable-libitm --disable-fixed-point | ||||
|           EOF | ||||
|            | ||||
|           # Build the toolchain | ||||
|           make -j$(nproc) | ||||
|           sudo make install | ||||
|            | ||||
|           # Verify installation | ||||
|           ls -la /opt/musl-${{ matrix.target }}/bin/ | ||||
|           /opt/musl-${{ matrix.target }}/bin/${{ matrix.target }}-gcc --version | ||||
|  | ||||
|       - name: Create toolchain artifact | ||||
|         run: | | ||||
|           sudo tar -czf musl-${{ matrix.target }}-toolchain.tar.gz -C /opt musl-${{ matrix.target }} | ||||
|  | ||||
|       - name: Upload toolchain artifact | ||||
|         uses: actions/upload-artifact@v4 | ||||
|         with: | ||||
|           name: musl-${{ matrix.target }}-toolchain | ||||
|           path: musl-${{ matrix.target }}-toolchain.tar.gz | ||||
|           retention-days: 1 | ||||
|  | ||||
|   release-binary: | ||||
|     name: Release Go Binary | ||||
|     needs: [build, build-musl-toolchain] | ||||
|     strategy: | ||||
|       fail-fast: false | ||||
|       matrix: | ||||
|         include: | ||||
|           - goos: linux | ||||
|             goarch: amd64 | ||||
|             cgo_enabled: "1" | ||||
|             musl_target: x86_64-linux-musl | ||||
|             cflags: "-O2 -static -fno-stack-protector" | ||||
|             ldflags: "-extldflags=-static" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: linux | ||||
|             goarch: 386 | ||||
|             cgo_enabled: "1" | ||||
|             musl_target: i686-linux-musl | ||||
|             cflags: "-O1 -march=i686 -mtune=generic -fno-stack-protector" | ||||
|             ldflags: "-extldflags=-static" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: linux | ||||
|             goarch: arm64 | ||||
|             cgo_enabled: "1" | ||||
|             musl_target: aarch64-linux-musl | ||||
|             cflags: "-O1 -fno-stack-protector" | ||||
|             ldflags: "-extldflags=-static" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: linux | ||||
|             goarch: riscv64 | ||||
|             cgo_enabled: "1" | ||||
|             musl_target: riscv64-linux-musl | ||||
|             cflags: "-O1 -fno-stack-protector" | ||||
|             ldflags: "-extldflags=-static" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: linux | ||||
|             goarch: mips64 | ||||
|             cgo_enabled: "1" | ||||
|             musl_target: mips64-linux-musl | ||||
|             cflags: "-O1 -fno-stack-protector" | ||||
|             ldflags: "-extldflags=-static" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: linux | ||||
|             goarch: mips64le | ||||
|             cgo_enabled: "1" | ||||
|             musl_target: mips64el-linux-musl | ||||
|             cflags: "-O1 -fno-stack-protector" | ||||
|             ldflags: "-extldflags=-static" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: linux | ||||
|             goarch: ppc64le | ||||
|             cgo_enabled: "1" | ||||
|             musl_target: powerpc64le-linux-musl | ||||
|             cflags: "-O1 -fno-stack-protector" | ||||
|             ldflags: "-extldflags=-static" | ||||
|             runner: ubuntu-latest | ||||
|            | ||||
|           - goos: linux | ||||
|             goarch: arm | ||||
|             cgo_enabled: "1" | ||||
|             musl_target: arm-linux-musleabihf | ||||
|             cflags: "-O1 -fno-stack-protector" | ||||
|             ldflags: "-extldflags=-latomic -static" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: windows | ||||
|             goarch: amd64 | ||||
|             cgo_enabled: "1" | ||||
|             cc: x86_64-w64-mingw32-gcc | ||||
|             cflags: "-O2 -static-libgcc -static-libstdc++" | ||||
|             ldflags: "-extldflags=-static" | ||||
|             packages: "build-essential gcc-mingw-w64-x86-64" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: windows | ||||
|             goarch: 386 | ||||
|             cgo_enabled: "1" | ||||
|             cc: i686-w64-mingw32-gcc | ||||
|             cflags: "-O2 -static-libgcc -static-libstdc++" | ||||
|             ldflags: "-extldflags=-static" | ||||
|             packages: "build-essential gcc-mingw-w64-i686" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: windows | ||||
|             goarch: arm64 | ||||
|             cgo_enabled: "0" | ||||
|             ldflags: "-s -w" | ||||
|             packages: "build-essential" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: darwin | ||||
|             goarch: amd64 | ||||
|             cgo_enabled: "0" | ||||
|             ldflags: "-s -w" | ||||
|             runner: macos-latest | ||||
|  | ||||
|           - goos: darwin | ||||
|             goarch: arm64 | ||||
|             cgo_enabled: "0" | ||||
|             ldflags: "-s -w" | ||||
|             runner: macos-latest | ||||
|  | ||||
|           - goos: linux | ||||
|             goarch: s390x | ||||
|             cgo_enabled: "0" | ||||
|             ldflags: "-s -w" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: linux | ||||
|             goarch: mips | ||||
|             cgo_enabled: "0" | ||||
|             ldflags: "-s -w" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: linux | ||||
|             goarch: mipsle | ||||
|             cgo_enabled: "0" | ||||
|             ldflags: "-s -w" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: linux | ||||
|             goarch: ppc64 | ||||
|             cgo_enabled: "0" | ||||
|             ldflags: "-s -w" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: freebsd | ||||
|             goarch: amd64 | ||||
|             cgo_enabled: "0" | ||||
|             ldflags: "-s -w" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: freebsd | ||||
|             goarch: arm64 | ||||
|             cgo_enabled: "0" | ||||
|             ldflags: "-s -w" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|     runs-on: ${{ matrix.runner }} | ||||
|     steps: | ||||
|       - name: Checkout code | ||||
|         uses: actions/checkout@v4 | ||||
|         with: | ||||
|           fetch-depth: 0 | ||||
|  | ||||
|       - name: Set up Go | ||||
|         uses: actions/setup-go@v5 | ||||
|         with: | ||||
|           go-version: 1.24.5 | ||||
|  | ||||
|       - name: Configure Git for Private Modules | ||||
|         run: | | ||||
|           git config --global url."https://${{ secrets.GHT }}@github.com/".insteadOf "https://github.com/" | ||||
|           git config --global url."git@github.com:".insteadOf "https://github.com/" | ||||
|         env: | ||||
|           GITHUB_TOKEN: ${{ secrets.GHT }} | ||||
|  | ||||
|       - name: Download musl toolchain | ||||
|         if: matrix.musl_target != '' | ||||
|         uses: actions/download-artifact@v4 | ||||
|         with: | ||||
|           name: musl-${{ matrix.musl_target }}-toolchain | ||||
|  | ||||
|       - name: Setup musl toolchain | ||||
|         if: matrix.musl_target != '' | ||||
|         run: | | ||||
|           sudo tar -xzf musl-${{ matrix.musl_target }}-toolchain.tar.gz -C /opt/ | ||||
|           echo "/opt/musl-${{ matrix.musl_target }}/bin" >> $GITHUB_PATH | ||||
|            | ||||
|           # Verify toolchain is working | ||||
|           /opt/musl-${{ matrix.musl_target }}/bin/${{ matrix.musl_target }}-gcc --version | ||||
|            | ||||
|           # Test compiler | ||||
|           echo 'int main() { return 0; }' > test.c | ||||
|           /opt/musl-${{ matrix.musl_target }}/bin/${{ matrix.musl_target }}-gcc ${{ matrix.cflags }} test.c -o test | ||||
|           rm -f test.c test | ||||
|  | ||||
|       - name: Install cross-compilation tools (non-musl) | ||||
|         if: matrix.runner != 'macos-latest' && matrix.musl_target == '' | ||||
|         run: | | ||||
|           sudo systemctl restart systemd-resolved || true | ||||
|           sudo apt-get update -qq || (sleep 10 && sudo apt-get update -qq) | ||||
|            | ||||
|           case "${{ matrix.goos }}-${{ matrix.goarch }}" in | ||||
|             windows-amd64|windows-386) | ||||
|               for i in 1 2 3; do | ||||
|                 sudo apt-get install -y build-essential gcc-mingw-w64-x86-64 gcc-mingw-w64-i686 && break || sleep 10 | ||||
|               done ;; | ||||
|             *) | ||||
|               sudo systemctl restart systemd-resolved || true | ||||
|               for i in 1 2 3; do | ||||
|                 sudo apt-get install -y build-essential && break || sleep 10 | ||||
|               done ;; | ||||
|           esac | ||||
|  | ||||
|       - name: Get latest tag | ||||
|         id: tag | ||||
|         run: | | ||||
|           TAG=$(git describe --tags --abbrev=0 2>/dev/null || echo "v0.1.0") | ||||
|           echo "tag=$TAG" >> $GITHUB_OUTPUT | ||||
|           echo "version=${TAG#v}" >> $GITHUB_OUTPUT | ||||
|  | ||||
|       - name: Build Binary | ||||
|         env: | ||||
|           CGO_ENABLED: ${{ matrix.cgo_enabled }} | ||||
|           GOOS: ${{ matrix.goos }} | ||||
|           GOARCH: ${{ matrix.goarch }} | ||||
|           CGO_CFLAGS: ${{ matrix.cflags }} | ||||
|           CGO_LDFLAGS: ${{ matrix.ldflags }} | ||||
|         run: | | ||||
|           go clean -cache -modcache -testcache | ||||
|            | ||||
|           # Set CC based on target | ||||
|           if [[ "${{ matrix.musl_target }}" != "" ]]; then | ||||
|             export CC="/opt/musl-${{ matrix.musl_target }}/bin/${{ matrix.musl_target }}-gcc" | ||||
|             export CXX="/opt/musl-${{ matrix.musl_target }}/bin/${{ matrix.musl_target }}-g++" | ||||
|           elif [[ "${{ matrix.cc }}" != "" ]]; then | ||||
|             export CC="${{ matrix.cc }}" | ||||
|           fi | ||||
|            | ||||
|           # 设置额外的环境变量 | ||||
|           if [[ -n "${{ matrix.goarm }}" ]]; then | ||||
|             export GOARM=${{ matrix.goarm }} | ||||
|           fi | ||||
|           if [[ -n "${{ matrix.gomips }}" ]]; then | ||||
|             export GOMIPS=${{ matrix.gomips }} | ||||
|           fi | ||||
|            | ||||
|           # 针对 Darwin 的特殊处理 | ||||
|           if [[ "${{ matrix.cgo_enabled }}" == "1" && "${{ matrix.goos }}" == "darwin" ]]; then | ||||
|             if [[ "${{ matrix.goarch }}" == "amd64" ]]; then | ||||
|               export CC="x86_64-apple-darwin21.4-clang" | ||||
|               export CXX="x86_64-apple-darwin21.4-clang++" | ||||
|             elif [[ "${{ matrix.goarch }}" == "arm64" ]]; then | ||||
|               export CC="aarch64-apple-darwin21.4-clang" | ||||
|               export CXX="aarch64-apple-darwin21.4-clang++" | ||||
|             fi | ||||
|             export OSXCROSS_ROOT="${OSXCROSS_ROOT}" | ||||
|           fi | ||||
|            | ||||
|           # 清理和准备 | ||||
|           rm -rf vendor/ | ||||
|           go mod download | ||||
|           go mod tidy | ||||
|           mkdir -p bin | ||||
|            | ||||
|           # 设置二进制文件名 | ||||
|           BINARY_NAME="goecs" | ||||
|           if [[ "${{ matrix.goos }}" == "windows" ]]; then | ||||
|             BINARY_NAME="${BINARY_NAME}.exe" | ||||
|           fi | ||||
|            | ||||
|           # 构建 LDFLAGS | ||||
|           LDFLAGS="-s -w -X main.version=${{ steps.tag.outputs.version }} -X main.arch=${{ matrix.goarch }}" | ||||
|           if [[ "${{ matrix.cgo_enabled }}" == "1" ]]; then | ||||
|             LDFLAGS="${LDFLAGS} -checklinkname=0 ${{ matrix.ldflags }}" | ||||
|           else | ||||
|             LDFLAGS="${LDFLAGS} -checklinkname=0 ${{ matrix.ldflags }}" | ||||
|           fi | ||||
|            | ||||
|           # 执行构建 | ||||
|           echo "Building for GOOS=$GOOS GOARCH=$GOARCH CGO_ENABLED=$CGO_ENABLED" | ||||
|           if [[ -n "$CC" ]]; then | ||||
|             echo "Using CC=$CC" | ||||
|           fi | ||||
|            | ||||
|           go build -a -o bin/$BINARY_NAME -ldflags="$LDFLAGS" -trimpath ./ | ||||
|            | ||||
|           # 验证文件是否存在 | ||||
|           [[ -f "bin/$BINARY_NAME" ]] || exit 1 | ||||
|            | ||||
|           # 显示构建信息 | ||||
|           echo "Built binary: bin/$BINARY_NAME" | ||||
|           ls -la bin/ | ||||
|           if command -v file >/dev/null 2>&1; then | ||||
|             file bin/$BINARY_NAME | ||||
|           fi | ||||
|  | ||||
|       - name: Create ZIP archive | ||||
|         run: | | ||||
|           cd bin | ||||
|           BINARY_NAME="goecs" | ||||
|           if [[ "${{ matrix.goos }}" == "windows" ]]; then | ||||
|             BINARY_NAME="${BINARY_NAME}.exe" | ||||
|           fi | ||||
|           ZIP_NAME="goecs_${{ matrix.goos }}_${{ matrix.goarch }}" | ||||
|           if [[ -n "${{ matrix.goarm }}" ]]; then | ||||
|             ZIP_NAME="${ZIP_NAME}v${{ matrix.goarm }}" | ||||
|           fi | ||||
|           if [[ -n "${{ matrix.gomips }}" ]]; then | ||||
|             ZIP_NAME="${ZIP_NAME}_${{ matrix.gomips }}" | ||||
|           fi | ||||
|           ZIP_NAME="${ZIP_NAME}.zip" | ||||
|           zip "$ZIP_NAME" "$BINARY_NAME" | ||||
|  | ||||
|       - name: Upload to Release | ||||
|         run: | | ||||
|           TAG="${{ steps.tag.outputs.tag }}" | ||||
|           RELEASE_ID=$(curl -s -H "Authorization: Bearer ${{ secrets.GHT }}" "https://api.github.com/repos/${{ github.repository }}/releases/tags/$TAG" | jq -r '.id') | ||||
|           cd bin | ||||
|           for file in *.zip; do | ||||
|             if [[ -f "$file" ]]; then | ||||
|               curl -s -H "Authorization: Bearer ${{ secrets.GHT }}" \ | ||||
|                 -H "Content-Type: application/zip" \ | ||||
|                 --data-binary @"$file" \ | ||||
|                 "https://uploads.github.com/repos/${{ github.repository }}/releases/$RELEASE_ID/assets?name=$file" | ||||
|             fi | ||||
|           done | ||||
|  | ||||
|   checksums: | ||||
|     name: Generate Checksums | ||||
|     runs-on: ubuntu-latest | ||||
|     needs: release-binary | ||||
|     steps: | ||||
|       - name: Checkout code | ||||
|         uses: actions/checkout@v4 | ||||
|         with: | ||||
|           fetch-depth: 0 | ||||
|  | ||||
|       - name: Get latest tag | ||||
|         id: tag | ||||
|         run: | | ||||
|           TAG=$(git describe --tags --abbrev=0 2>/dev/null || echo "v0.1.0") | ||||
|           echo "tag=$TAG" >> $GITHUB_OUTPUT | ||||
|  | ||||
|       - name: Download release assets | ||||
|         run: | | ||||
|           TAG="${{ steps.tag.outputs.tag }}" | ||||
|           RELEASE_ID=$(curl -s -H "Authorization: Bearer ${{ secrets.GHT }}" "https://api.github.com/repos/${{ github.repository }}/releases/tags/$TAG" | jq -r '.id') | ||||
|           mkdir -p assets | ||||
|           ASSETS=$(curl -s -H "Accept: application/vnd.github.v3+json" "https://api.github.com/repos/${{ github.repository }}/releases/$RELEASE_ID/assets") | ||||
|           echo "$ASSETS" | jq -r '.[] | select(.name | endswith(".zip")) | .browser_download_url' | while read url; do | ||||
|             filename=$(basename "$url") | ||||
|             curl -L -H "Authorization: Bearer ${{ secrets.GHT }}" "$url" -o "assets/$filename" | ||||
|           done | ||||
|  | ||||
|       - name: Generate checksums | ||||
|         run: | | ||||
|           cd assets | ||||
|           sha256sum *.zip > checksums.txt | ||||
|           if [[ -f "../goecs.sh" ]]; then | ||||
|             sha256sum ../goecs.sh >> checksums.txt | ||||
|           fi | ||||
|  | ||||
|       - name: Upload checksums | ||||
|         run: | | ||||
|           TAG="${{ steps.tag.outputs.tag }}" | ||||
|           RELEASE_ID=$(curl -s -H "Authorization: Bearer ${{ secrets.GHT }}" "https://api.github.com/repos/${{ github.repository }}/releases/tags/$TAG" | jq -r '.id') | ||||
|           curl -s -H "Authorization: Bearer ${{ secrets.GHT }}" \ | ||||
|             -H "Content-Type: text/plain" \ | ||||
|             --data-binary @assets/checksums.txt \ | ||||
|             "https://uploads.github.com/repos/${{ github.repository }}/releases/$RELEASE_ID/assets?name=checksums.txt" | ||||
|  | ||||
|   update-script: | ||||
|     name: Update Script Version | ||||
|     runs-on: ubuntu-latest | ||||
|     needs: checksums | ||||
|     steps: | ||||
|       - name: Checkout code | ||||
|         uses: actions/checkout@v4 | ||||
|         with: | ||||
|           fetch-depth: 0 | ||||
|  | ||||
|       - name: Get latest tag | ||||
|         id: tag | ||||
|         run: | | ||||
|           TAG=$(git describe --tags --abbrev=0 2>/dev/null || echo "v0.1.0") | ||||
|           echo "tag=$TAG" >> $GITHUB_OUTPUT | ||||
|           echo "version=${TAG#v}" >> $GITHUB_OUTPUT | ||||
|  | ||||
|       - name: Update goecs.sh version | ||||
|         run: | | ||||
|           VERSION="${{ steps.tag.outputs.version }}" | ||||
|           BRANCH="master" | ||||
|           git config --global user.name "github-actions[bot]" | ||||
|           git config --global user.email "github-actions[bot]@users.noreply.github.com" | ||||
|           git config --global --unset url."git@github.com:".insteadOf || true | ||||
|           git fetch origin $BRANCH | ||||
|           git checkout $BRANCH | ||||
|           if [ -f "goecs.sh" ]; then | ||||
|             sed -i "s/\(_yellow \"Unable to get version info, using default version \).*\(\".*\)/\1$VERSION\2/" "goecs.sh" | ||||
|             sed -i "s/\(ECS_VERSION=\"\).*\(\"\)/\1$VERSION\2/" "goecs.sh" | ||||
|             if ! git diff --quiet "goecs.sh"; then | ||||
|               git add "goecs.sh" | ||||
|               git commit -m "chore: update ECS_VERSION to $VERSION in goecs.sh" | ||||
|               git push origin $BRANCH | ||||
|             fi | ||||
|           fi | ||||
|         env: | ||||
|           GITHUB_TOKEN: ${{ secrets.GHT }} | ||||
| @@ -7,5 +7,5 @@ import ( | ||||
|  | ||||
| func ComMediaTest(language string) { | ||||
| 	res := commediatests.MediaTests(language) | ||||
| 	fmt.Printf(res) | ||||
| 	fmt.Print(res) | ||||
| } | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| package network1 | ||||
|  | ||||
| import "github.com/oneclickvirt/security/network" | ||||
| import "github.com/oneclickvirt/basics/network" | ||||
|  | ||||
| // 本包在main中不使用 | ||||
| func NetworkCheck(checkType string, enableSecurityCheck bool, language string) (string, string, error) { | ||||
|   | ||||
| @@ -15,8 +15,8 @@ func TestIpv4SecurityCheck(t *testing.T) { | ||||
| 	// 全项测试 | ||||
| 	ipInfo, securityInfo, _ := NetworkCheck("both", true, "zh") | ||||
| 	fmt.Println("--------------------------------------------------") | ||||
| 	fmt.Printf(ipInfo) | ||||
| 	fmt.Print(ipInfo) | ||||
| 	fmt.Println("--------------------------------------------------") | ||||
| 	fmt.Printf(securityInfo) | ||||
| 	fmt.Print(securityInfo) | ||||
| 	fmt.Println("--------------------------------------------------") | ||||
| } | ||||
|   | ||||
| @@ -1,9 +0,0 @@ | ||||
| package ntrace | ||||
|  | ||||
| import ( | ||||
| 	"github.com/oneclickvirt/nt3/nt" | ||||
| ) | ||||
|  | ||||
| func TraceRoute3(language, location, checkType string) { | ||||
| 	nt.TraceRoute(language, location, checkType) | ||||
| } | ||||
| @@ -1,9 +0,0 @@ | ||||
| package ntrace | ||||
|  | ||||
| import "testing" | ||||
|  | ||||
| // https://github.com/nxtrace/NTrace-core/blob/main/fast_trace/fast_trace.go | ||||
| // 本包仅测试无实际使用 | ||||
| func TestTraceRoute(t *testing.T) { | ||||
| 	TraceRoute3("en", "GZ", "ipv4") | ||||
| } | ||||
							
								
								
									
										439
									
								
								.github/workflows/build_binary.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										439
									
								
								.github/workflows/build_binary.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -1,235 +1,26 @@ | ||||
| name: Build and Release | ||||
|  | ||||
| on: | ||||
|   workflow_dispatch: | ||||
|     tags: | ||||
|       - "v*.*.*" | ||||
|  | ||||
| jobs: | ||||
|   build: | ||||
|     name: Release Check And Build | ||||
|   goreleaser: | ||||
|     runs-on: ubuntu-latest | ||||
|     container: | ||||
|       # 1.20 是 Windows 7/8 Server 2008/2012 最后一个支持版本 | ||||
|       image: goreleaser/goreleaser-cross:v1.20 | ||||
|     steps: | ||||
|       - name: Checkout code | ||||
|         uses: actions/checkout@v4 | ||||
|         with: | ||||
|           fetch-depth: 0 | ||||
|  | ||||
|       - name: Get latest tag | ||||
|         id: tag | ||||
|         run: | | ||||
|           TAG=$(git describe --tags --abbrev=0 2>/dev/null || echo "v0.1.0") | ||||
|           echo "tag=$TAG" >> $GITHUB_OUTPUT | ||||
|           echo "version=${TAG#v}" >> $GITHUB_OUTPUT | ||||
|  | ||||
|       - name: Generate changelog | ||||
|         id: changelog | ||||
|         run: | | ||||
|           TAG="${{ steps.tag.outputs.tag }}" | ||||
|           PREV_TAG=$(git describe --tags --abbrev=0 "$TAG^" 2>/dev/null || echo "") | ||||
|           if [ -z "$PREV_TAG" ]; then | ||||
|             CHANGELOG=$(git log --oneline --pretty=format:"* %H %s" "$TAG" | head -20) | ||||
|           else | ||||
|             CHANGELOG=$(git log --oneline --pretty=format:"* %H %s" "$PREV_TAG..$TAG") | ||||
|           fi | ||||
|           FULL_CHANGELOG="## Changelog"$'\n'"$CHANGELOG" | ||||
|           echo "$FULL_CHANGELOG" > changelog.txt | ||||
|           echo "changelog<<EOF" >> $GITHUB_OUTPUT | ||||
|           echo "$FULL_CHANGELOG" >> $GITHUB_OUTPUT | ||||
|           echo "EOF" >> $GITHUB_OUTPUT | ||||
|  | ||||
|       - name: Create release if not exists | ||||
|         run: | | ||||
|           TAG="${{ steps.tag.outputs.tag }}" | ||||
|           RELEASE_EXISTS=$(curl -s -H "Authorization: Bearer ${{ secrets.GHT }}" "https://api.github.com/repos/${{ github.repository }}/releases/tags/$TAG" | jq -r '.id // empty') | ||||
|           if [ -z "$RELEASE_EXISTS" ]; then | ||||
|             CHANGELOG_BODY=$(cat changelog.txt | jq -Rs .) | ||||
|             curl -s -X POST -H "Authorization: Bearer ${{ secrets.GHT }}" \ | ||||
|               -H "Content-Type: application/json" \ | ||||
|               -d "{\"tag_name\":\"$TAG\",\"name\":\"$TAG\",\"body\":$CHANGELOG_BODY,\"draft\":false,\"prerelease\":false}" \ | ||||
|               "https://api.github.com/repos/${{ github.repository }}/releases" > /dev/null | ||||
|           fi | ||||
|  | ||||
|       - name: Delete existing release assets | ||||
|         run: | | ||||
|           TAG="${{ steps.tag.outputs.tag }}" | ||||
|           RELEASE_ID=$(curl -s -H "Authorization: Bearer ${{ secrets.GHT }}" "https://api.github.com/repos/${{ github.repository }}/releases/tags/$TAG" | jq -r '.id') | ||||
|           if [ "$RELEASE_ID" != "null" ]; then | ||||
|             ASSETS=$(curl -s -H "Accept: application/vnd.github.v3+json" "https://api.github.com/repos/${{ github.repository }}/releases/$RELEASE_ID/assets" | jq -r '.[] | .id') | ||||
|             for asset in $ASSETS; do | ||||
|               curl -s -X DELETE -H "Authorization: Bearer ${{ secrets.GHT }}" "https://api.github.com/repos/${{ github.repository }}/releases/assets/$asset" | ||||
|             done | ||||
|             sleep 30 | ||||
|           fi | ||||
|  | ||||
|   release-binary: | ||||
|     name: Release Go Binary | ||||
|     needs: build | ||||
|     strategy: | ||||
|       fail-fast: false | ||||
|       matrix: | ||||
|         include: | ||||
|           - goos: linux | ||||
|             goarch: amd64 | ||||
|             cgo_enabled: "1" | ||||
|             cc: x86_64-linux-gnu-gcc | ||||
|             cflags: "-O2 -static -fno-stack-protector" | ||||
|             ldflags: "-extldflags=-static -s -w" | ||||
|             packages: "build-essential gcc" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: linux | ||||
|             goarch: 386 | ||||
|             cgo_enabled: "1" | ||||
|             cc: x86_64-linux-gnu-gcc | ||||
|             cflags: "-m32 -O1 -march=i686 -mtune=generic -fno-stack-protector" | ||||
|             ldflags: "-extldflags=-static -s -w" | ||||
|             packages: "build-essential gcc-multilib" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: linux | ||||
|             goarch: arm64 | ||||
|             cgo_enabled: "1" | ||||
|             cc: aarch64-linux-gnu-gcc | ||||
|             cflags: "-O1 -fno-stack-protector" | ||||
|             ldflags: "-extldflags=-static -s -w" | ||||
|             packages: "build-essential gcc-aarch64-linux-gnu" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: linux | ||||
|             goarch: riscv64 | ||||
|             cgo_enabled: "1" | ||||
|             cc: riscv64-linux-gnu-gcc | ||||
|             cflags: "-O1 -fno-stack-protector" | ||||
|             ldflags: "-extldflags=-static -s -w" | ||||
|             packages: "build-essential gcc-riscv64-linux-gnu" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: linux | ||||
|             goarch: mips64 | ||||
|             cgo_enabled: "1" | ||||
|             cc: mips64-linux-gnuabi64-gcc | ||||
|             cflags: "-O1 -fno-stack-protector" | ||||
|             ldflags: "-extldflags=-static -s -w" | ||||
|             packages: "build-essential gcc-mips64-linux-gnuabi64" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: linux | ||||
|             goarch: mips64le | ||||
|             cgo_enabled: "1" | ||||
|             cc: mips64el-linux-gnuabi64-gcc | ||||
|             cflags: "-O1 -fno-stack-protector" | ||||
|             ldflags: "-extldflags=-static -s -w" | ||||
|             packages: "build-essential gcc-mips64el-linux-gnuabi64" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: linux | ||||
|             goarch: ppc64le | ||||
|             cgo_enabled: "1" | ||||
|             cc: powerpc64le-linux-gnu-gcc | ||||
|             cflags: "-O1 -fno-stack-protector" | ||||
|             ldflags: "-extldflags=-static -s -w" | ||||
|             packages: "build-essential gcc-powerpc64le-linux-gnu" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: windows | ||||
|             goarch: amd64 | ||||
|             cgo_enabled: "1" | ||||
|             cc: x86_64-w64-mingw32-gcc | ||||
|             cflags: "-O2 -static-libgcc -static-libstdc++" | ||||
|             ldflags: "-extldflags=-static -s -w" | ||||
|             packages: "build-essential gcc-mingw-w64-x86-64" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: windows | ||||
|             goarch: 386 | ||||
|             cgo_enabled: "1" | ||||
|             cc: i686-w64-mingw32-gcc | ||||
|             cflags: "-O2 -static-libgcc -static-libstdc++" | ||||
|             ldflags: "-extldflags=-static -s -w" | ||||
|             packages: "build-essential gcc-mingw-w64-i686" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: darwin | ||||
|             goarch: amd64 | ||||
|             cgo_enabled: "0" | ||||
|             ldflags: "-s -w" | ||||
|             runner: macos-latest | ||||
|  | ||||
|           - goos: darwin | ||||
|             goarch: arm64 | ||||
|             cgo_enabled: "0" | ||||
|             ldflags: "-s -w" | ||||
|             runner: macos-latest | ||||
|  | ||||
|           - goos: linux | ||||
|             goarch: arm | ||||
|             cgo_enabled: "0" | ||||
|             ldflags: "-s -w" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: linux | ||||
|             goarch: s390x | ||||
|             cgo_enabled: "0" | ||||
|             ldflags: "-s -w" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: linux | ||||
|             goarch: mips | ||||
|             cgo_enabled: "0" | ||||
|             ldflags: "-s -w" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: linux | ||||
|             goarch: mipsle | ||||
|             cgo_enabled: "0" | ||||
|             ldflags: "-s -w" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: linux | ||||
|             goarch: ppc64 | ||||
|             cgo_enabled: "0" | ||||
|             ldflags: "-s -w" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: windows | ||||
|             goarch: arm64 | ||||
|             cgo_enabled: "0" | ||||
|             ldflags: "-s -w" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: freebsd | ||||
|             goarch: amd64 | ||||
|             cgo_enabled: "0" | ||||
|             ldflags: "-s -w" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: freebsd | ||||
|             goarch: arm64 | ||||
|             cgo_enabled: "0" | ||||
|             ldflags: "-s -w" | ||||
|             runner: ubuntu-latest | ||||
|            | ||||
|           - goos: openbsd | ||||
|             goarch: amd64 | ||||
|             cgo_enabled: "0" | ||||
|             ldflags: "-s -w" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|           - goos: openbsd | ||||
|             goarch: arm64 | ||||
|             cgo_enabled: "0" | ||||
|             ldflags: "-s -w" | ||||
|             runner: ubuntu-latest | ||||
|  | ||||
|     runs-on: ${{ matrix.runner }} | ||||
|     steps: | ||||
|       - name: Checkout code | ||||
|         uses: actions/checkout@v4 | ||||
|       - run: | | ||||
|           git config --global --add safe.directory /__w/ecs/ecs | ||||
|       - name: Checkout | ||||
|         uses: actions/checkout@v3 | ||||
|         with: | ||||
|           fetch-depth: 0 | ||||
|  | ||||
|       - name: Set up Go | ||||
|         uses: actions/setup-go@v5 | ||||
|         uses: actions/setup-go@v4 | ||||
|         with: | ||||
|           go-version: 1.24.5 | ||||
|  | ||||
| @@ -240,200 +31,44 @@ jobs: | ||||
|         env: | ||||
|           GITHUB_TOKEN: ${{ secrets.GHT }} | ||||
|  | ||||
|       - name: Install cross-compilation tools | ||||
|         if: matrix.runner != 'macos-latest' | ||||
|         run: | | ||||
|           sudo apt-get update -qq | ||||
|           case "${{ matrix.goos }}-${{ matrix.goarch }}" in | ||||
|             linux-386) | ||||
|               sudo apt-get install -y build-essential gcc-multilib g++-multilib ;; | ||||
|             linux-arm64) | ||||
|               sudo apt-get install -y build-essential gcc-aarch64-linux-gnu ;; | ||||
|             linux-riscv64) | ||||
|               sudo apt-get install -y build-essential gcc-riscv64-linux-gnu ;; | ||||
|             linux-mips64) | ||||
|               sudo apt-get install -y build-essential gcc-mips64-linux-gnuabi64 ;; | ||||
|             linux-mips64le) | ||||
|               sudo apt-get install -y build-essential gcc-mips64el-linux-gnuabi64 ;; | ||||
|             linux-ppc64le) | ||||
|               sudo apt-get install -y build-essential gcc-powerpc64le-linux-gnu ;; | ||||
|             windows-amd64|windows-386) | ||||
|               sudo apt-get install -y build-essential gcc-mingw-w64-x86-64 gcc-mingw-w64-i686 ;; | ||||
|             *) | ||||
|               sudo apt-get install -y build-essential ;; | ||||
|           esac | ||||
|       - name: Run GoReleaser | ||||
|         uses: goreleaser/goreleaser-action@v6 | ||||
|         with: | ||||
|           distribution: goreleaser | ||||
|           # version: latest | ||||
|           version: '~> v2' | ||||
|           args: release | ||||
|         env: | ||||
|           GITHUB_TOKEN: ${{ secrets.GHT }} | ||||
|           GOPRIVATE: github.com/oneclickvirt/security | ||||
|  | ||||
|       - name: Get latest tag | ||||
|         id: tag | ||||
|       - name: Update goecs.sh with new version | ||||
|         run: | | ||||
|           TAG=$(git describe --tags --abbrev=0 2>/dev/null || echo "v0.1.0") | ||||
|           echo "tag=$TAG" >> $GITHUB_OUTPUT | ||||
|           echo "version=${TAG#v}" >> $GITHUB_OUTPUT | ||||
|  | ||||
|       - name: Build Binary | ||||
|         run: | | ||||
|           go clean -cache -modcache -testcache | ||||
|           export CGO_ENABLED=${{ matrix.cgo_enabled }} | ||||
|           export GOOS=${{ matrix.goos }} | ||||
|           export GOARCH=${{ matrix.goarch }} | ||||
|           if [[ -n "${{ matrix.goarm }}" ]]; then | ||||
|             export GOARM=${{ matrix.goarm }} | ||||
|           fi | ||||
|           if [[ -n "${{ matrix.gomips }}" ]]; then | ||||
|             export GOMIPS=${{ matrix.gomips }} | ||||
|           fi | ||||
|           if [[ "${{ matrix.cgo_enabled }}" == "1" && "${{ matrix.goos }}" == "darwin" ]]; then | ||||
|             if [[ "${{ matrix.goarch }}" == "amd64" ]]; then | ||||
|               export CC="x86_64-apple-darwin21.4-clang" | ||||
|               export CXX="x86_64-apple-darwin21.4-clang++" | ||||
|             elif [[ "${{ matrix.goarch }}" == "arm64" ]]; then | ||||
|               export CC="aarch64-apple-darwin21.4-clang" | ||||
|               export CXX="aarch64-apple-darwin21.4-clang++" | ||||
|             fi | ||||
|             export CGO_CFLAGS="${{ matrix.cflags }}" | ||||
|             export CGO_LDFLAGS="${{ matrix.ldflags }}" | ||||
|             export OSXCROSS_ROOT="${OSXCROSS_ROOT}" | ||||
|           elif [[ "${{ matrix.cgo_enabled }}" == "1" && "${{ matrix.runner }}" != "macos-latest" ]]; then | ||||
|             export CC="${{ matrix.cc }}" | ||||
|             export CGO_CFLAGS="${{ matrix.cflags }}" | ||||
|             if [[ "${{ matrix.goos }}" == "windows" ]]; then | ||||
|               export CGO_LDFLAGS="-static-libgcc -static-libstdc++" | ||||
|             fi | ||||
|           fi | ||||
|           if [[ "${{ matrix.cgo_enabled }}" == "1" ]]; then | ||||
|             echo 'int main() { return 0; }' > test.c | ||||
|             $CC $CGO_CFLAGS test.c -o test || exit 1 | ||||
|             rm -f test.c test | ||||
|           fi | ||||
|           rm -rf vendor/ | ||||
|           go mod download | ||||
|           go mod tidy | ||||
|           mkdir -p bin | ||||
|           BINARY_NAME="goecs" | ||||
|           if [[ "${{ matrix.goos }}" == "windows" ]]; then | ||||
|             BINARY_NAME="${BINARY_NAME}.exe" | ||||
|           fi | ||||
|           LDFLAGS="-s -w -X main.version=${{ steps.tag.outputs.version }} -X main.arch=${{ matrix.goarch }}" | ||||
|           if [[ "${{ matrix.cgo_enabled }}" == "1" ]]; then | ||||
|             LDFLAGS="${LDFLAGS} -checklinkname=0 ${{ matrix.ldflags }}" | ||||
|             env CGO_ENABLED=1 GOOS=$GOOS GOARCH=$GOARCH CC="$CC" CGO_CFLAGS="$CGO_CFLAGS" CGO_LDFLAGS="$CGO_LDFLAGS" \ | ||||
|               go build -a -o bin/$BINARY_NAME -ldflags="$LDFLAGS" -trimpath ./ | ||||
|           if [[ "$GITHUB_REF" == refs/tags/* ]]; then | ||||
|             VERSION="${GITHUB_REF#refs/tags/v}" | ||||
|           else | ||||
|             LDFLAGS="${LDFLAGS} -checklinkname=0 ${{ matrix.ldflags }}" | ||||
|             env CGO_ENABLED=0 GOOS=$GOOS GOARCH=$GOARCH \ | ||||
|               go build -o bin/$BINARY_NAME -ldflags="$LDFLAGS" -trimpath ./ | ||||
|             VERSION=$(git describe --tags --abbrev=0 2>/dev/null | sed 's/^v//' || echo "0.1.37") | ||||
|           fi | ||||
|           [[ -f "bin/$BINARY_NAME" ]] || exit 1 | ||||
|  | ||||
|       - name: Create ZIP archive | ||||
|         run: | | ||||
|           cd bin | ||||
|           BINARY_NAME="goecs" | ||||
|           if [[ "${{ matrix.goos }}" == "windows" ]]; then | ||||
|             BINARY_NAME="${BINARY_NAME}.exe" | ||||
|           fi | ||||
|           ZIP_NAME="goecs_${{ matrix.goos }}_${{ matrix.goarch }}" | ||||
|           if [[ -n "${{ matrix.goarm }}" ]]; then | ||||
|             ZIP_NAME="${ZIP_NAME}v${{ matrix.goarm }}" | ||||
|           fi | ||||
|           if [[ -n "${{ matrix.gomips }}" ]]; then | ||||
|             ZIP_NAME="${ZIP_NAME}_${{ matrix.gomips }}" | ||||
|           fi | ||||
|           ZIP_NAME="${ZIP_NAME}.zip" | ||||
|           zip "$ZIP_NAME" "$BINARY_NAME" | ||||
|  | ||||
|       - name: Upload to Release | ||||
|         run: | | ||||
|           TAG="${{ steps.tag.outputs.tag }}" | ||||
|           RELEASE_ID=$(curl -s -H "Authorization: Bearer ${{ secrets.GHT }}" "https://api.github.com/repos/${{ github.repository }}/releases/tags/$TAG" | jq -r '.id') | ||||
|           cd bin | ||||
|           for file in *.zip; do | ||||
|             if [[ -f "$file" ]]; then | ||||
|               curl -s -H "Authorization: Bearer ${{ secrets.GHT }}" \ | ||||
|                 -H "Content-Type: application/zip" \ | ||||
|                 --data-binary @"$file" \ | ||||
|                 "https://uploads.github.com/repos/${{ github.repository }}/releases/$RELEASE_ID/assets?name=$file" | ||||
|             fi | ||||
|           done | ||||
|  | ||||
|   checksums: | ||||
|     name: Generate Checksums | ||||
|     runs-on: ubuntu-latest | ||||
|     needs: release-binary | ||||
|     steps: | ||||
|       - name: Checkout code | ||||
|         uses: actions/checkout@v4 | ||||
|         with: | ||||
|           fetch-depth: 0 | ||||
|  | ||||
|       - name: Get latest tag | ||||
|         id: tag | ||||
|         run: | | ||||
|           TAG=$(git describe --tags --abbrev=0 2>/dev/null || echo "v0.1.0") | ||||
|           echo "tag=$TAG" >> $GITHUB_OUTPUT | ||||
|  | ||||
|       - name: Download release assets | ||||
|         run: | | ||||
|           TAG="${{ steps.tag.outputs.tag }}" | ||||
|           RELEASE_ID=$(curl -s -H "Authorization: Bearer ${{ secrets.GHT }}" "https://api.github.com/repos/${{ github.repository }}/releases/tags/$TAG" | jq -r '.id') | ||||
|           mkdir -p assets | ||||
|           ASSETS=$(curl -s -H "Accept: application/vnd.github.v3+json" "https://api.github.com/repos/${{ github.repository }}/releases/$RELEASE_ID/assets") | ||||
|           echo "$ASSETS" | jq -r '.[] | select(.name | endswith(".zip")) | .browser_download_url' | while read url; do | ||||
|             filename=$(basename "$url") | ||||
|             curl -L -H "Authorization: Bearer ${{ secrets.GHT }}" "$url" -o "assets/$filename" | ||||
|           done | ||||
|  | ||||
|       - name: Generate checksums | ||||
|         run: | | ||||
|           cd assets | ||||
|           sha256sum *.zip > checksums.txt | ||||
|           if [[ -f "../goecs.sh" ]]; then | ||||
|             sha256sum ../goecs.sh >> checksums.txt | ||||
|           fi | ||||
|  | ||||
|       - name: Upload checksums | ||||
|         run: | | ||||
|           TAG="${{ steps.tag.outputs.tag }}" | ||||
|           RELEASE_ID=$(curl -s -H "Authorization: Bearer ${{ secrets.GHT }}" "https://api.github.com/repos/${{ github.repository }}/releases/tags/$TAG" | jq -r '.id') | ||||
|           curl -s -H "Authorization: Bearer ${{ secrets.GHT }}" \ | ||||
|             -H "Content-Type: text/plain" \ | ||||
|             --data-binary @assets/checksums.txt \ | ||||
|             "https://uploads.github.com/repos/${{ github.repository }}/releases/$RELEASE_ID/assets?name=checksums.txt" | ||||
|  | ||||
|   update-script: | ||||
|     name: Update Script Version | ||||
|     runs-on: ubuntu-latest | ||||
|     needs: checksums | ||||
|     steps: | ||||
|       - name: Checkout code | ||||
|         uses: actions/checkout@v4 | ||||
|         with: | ||||
|           fetch-depth: 0 | ||||
|  | ||||
|       - name: Get latest tag | ||||
|         id: tag | ||||
|         run: | | ||||
|           TAG=$(git describe --tags --abbrev=0 2>/dev/null || echo "v0.1.0") | ||||
|           echo "tag=$TAG" >> $GITHUB_OUTPUT | ||||
|           echo "version=${TAG#v}" >> $GITHUB_OUTPUT | ||||
|  | ||||
|       - name: Update goecs.sh version | ||||
|         run: | | ||||
|           VERSION="${{ steps.tag.outputs.version }}" | ||||
|           echo "Using version: $VERSION" | ||||
|           FILE="goecs.sh" | ||||
|           BRANCH="master" | ||||
|           git config --global user.name "github-actions[bot]" | ||||
|           git config --global user.email "github-actions[bot]@users.noreply.github.com" | ||||
|           git config --global --unset url."git@github.com:".insteadOf || true | ||||
|           git fetch origin $BRANCH | ||||
|           git checkout $BRANCH | ||||
|           if [ -f "goecs.sh" ]; then | ||||
|             sed -i "s/\(_yellow \"Unable to get version info, using default version \).*\(\".*\)/\1$VERSION\2/" "goecs.sh" | ||||
|             sed -i "s/\(ECS_VERSION=\"\).*\(\"\)/\1$VERSION\2/" "goecs.sh" | ||||
|             if ! git diff --quiet "goecs.sh"; then | ||||
|               git add "goecs.sh" | ||||
|           if [ ! -f "$FILE" ]; then | ||||
|             echo "Error: $FILE not found" | ||||
|             exit 1 | ||||
|           fi | ||||
|           sed -i "s/\(_yellow \"Unable to get version info, using default version \).*\(\".*\)/\1$VERSION\2/" "$FILE" | ||||
|           sed -i "s/\(ECS_VERSION=\"\).*\(\"\)/\1$VERSION\2/" "$FILE" | ||||
|           if git diff --quiet "$FILE"; then | ||||
|             echo "No changes detected in $FILE" | ||||
|             exit 0 | ||||
|           fi | ||||
|           git add "$FILE" | ||||
|           git commit -m "chore: update ECS_VERSION to $VERSION in goecs.sh" | ||||
|           git push origin $BRANCH | ||||
|             fi | ||||
|           fi | ||||
|         env: | ||||
|           GITHUB_TOKEN: ${{ secrets.GHT }} | ||||
|   | ||||
							
								
								
									
										20
									
								
								.github/workflows/build_docker.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										20
									
								
								.github/workflows/build_docker.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -28,12 +28,19 @@ jobs: | ||||
|           username: ${{ secrets.DOCKER_USERNAME }} | ||||
|           password: ${{ secrets.DOCKER_PASSWORD }} | ||||
|  | ||||
|       - name: Login to CNB Registry | ||||
|       # - name: Login to CNB Registry | ||||
|       #   uses: docker/login-action@v2 | ||||
|       #   with: | ||||
|       #     registry: ${{ secrets.CNB_DOCKER_REGISTRY }} | ||||
|       #     username: ${{ secrets.CNB_USERNAME }} | ||||
|       #     password: ${{ secrets.CNB_TOKEN }} | ||||
|        | ||||
|       - name: Login to Aliyun Container Registry | ||||
|         uses: docker/login-action@v2 | ||||
|         with: | ||||
|           registry: ${{ secrets.CNB_DOCKER_REGISTRY }} | ||||
|           username: ${{ secrets.CNB_USERNAME }} | ||||
|           password: ${{ secrets.CNB_TOKEN }} | ||||
|           registry: crpi-8tmognxgyb86bm61.cn-guangzhou.personal.cr.aliyuncs.com | ||||
|           username: ${{ secrets.ALIYUN_USERNAME }} | ||||
|           password: ${{ secrets.ALIYUN_PASSWORD }} | ||||
|  | ||||
|       - name: Build and push Docker images | ||||
|         uses: docker/build-push-action@v4 | ||||
| @@ -41,8 +48,9 @@ jobs: | ||||
|           context: . | ||||
|           file: ./Dockerfile | ||||
|           platforms: linux/amd64,linux/arm/v6,linux/arm/v7,linux/arm64,linux/386,linux/riscv64 | ||||
|           # linux/mips,linux/mipsle 暂不支持 alpine, linux/s390x 编译卡死 | ||||
|           # linux/mips,linux/mipsle 暂不支持 alpine, linux/s390x 编译卡死,cnb组织空间不足无法推送 | ||||
|           # ${{ secrets.CNB_DOCKER_REGISTRY }}/oneclickvirt/ecs:latest | ||||
|           push: true | ||||
|           tags: | | ||||
|             ${{ secrets.DOCKER_USERNAME }}/goecs:latest | ||||
|             ${{ secrets.CNB_DOCKER_REGISTRY }}/oneclickvirt/ecs:latest | ||||
|             crpi-8tmognxgyb86bm61.cn-guangzhou.personal.cr.aliyuncs.com/oneclickvirt/ecs:latest | ||||
|   | ||||
							
								
								
									
										108
									
								
								.github/workflows/build_public.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										108
									
								
								.github/workflows/build_public.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,108 @@ | ||||
| name: Public Build | ||||
| on: | ||||
|   workflow_run: | ||||
|     workflows: ["Build and Release"] | ||||
|     types: | ||||
|       - completed | ||||
|   workflow_dispatch: | ||||
| jobs: | ||||
|   build: | ||||
|     runs-on: ubuntu-latest | ||||
|      | ||||
|     steps: | ||||
|     - uses: actions/checkout@v4 | ||||
|       with: | ||||
|         fetch-depth: 0 | ||||
|     - name: Set up Go | ||||
|       uses: actions/setup-go@v5 | ||||
|       with: | ||||
|         go-version: '1.24.5' | ||||
|          | ||||
|     - name: Update master branch README files | ||||
|       run: | | ||||
|         git config --global user.name 'GitHub Actions' | ||||
|         git config --global user.email 'actions@github.com' | ||||
|          | ||||
|         if [ -f "go.mod" ]; then | ||||
|           GO_VERSION=$(grep "^go " go.mod | head -n 1 | awk '{print $2}') | ||||
|           echo "提取到的 Go 版本: $GO_VERSION" | ||||
|           if [ -n "$GO_VERSION" ] && [[ "$GO_VERSION" =~ ^[0-9]+\.[0-9]+(\.[0-9]+)?$ ]]; then | ||||
|             echo "版本验证成功,开始替换..." | ||||
|             if [ -f "README.md" ]; then | ||||
|               sed -i "s/选择 go [0-9]\+\.[0-9]\+\.[0-9]\+ 的版本进行安装/选择 go $GO_VERSION 的版本进行安装/g" README.md | ||||
|               sed -i 's|但二进制文件编译至 \[securityCheck\].*)|但已开源|g' README.md | ||||
|               sed -i 's|security.*Enable/Disable security test (default true)|security        Enable/Disable security test (default false)|g' README.md | ||||
|               echo "已更新 README.md" | ||||
|             fi | ||||
|             if [ -f "README_EN.md" ]; then | ||||
|               sed -i "s/Select go [0-9]\+\.[0-9]\+\.[0-9]\+ version to install/Select go $GO_VERSION version to install/g" README_EN.md | ||||
|               sed -i 's|but binary files compiled in \[securityCheck\].*)|but open sourced|g' README_EN.md | ||||
|               sed -i 's|security.*Enable/Disable security test (default true)|security        Enable/Disable security test (default false)|g' README_EN.md | ||||
|               echo "已更新 README_EN.md" | ||||
|             fi | ||||
|             git add README.md README_EN.md | ||||
|             git commit -m "Auto update README files" || echo "No changes to commit" | ||||
|             git push origin ${{ github.ref_name }} | ||||
|           else | ||||
|             echo "错误:未能提取到有效的 Go 版本号或版本号格式不正确" | ||||
|             exit 1 | ||||
|           fi | ||||
|         else | ||||
|           echo "错误:未找到 go.mod 文件" | ||||
|           exit 1 | ||||
|         fi | ||||
|          | ||||
|     - name: Create public branch | ||||
|       run: | | ||||
|         git checkout -b public || git checkout public | ||||
|         git merge ${{ github.ref_name }} --no-edit || true | ||||
|          | ||||
|     - name: Remove security package references | ||||
|       run: | | ||||
|         find . -type f -name "*.go" -exec sed -i 's|"github.com/oneclickvirt/security/network"|"github.com/oneclickvirt/basics/network"|g' {} + | ||||
|         sed -i '/SecurityUploadToken/d' utils/utils.go | ||||
|         sed -i 's|"github.com/oneclickvirt/security/network"|"github.com/oneclickvirt/basics/network"|g' utils/utils.go | ||||
|         sed -i '/^import/,/^)/{/^)/a\'$'\n''const token = "OvwKx5qgJtf7PZgCKbtyojSU.MTcwMTUxNzY1MTgwMw"'$'\n''}' utils/utils.go | ||||
|         sed -i '/github.com\/oneclickvirt\/security/d' go.mod | ||||
|         sed -i 's|var securityFlag = flag.Bool("security", true,|var securityFlag = flag.Bool("security", false,|g' goecs.go | ||||
|         sed -i 's|VPS融合怪测试|VPS融合怪测试(非官方编译)|g' utils/utils.go | ||||
|         sed -i 's|VPS Fusion Monster Test|VPS Fusion Monster Test (Unofficial)|g' utils/utils.go | ||||
|         go mod tidy | ||||
|         sed -i 's|但二进制文件编译至 \[securityCheck\].*)|但已开源|g' README.md | ||||
|         sed -i 's|but binary files compiled in \[securityCheck\].*)|but open sourced|g' README_EN.md | ||||
|         sed -i 's|security.*Enable/Disable security test (default true)|security        Enable/Disable security test (default false)|g' README.md | ||||
|         sed -i 's|security.*Enable/Disable security test (default true)|security        Enable/Disable security test (default false)|g' README_EN.md | ||||
|     - name: Update Go version in README files | ||||
|       run: | | ||||
|         if [ -f "go.mod" ]; then | ||||
|           GO_VERSION=$(grep "^go " go.mod | head -n 1 | awk '{print $2}') | ||||
|           echo "提取到的 Go 版本: $GO_VERSION" | ||||
|           if [ -n "$GO_VERSION" ] && [[ "$GO_VERSION" =~ ^[0-9]+\.[0-9]+(\.[0-9]+)?$ ]]; then | ||||
|             echo "版本验证成功,开始替换..." | ||||
|             if [ -f "README.md" ]; then | ||||
|               sed -i "s/选择 go [0-9]\+\.[0-9]\+\.[0-9]\+ 的版本进行安装/选择 go $GO_VERSION 的版本进行安装/g" README.md | ||||
|               echo "已更新 README.md" | ||||
|             fi | ||||
|             if [ -f "README_EN.md" ]; then | ||||
|               sed -i "s/Select go [0-9]\+\.[0-9]\+\.[0-9]\+ version to install/Select go $GO_VERSION version to install/g" README_EN.md | ||||
|               echo "已更新 README_EN.md" | ||||
|             fi | ||||
|           else | ||||
|             echo "错误:未能提取到有效的 Go 版本号或版本号格式不正确" | ||||
|             exit 1 | ||||
|           fi | ||||
|         else | ||||
|           echo "错误:未找到 go.mod 文件" | ||||
|           exit 1 | ||||
|         fi | ||||
|     - name: Build and Test | ||||
|       run: | | ||||
|         go build -o maintest | ||||
|         ./maintest -menu=false -l en -security=false -upload=false || exit 1 | ||||
|         rm -rf maintest | ||||
|          | ||||
|     - name: Commit and push changes | ||||
|       run: | | ||||
|         git add . | ||||
|         git commit -m "Auto update public version (no security package)" || echo "No changes to commit" | ||||
|         git push -f origin public | ||||
							
								
								
									
										81
									
								
								.github/workflows/public_build.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										81
									
								
								.github/workflows/public_build.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,81 +0,0 @@ | ||||
| name: Public Build | ||||
|  | ||||
| on: | ||||
|   workflow_run: | ||||
|     workflows: ["Build and Release"] | ||||
|     types: | ||||
|       - completed | ||||
|   workflow_dispatch: | ||||
|  | ||||
| jobs: | ||||
|   build: | ||||
|     runs-on: ubuntu-latest | ||||
|      | ||||
|     steps: | ||||
|     - uses: actions/checkout@v4 | ||||
|       with: | ||||
|         fetch-depth: 0 | ||||
|  | ||||
|     - name: Set up Go | ||||
|       uses: actions/setup-go@v5 | ||||
|       with: | ||||
|         go-version: '1.23.4' | ||||
|          | ||||
|     - name: Create public branch | ||||
|       run: | | ||||
|         git config --global user.name 'GitHub Actions' | ||||
|         git config --global user.email 'actions@github.com' | ||||
|         git checkout -b public || git checkout public | ||||
|         git merge ${{ github.ref_name }} --no-edit || true | ||||
|          | ||||
|     - name: Remove security package references | ||||
|       run: | | ||||
|         # 移除 network 包中对 security 的引用 | ||||
|         find . -type f -name "*.go" -exec sed -i 's|"github.com/oneclickvirt/security/network"|"github.com/oneclickvirt/basics/network"|g' {} + | ||||
|         # 修改 back/network/network.go | ||||
|         cat > back/network/network.go << 'EOF' | ||||
|         package network1 | ||||
|  | ||||
|         import "github.com/oneclickvirt/basics/network" | ||||
|  | ||||
|         func NetworkCheck(checkType string, enableSecurityCheck bool, language string) (string, string, error) { | ||||
|             ipInfo, _, err := network.NetworkCheck(checkType, false, language) | ||||
|             return ipInfo, "", err | ||||
|         } | ||||
|         EOF | ||||
|          | ||||
|         # 修改 utils/utils.go 中的 BasicsAndSecurityCheck 函数 | ||||
|         sed -i '/SecurityUploadToken/d' utils/utils.go | ||||
|         sed -i 's|"github.com/oneclickvirt/security/network"|"github.com/oneclickvirt/basics/network"|g' utils/utils.go | ||||
|          | ||||
|         # 在 utils/utils.go 中添加 token 常量(在 import 语句之后) | ||||
|         sed -i '/^import/,/^)/{/^)/a\'$'\n''const token = "OvwKx5qgJtf7PZgCKbtyojSU.MTcwMTUxNzY1MTgwMw"'$'\n''}' utils/utils.go | ||||
|          | ||||
|         # 修改 go.mod,移除私有仓库依赖 | ||||
|         sed -i '/github.com\/oneclickvirt\/security/d' go.mod | ||||
|          | ||||
|         # 修改 goecs.go,禁用 security 检测 | ||||
|         sed -i 's|var securityFlag = flag.Bool("security", true,|var securityFlag = flag.Bool("security", false,|g' goecs.go | ||||
|          | ||||
|         # 更新依赖 | ||||
|         go mod tidy | ||||
|          | ||||
|         # 修改 README.md 和 README_EN.md 中的敏感信息 | ||||
|         sed -i 's|但二进制文件编译至 \[securityCheck\].*)|但已开源|g' README.md | ||||
|         sed -i 's|but binary files compiled in \[securityCheck\].*)|but open sourced|g' README_EN.md | ||||
|          | ||||
|         # 修改命令行帮助信息 | ||||
|         sed -i 's|security.*Enable/Disable security test (default true)|security        Enable/Disable security test (default false)|g' README.md | ||||
|         sed -i 's|security.*Enable/Disable security test (default true)|security        Enable/Disable security test (default false)|g' README_EN.md | ||||
|  | ||||
|     - name: Build and Test | ||||
|       run: | | ||||
|         env CGO_ENABLED=1 go build -o maintest | ||||
|         ./maintest -menu=false -l en -security=false -upload=false || exit 1 | ||||
|         rm -rf maintest | ||||
|          | ||||
|     - name: Commit and push changes | ||||
|       run: | | ||||
|         git add . | ||||
|         git commit -m "Auto update public version (no security package)" || echo "No changes to commit" | ||||
|         git push -f origin public | ||||
							
								
								
									
										82
									
								
								.goreleaser.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										82
									
								
								.goreleaser.yaml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,82 @@ | ||||
| before: | ||||
|   hooks: | ||||
|     - go mod tidy -v | ||||
| builds: | ||||
|   - id: universal | ||||
|     env: | ||||
|       - CGO_ENABLED=0 | ||||
|     ldflags: | ||||
|       - -s -w -X main.version={{.Version}} -X main.arch={{.Arch}} -checklinkname=0 | ||||
|     goos: | ||||
|       - linux | ||||
|       - windows | ||||
|       - freebsd | ||||
|     goarch: | ||||
|       - arm | ||||
|       - arm64 | ||||
|       - 386 | ||||
|       - amd64 | ||||
|       - mips | ||||
|       - mipsle | ||||
|       - mips64 | ||||
|       - mips64le | ||||
|       - ppc64 | ||||
|       - ppc64le | ||||
|       - s390x | ||||
|       - riscv64 | ||||
|     gomips: | ||||
|       - softfloat | ||||
|     ignore: | ||||
|       - goos: windows | ||||
|         goarch: arm | ||||
|     main: ./ | ||||
|     binary: goecs | ||||
|   - id: darwin-amd64 | ||||
|     env: | ||||
|       - CGO_ENABLED=1 | ||||
|       - CC=o64-clang | ||||
|       - CXX=o64-clang++ | ||||
|     ldflags: | ||||
|       - -s -w -X main.version={{.Version}} -X main.arch={{.Arch}} -checklinkname=0 | ||||
|     goos: | ||||
|       - darwin | ||||
|     goarch: | ||||
|       - amd64 | ||||
|     main: ./ | ||||
|     binary: goecs | ||||
|   - id: darwin-arm64 | ||||
|     env: | ||||
|       - CGO_ENABLED=1 | ||||
|       - CC=oa64-clang | ||||
|       - CXX=oa64-clang++ | ||||
|     ldflags: | ||||
|       - -s -w -X main.version={{.Version}} -X main.arch={{.Arch}} -checklinkname=0 | ||||
|     goos: | ||||
|       - darwin | ||||
|     goarch: | ||||
|       - arm64 | ||||
|     main: ./ | ||||
|     binary: goecs | ||||
| universal_binaries: | ||||
|   - name_template: "goecs" | ||||
|     replace: false | ||||
| checksum: | ||||
|   name_template: "checksums.txt" | ||||
| snapshot: | ||||
|   name_template: "goecs" | ||||
| archives: | ||||
|   - name_template: "goecs_{{ .Os }}_{{ .Arch }}" | ||||
|     format: zip | ||||
|     files: | ||||
|       - none* | ||||
| changelog: | ||||
|   sort: asc | ||||
|   filters: | ||||
|     exclude: | ||||
|       - "^docs:" | ||||
|       - "^test:" | ||||
|       - "^chore" | ||||
|       - Merge pull request | ||||
|       - Merge branch | ||||
|       - go mod tidy | ||||
|       - New translations | ||||
							
								
								
									
										64
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										64
									
								
								README.md
									
									
									
									
									
								
							| @@ -1,14 +1,14 @@ | ||||
| # ECS | ||||
|  | ||||
| [](https://github.com/oneclickvirt/ecs/actions/workflows/main.yaml) | ||||
| [](https://github.com/oneclickvirt/ecs/actions/workflows/build_binary.yaml) | ||||
|  | ||||
| [](https://app.fossa.com/projects/git%2Bgithub.com%2Foneclickvirt%2Fecs?ref=badge_shield) | ||||
|  | ||||
| [](https://hits.spiritlhl.net) | ||||
| [](https://hits.spiritlhl.net) [](https://github.com/oneclickvirt/ecs/releases) | ||||
|  | ||||
| 融合怪测评项目 - GO版本 | ||||
|  | ||||
| (环境安装[非必须]使用shell外无额外shell文件依赖,环境安装只是为了测的更准,极端情况下无环境依赖安装也可全测项目) | ||||
| (仅环境安装[非必须]使用shell外无额外shell文件依赖,环境安装只是为了测的更准,极端情况下无环境依赖安装也可全测项目) | ||||
|  | ||||
| 如有问题请 [issues](https://github.com/oneclickvirt/ecs/issues) 反馈。 | ||||
|  | ||||
| @@ -28,23 +28,25 @@ Shell 版本:[https://github.com/spiritLHLS/ecs](https://github.com/spiritLHLS | ||||
|  | ||||
| ### **编译与测试支持情况** | ||||
| | 编译支持的架构             | 测试支持的架构 | 编译支持的系统             | 测试支持的系统 | | ||||
| |---------------------------|-----------|---------|-----------| | ||||
| |---------------------------|--------------|---------------------------|---------------| | ||||
| | amd64                     | amd64        | Linux                     | Linux         | | ||||
| | arm                       |           | Windows | Windows   | | ||||
| | arm64                     | arm64     | FreeBSD |           | | ||||
| | 386                       |           | OpenBSD |           | | ||||
| | mips                      |           | MacOS   | MacOS     | | ||||
| | mipsle                    |           |         |           | | ||||
| | arm64                     | arm64        | Windows                   | Windows       | | ||||
| | arm                       |              | MacOS(Darwin)             | MacOS         | | ||||
| | 386                       |              | FreeBSD                   |               | | ||||
| | mips,mipsle               |              | Android                   |               | | ||||
| | mips64,mips64le           |              |                           |               |  | ||||
| | ppc64,ppc64le             |              |                           |               | | ||||
| | s390x                     | s390x        |                           |               | | ||||
| | riscv64                   |              |                           |               | | ||||
|  | ||||
|  | ||||
| > 更多架构与系统请自行测试,如有问题请开 issues。 | ||||
| > 更多架构与系统请自行测试或编译,如有问题请开 issues。 | ||||
|  | ||||
| ### **待支持的系统** | ||||
|  | ||||
| | 系统           | 说明                       | | ||||
| |-----|---------------------------| | ||||
| |----------------|---------------------------| | ||||
| | Android(arm64) | 存在权限问题未修复,非安卓系统的ARM架构无问题      | | ||||
| | OpenBSD/NetBSD | 部分Goalng的官方库未支持本系统(尤其是net相关项目)  | | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -52,13 +54,13 @@ Shell 版本:[https://github.com/spiritLHLS/ecs](https://github.com/spiritLHLS | ||||
|  | ||||
| - 系统基础信息查询,IP基础信息并发查询:[basics](https://github.com/oneclickvirt/basics)、[gostun](https://github.com/oneclickvirt/gostun) | ||||
| - CPU 测试:[cputest](https://github.com/oneclickvirt/cputest),支持 sysbench(lua/golang版本)、geekbench、winsat | ||||
| - 内存测试:[memorytest](https://github.com/oneclickvirt/memorytest),支持 sysbench、dd | ||||
| - 内存测试:[memorytest](https://github.com/oneclickvirt/memorytest),支持 sysbench、dd、winsat、mbw、stream | ||||
| - 硬盘测试:[disktest](https://github.com/oneclickvirt/disktest),支持 dd、fio、winsat | ||||
| - 流媒体解锁信息并发查询:[netflix-verify](https://github.com/sjlleo/netflix-verify) 等逻辑,开发至 [CommonMediaTests](https://github.com/oneclickvirt/CommonMediaTests) | ||||
| - 常见流媒体测试并发查询:[UnlockTests](https://github.com/oneclickvirt/UnlockTests),逻辑借鉴 [RegionRestrictionCheck](https://github.com/lmc999/RegionRestrictionCheck) 等 | ||||
| - IP 质量/安全信息并发查询:二进制文件编译至 [securityCheck](https://github.com/oneclickvirt/securityCheck) | ||||
| - 邮件端口测试:[portchecker](https://github.com/oneclickvirt/portchecker) | ||||
| - 三网回程测试:借鉴 [zhanghanyun/backtrace](https://github.com/zhanghanyun/backtrace),二次开发至 [oneclickvirt/backtrace](https://github.com/oneclickvirt/backtrace) | ||||
| - 上游及回程路由线路检测:借鉴 [zhanghanyun/backtrace](https://github.com/zhanghanyun/backtrace),二次开发至 [oneclickvirt/backtrace](https://github.com/oneclickvirt/backtrace) | ||||
| - 三网路由测试:基于 [NTrace-core](https://github.com/nxtrace/NTrace-core),二次开发至 [nt3](https://github.com/oneclickvirt/nt3) | ||||
| - 网速测试:基于 [speedtest.net](https://github.com/spiritLHLS/speedtest.net-CN-ID) 和 [speedtest.cn](https://github.com/spiritLHLS/speedtest.cn-CN-ID) 数据,开发至 [oneclickvirt/speedtest](https://github.com/oneclickvirt/speedtest) | ||||
| - 三网 Ping 值测试:借鉴 [ecsspeed](https://github.com/spiritLHLS/ecsspeed),二次开发至 [pingtest](https://github.com/oneclickvirt/pingtest) | ||||
| @@ -70,7 +72,7 @@ Shell 版本:[https://github.com/spiritLHLS/ecs](https://github.com/spiritLHLS | ||||
|  | ||||
| ## **使用说明** | ||||
|  | ||||
| ### **Linux/FreeBSD/OpenBSD/MacOS** | ||||
| ### **Linux/FreeBSD/MacOS** | ||||
|  | ||||
| #### **一键命令** | ||||
|  | ||||
| @@ -79,25 +81,25 @@ Shell 版本:[https://github.com/spiritLHLS/ecs](https://github.com/spiritLHLS | ||||
| - **国际用户无加速:** | ||||
|  | ||||
|   ```bash | ||||
|   export noninteractive=true && curl -L https://raw.githubusercontent.com/oneclickvirt/ecs/master/goecs.sh -o goecs.sh && chmod +x goecs.sh && bash goecs.sh env && bash goecs.sh install && goecs | ||||
|   export noninteractive=true && curl -L https://raw.githubusercontent.com/oneclickvirt/ecs/master/goecs.sh -o goecs.sh && chmod +x goecs.sh && ./goecs.sh env && ./goecs.sh install && goecs | ||||
|   ``` | ||||
|  | ||||
| - **国际/国内使用 CDN 加速:** | ||||
|  | ||||
|   ```bash | ||||
|   export noninteractive=true && curl -L https://cdn.spiritlhl.net/https://raw.githubusercontent.com/oneclickvirt/ecs/master/goecs.sh -o goecs.sh && chmod +x goecs.sh && bash goecs.sh env && bash goecs.sh install && goecs | ||||
|   export noninteractive=true && curl -L https://cdn.spiritlhl.net/https://raw.githubusercontent.com/oneclickvirt/ecs/master/goecs.sh -o goecs.sh && chmod +x goecs.sh && ./goecs.sh env && ./goecs.sh install && goecs | ||||
|   ``` | ||||
|  | ||||
| - **国内用户使用 CNB 加速:** | ||||
|  | ||||
|   ```bash | ||||
|   export noninteractive=true && curl -L https://cnb.cool/oneclickvirt/ecs/-/git/raw/main/goecs.sh -o goecs.sh && chmod +x goecs.sh && bash goecs.sh env && bash goecs.sh install && goecs | ||||
|   export noninteractive=true && curl -L https://cnb.cool/oneclickvirt/ecs/-/git/raw/main/goecs.sh -o goecs.sh && chmod +x goecs.sh && ./goecs.sh env && ./goecs.sh install && goecs | ||||
|   ``` | ||||
|  | ||||
| - **短链接:** | ||||
|  | ||||
|   ```bash | ||||
|   export noninteractive=true && curl -L https://bash.spiritlhl.net/goecs -o goecs.sh && chmod +x goecs.sh && bash goecs.sh env && bash goecs.sh install && goecs | ||||
|   export noninteractive=true && curl -L https://bash.spiritlhl.net/goecs -o goecs.sh && chmod +x goecs.sh && ./goecs.sh env && ./goecs.sh install && goecs | ||||
|   ``` | ||||
|  | ||||
| #### **详细说明** | ||||
| @@ -265,22 +267,24 @@ docker run --rm spiritlhl/goecs:latest -menu=false -l zh | ||||
|  | ||||
| 使用Docker执行测试,硬件测试会有一些偏差和虚拟化架构判断失效,还是推荐直接测试而不使用Docker测试。 | ||||
|  | ||||
| 国内镜像地址:https://cnb.cool/oneclickvirt/ecs/-/packages/docker/ecs | ||||
| 国内阿里云镜像加速 | ||||
|  | ||||
| 请确保执行下述命令前本机已安装Docker | ||||
|  | ||||
| 特权模式+host网络 | ||||
|  | ||||
| ```shell | ||||
| docker run --rm --privileged --network host docker.cnb.cool/oneclickvirt/ecs:latest -menu=false -l zh | ||||
| docker run --rm --privileged --network host crpi-8tmognxgyb86bm61.cn-guangzhou.personal.cr.aliyuncs.com/oneclickvirt/ecs:latest -menu=false -l zh | ||||
| ``` | ||||
|  | ||||
| 非特权模式+非host网络 | ||||
|  | ||||
| ```shell | ||||
| docker run --rm docker.cnb.cool/oneclickvirt/ecs:latest -menu=false -l zh | ||||
| docker run --rm crpi-8tmognxgyb86bm61.cn-guangzhou.personal.cr.aliyuncs.com/oneclickvirt/ecs:latest -menu=false -l zh | ||||
| ``` | ||||
|  | ||||
| 实际上还有CNB镜像地址 https://cnb.cool/oneclickvirt/ecs/-/packages/docker/ecs 但很可惜组织空间不足无法推送了,更推荐使用阿里云镜像加速 | ||||
|  | ||||
| </details> | ||||
|  | ||||
| --- | ||||
| @@ -297,20 +301,18 @@ cd ecs | ||||
| ``` | ||||
|  | ||||
| 2. 安装 Go 环境(如已安装可跳过) | ||||
|  | ||||
| 选择 go 1.24.5 的版本进行安装 | ||||
|  | ||||
| ```bash | ||||
| # 下载并安装 Go | ||||
| wget https://go.dev/dl/go1.23.4.linux-amd64.tar.gz | ||||
| rm -rf /usr/local/go && tar -C /usr/local -xzf go1.23.4.linux-amd64.tar.gz | ||||
| export PATH=$PATH:/usr/local/go/bin | ||||
| curl -L https://cdn.spiritlhl.net/https://raw.githubusercontent.com/spiritLHLS/one-click-installation-script/main/install_scripts/go.sh -o go.sh && chmod +x go.sh && bash go.sh  | ||||
| ``` | ||||
|  | ||||
| 3. 编译 | ||||
| ```bash | ||||
| env CGO_ENABLED=1 go build -o goecs | ||||
| go build -o goecs | ||||
| ``` | ||||
|  | ||||
| CGO在某些系统和架构中可能需要设置为0,请都进行尝试直到编译成功无报错。 | ||||
|  | ||||
| 4. 运行测试 | ||||
| ```bash | ||||
| ./goecs -menu=false -l zh | ||||
| @@ -395,6 +397,10 @@ GOOS=darwin GOARCH=amd64 go build -o goecs_darwin | ||||
|   <img src="https://console.zmto.com/templates/2019/dist/images/logo_dark.svg" alt="zmto" style="height: 50px;"> | ||||
| </a> | ||||
|  | ||||
| ## History Usage | ||||
|  | ||||
|  | ||||
|  | ||||
| ## Stargazers over time | ||||
|  | ||||
| [](https://www.spiritlhl.net) | ||||
|   | ||||
							
								
								
									
										54
									
								
								README_EN.md
									
									
									
									
									
								
							
							
						
						
									
										54
									
								
								README_EN.md
									
									
									
									
									
								
							| @@ -1,8 +1,10 @@ | ||||
| # ecs | ||||
| # ECS | ||||
|  | ||||
| [](https://github.com/oneclickvirt/ecs/actions/workflows/main.yaml)  | ||||
| [](https://github.com/oneclickvirt/ecs/actions/workflows/build_binary.yaml) | ||||
|  | ||||
| [](https://hits.spiritlhl.net) | ||||
| [](https://app.fossa.com/projects/git%2Bgithub.com%2Foneclickvirt%2Fecs?ref=badge_shield) | ||||
|  | ||||
| [](https://hits.spiritlhl.net) [](https://github.com/oneclickvirt/ecs/releases) | ||||
|  | ||||
| Fusion Monster Evaluation Project - GO Version | ||||
|  | ||||
| @@ -28,27 +30,31 @@ Shell version: [https://github.com/spiritLHLS/ecs/blob/main/README_EN.md](https: | ||||
| | Supported for Compilation | Tested on | Supported OS for Compilation | Tested OS | | ||||
| |---------------------------|-----------|------------------------------|-----------| | ||||
| | amd64                     | amd64     | Linux                        | Linux     | | ||||
| | arm                       |           | Windows                      | Windows   | | ||||
| | arm64                     | arm64     | FreeBSD                      |           | | ||||
| | 386                       |           | OpenBSD                      |           | | ||||
| | mips                      |           | MacOS(darwin)                | MacOS     | | ||||
| | mipsle                    |           |                              |           | | ||||
| | arm64                     | arm64     | Windows                      | Windows   | | ||||
| | arm                       |           | MacOS(Darwin)                | MacOS     | | ||||
| | 386                       |           | FreeBSD                      |           | | ||||
| | mips,mipsle               |           | Android                      |           | | ||||
| | mips64,mips64le           |           |                              |           |  | ||||
| | ppc64,ppc64le             |           |                              |           | | ||||
| | s390x                     | s390x     |                              |           | | ||||
| | riscv64                   |           |                              |           | | ||||
|  | ||||
| > Please test additional architectures and systems yourself. If you encounter any issues, please open an issue. | ||||
| > For more information about the architecture and system, please test or compile it yourself, and open issues if you have any questions. | ||||
|  | ||||
| ### **Systems Pending Support** | ||||
|  | ||||
| | OS     | Notes                                                                                           | | ||||
| |--------|-------------------------------------------------------------------------------------------------| | ||||
| | Android(arm64) | Permission issues that are not fixed, no problems with ARM architecture for non-Android systems    | | ||||
| | OpenBSD/NetBSD | Some of Goalng's official libraries do not support this system (especially net-related items) | | ||||
|  | ||||
| --- | ||||
|  | ||||
| ## **Features** | ||||
|  | ||||
| - System basic information query and concurrent IP basic information query: Self-developed [basics](https://github.com/oneclickvirt/basics), [gostun](https://github.com/oneclickvirt/gostun) | ||||
| - CPU test: Self-developed [cputest](https://github.com/oneclickvirt/cputest) supporting sysbench(lua/golang version), geekbench, winsat | ||||
| - Memory test: Self-developed [memorytest](https://github.com/oneclickvirt/memorytest) supporting sysbench, dd | ||||
| - Memory test: Self-developed [memorytest](https://github.com/oneclickvirt/memorytest) supporting sysbench, dd, winsat, mbw, stream | ||||
| - Disk test: Self-developed [disktest](https://github.com/oneclickvirt/disktest) supporting dd, fio, winsat | ||||
| - Streaming media unlock information concurrent query: Modified from [netflix-verify](https://github.com/sjlleo/netflix-verify) and more to [CommonMediaTests](https://github.com/oneclickvirt/CommonMediaTests) | ||||
| - Common streaming media tests concurrent query: Self-developed to [UnlockTests](https://github.com/oneclickvirt/UnlockTests), logic modified from [RegionRestrictionCheck](https://github.com/lmc999/RegionRestrictionCheck) and others | ||||
| @@ -66,7 +72,7 @@ Shell version: [https://github.com/spiritLHLS/ecs/blob/main/README_EN.md](https: | ||||
|  | ||||
| ## **Instructions for Use** | ||||
|  | ||||
| ### **Linux/FreeBSD/OpenBSD/MacOS** | ||||
| ### **Linux/FreeBSD/MacOS** | ||||
|  | ||||
| #### **One-click command** | ||||
|  | ||||
| @@ -75,19 +81,19 @@ Shell version: [https://github.com/spiritLHLS/ecs/blob/main/README_EN.md](https: | ||||
| - **International users without acceleration:** | ||||
|  | ||||
|   ```bash | ||||
|   export noninteractive=true && curl -L https://raw.githubusercontent.com/oneclickvirt/ecs/master/goecs.sh -o goecs.sh && chmod +x goecs.sh && bash goecs.sh env && bash goecs.sh install && goecs -l en | ||||
|   export noninteractive=true && curl -L https://raw.githubusercontent.com/oneclickvirt/ecs/master/goecs.sh -o goecs.sh && chmod +x goecs.sh && ./goecs.sh env && ./goecs.sh install && goecs | ||||
|   ``` | ||||
|  | ||||
| - **International/domestic users with CDN acceleration:** | ||||
|  | ||||
|   ```bash | ||||
|   export noninteractive=true && curl -L https://cdn.spiritlhl.net/https://raw.githubusercontent.com/oneclickvirt/ecs/master/goecs.sh -o goecs.sh && chmod +x goecs.sh && bash goecs.sh env && bash goecs.sh install && goecs -l en | ||||
|   export noninteractive=true && curl -L https://cdn.spiritlhl.net/https://raw.githubusercontent.com/oneclickvirt/ecs/master/goecs.sh -o goecs.sh && chmod +x goecs.sh && ./goecs.sh env && ./goecs.sh install && goecs | ||||
|   ``` | ||||
|  | ||||
| - **Domestic users with CNB acceleration:** | ||||
|  | ||||
|   ```bash | ||||
|   export noninteractive=true && curl -L https://cnb.cool/oneclickvirt/ecs/-/git/raw/main/goecs.sh -o goecs.sh && chmod +x goecs.sh && bash goecs.sh env && bash goecs.sh install && goecs -l en | ||||
|   export noninteractive=true && curl -L https://cnb.cool/oneclickvirt/ecs/-/git/raw/main/goecs.sh -o goecs.sh && chmod +x goecs.sh && ./goecs.sh env && ./goecs.sh install && goecs | ||||
|   ``` | ||||
|  | ||||
| - **Short Link:** | ||||
| @@ -292,20 +298,18 @@ cd ecs | ||||
| ``` | ||||
|  | ||||
| 2. Install Go environment (skip if already installed) | ||||
|  | ||||
| Select go 1.24.5 version to install  | ||||
|  | ||||
| ```bash | ||||
| # Download and install Go | ||||
| wget https://go.dev/dl/go1.23.4.linux-amd64.tar.gz | ||||
| rm -rf /usr/local/go && tar -C /usr/local -xzf go1.23.4.linux-amd64.tar.gz | ||||
| export PATH=$PATH:/usr/local/go/bin | ||||
| curl -L https://cdn.spiritlhl.net/https://raw.githubusercontent.com/spiritLHLS/one-click-installation-script/main/install_scripts/go.sh -o go.sh && chmod +x go.sh && bash go.sh  | ||||
| ``` | ||||
|  | ||||
| 3. Compile | ||||
| ```bash | ||||
| env CGO_ENABLED=1 go build -o goecs | ||||
| go build -o goecs | ||||
| ``` | ||||
|  | ||||
| CGO may need to be set to 0 in some systems and architectures, please try them all until they compile successfully without errors. | ||||
|  | ||||
| 4. Run test | ||||
| ```bash | ||||
| ./goecs -menu=false -l en | ||||
| @@ -340,6 +344,7 @@ GOOS=darwin GOARCH=amd64 go build -o goecs_darwin | ||||
| | Test stability | Core test components unchanged for 10+ years | Each major version updates test items, making scores hard to compare between versions (each version benchmarks against current best CPUs) | | ||||
| | Test content | Only tests computing performance | Covers multiple performance aspects with weighted scores, though some tests aren't commonly used | | ||||
| | Suitable scenarios | Good for quick tests, focuses on computing performance | Good for comprehensive testing | | ||||
| | Ranking         | [sysbench.spiritlhl.net](https://sysbench.spiritlhl.net/) | [browser.geekbench.com](https://browser.geekbench.com/) | | ||||
|  | ||||
| Note that `goecs` allows you to specify CPU test method via parameters. The default is chosen for faster testing across more systems. | ||||
|  | ||||
| @@ -389,6 +394,13 @@ Thanks also to the following platforms for editorial and testing support | ||||
|   <img src="https://console.zmto.com/templates/2019/dist/images/logo_dark.svg" alt="zmto" style="height: 50px;"> | ||||
| </a> | ||||
|  | ||||
| ## History Usage | ||||
|  | ||||
|  | ||||
|  | ||||
| ## Stargazers over time | ||||
|  | ||||
| [](https://www.spiritlhl.net) | ||||
|  | ||||
| ## License | ||||
| [](https://app.fossa.com/projects/git%2Bgithub.com%2Foneclickvirt%2Fecs?ref=badge_large) | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| ## 目录 / Table of Contents / 目次 | ||||
|  | ||||
| [](https://hits.spiritlhl.net) | ||||
| [](https://hits.spiritlhl.net) [](https://github.com/oneclickvirt/ecs/releases) | ||||
|  | ||||
| ## 语言 / Languages / 言語 | ||||
| - [中文](#中文) | ||||
| @@ -15,7 +15,7 @@ | ||||
| - [流媒体解锁](#流媒体解锁) | ||||
| - [IP质量检测](#IP质量检测) | ||||
| - [邮件端口检测](#邮件端口检测) | ||||
| - [三网回城线路检测](#三网回城线路检测) | ||||
| - [上游及回程线路检测](#上游及回程线路检测) | ||||
| - [三网回程路由检测](#三网回程路由检测) | ||||
| - [就近测速](#就近测速) | ||||
|  | ||||
| @@ -27,6 +27,7 @@ | ||||
| - [Streaming Media Unlocking](#Streaming-Media-Unlocking) | ||||
| - [IP Quality Detection](#IP-Quality-Detection) | ||||
| - [Email Port Detection](#Email-Port-Detection) | ||||
| - [Nearby Speed Testing](#Nearby-Speed-Testing) | ||||
|  | ||||
| ## 日本語 | ||||
| - [システム基本情報](#システム基本情報) | ||||
| @@ -36,6 +37,7 @@ | ||||
| - [ストリーミングメディアロック解除](#ストリーミングメディアロック解除) | ||||
| - [IP品質検出](#IP品質検出) | ||||
| - [メールポート検出](#メールポート検出) | ||||
| - [近隣スピードテスト](#近隣スピードテスト) | ||||
|  | ||||
| --- | ||||
|  | ||||
| @@ -43,23 +45,32 @@ | ||||
|  | ||||
| ### **系统基础信息** | ||||
|  | ||||
| CPU型号: 不必多说,大概的说,按CPU的发布时间,都是新款则AMD好于Intel,都是旧款则Intel好于AMD。 | ||||
| 依赖项目:[https://github.com/oneclickvirt/basics](https://github.com/oneclickvirt/basics) [https://github.com/oneclickvirt/gostun](https://github.com/oneclickvirt/gostun) | ||||
|  | ||||
| CPU数量: 会检测是物理核心还是逻辑核心,优先展示物理核心,查不到物理核心才去展示逻辑核心。在服务器实际使用过程中,程序一般是按逻辑核心分配执行的,非视频转码和科学计算,物理核心一般都是开超线程成逻辑核心用,横向比较的时候,对应类型的核心数量才有比较的意义。 | ||||
| CPU型号: 一般来说,按CPU的发布时间,都是新款则AMD好于Intel,都是旧款则Intel好于AMD,而Apple的M系列芯片则是断层式领先。 | ||||
|  | ||||
| CPU缓存:显示的宿主机的CPU三级缓存信息。 | ||||
| CPU数量: 会检测是物理核心还是逻辑核心,优先展示物理核心,查不到物理核心才去展示逻辑核心。在服务器实际使用过程中,程序一般是按逻辑核心分配执行的,非视频转码和科学计算,物理核心一般都是开超线程成逻辑核心用,横向比较的时候,对应类型的核心数量才有比较的意义。当然如果一个是物理核一个是虚拟核,在后面CPU测试得分类似的情况下,肯定是物理核更优,无需担忧CPU性能被共享的问题。 | ||||
|  | ||||
| AES-NI: 指令集是加密解密加速用的,有的话常规网络请求会更快一些,性能更高一些,没有的话会影响网络请求(含代理用途)。 | ||||
| CPU缓存:显示的宿主机的CPU三级缓存信息。对普通应用可能影响不大,但对数据库、编译、大规模并发请求等场景,L3 大小显著影响性能。 | ||||
|  | ||||
| AES-NI: AES指令集是加密解密加速用的,对于HTTPS(TLS/SSL)网络请求、VPN代理(配置使用的AES加密的)、磁盘加密这些场景有明显的优化,更快更省资源。 | ||||
|  | ||||
| VM-x/AMD-V/Hyper-V: 是当前测试宿主机是否支持嵌套虚拟化的指标,如果测试环境是套在docker里测或者没有root权限,那么这个默认就是检测不到显示不支持嵌套虚拟化。这个指标在你需要在宿主机上开设虚拟机(如 KVM、VirtualBox、VMware)的时候有用,其他用途该指标用处不大。 | ||||
|  | ||||
| 内存: 显示内存 正在使用的大小/总大小 ,不含虚拟内存。 | ||||
|  | ||||
| 气球驱动: 显示宿主机是否使用了气球驱动,使用了证明母机有共享内存使用,需要结合下面的内存读写测试查看是否有超售/严格的限制。 | ||||
| 气球驱动: 显示宿主机是否启用了气球驱动。气球驱动用于宿主机和虚拟机之间动态调节内存分配:宿主机可以通过驱动要求虚拟机“放气”释放部分内存,或“充气”占用更多内存。启用它通常意味着宿主机具备内存超售能力,但是否真的存在超售,需要结合下面的内存读写测试查看是否有超售/严格的限制。 | ||||
|  | ||||
| 内核页合并:显示宿主机是否使用了KSM内存融合,使用了证明母机有共享内存使用,需要结合下面的内存读写测试查看是否有超售/严格的限制。 | ||||
| 内核页合并:显示宿主机是否启用了内核页合并机制。KSM 会将多个进程中内容相同的内存页合并为一份,以减少物理内存占用。启用它通常意味着宿主机可能在进行内存节省或存在一定程度的内存超售。是否真正造成性能影响或内存紧张,需要结合下面的内存读写测试查看是否有超售/严格的限制。 | ||||
|  | ||||
| 虚拟内存: 显示 SWAP虚拟内存 | ||||
| 虚拟内存: swap虚拟内存 是磁盘上划出的虚拟内存空间,用来在物理内存不足时临时存放数据。它能防止内存不足导致程序崩溃,但频繁使用会明显拖慢系统,Linux 官方推荐的 swap 配置如下: | ||||
|  | ||||
| | 物理内存大小             | 推荐 SWAP 大小 | | ||||
| | ------------------ | ---------- | | ||||
| | ≤ 2G               | 内存的 2 倍    | | ||||
| | 2G < 内存 ≤ 8G       | 等于物理内存大小   | | ||||
| | ≥ 8G               | 约 8G 即可    | | ||||
| | 需要休眠 (hibernation) | 至少等于物理内存大小 | | ||||
|  | ||||
| 硬盘空间: 显示硬盘 正在使用的大小/总大小 | ||||
|  | ||||
| @@ -69,15 +80,15 @@ VM-x/AMD-V/Hyper-V: 是当前测试宿主机是否支持嵌套虚拟化的指标 | ||||
|  | ||||
| 内核: 显示系统内核版本 | ||||
|  | ||||
| 系统在线时间: 显示宿主机自从开机到测试时已在线时长 | ||||
| 系统在线时间: 显示宿主机自从开机到测试时的已在线时长 | ||||
|  | ||||
| 时区: 显示宿主机系统时区 | ||||
|  | ||||
| 负载: 显示系统负载 | ||||
|  | ||||
| 虚拟化架构: 显示宿主机来自什么虚拟化架构,一般来说推荐```Dedicated > KVM > Xen```虚拟化,其他虚拟化都会存在性能损耗,导致使用的时候存在性能共享/损耗,但这个也说不准,独立服务器才拥有完全独立的资源占用,其他虚拟化基本都会有资源共享,取决于宿主机的售卖者是否有良心,具体性能优劣还是得看后面的专项测试。 | ||||
| 虚拟化架构: 显示宿主机来自什么虚拟化架构,一般来说推荐```Dedicated > KVM > Xen```虚拟化,其他虚拟化都会存在性能损耗,导致使用的时候存在性能共享/损耗,但这个也说不准,独立服务器才拥有完全独立的资源占用,其他虚拟化基本都会有资源共享,取决于宿主机的持有者对这个虚拟机是否有良心,具体性能优劣还是得看后面的专项性能测试。 | ||||
|  | ||||
| NAT类型: 显示NAT类型,具体推荐```Full Cone > Restricted Cone > Port Restricted Cone > Symmetric```,测不出来时会显示```Inconclusive```,一般来说不拿来做特殊用途(有关于特殊的代理和实时通讯需求的),都不用关注本指标。 | ||||
| NAT类型: 显示NAT类型,具体推荐```Full Cone > Restricted Cone > Port Restricted Cone > Symmetric```,测不出来或者非正规协议的类型会显示```Inconclusive```,一般来说只有特殊用途,比如有特殊的代理、实时通讯、做FRP内穿端口等需求才需要特别关注,其他一般情况下都不用关注本指标。 | ||||
|  | ||||
| TCP加速方式:一般是```cubic/bbr```拥塞控制协议,一般来说做代理服务器用bbr可以改善网速,普通用途不必关注此指标。 | ||||
|  | ||||
| @@ -85,12 +96,14 @@ IPV4/IPV6 ASN: 显示宿主机IP所属的ASN组织ID和名字,同一个IDC可 | ||||
|  | ||||
| IPV4/IPV6 Location: 显示对应协议的IP在数据库中的地理位置。 | ||||
|  | ||||
| IPV4 Active IPs: 根据 bgp.tools 信息查询当前CIDR分块中 活跃邻居数量/总邻居数量 | ||||
| IPV4 Active IPs: 根据 bgp.tools 信息查询当前CIDR分块中 活跃邻居数量/总邻居数量 由于是非实时的,可能存在延迟。 | ||||
|  | ||||
| IPV6 子网掩码:根据宿主机信息查询的本机IPV6子网大小 | ||||
|  | ||||
| ### **CPU测试** | ||||
|  | ||||
| 依赖项目:[https://github.com/oneclickvirt/cputest](https://github.com/oneclickvirt/cputest) | ||||
|  | ||||
| 支持通过命令行参数选择```GeekBench```和```Sysbench```进行测试: | ||||
|  | ||||
| | 比较项             | sysbench | geekbench | | ||||
| @@ -101,6 +114,7 @@ IPV6 子网掩码:根据宿主机信息查询的本机IPV6子网大小 | ||||
| | 测试稳定性       | 核心测试组件10年以上未变 | 每个大版本更新测试项,分数不同版本间难以对比(每个版本对标当前最好的CPU) | | ||||
| | 测试内容         | 仅测试计算性能,基于素数计算 | 覆盖多种性能测试,分数加权计算,但部分测试实际不常用 | | ||||
| | 适用场景         | 适合快速测试,仅测试计算性能 | 适合综合全面的测试 | | ||||
| | 排行榜         | [sysbench.spiritlhl.net](https://sysbench.spiritlhl.net/) | [browser.geekbench.com](https://browser.geekbench.com/) | | ||||
|  | ||||
| 默认使用```Sysbench```进行测试,基准大致如下: | ||||
|  | ||||
| @@ -116,16 +130,51 @@ AMD的7950x单核满血性能得分在6500左右,AMD的5950x单核满血性能 | ||||
|  | ||||
| 多说一句,```GeekBench```测的很多内容,实际在服务器使用过程中根本用不到,测试仅供参考。当然```Sysbench```非常不全面,但它基于最基础的计算性能可以大致比较CPU的性能。 | ||||
|  | ||||
| 实际上CPU性能测试够用就行,除非是科学计算以及视频转码,一般不需要特别追求高性能CPU。 | ||||
| 实际上CPU性能测试够用就行,除非是科学计算以及视频转码,一般不需要特别追求高性能CPU。如果有性能需求,那么需要关注程序本身吃的是多核还是单核,对应看多核还是单核得分。 | ||||
|  | ||||
| ### **内存测试** | ||||
|  | ||||
| 一般来说,只需要判断IO速度是否低于```10240MB/s```,如果低于这个值那么证明内存性能不佳,极大概率存在超售超卖问题。 | ||||
| 依赖项目:[https://github.com/oneclickvirt/memorytest](https://github.com/oneclickvirt/memorytest) | ||||
|  | ||||
| 至于超开的原因可能是开了虚拟内存(硬盘当内存用)、可能是开了ZRAM(牺牲CPU性能)、可能是开了气球驱动、可能是开了KSM内存融合,原因多种多样。 | ||||
| 一般来说,只需要判断 IO 速度是否低于 `10240 MB/s (≈10 GB/s)` | ||||
|  | ||||
| 如果低于这个值,那么证明内存性能不佳,极大概率存在超售超卖问题。 | ||||
|  | ||||
| 至于超开的原因可能是: | ||||
|  | ||||
| * 开了虚拟内存 (硬盘当内存用) | ||||
| * 开了 ZRAM (牺牲 CPU 性能) | ||||
| * 开了气球驱动 (Balloon Driver) | ||||
| * 开了 KSM 内存融合 | ||||
|  | ||||
| 原因多种多样。 | ||||
|  | ||||
| | 内存类型 | 典型频率 (MHz)   | 单通道带宽                                 | 双通道带宽                                   | | ||||
| | ---- | ------------ | ------------------------------------- | --------------------------------------- | | ||||
| | DDR3 | 1333 \~ 2133 | 10 \~ 17 GB/s (≈ 10240 \~ 17408 MB/s) | 20 \~ 34 GB/s (≈ 20480 \~ 34816 MB/s)   | | ||||
| | DDR4 | 2133 \~ 3200 | 17 \~ 25 GB/s (≈ 17408 \~ 25600 MB/s) | 34 \~ 50 GB/s (≈ 34816 \~ 51200 MB/s)   | | ||||
| | DDR5 | 4800 \~ 7200 | 38 \~ 57 GB/s (≈ 38912 \~ 58368 MB/s) | 76 \~ 114 GB/s (≈ 77824 \~ 116736 MB/s) | | ||||
|  | ||||
| 根据上表内容,本项目测试的粗略判断方法: | ||||
|  | ||||
| * **< 20 GB/s (20480 MB/s)** → 可能是 DDR3(或 DDR4 单通道 / 低频) | ||||
| * **20 \~ 40 GB/s (20480 \~ 40960 MB/s)** → 大概率 DDR4 | ||||
| * **≈ 50 GB/s (≈ 51200 MB/s)** → 基本就是 DDR5 | ||||
|  | ||||
| 对于各种测试参数对应方法的比较: | ||||
|  | ||||
| | 参数方法    | 测试准确性 | 测试速度 | 适应架构 | 依赖情况 | | ||||
| |------------|------------|----------|----------|----------|----------| | ||||
| | **stream** | 高 —— 结果稳定,更符合实际情况 | 快速 | 跨平台(Linux/Windows/Unix) | 自带依赖无需额外安装 | | ||||
| | **sysbench** | 高 —— 可测试 CPU、内存、IO、线程等,结果较可靠 | 中等 | 跨平台(Linux、Windows 部分支持) | 需环境额外安装 | | ||||
| | **winsat** | 中等偏高 —— Windows 内置工具,结果和环境相关 | 中等 | 仅限 Windows | 物理机器上自带,虚拟机不可用 | | ||||
| | **mbw**    | 中等 —— 结果可能受缓存/调度影响 | 非常快 | 跨平台(几乎所有类 Unix 系统) | 自带依赖无需额外安装 | | ||||
| | **dd**     | 低 —— 结果受缓存影响 | 快速 | 跨平台(几乎所有类 Unix 系统) | 需环境额外安装 | | ||||
|  | ||||
| ### **硬盘测试** | ||||
|  | ||||
| 依赖项目:[https://github.com/oneclickvirt/disktest](https://github.com/oneclickvirt/disktest) | ||||
|  | ||||
| ```dd```测试可能误差偏大但测试速度快无硬盘大小限制,```fio```测试真实一些但测试速度慢有硬盘以及内存大小的最低需求。 | ||||
|  | ||||
| 同时,服务器可能有不同的文件系统,某些文件系统的IO引擎在同样的硬件条件下测试的读写速度更快,这是正常的。项目默认使用```fio```进行测试,测试使用的IO引擎优先级为```libaio > posixaio > psync```,备选项```dd```测试在```fio```测试不可用时自动替换。 | ||||
| @@ -134,7 +183,7 @@ AMD的7950x单核满血性能得分在6500左右,AMD的5950x单核满血性能 | ||||
|  | ||||
| | 操作系统类型 | 主要指标 | 次要指标 | | ||||
| |---------|-------------------|---------------------| | ||||
| | Windows/Mac | 4K读 → 64K读 → 写入测试 | 图形界面系统优先考虑读取性能 | | ||||
| | Windows/MacOS | 4K读 → 64K读 → 写入测试 | 图形界面系统优先考虑读取性能 | | ||||
| | Linux (无图形界面) | 4K读 + 4K写 + 1M读写| 读/写值通常相似 | | ||||
|  | ||||
| 以下硬盘类型对于指标值指 常规~满血 性能状态,指```libaio```作为IO测试引擎,指在```Linux```下进行测试 | ||||
| @@ -168,14 +217,40 @@ AMD的7950x单核满血性能得分在6500左右,AMD的5950x单核满血性能 | ||||
|  | ||||
| ### **流媒体解锁** | ||||
|  | ||||
| 检索常见的流媒体平台解锁,当然也不全是流媒体,还有一些常见的别的平台的解锁也纳入了。一般来说,IP解锁地区都是一致的不会到处乱飘,如果发现多家平台解锁地区不一致,那么IP大概率是租赁的IPXO等平台的,各平台数据库识别缓慢,IP质量一般来说也好不到哪里去。 | ||||
| 依赖项目:[https://github.com/oneclickvirt/CommonMediaTests](https://github.com/oneclickvirt/CommonMediaTests) [https://github.com/oneclickvirt/UnlockTests](https://github.com/oneclickvirt/UnlockTests) | ||||
|  | ||||
| 默认只检测跨国流媒体解锁。 | ||||
|  | ||||
| 一般来说,正常的情况下,一个IP多个流媒体的解锁地区都是一致的不会到处乱飘,如果发现多家平台解锁地区不一致,那么IP大概率来自IPXO等平台租赁或者是刚刚宣告和被使用,未被流媒体普通的数据库所识别修正地域。由于各平台的IP数据库识别速度不一致,所以有时候有的平台解锁区域正常,有的飘到路由上的某个位置,有的飘到IP未被你使用前所在的位置。 | ||||
|  | ||||
| | DNS 类型       | 解锁方式判断是否必要 | DNS 对解锁影响 | 说明                                      | | ||||
| | ------------ | ---------- | --------- | --------------------------------------- | | ||||
| | 官方主流 DNS     | 否          | 小         | 流媒体解锁主要依赖节点 IP,DNS 解析基本不会干扰解锁。          | | ||||
| | 非主流 / 自建 DNS | 是          | 大         | 流媒体解锁结果受 DNS 解析影响较大,需要判断是原生解锁还是 DNS 解锁。 | | ||||
|  | ||||
| 所以测试过程中,如果宿主机当前使用的是官方主流的DNS,不会进行是否为原生解锁的判断。 | ||||
|  | ||||
| ### **IP质量检测** | ||||
|  | ||||
| 依赖项目:[https://github.com/oneclickvirt/securityCheck](https://github.com/oneclickvirt/securityCheck) | ||||
|  | ||||
| 检测14个数据库的IP相关信息,一般来说看使用类型和公司类型还有安全信息的其他判别足矣,安全得分真的图一乐。多个平台比较对应检测项目都为对应值,证明当前IP确实如此,不要仅相信一个数据库源的信息。 | ||||
|  | ||||
| * **使用类型 & 公司类型**:显示IP归属和使用场景,例如是否属于家庭用户、企业办公、托管服务或云/数据中心。 | ||||
| * **云提供商 / 数据中心 / 移动设备**:判断IP是否来自云服务、数据中心或移动网络,帮助识别共享或高风险IP。 | ||||
| * **代理 / VPN / Tor / Tor出口**:检测IP是否用于隐藏真实身份或位置,可能涉及匿名访问或滥用行为。 | ||||
| * **网络爬虫 / 机器人**:识别自动化访问或采集程序,对安全风险评估有参考价值。 | ||||
| * **匿名 / 滥用者 / 威胁 / 中继 / Bogon**:显示IP历史行为特征和是否属于保留/未分配IP,辅助判断IP可信度。 | ||||
| * **安全得分、声誉、信任得分、威胁得分、欺诈得分、滥用得分**:各数据库对IP的量化安全评价,仅供参考。 | ||||
| * **社区投票 & 黑名单记录**:展示用户反馈及公共黑名单信息,可快速识别潜在风险。 | ||||
| * **Google搜索可行性**:检测IP访问Google搜索服务的可行性,间接反映网络限制或屏蔽情况。 | ||||
|  | ||||
| 多平台对比更可靠,不同数据库算法和更新频率不同,单一来源可能存在误判。多个数据库显示相似结果,说明这个结果更可靠。 | ||||
|  | ||||
| ### **邮件端口检测** | ||||
|  | ||||
| 依赖项目:[https://github.com/oneclickvirt/portchecker](https://github.com/oneclickvirt/portchecker) | ||||
|  | ||||
| - **SMTP(25)**:用于邮件服务器之间传输邮件(发送邮件)。 | ||||
| - **SMTPS(465)**:用于加密的 SMTP 发送邮件(SSL/TLS 方式)。 | ||||
| - **SMTP(587)**:用于客户端向邮件服务器发送邮件,支持 STARTTLS 加密。 | ||||
| @@ -184,13 +259,43 @@ AMD的7950x单核满血性能得分在6500左右,AMD的5950x单核满血性能 | ||||
| - **IMAP(143)**:用于邮件客户端在线管理邮件(查看、同步邮件),不加密。 | ||||
| - **IMAPS(993)**:用于加密的 IMAP,安全地管理邮件(SSL/TLS 方式)。 | ||||
|  | ||||
| 具体当前宿主机不做邮局或者不收电子邮件,那么该项目指标不用理会。 | ||||
| 具体当前宿主机不做邮局,不收发电子邮件,那么该项目指标不需要理会。 | ||||
|  | ||||
| ### **三网回程线路检测** | ||||
| ### **上游及回程线路检测** | ||||
|  | ||||
| 检测当前的宿主机的IP地址 到 四个主要POP点城市的三个主要运营商的接入点的IP地址 的线路,具体来说 | ||||
| 依赖项目:[https://github.com/oneclickvirt/backtrace](https://github.com/oneclickvirt/backtrace) | ||||
|  | ||||
| 电信163、联通4837、移动CMI 是常见的线路 | ||||
| #### 上游类型与运营商等级说明 | ||||
|  | ||||
| - **直接上游(Direct Upstream)**   | ||||
|   当前运营商直接购买网络服务的上级运营商,通常是 BGP 邻居。 | ||||
|  | ||||
| - **间接上游(Indirect Upstream)**   | ||||
|   直接上游的上级,形成层层向上的关系链。可通过 BGP 路由路径中的多跳信息识别。 | ||||
|  | ||||
| | 等级 | 描述 | | ||||
| |------|------| | ||||
| | **Tier 1 Global** | 全球顶级运营商(如 AT&T、Verizon、NTT、Telia 等),之间免费互联(Settlement-Free Peering),不依赖他人即可访问全球任意网络。 | | ||||
| | **Tier 1 Regional** | 区域性顶级运营商,在特定区域具有一级能力,但在全球范围互联性稍弱。 | | ||||
| | **Tier 1 Indirect** | 间接连接的 Tier 1(非直接购买),通过中间上游间接接入 Tier 1 网络。 | | ||||
| | **Tier 2** | 需要向 Tier 1 付费购买上网能力的二级运营商,通常是各国主流电信商或 ISP。 | | ||||
| | **CDN Provider** | 内容分发网络提供商,如 Cloudflare、Akamai、Fastly 等,主要用于内容加速而非传统上游。 | | ||||
| | **Direct/Indirect Others** | 其他类型的直接或间接连接,如 IX(Internet Exchange)成员、私有对等互联等。 | | ||||
|  | ||||
| 上游质量判断:直接接入的高等级上游(特别是 Tier 1 Global)越多,通常网络连通性越好。但实际网络质量也受到以下因素影响: | ||||
|  | ||||
|   - 上下游之间的商业结算关系; | ||||
|   - 购买的带宽套餐和服务质量; | ||||
|   - 对等端口(Peering Ports)大小和负载; | ||||
|   - 网络拥塞、路由策略、延迟路径等。 | ||||
|  | ||||
| 无法完全从 BGP 路由中判断。 | ||||
|  | ||||
| 一般来说,**接入高质量上游越多,网络连通性越优**。但由于存在诸多不可见的商业和技术因素,**无法仅凭上游等级准确判断网络质量**,上游检测约等于图一乐,实际得看对应的路由情况和长时间Ping的情况。 | ||||
|  | ||||
| 然后是检测当前的宿主机的IP地址 到 四个主要POP点城市的三个主要运营商的接入点的IP地址 的线路,具体来说 | ||||
|  | ||||
| 电信163、联通4837、移动CMI 是常见的线路,移动CMI对两广地区的移动运营商特供延迟低,也能算优质,仅限两广移动。 | ||||
|  | ||||
| 电信CN2GIA > 电信CN2GT 移动CMIN2 联通9929 算优质的线路 | ||||
|  | ||||
| @@ -198,310 +303,484 @@ AMD的7950x单核满血性能得分在6500左右,AMD的5950x单核满血性能 | ||||
|  | ||||
| ### **三网回程路由检测** | ||||
|  | ||||
| 默认检测广州为目的地,实际可使用命令行参数指定目的地,见对应的说明。 | ||||
| 依赖项目:[https://github.com/oneclickvirt/nt3](https://github.com/oneclickvirt/nt3) | ||||
|  | ||||
| 主要就是看是不是直连,是不是延迟低,是不是没有隐藏路由信息。如果路由全球跑,延迟起飞,那么线路自然不会好到哪里去。 | ||||
| 默认检测广州为目的地,实际可使用命令行参数指定目的地,见对应的参数说明。 | ||||
|  | ||||
| 主要就是看是不是直连,是不是延迟低,是不是没有隐藏路由信息,有没有一些优质线路或IX链接。 | ||||
|  | ||||
| 如果路由全球跑,延迟起飞,那么线路自然不会好到哪里去。 | ||||
|  | ||||
| 有时候路由信息完全藏起来了,只知道实际使用的延迟低,实际可能也是优质线路只是查不到信息,这就没办法直接识别了。 | ||||
|  | ||||
| ### **就近测速** | ||||
|  | ||||
| 依赖项目:[https://github.com/oneclickvirt/speedtest](https://github.com/oneclickvirt/speedtest) | ||||
|  | ||||
| 先测的官方推荐的测速点,然后测有代表性的国际测速点,最后测国内三大运营商ping值最低的测速点。 | ||||
|  | ||||
| 境内使用为主就看境内测速即可,境外使用看境外测速,官方测速点可以代表受测的宿主机本地带宽基准。 | ||||
|  | ||||
| 一般来说境外的服务器的带宽100Mbps起步,境内的服务器1Mbps带宽起步,具体看线路优劣,带宽特别大有时候未必用得上,够用就行了。 | ||||
| 一般来说中国境外的服务器的带宽100Mbps起步,中国境内的服务器1Mbps带宽起步,具体看线路优劣,带宽特别大有时候未必用得上,够用就行了。 | ||||
|  | ||||
| 日常我偏向使用1Gbps带宽的服务器,至少下载依赖什么的速度足够快,境内小水管几Mbps真的下半天下不完,恨不得到机房插个U盘转移数据。 | ||||
|  | ||||
| --- | ||||
|  | ||||
| ## English | ||||
|  | ||||
| ### **Basic System Information** | ||||
| ### Basic System Information | ||||
|  | ||||
| CPU Model: Simply put, generally speaking, based on CPU release dates, newer AMD models are better than Intel, while for older models, Intel is better than AMD. | ||||
| Dependency project: [https://github.com/oneclickvirt/basics](https://github.com/oneclickvirt/basics) [https://github.com/oneclickvirt/gostun](https://github.com/oneclickvirt/gostun) | ||||
|  | ||||
| CPU Count: It will detect whether these are physical cores or logical cores, prioritizing display of physical cores, only showing logical cores if physical core information is unavailable. In actual server usage, programs are generally allocated by logical cores. Except for video transcoding and scientific computing, physical cores are usually enabled with hyperthreading to function as logical cores. When making comparisons, only cores of the corresponding type have meaningful comparison value. | ||||
| **CPU Model**: Generally speaking, based on CPU release time, newer models favor AMD over Intel, while older models favor Intel over AMD. Apple's M-series chips are in a league of their own. | ||||
|  | ||||
| CPU Cache: Displays the host machine's three-level CPU cache information. | ||||
| **CPU Count**: Detects whether cores are physical or logical, prioritizing physical cores display. Logical cores are shown only when physical cores cannot be detected. In actual server usage, programs are generally allocated based on logical cores. For non-video transcoding and scientific computing, physical cores usually have hyperthreading enabled to function as logical cores. When making comparisons, only corresponding core types are meaningful. Of course, if one is physical and the other is virtual, with similar CPU test scores, physical cores are definitely better without worrying about CPU performance sharing issues. | ||||
|  | ||||
| AES-NI: This instruction set is used for encryption/decryption acceleration. With it, normal network requests will be faster and performance will be higher. Without it, network requests (including proxy usage) will be affected. | ||||
| **CPU Cache**: Displays the host's CPU L3 cache information. While it may not significantly impact regular applications, for databases, compilation, and large-scale concurrent requests, L3 cache size significantly affects performance. | ||||
|  | ||||
| VM-x/AMD-V/Hyper-V: This indicates whether the current host machine supports nested virtualization. If the test environment is running inside Docker or doesn't have root privileges, then by default this will be undetectable and will show as not supporting nested virtualization. This metric is useful when you need to set up virtual machines (such as KVM, VirtualBox, VMware) on the host machine; for other purposes, this metric is not very useful. | ||||
| **AES-NI**: AES instruction set is used for encryption/decryption acceleration, providing significant optimization for HTTPS (TLS/SSL) network requests, VPN proxies (configured with AES encryption), and disk encryption scenarios - faster and more resource-efficient. | ||||
|  | ||||
| Memory: Displays memory size in format "currently used size/total size", not including virtual memory. | ||||
| **VM-x/AMD-V/Hyper-V**: Indicators showing whether the current test host supports nested virtualization. If the test environment runs in Docker or lacks root permissions, this will show as unsupported by default. This indicator is useful when you need to create virtual machines (like KVM, VirtualBox, VMware) on the host; otherwise, it has limited utility. | ||||
|  | ||||
| Balloon Driver: Shows whether the host machine is using a balloon driver. If used, it proves the parent machine has shared memory usage, which should be examined alongside the memory read/write test below to check for overselling/strict limitations. | ||||
| **Memory**: Shows memory usage as currently used size/total size, excluding virtual memory. | ||||
|  | ||||
| Kernel Same-page Merging: Shows whether the host machine is using KSM memory fusion. If used, it proves the parent machine has shared memory usage, which should be examined alongside the memory read/write test below to check for overselling/strict limitations. | ||||
| **Balloon Driver**: Shows whether the host has balloon driver enabled. Balloon driver is used for dynamic memory allocation between host and virtual machines: the host can request virtual machines to "deflate" and release some memory, or "inflate" to occupy more memory. Enabling it usually means the host has memory overselling capability, but whether actual overselling exists needs to be checked with memory read/write tests below for overselling/strict limitations. | ||||
|  | ||||
| Virtual Memory: Displays SWAP virtual memory. | ||||
| **Kernel Page Merging**: Shows whether the host has kernel page merging mechanism enabled. KSM merges memory pages with identical content from multiple processes into a single copy to reduce physical memory usage. Enabling it usually means the host may be implementing memory savings or has some degree of memory overselling. Whether it actually causes performance impact or memory pressure needs to be checked with memory read/write tests below for overselling/strict limitations. | ||||
|  | ||||
| Disk Space: Displays disk usage in format "currently used size/total size". | ||||
| **Virtual Memory**: Swap virtual memory is virtual memory space allocated on disk, used to temporarily store data when physical memory is insufficient. It prevents program crashes due to insufficient memory, but frequent use will noticeably slow down the system. Linux officially recommends swap configuration as follows: | ||||
|  | ||||
| Boot Disk Path: Shows the path of the boot disk. | ||||
| | Physical Memory Size | Recommended SWAP Size | | ||||
| | -------------------- | --------------------- | | ||||
| | ≤ 2G                | 2x memory size        | | ||||
| | 2G < memory ≤ 8G    | Equal to physical memory | | ||||
| | ≥ 8G                | About 8G is sufficient | | ||||
| | Hibernation needed  | At least equal to physical memory | | ||||
|  | ||||
| System: Displays system name and architecture. | ||||
| **Disk Space**: Shows disk usage as currently used size/total size | ||||
|  | ||||
| Kernel: Displays system kernel version. | ||||
| **Boot Disk Path**: Shows the boot disk path | ||||
|  | ||||
| System Uptime: Shows how long the host machine has been online since boot until testing time. | ||||
| **System**: Shows system name and architecture | ||||
|  | ||||
| Timezone: Displays the host machine's system timezone. | ||||
| **Kernel**: Shows system kernel version | ||||
|  | ||||
| Load: Displays system load. | ||||
| **System Uptime**: Shows host uptime from boot to test time | ||||
|  | ||||
| Virtualization Architecture: Shows what virtualization architecture the host machine uses. Generally speaking, the recommended order is `Dedicated > KVM > Xen` virtualization. Other virtualization will have performance losses, leading to shared/degraded performance during use. However, this is not definitive. Only dedicated servers have completely independent resource usage; other virtualization methods basically all have resource sharing, depending on whether the host machine seller has a conscience. The specific performance merits still depend on the specialized tests that follow. | ||||
| **Time Zone**: Shows host system time zone | ||||
|  | ||||
| NAT Type: Displays NAT type. Specifically recommended in order: ```Full Cone > Restricted Cone > Port Restricted Cone > Symmetric```. When not detectable, it will show ```Inconclusive```. Generally speaking, if you're not using it for special purposes (related to special proxy and real-time communication needs), you don't need to pay attention to this metric. | ||||
| **Load**: Shows system load | ||||
|  | ||||
| TCP Acceleration Method: Generally this is the ```cubic/bbr``` congestion control protocol. Generally speaking, using bbr for proxy servers can improve network speed; for ordinary purposes, you don't need to pay attention to this indicator. | ||||
| **Virtualization Architecture**: Shows what virtualization architecture the host comes from. Generally recommended: `Dedicated > KVM > Xen` virtualization. Other virtualization types have performance losses, causing performance sharing/loss during use. However, this isn't definitive - only dedicated servers have completely independent resource usage. Other virtualization basically involves resource sharing, depending on whether the host holder is conscientious about this virtual machine. Actual performance superiority still depends on subsequent specialized performance tests. | ||||
|  | ||||
| IPv4/IPv6 ASN: Displays the ASN organization ID and name that the host machine's IP belongs to. The same IDC may have multiple ASNs, and an ASN may have multiple vendors selling servers with different IP segments. The specific upstream and downstream relationships are complex and can be further viewed using bgp.tool. | ||||
| **NAT Type**: Shows NAT type. Specifically recommended: `Full Cone > Restricted Cone > Port Restricted Cone > Symmetric`. Undetectable or non-standard protocol types show as `Inconclusive`. Generally, only special purposes like specific proxies, real-time communication, or FRP port forwarding need special attention to this indicator; other general situations don't need to focus on this metric. | ||||
|  | ||||
| IPv4/IPv6 Location: Shows the geographic location of the corresponding protocol's IP in the database. | ||||
| **TCP Acceleration Method**: Usually `cubic/bbr` congestion control protocols. Generally speaking, using bbr for proxy servers can improve network speed; regular usage doesn't need to focus on this indicator. | ||||
|  | ||||
| IPV4 Active IPs: Query the number of active neighbours/total number of neighbours in the current CIDR chunk based on the bgp.tools information. | ||||
| **IPV4/IPV6 ASN**: Shows the ASN organization ID and name that the host IP belongs to. The same IDC may have multiple ASNs, and ASNs may have multiple merchants selling servers with different IP segments. The specific upstream and downstream relationships are complex; use bgp.tool for further investigation. | ||||
|  | ||||
| ### **CPU Testing** | ||||
| **IPV4/IPV6 Location**: Shows the geographic location of the corresponding protocol's IP in the database. | ||||
|  | ||||
| Supports selecting `GeekBench` and `Sysbench` for testing through command line parameters: | ||||
| **IPV4 Active IPs**: Based on bgp.tools information, queries active neighbor count/total neighbor count in the current CIDR block. Since this is non-real-time, there may be delays. | ||||
|  | ||||
| **IPV6 Subnet Mask**: Queries the local IPV6 subnet size based on host information. | ||||
|  | ||||
| ### CPU Testing | ||||
|  | ||||
| Dependency project: [https://github.com/oneclickvirt/cputest](https://github.com/oneclickvirt/cputest) | ||||
|  | ||||
| Supports command-line parameter selection between `GeekBench` and `Sysbench` for testing: | ||||
|  | ||||
| | Comparison Item | sysbench | geekbench | | ||||
| |------------------|----------|-----------| | ||||
| | Application Range | Lightweight, can run on almost any server | Heavyweight, cannot run on small machines | | ||||
| | Test Requirements | No network needed, no special hardware requirements | Requires network, IPv4 environment, at least 1GB memory | | ||||
| | Open Source Status | Based on LUA, open source, can compile versions for various architectures (this project has been rebuilt in Go version built-in) | Official binary closed source code, does not support self-compilation | | ||||
| | Test Stability | Core test components unchanged for over 10 years | Test items updated with each major version, scores difficult to compare between different versions (each version benchmarks against current best CPUs) | | ||||
| | Test Content | Only tests computational performance, based on prime number calculation | Covers multiple performance tests, weighted score calculation, but some tests are not commonly used in practice | | ||||
| | Applicable Scenarios | Suitable for quick testing, only tests computational performance | Suitable for comprehensive testing | | ||||
| |----------------|----------|-----------| | ||||
| | Applicability | Lightweight, runs on almost any server | Heavy, cannot run on small machines | | ||||
| | Test Requirements | No network needed, no special hardware requirements | Requires internet, IPV4 environment, at least 1G memory | | ||||
| | Open Source | LUA-based, open source, can compile for various architectures (this project has Go version built-in) | Official binary closed source, doesn't support self-compilation | | ||||
| | Test Stability | Core test components unchanged for 10+ years | Updates test items with each major version, scores difficult to compare across versions (each version benchmarks against current best CPUs) | | ||||
| | Test Content | Only tests computational performance, based on prime calculations | Covers various performance tests, weighted score calculation, but some tests aren't commonly used | | ||||
| | Use Case | Suitable for quick testing, only tests computational performance | Suitable for comprehensive testing | | ||||
| | Leaderboard | [sysbench.spiritlhl.net](https://sysbench.spiritlhl.net/) | [browser.geekbench.com](https://browser.geekbench.com/) | | ||||
|  | ||||
| By default, ```Sysbench``` is used for testing, with the baseline roughly as follows: | ||||
| Default uses `Sysbench` for testing, with rough benchmarks as follows: | ||||
|  | ||||
| CPU test single-core ```Sysbench``` scores above 5000 can be considered first tier, 4000 to 5000 points second tier, with roughly one tier per 1000 points. | ||||
| CPU test single-core `Sysbench` scores above 5000 can be considered first-tier, 4000-5000 points second-tier, roughly one tier per 1000 points. | ||||
|  | ||||
| AMD's 7950x single-core full performance score is around 6500, AMD's 5950x single-core full performance score is around 5700, Intel's ordinary CPUs (E5 series, etc.) are around 1000~800, and single-core CPUs scoring below 500 can be said to have relatively poor performance. | ||||
| AMD 7950x single-core full performance scores around 6500, AMD 5950x single-core full performance scores around 5700, Intel regular CPUs (E5 series) around 1000-800, single-core CPUs below 500 can be considered poor performance. | ||||
|  | ||||
| Sometimes multi-core scores are the same as single-core scores, proving that the vendor is limiting program concurrent use of CPU, a typical example being Tencent Cloud. | ||||
| Sometimes multi-core and single-core scores are identical, proving the merchant is limiting program concurrent CPU usage, typical example being Tencent Cloud. | ||||
|  | ||||
| Benchmarks for ```Sysbench``` can be found in the [CPU Performance Ladder For Sysbench](https://sysbench.spiritlhl.net/) ladder chart, with specific scores regardless of the version of sysbench tested. | ||||
| `Sysbench` benchmarks can be seen in the [CPU Performance Ladder For Sysbench](https://sysbench.spiritlhl.net/) tier chart. Specific scores depend on the sysbench version tested. | ||||
|  | ||||
| For ```GeekBench``` baselines, see the [official website](https://browser.geekbench.com/processor-benchmarks/) ladder chart. Specific scores differ for each ```GeekBench``` version, so note which ```GeekBench``` version is being used when testing. | ||||
| `GeekBench` benchmarks can be seen in the [official website](https://browser.geekbench.com/processor-benchmarks/) tier chart. Specific scores differ for each `GeekBench` version, pay attention to which `GeekBench` version was used during testing. | ||||
|  | ||||
| As an additional note, many things tested by `GeekBench` are not actually used in server usage processes, so the test is for reference only. Of course, `Sysbench` is very incomplete, but it can roughly compare CPU performance based on the most basic computational performance. | ||||
| One more thing: `GeekBench` tests many things that are actually unused in server operations, tests are for reference only. Of course, `Sysbench` is very incomplete, but it can roughly compare CPU performance based on the most basic computational performance. | ||||
|  | ||||
| In practice, CPU performance just needs to be sufficient. Unless you're doing scientific computing or video transcoding, you generally don't need to pursue high-performance CPUs. | ||||
| Actually, CPU performance testing should be sufficient. Unless for scientific computing and video transcoding, generally no need to particularly pursue high-performance CPUs. If there are performance requirements, need to focus on whether the program itself uses multi-core or single-core, and look at multi-core or single-core scores accordingly. | ||||
|  | ||||
| ### **Memory Testing** | ||||
| ### Memory Testing | ||||
|  | ||||
| Generally speaking, you only need to determine whether the IO speed is below `10240MB/s`. If it's below this value, it proves that memory performance is poor, with an extremely high probability of overselling issues. | ||||
| Dependency project: [https://github.com/oneclickvirt/memorytest](https://github.com/oneclickvirt/memorytest) | ||||
|  | ||||
| As for the reasons for oversubscription, it could be that virtual memory is enabled (using disk as memory), ZRAM might be enabled (sacrificing CPU performance), balloon drivers might be enabled, or KSM memory fusion might be enabled - there are various possible reasons. | ||||
| Generally speaking, you only need to determine if IO speed is below `10240 MB/s (≈10 GB/s)`. | ||||
|  | ||||
| ### **Disk Testing** | ||||
| If below this value, it proves poor memory performance with high probability of overselling issues. | ||||
|  | ||||
| The `dd` test may have larger errors but is faster to test with no disk size limitations. The `fio` test is more realistic but slower to test and has minimum requirements for disk and memory size. | ||||
| Possible reasons for overselling: | ||||
|  | ||||
| At the same time, servers may have different file systems, and certain file systems' IO engines may test faster read/write speeds under the same hardware conditions, which is normal. The project uses `fio` for testing by default, with IO engine priority being `libaio > posixaio > psync`. The alternative `dd` test automatically replaces when `fio` testing is not available. | ||||
| * Virtual memory enabled (using disk as memory) | ||||
| * ZRAM enabled (sacrificing CPU performance) | ||||
| * Balloon driver enabled | ||||
| * KSM memory fusion enabled | ||||
|  | ||||
| Using `fio` test results as an example, the baseline is as follows: | ||||
| Various reasons exist. | ||||
|  | ||||
| | Memory Type | Typical Frequency (MHz) | Single Channel Bandwidth | Dual Channel Bandwidth | | ||||
| | ----------- | ----------------------- | ------------------------ | ---------------------- | | ||||
| | DDR3 | 1333 ~ 2133 | 10 ~ 17 GB/s (≈ 10240 ~ 17408 MB/s) | 20 ~ 34 GB/s (≈ 20480 ~ 34816 MB/s) | | ||||
| | DDR4 | 2133 ~ 3200 | 17 ~ 25 GB/s (≈ 17408 ~ 25600 MB/s) | 34 ~ 50 GB/s (≈ 34816 ~ 51200 MB/s) | | ||||
| | DDR5 | 4800 ~ 7200 | 38 ~ 57 GB/s (≈ 38912 ~ 58368 MB/s) | 76 ~ 114 GB/s (≈ 77824 ~ 116736 MB/s) | | ||||
|  | ||||
| Based on the above table, this project's rough judgment method: | ||||
|  | ||||
| * **< 20 GB/s (20480 MB/s)** → Possibly DDR3 (or DDR4 single channel / low frequency) | ||||
| * **20 ~ 40 GB/s (20480 ~ 40960 MB/s)** → Most likely DDR4 | ||||
| * **≈ 50 GB/s (≈ 51200 MB/s)** → Basically DDR5 | ||||
|  | ||||
| | Parameter Method | Test Accuracy | Test Speed | Architecture Compatibility | Dependency Requirements | | ||||
| |------------------|---------------|------------|---------------------------|------------------------| | ||||
| | **stream** | High — Stable results, more realistic | Fast | Cross-platform (Linux/Windows/Unix) | Built-in dependencies, no additional installation required | | ||||
| | **sysbench** | High — Can test CPU, memory, IO, threads, etc., reliable results | Medium | Cross-platform (Linux, partial Windows support) | Requires additional environment installation | | ||||
| | **winsat** | Medium-High — Windows built-in tool, results environment-dependent | Medium | Windows only | Built-in on physical machines, unavailable on virtual machines | | ||||
| | **mbw** | Medium — Results may be affected by cache/scheduling | Very Fast | Cross-platform (almost all Unix-like systems) | Built-in dependencies, no additional installation required | | ||||
| | **dd** | Low — Results affected by cache | Fast | Cross-platform (almost all Unix-like systems) | Requires additional environment installation | | ||||
|  | ||||
| ### Disk Testing | ||||
|  | ||||
| Dependency project: [https://github.com/oneclickvirt/disktest](https://github.com/oneclickvirt/disktest) | ||||
|  | ||||
| `dd` testing may have larger errors but tests quickly with no disk size limitations. `fio` testing is more realistic but tests slowly with minimum disk and memory size requirements. | ||||
|  | ||||
| Additionally, servers may have different file systems. Certain file systems' IO engines achieve faster read/write speeds under the same hardware conditions, which is normal. The project defaults to using `fio` for testing, with IO engine priority: `libaio > posixaio > psync`. Alternative `dd` testing automatically replaces when `fio` testing is unavailable. | ||||
|  | ||||
| Using `fio` test results as benchmark examples: | ||||
|  | ||||
| | OS Type | Primary Metrics | Secondary Metrics | | ||||
| |---------|-------------------|---------------------| | ||||
| | Windows/Mac | 4K read → 64K read → Write test | Graphical systems prioritize read performance | | ||||
| | Linux (without GUI) | 4K read + 4K write + 1M read/write | Read/write values usually similar | | ||||
| |---------|-----------------|-------------------| | ||||
| | Windows/MacOS | 4K Read → 64K Read → Write Tests | GUI systems prioritize read performance | | ||||
| | Linux (No GUI) | 4K Read + 4K Write + 1M Read/Write | Read/Write values usually similar | | ||||
|  | ||||
| The following disk types refer to metric values indicating normal~full-power performance states, using `libaio` as the IO test engine, testing under `Linux` | ||||
| The following disk types refer to regular~full performance states, using `libaio` as IO test engine, tested under `Linux`: | ||||
|  | ||||
| | Drive Type | 4K (IOPS) Performance | 1M (IOPS) Performance | | ||||
| |------------|--------------------------|----------------------| | ||||
| |------------|----------------------|----------------------| | ||||
| | NVMe SSD | ≥ 200 MB/s | 5-10 GB/s | | ||||
| | Standard SSD | 50-100 MB/s | 2-3 GB/s | | ||||
| | HDD (Mechanical) | 10-40 MB/s | 500-600 MB/s | | ||||
| | Poor Performance | < 10 MB/s | < 200 MB/s | | ||||
|  | ||||
| Quick assessment: | ||||
| Quick Assessment: | ||||
|  | ||||
| 1. **Primary Check**: 4K read(IOPS) 4K write(IOPS) | ||||
|     - Almost identical with little difference | ||||
| 1. **Primary Check**: 4K Read (IOPS) 4K Write (IOPS) | ||||
|     - Nearly identical with little difference | ||||
|     - ≥ 200 MB/s = NVMe SSD | ||||
|     - 50-100 MB/s = Standard SSD | ||||
|     - 10-40 MB/s = HDD (Mechanical) | ||||
|     - < 10 MB/s = Poor performance, severe overselling/restriction | ||||
|     - < 10 MB/s = Poor performance, severe overselling/restrictions | ||||
|  | ||||
| 2. **Secondary Check**: 1M total(IOPS) | ||||
|     - IO limit set by provider | ||||
| 2. **Secondary Check**: 1M Total (IOPS) | ||||
|     - Provider's IO limitations | ||||
|     - Resource overselling situation | ||||
|     - Higher value is better | ||||
|     - NVMe SSD typically reaches 4-6 GB/s | ||||
|     - Standard SSD typically reaches 1-2 GB/s | ||||
|     - Higher values are better | ||||
|     - NVMe SSD usually reaches 4-6 GB/s | ||||
|     - Standard SSD usually reaches 1-2 GB/s | ||||
|  | ||||
| If NVMe SSD's 1M(IOPS) value < 1GB/s, it indicates severe resource overselling. | ||||
| If NVMe SSD's 1M (IOPS) value < 1GB/s indicates serious resource overselling. | ||||
|  | ||||
| Note that this is testing real IO, limited to this project only. The baseline may not be universal for tests not from this project, because they might not use the same parameters when testing, might not set direct IO reading/writing, might use inconsistent IO engines, or might set inconsistent test times, all of which will cause baseline deviations. | ||||
| Note: This tests real IO, limited to this project. IO tests from other projects don't guarantee universal benchmarks because they may use different parameters, may not set direct IO read/write, may have inconsistent IO engines, or inconsistent test times, all causing benchmark deviations. | ||||
|  | ||||
| ### **Streaming Media Unlocking** | ||||
| ### Streaming Media Unlocking | ||||
|  | ||||
| Checks common streaming media platform unlocking, though not all are streaming media - some other common platform unlocks are also included. Generally speaking, IP unlocking regions are consistent and don't randomly fluctuate. If you find that multiple platforms have inconsistent unlocking regions, then the IP is likely rented from platforms like IPXO, with slow recognition in various platform databases. Generally speaking, the IP quality won't be good either. | ||||
| Dependency project: [https://github.com/oneclickvirt/CommonMediaTests](https://github.com/oneclickvirt/CommonMediaTests) [https://github.com/oneclickvirt/UnlockTests](https://github.com/oneclickvirt/UnlockTests) | ||||
|  | ||||
| ### **IP Quality Detection** | ||||
| Default only checks cross-border streaming media unlocking. | ||||
|  | ||||
| Checks IP-related information from 14 databases. Generally speaking, it's sufficient to look at usage type, company type, and other security information judgments. The security score is really just for amusement. When multiple platforms compare corresponding detection items to corresponding values, it proves that the current IP is indeed as such. Don't just trust information from a single database source. | ||||
| Generally speaking, under normal circumstances, multiple streaming services for one IP should have consistent unlock regions without scattered locations. If multiple platforms show inconsistent unlock regions, the IP likely comes from platforms like IPXO rentals or has been recently announced and used, not yet recognized and corrected by streaming media common databases. Due to inconsistent IP database recognition speeds across platforms, sometimes some platforms unlock regions normally, some drift to certain router locations, and some drift to where the IP was before you used it. | ||||
|  | ||||
| ### **Email Port Detection** | ||||
| | DNS Type | Unlock Method Judgment Necessary | DNS Impact on Unlocking | Description | | ||||
| | -------- | ------------------------------- | ----------------------- | ----------- | | ||||
| | Official Mainstream DNS | No | Small | Streaming unlock mainly relies on node IP, DNS resolution basically doesn't interfere with unlocking | | ||||
| | Non-mainstream / Self-built DNS | Yes | Large | Streaming unlock results greatly affected by DNS resolution, need to judge if it's native unlock or DNS unlock | | ||||
|  | ||||
| - **SMTP (25)**: Used for email transmission between mail servers (sending mail). | ||||
| So during testing, if the host currently uses official mainstream DNS, no judgment of whether it's native unlocking will be performed. | ||||
|  | ||||
| ### IP Quality Detection | ||||
|  | ||||
| Dependency project: [https://github.com/oneclickvirt/securityCheck](https://github.com/oneclickvirt/securityCheck) | ||||
|  | ||||
| Detects IP-related information from 14 databases. Generally speaking, looking at usage type, company type, and security information's other discriminators is sufficient. Security scores are really just for fun. When multiple platforms show corresponding detection items all having corresponding values, it proves the current IP is indeed as such - don't trust information from just one database source. | ||||
|  | ||||
| * **Usage Type & Company Type**: Shows IP attribution and usage scenarios, such as whether it belongs to home users, enterprise office, hosting services, or cloud/data centers. | ||||
| * **Cloud Provider / Data Center / Mobile Device**: Determines if IP comes from cloud services, data centers, or mobile networks, helping identify shared or high-risk IPs. | ||||
| * **Proxy / VPN / Tor / Tor Exit**: Detects if IP is used to hide real identity or location, possibly involving anonymous access or abuse behavior. | ||||
| * **Web Crawler / Bot**: Identifies automated access or collection programs, with reference value for security risk assessment. | ||||
| * **Anonymous / Abuser / Threat / Relay / Bogon**: Shows IP historical behavior characteristics and whether it belongs to reserved/unallocated IPs, assisting in judging IP credibility. | ||||
| * **Security Score, Reputation, Trust Score, Threat Score, Fraud Score, Abuse Score**: Various databases' quantified security evaluations of IPs, for reference only. | ||||
| * **Community Voting & Blacklist Records**: Shows user feedback and public blacklist information, can quickly identify potential risks. | ||||
| * **Google Search Feasibility**: Tests IP's feasibility for accessing Google search services, indirectly reflecting network restrictions or blocking situations. | ||||
|  | ||||
| Multi-platform comparison is more reliable. Different databases have different algorithms and update frequencies; single sources may misjudge. Similar results from multiple databases indicate higher reliability. | ||||
|  | ||||
| ### Email Port Detection | ||||
|  | ||||
| Dependency project: [https://github.com/oneclickvirt/portchecker](https://github.com/oneclickvirt/portchecker) | ||||
|  | ||||
| - **SMTP (25)**: Used for mail transfer between mail servers (sending mail). | ||||
| - **SMTPS (465)**: Used for encrypted SMTP mail sending (SSL/TLS method). | ||||
| - **SMTP (587)**: Used for clients to send email to mail servers, supports STARTTLS encryption. | ||||
| - **POP3 (110)**: Used for email clients to download mail from servers, unencrypted. | ||||
| - **SMTP (587)**: Used for clients sending mail to mail servers, supports STARTTLS encryption. | ||||
| - **POP3 (110)**: Used for mail clients downloading mail from servers, unencrypted. | ||||
| - **POP3S (995)**: Used for encrypted POP3, securely downloading mail (SSL/TLS method). | ||||
| - **IMAP (143)**: Used for email clients to manage mail online (view, sync mail), unencrypted. | ||||
| - **IMAP (143)**: Used for mail clients managing mail online (viewing, syncing mail), unencrypted. | ||||
| - **IMAPS (993)**: Used for encrypted IMAP, securely managing mail (SSL/TLS method). | ||||
|  | ||||
| Specifically, if the current host machine is not being used as a mail server or not receiving electronic mail, then this project metric can be disregarded. | ||||
| If the current host doesn't function as a mail server and doesn't send/receive emails, this project indicator can be ignored. | ||||
|  | ||||
| ### Nearby Speed Testing | ||||
|  | ||||
| Dependency project: [https://github.com/oneclickvirt/speedtest](https://github.com/oneclickvirt/speedtest) | ||||
|  | ||||
| First test the officially recommended speed test points, then test representative international speed test points. | ||||
|  | ||||
| Official speed test points can represent the local bandwidth baseline of the host machine being tested. | ||||
|  | ||||
| In daily use, I prefer to use servers with 1Gbps bandwidth, at least the speed of downloading dependencies is fast enough. | ||||
|  | ||||
| --- | ||||
|  | ||||
| ## 日本語 | ||||
|  | ||||
| ### **システム基本情報** | ||||
| ### システム基本情報 | ||||
|  | ||||
| CPU型番: 簡単に言えば、CPUの発売時期によって、新しいモデルならAMDがIntelより優れ、古いモデルならIntelがAMDより優れています。 | ||||
| 依存プロジェクト:[https://github.com/oneclickvirt/basics](https://github.com/oneclickvirt/basics) [https://github.com/oneclickvirt/gostun](https://github.com/oneclickvirt/gostun) | ||||
|  | ||||
| CPUコア数: 物理コアか論理コアかを検出し、優先的に物理コアを表示します。物理コアが検出できない場合のみ論理コアを表示します。サーバーの実際の使用では、プログラムは通常、論理コアに基づいて実行されます。ビデオエンコードや科学計算以外では、物理コアは通常ハイパースレッディングを有効にして論理コアとして使用されます。比較する際は、同じタイプのコア数を比較することが意味を持ちます。 | ||||
| **CPUモデル**: 一般的に、CPUのリリース時期に基づいて、新しいモデルではAMDがIntelより優れており、古いモデルではIntelがAMDより優れています。一方、AppleのMシリーズチップは圧倒的に優位に立っています。 | ||||
|  | ||||
| CPUキャッシュ:ホストマシンのCPU L1/L2/L3キャッシュ情報を表示します。 | ||||
| **CPU数量**: 物理コアか論理コアかを検出し、物理コアの表示を優先します。物理コアが検出できない場合のみ論理コアを表示します。実際のサーバー使用において、プログラムは一般的に論理コアベースで実行が割り当てられます。動画変換や科学計算以外では、物理コアは通常ハイパースレッディングを有効にして論理コアとして使用されます。比較する際は、対応するコアタイプの数量のみが比較意義を持ちます。もちろん、一つが物理コア、もう一つが仮想コアで、CPUテストスコアが似ている場合、物理コアの方が明らかに優れており、CPU性能共有の問題を心配する必要はありません。 | ||||
|  | ||||
| AES-NI: 暗号化/復号化を高速化する命令セットです。これがあれば通常のネットワークリクエストがより速く、パフォーマンスが高くなります。ない場合はネットワークリクエスト(プロキシ用途を含む)に影響します。 | ||||
| **CPUキャッシュ**: ホストのCPU L3キャッシュ情報を表示します。一般的なアプリケーションにはあまり影響しないかもしれませんが、データベース、コンパイル、大規模な並行リクエストなどのシナリオでは、L3キャッシュサイズが性能に大きく影響します。 | ||||
|  | ||||
| VM-x/AMD-V/Hyper-V: 現在のテスト環境がネステッド仮想化をサポートしているかどうかを示す指標です。テスト環境がDockerコンテナ内にあるか、root権限がない場合、デフォルトでは検出できず、ネステッド仮想化をサポートしていないと表示されます。この指標は、ホストマシン上で仮想マシン(KVM、VirtualBox、VMwareなど)を設定する必要がある場合に役立ちますが、他の用途ではあまり重要ではありません。 | ||||
| **AES-NI**: AES命令セットは暗号化・復号化の高速化に使用され、HTTPS(TLS/SSL)ネットワークリクエスト、VPNプロキシ(AES暗号化設定使用)、ディスク暗号化などのシナリオで明らかな最適化を提供し、より高速でリソース効率的です。 | ||||
|  | ||||
| メモリ: 使用中サイズ/総サイズ のメモリを表示します。仮想メモリは含まれません。 | ||||
| **VM-x/AMD-V/Hyper-V**: 現在のテストホストがネスト仮想化をサポートしているかどうかの指標です。テスト環境がDockerで実行されているか、root権限がない場合、デフォルトではネスト仮想化不サポートとして表示されます。この指標は、ホスト上で仮想マシン(KVM、VirtualBox、VMwareなど)を作成する必要がある場合に有用で、その他の用途では限定的です。 | ||||
|  | ||||
| バルーンドライバ: ホストマシンがバルーンドライバを使用しているかどうかを表示します。使用している場合は、親マシンがメモリを共有していることを示し、以下のメモリ読み書きテストと合わせて、オーバーセリング/厳しい制限があるかどうかを確認する必要があります。 | ||||
| **メモリ**: メモリ使用量を現在使用中サイズ/総サイズで表示します。仮想メモリは含まれません。 | ||||
|  | ||||
| Kernel Same-page Merging: ホストマシンがKSMメモリマージを使用しているかどうかを表示します。使用している場合は、親マシンがメモリを共有していることを示し、以下のメモリ読み書きテストと合わせて、オーバーセリング/厳しい制限があるかどうかを確認する必要があります。 | ||||
| **バルーンドライバー**: ホストでバルーンドライバーが有効になっているかどうかを表示します。バルーンドライバーはホストと仮想マシン間での動的メモリ割り当てに使用され、ホストは仮想マシンに「収縮」してメモリの一部を解放するか、「膨張」してより多くのメモリを占有するよう要求できます。有効化は通常、ホストがメモリオーバーセリング機能を持っていることを意味しますが、実際にオーバーセリングが存在するかどうかは、下記のメモリ読み書きテストでオーバーセリング/厳格な制限を確認する必要があります。 | ||||
|  | ||||
| 仮想メモリ: SWAP仮想メモリを表示します | ||||
| **カーネルページマージ**: ホストでカーネルページマージ機能が有効になっているかどうかを表示します。KSMは複数のプロセスから同一内容のメモリページを1つにマージして物理メモリ使用量を削減します。有効化は通常、ホストがメモリ節約を実施しているか、ある程度のメモリオーバーセリングがあることを意味します。実際に性能影響やメモリ不足を引き起こすかどうかは、下記のメモリ読み書きテストでオーバーセリング/厳格な制限を確認する必要があります。 | ||||
|  | ||||
| ディスク容量: 使用中サイズ/総サイズ のディスク容量を表示します | ||||
| **仮想メモリ**: スワップ仮想メモリは、ディスク上に割り当てられた仮想メモリ空間で、物理メモリ不足時にデータを一時的に格納するために使用されます。メモリ不足によるプログラムクラッシュを防ぎますが、頻繁な使用はシステムを著しく遅くします。Linuxが公式推奨するスワップ設定は以下の通りです: | ||||
|  | ||||
| ブートディスクパス:ブートディスクのパスを表示します | ||||
| | 物理メモリサイズ | 推奨SWAPサイズ | | ||||
| | ---------------- | -------------- | | ||||
| | ≤ 2G            | メモリの2倍    | | ||||
| | 2G < メモリ ≤ 8G | 物理メモリサイズと同等 | | ||||
| | ≥ 8G            | 約8Gで十分     | | ||||
| | 休止状態(hibernation)必要 | 最低でも物理メモリサイズと同等 | | ||||
|  | ||||
| OS: システム名とアーキテクチャを表示します | ||||
| **ディスク容量**: ディスク使用量を現在使用中サイズ/総サイズで表示します | ||||
|  | ||||
| カーネル: システムカーネルバージョンを表示します | ||||
| **起動ディスクパス**: 起動ディスクのパスを表示します | ||||
|  | ||||
| システム稼働時間: ホストマシンが起動してからテスト時までの稼働時間を表示します | ||||
| **システム**: システム名とアーキテクチャを表示します | ||||
|  | ||||
| タイムゾーン: ホストマシンのシステムタイムゾーンを表示します | ||||
| **カーネル**: システムカーネルバージョンを表示します | ||||
|  | ||||
| 負荷: システム負荷を表示します | ||||
| **システム稼働時間**: ホストの起動からテスト時点までの稼働時間を表示します | ||||
|  | ||||
| 仮想化アーキテクチャ: ホストマシンがどの仮想化アーキテクチャから来ているかを表示します。一般的に ```Dedicated > KVM > Xen``` 仮想化が推奨されます。他の仮想化はパフォーマンス低下を引き起こし、使用時にパフォーマンス共有/損失が発生しますが、これも確実ではありません。専用サーバーのみが完全に独立したリソース占有を持ち、他の仮想化はほとんどリソース共有があります。これはホストマシンの販売者が良心的かどうかによって異なります。具体的なパフォーマンスの優劣は、後の専門テストを見る必要があります。 | ||||
| **タイムゾーン**: ホストシステムのタイムゾーンを表示します | ||||
|  | ||||
| NAT種類: NAT種類を表示します。具体的には ```Full Cone > Restricted Cone > Port Restricted Cone > Symmetric``` が推奨されます。検出できない場合は ```Inconclusive``` と表示されます。一般的に特別な用途(特殊なプロキシとリアルタイム通信の要件に関連する)に使用しない限り、この指標を気にする必要はありません。 | ||||
| **負荷**: システム負荷を表示します | ||||
|  | ||||
| TCP加速方式:一般的に ```cubic/bbr``` 輻輳制御プロトコルです。一般的にプロキシサーバーとして使用する場合、bbrを使用するとネットワーク速度が改善されますが、通常の用途ではこの指標に注目する必要はありません。 | ||||
| **仮想化アーキテクチャ**: ホストがどの仮想化アーキテクチャから来ているかを表示します。一般的に推奨順序:`Dedicated > KVM > Xen`仮想化。その他の仮想化は性能損失があり、使用時に性能共有/損失が発生しますが、これは断定的ではありません。専用サーバーのみが完全に独立したリソース占有を持ちます。その他の仮想化は基本的にリソース共有があり、ホスト保有者がこの仮想マシンに対して良心的かどうかに依存します。実際の性能優劣は後続の専門性能テストを見る必要があります。 | ||||
|  | ||||
| IPV4/IPV6 ASN: ホストマシンのIPが属するASN組織IDと名前を表示します。同じIDCに複数のASNがある可能性があり、1つのASNの下に異なるIPセグメントのサーバーを販売する複数の業者がいる可能性があります。具体的な上流/下流関係は複雑です。bgp.toolを使用してさらに詳しく調べることができます。 | ||||
| **NATタイプ**: NATタイプを表示します。具体的な推奨順序:`Full Cone > Restricted Cone > Port Restricted Cone > Symmetric`。検出不可能または非標準プロトコルタイプは`Inconclusive`と表示されます。一般的に、特殊な用途、例えば特殊なプロキシ、リアルタイム通信、FRPポート転送などの需要がある場合のみ特別に注意が必要で、その他の一般状況では本指標に注意する必要はありません。 | ||||
|  | ||||
| IPV4/IPV6 ロケーション: データベース内の対応するプロトコルのIPの地理的位置を表示します。 | ||||
| **TCP高速化方式**: 通常`cubic/bbr`輻輳制御プロトコルです。一般的に、プロキシサーバーでbbrを使用すると通信速度を改善できますが、普通の用途では本指標に注意する必要はありません。 | ||||
|  | ||||
| IPV4 アクティブIP: bgp.tools情報に基づいて、現在のCIDRチャンクのアクティブなネイバー数/総ネイバー数を照会する。 | ||||
| **IPV4/IPV6 ASN**: ホストIPが属するASN組織IDと名前を表示します。同じIDCが複数のASNを持つ可能性があり、ASNの下に異なるIPセグメントのサーバーを販売する複数の業者がいる可能性があります。具体的な上下流関係は複雑で、bgp.toolでさらなる調査が可能です。 | ||||
|  | ||||
| ### **CPUテスト** | ||||
| **IPV4/IPV6 Location**: 対応プロトコルのIPのデータベース内地理位置を表示します。 | ||||
|  | ||||
| コマンドラインパラメータを通じて```GeekBench```と```Sysbench```のテストを選択できます: | ||||
| **IPV4 Active IPs**: bgp.tools情報に基づき、現在のCIDRブロック内のアクティブ近隣数/総近隣数を照会します。非リアルタイムのため遅延がある可能性があります。 | ||||
|  | ||||
| **IPV6 サブネットマスク**: ホスト情報に基づいてローカルIPV6サブネットサイズを照会します。 | ||||
|  | ||||
| ### CPUテスト | ||||
|  | ||||
| 依存プロジェクト:[https://github.com/oneclickvirt/cputest](https://github.com/oneclickvirt/cputest) | ||||
|  | ||||
| コマンドラインパラメータで`GeekBench`と`Sysbench`のテスト選択をサポートします: | ||||
|  | ||||
| | 比較項目 | sysbench | geekbench | | ||||
| |------------------|----------|-----------| | ||||
| |----------|----------|-----------| | ||||
| | 適用範囲 | 軽量、ほぼすべてのサーバーで実行可能 | 重量級、小型マシンでは実行不可 | | ||||
| | テスト要件 | ネットワーク不要、特別なハードウェア要件なし | ネットワーク必要、IPV4環境、最低1Gメモリ | | ||||
| | オープンソース状況 | LUAベース、オープンソース、各アーキテクチャ版をコンパイル可能(本プロジェクトではGoに再構築して内蔵) | 公式バイナリはクローズドソース、自己コンパイル不可 | | ||||
| | テスト安定性 | コアテストコンポーネントは10年以上変更なし | 各メジャーバージョンでテスト項目更新、スコアはバージョン間で比較困難(各バージョンは当時最高のCPUを基準) | | ||||
| | テスト内容 | 計算性能のみテスト、素数計算ベース | 多様な性能テストカバー、スコアは重み付け計算、一部テストは実際にはあまり使用されない | | ||||
| | 適用シーン | 迅速なテストに適合、計算性能のみテスト | 総合的な全面テストに適合 | | ||||
| | テスト要件 | ネットワーク不要、特殊ハードウェア不要 | インターネット必要、IPV4環境、最低1Gメモリ | | ||||
| | オープンソース状況 | LUAベース、オープンソース、各アーキテクチャ版を自分でコンパイル可能(本プロジェクトはGo版を内蔵) | 公式バイナリクローズドソース、自分でコンパイル不可 | | ||||
| | テスト安定性 | コアテストコンポーネント10年以上変更なし | 各メジャーバージョン更新でテスト項目変更、異なるバージョン間でスコア比較困難(各バージョンは現在最高のCPUを基準) | | ||||
| | テスト内容 | 計算性能のみテスト、素数計算ベース | 多種性能テストをカバー、スコア加重計算、但し一部テストは実際には非常用 | | ||||
| | 適用シナリオ | 迅速テストに適し、計算性能のみテスト | 総合的で全面的なテストに適用 | | ||||
| | ランキング | [sysbench.spiritlhl.net](https://sysbench.spiritlhl.net/) | [browser.geekbench.com](https://browser.geekbench.com/) | | ||||
|  | ||||
| デフォルトでは```Sysbench```を使用してテストを行います。基準は概ね以下の通りです: | ||||
| デフォルトで`Sysbench`でテストを行い、基準は大まかに以下の通りです: | ||||
|  | ||||
| CPUテストのシングルコア```Sysbench```スコアが5000以上なら第一ティア、4000〜5000点なら第二ティア、1000点ごとに大体一ランクと考えられます。 | ||||
| CPUテストシングルコア`Sysbench`スコア5000以上は第一階層、4000-5000点は第二階層、1000点毎に大体一階層と考えられます。 | ||||
|  | ||||
| AMDの7950xシングルコアのフルパフォーマンススコアは約6500、AMDの5950xシングルコアのフルパフォーマンススコアは約5700、Intelの通常のCPU(E5など)は約1000〜800、500未満のシングルコアCPUはパフォーマンスが比較的低いと言えます。 | ||||
| AMD 7950xシングルコアフル性能スコアは6500前後、AMD 5950xシングルコアフル性能スコアは5700前後、Intel普通のCPU(E5系など)は1000~800前後、500以下のシングルコアCPUは性能が比較的劣ると言えます。 | ||||
|  | ||||
| 時々、マルチコアスコアとシングルコアスコアが同じ場合があります。これは販売者がプログラムの並列CPU使用を制限していることを示しています。典型的な例はTencent Cloudです。 | ||||
| 時々マルチコアスコアとシングルコアスコアが同じになることがあり、これは業者がプログラムの並行CPU使用を制限していることを証明します。典型例はTencent Cloudです。 | ||||
|  | ||||
| Sysbenchのベンチマークは[CPU Performance Ladder For Sysbench](https://sysbench.spiritlhl.net/)のラダーチャートで見ることができる。 | ||||
| `Sysbench`の基準は[CPU Performance Ladder For Sysbench](https://sysbench.spiritlhl.net/)階層図で確認可能、具体的なスコアはテストしたsysbenchのバージョンに依存しません。 | ||||
|  | ||||
| ```GeekBench```の基準は[公式ウェブサイト](https://browser.geekbench.com/processor-benchmarks/)の階層チャートを参照してください。具体的なスコアは各```GeekBench```バージョンで異なるため、テスト時の```GeekBench```バージョンに注意してください。 | ||||
| `GeekBench`の基準は[公式サイト](https://browser.geekbench.com/processor-benchmarks/)階層図で確認可能、具体的なスコアは各`GeekBench`バージョンで異なり、使用時にテストした`GeekBench`バージョンに注意してください。 | ||||
|  | ||||
| 補足ですが、```GeekBench```がテストする多くの内容は、サーバー使用過程で実際には必要ないことが多いです。テストは参考程度にしてください。もちろん```Sysbench```は非常に包括的ではありませんが、基本的な計算性能に基づいてCPUのパフォーマンスを大まかに比較できます。 | ||||
| もう一つ付け加えると、`GeekBench`はテストする多くの内容が実際のサーバー使用過程では全く使われないため、テストは参考のみです。もちろん`Sysbench`は非常に不完全ですが、最も基本的な計算性能に基づいてCPUの性能を大まかに比較できます。 | ||||
|  | ||||
| 実際にはCPUパフォーマンスは十分であれば良く、科学計算やビデオエンコード以外では、特に高性能CPUを追求する必要はありません。 | ||||
| 実際にはCPU性能テストは十分であれば良く、科学計算や動画変換以外では、特に高性能CPUを追求する必要は一般的にありません。性能需要がある場合は、プログラム自体がマルチコアかシングルコアを使うかに注意し、対応してマルチコアかシングルコアのスコアを見る必要があります。 | ||||
|  | ||||
| ### **メモリテスト** | ||||
| ### メモリテスト | ||||
|  | ||||
| 一般的に、IO速度が```10240MB/s```未満かどうかを判断するだけで十分です。この値を下回る場合、メモリパフォーマンスが良くなく、オーバーセリング/オーバーコミットの問題がある可能性が非常に高いです。 | ||||
| 依存プロジェクト:[https://github.com/oneclickvirt/memorytest](https://github.com/oneclickvirt/memorytest) | ||||
|  | ||||
| オーバーコミットの原因は、仮想メモリの使用(ディスクをメモリとして使用)、ZRAM(CPUパフォーマンスを犠牲)、バルーンドライバの使用、KSMメモリマージの使用など、様々な可能性があります。 | ||||
| 一般的に、IO速度が`10240 MB/s (≈10 GB/s)`を下回るかどうかを判断するだけで十分です。 | ||||
|  | ||||
| ### **ディスクテスト** | ||||
| この値を下回る場合、メモリ性能が不良で、オーバーセリング問題が存在する可能性が極めて高いことを証明します。 | ||||
|  | ||||
| ```dd```テストは誤差が大きい可能性がありますが、テスト速度が速くディスクサイズ制限がありません。```fio```テストはより現実的ですが、テスト速度が遅く、ディスクおよびメモリサイズの最低要件があります。 | ||||
| オーバーセリングの原因は以下が考えられます: | ||||
|  | ||||
| 同時に、サーバーには異なるファイルシステムがある可能性があり、特定のファイルシステムのIOエンジンは同じハードウェア条件下でも読み書き速度が速くなる場合があります。これは正常です。プロジェクトはデフォルトで```fio```を使用してテストを行います。テストに使用されるIOエンジンの優先順位は```libaio > posixaio > psync```です。代替オプションの```dd```テストは```fio```テストが利用できない場合に自動的に置き換えられます。 | ||||
| * 仮想メモリの有効化(ディスクをメモリとして使用) | ||||
| * ZRAMの有効化(CPU性能を犠牲) | ||||
| * バルーンドライバーの有効化 | ||||
| * KSMメモリ融合の有効化 | ||||
|  | ||||
| ```fio```テスト結果を例に基準は以下の通りです: | ||||
| 原因は多種多様です。 | ||||
|  | ||||
| | OSタイプ | 主要指標 | 副次指標 | | ||||
| |---------|-------------------|---------------------| | ||||
| | Windows/Mac | 4K読み → 64K読み → 書き込みテスト | グラフィカルインターフェースシステムは読み取りパフォーマンスを優先 | | ||||
| | Linux (GUIなし) | 4K読み + 4K書き + 1M読み書き| 読み/書き値は通常類似 | | ||||
| | メモリタイプ | 典型的周波数 (MHz) | シングルチャネル帯域幅 | デュアルチャネル帯域幅 | | ||||
| | ----------- | ------------------ | -------------------- | -------------------- | | ||||
| | DDR3 | 1333 ~ 2133 | 10 ~ 17 GB/s (≈ 10240 ~ 17408 MB/s) | 20 ~ 34 GB/s (≈ 20480 ~ 34816 MB/s) | | ||||
| | DDR4 | 2133 ~ 3200 | 17 ~ 25 GB/s (≈ 17408 ~ 25600 MB/s) | 34 ~ 50 GB/s (≈ 34816 ~ 51200 MB/s) | | ||||
| | DDR5 | 4800 ~ 7200 | 38 ~ 57 GB/s (≈ 38912 ~ 58368 MB/s) | 76 ~ 114 GB/s (≈ 77824 ~ 116736 MB/s) | | ||||
|  | ||||
| 以下のディスクタイプの指標値は、通常〜フルパフォーマンス状態を示し、```libaio```をIOテストエンジンとして使用し、```Linux```でテストを実施した場合を指します: | ||||
| 上表内容に基づく、本プロジェクトテストの粗略判断方法: | ||||
|  | ||||
| | ドライブタイプ | 4K(IOPS)パフォーマンス | 1M(IOPS)パフォーマンス | | ||||
| |------------|--------------------------|----------------------| | ||||
| * **< 20 GB/s (20480 MB/s)** → DDR3の可能性(またはDDR4シングルチャネル/低周波数) | ||||
| * **20 ~ 40 GB/s (20480 ~ 40960 MB/s)** → 高確率でDDR4 | ||||
| * **≈ 50 GB/s (≈ 51200 MB/s)** → 基本的にDDR5 | ||||
|  | ||||
| | パラメータ方法 | テスト精度 | テスト速度 | アーキテクチャ対応 | 依存関係 | | ||||
| |------------|------------|----------|----------|----------| | ||||
| | **stream** | 高 — 結果が安定し、実際の状況により適合 | 高速 | クロスプラットフォーム(Linux/Windows/Unix) | 内蔵依存関係、追加インストール不要 | | ||||
| | **sysbench** | 高 — CPU、メモリ、IO、スレッドなどをテスト可能、結果が信頼性高い | 中程度 | クロスプラットフォーム(Linux、Windows部分対応) | 環境への追加インストールが必要 | | ||||
| | **winsat** | 中程度やや高 — Windows内蔵ツール、結果は環境に依存 | 中程度 | Windows限定 | 物理マシンに内蔵、仮想マシンでは利用不可 | | ||||
| | **mbw** | 中程度 — 結果がキャッシュ/スケジューリングの影響を受ける可能性 | 非常に高速 | クロスプラットフォーム(ほぼ全てのUnix系システム) | 内蔵依存関係、追加インストール不要 | | ||||
| | **dd** | 低 — 結果がキャッシュの影響を受ける | 高速 | クロスプラットフォーム(ほぼ全てのUnix系システム) | 環境への追加インストールが必要 | | ||||
|  | ||||
| ### ディスクテスト | ||||
|  | ||||
| 依存プロジェクト:[https://github.com/oneclickvirt/disktest](https://github.com/oneclickvirt/disktest) | ||||
|  | ||||
| `dd`テストは誤差が大きい可能性がありますが、テスト速度が速くディスクサイズ制限がありません。`fio`テストはより現実的ですが、テスト速度が遅く、ディスクおよびメモリサイズの最低要件があります。 | ||||
|  | ||||
| 同時に、サーバーは異なるファイルシステムを持つ可能性があり、一部のファイルシステムのIOエンジンは同じハードウェア条件下でテストの読み書き速度がより速く、これは正常です。プロジェクトはデフォルトで`fio`でテストを行い、使用するIOエンジンの優先度は`libaio > posixaio > psync`、代替オプション`dd`テストは`fio`テストが使用不可能時に自動置換されます。 | ||||
|  | ||||
| `fio`テスト結果を例に基準は以下の通りです: | ||||
|  | ||||
| | OS タイプ | 主要指標 | 次要指標 | | ||||
| |----------|----------|----------| | ||||
| | Windows/MacOS | 4K読み取り → 64K読み取り → 書き込みテスト | グラフィカルインターフェースシステムは読み取り性能を優先考慮 | | ||||
| | Linux(グラフィカルインターフェースなし)| 4K読み取り + 4K書き込み + 1M読み書き | 読み取り/書き込み値は通常類似 | | ||||
|  | ||||
| 以下のディスクタイプは通常~フル血状態の性能を指し、`libaio`をIOテストエンジンとし、`Linux`下でテストを行うことを指します: | ||||
|  | ||||
| | ドライブタイプ | 4K (IOPS) 性能 | 1M (IOPS) 性能 | | ||||
| |---------------|----------------|----------------| | ||||
| | NVMe SSD | ≥ 200 MB/s | 5-10 GB/s | | ||||
| | 標準SSD | 50-100 MB/s | 2-3 GB/s | | ||||
| | HDD (機械式ハードディスク) | 10-40 MB/s | 500-600 MB/s | | ||||
| | HDD(機械式ハードディスク)| 10-40 MB/s | 500-600 MB/s | | ||||
| | 性能不良 | < 10 MB/s | < 200 MB/s | | ||||
|  | ||||
| 迅速な評価: | ||||
| 迅速評価: | ||||
|  | ||||
| 1. **主要チェック**: 4K読み(IOPS) 4K書き(IOPS) | ||||
|     - ほぼ同じで大きな差がない | ||||
| 1. **主要チェック**: 4K読み取り (IOPS) 4K書き込み (IOPS) | ||||
|     - ほぼ同じで差は小さい | ||||
|     - ≥ 200 MB/s = NVMe SSD | ||||
|     - 50-100 MB/s = 標準SSD | ||||
|     - 10-40 MB/s = HDD (機械式ハードディスク) | ||||
|     - < 10 MB/s = 性能不良、オーバーセリング/制限が深刻 | ||||
|     - 10-40 MB/s = HDD(機械式ハードディスク) | ||||
|     - < 10 MB/s = ゴミ性能、オーバーセリング/制限が深刻 | ||||
|  | ||||
| 2. **副次チェック**: 1M合計(IOPS) | ||||
|     - プロバイダが設定したIO制限 | ||||
|     - リソースのオーバーコミット状況 | ||||
|     - 値が高いほど良い | ||||
|     - NVMe SSDは通常4-6 GB/s達成 | ||||
|     - 標準SSDは通常1-2 GB/s達成 | ||||
| 2. **次要チェック**: 1M総計 (IOPS) | ||||
|     - プロバイダーが設定したIO制限 | ||||
|     - リソースオーバーセリング状況 | ||||
|     - 数値が高いほど良い | ||||
|     - NVMe SSDは通常4-6 GB/sに達する | ||||
|     - 標準SSDは通常1-2 GB/sに達する | ||||
|  | ||||
| NVMe SSDの1M(IOPS)値が< 1GB/sの場合、深刻なリソースオーバーコミットが存在することを示します。 | ||||
| NVMe SSDの1M (IOPS)値 < 1GB/s の場合、深刻なリソースオーバーセリングが存在することを示します。 | ||||
|  | ||||
| 注意:ここでテストされるのは実際のIOであり、本プロジェクトに限定されます。本プロジェクト以外のテストによるIOは基準の普遍性を保証できません。他のテストでは同じパラメータを使用していない可能性があり、IO直接読み書きを設定していない可能性、IOエンジンの設定が一致しない可能性、テスト時間の設定が一致しない可能性があり、これらはすべて基準にズレを生じさせる原因となります。 | ||||
| 注意:ここでテストするのは真のIOで、本プロジェクト限定です。本プロジェクト以外でテストしたIOは基準の汎用性を保証しません。彼らがテスト時に同じパラメータを使用していない可能性、IO直接読み書きを設定していない可能性、IOエンジン設定が一致しない可能性、テスト時間設定が一致しない可能性があり、すべて基準の偏差を引き起こします。 | ||||
|  | ||||
| ### **ストリーミングメディアロック解除** | ||||
| ### ストリーミングメディアロック解除 | ||||
|  | ||||
| 一般的なストリーミングメディアプラットフォームのロック解除を検索します。もちろん、すべてがストリーミングメディアというわけではなく、他の一般的なプラットフォームのロック解除も含まれています。一般的に、IPのロック解除地域は一貫しており、あちこちに変動することはありません。複数のプラットフォームでロック解除地域が一致しない場合、そのIPはIPXOなどのプラットフォームからレンタルされている可能性が高く、各プラットフォームのデータベース識別が遅いため、IP品質も一般的に良くないと考えられます。 | ||||
| 依存プロジェクト:[https://github.com/oneclickvirt/CommonMediaTests](https://github.com/oneclickvirt/CommonMediaTests) [https://github.com/lmc999/RegionRestrictionCheck](https://github.com/lmc999/RegionRestrictionCheck) | ||||
|  | ||||
| ### **IP品質検出** | ||||
| デフォルトでは国境を越えるストリーミングメディアのロック解除のみをチェックします。 | ||||
|  | ||||
| 14のデータベースのIP関連情報を検出します。一般的に、使用タイプ、会社タイプ、およびその他のセキュリティ情報の判別を見るだけで十分です。セキュリティスコアは参考程度です。複数のプラットフォームで対応する検出項目が一致している場合、現在のIPが確かにそうであることを証明しています。単一のデータベースソースの情報だけを信頼しないでください。 | ||||
| 一般的に、正常な状況下では、一つのIPの複数のストリーミングメディアのロック解除地域はすべて一致し、あちこち飛び回ることはありません。複数のプラットフォームでロック解除地域が一致しない場合、IPはIPXOなどのプラットフォームからのレンタルか、最近宣告され使用されたもので、ストリーミングメディアの一般的なデータベースに認識修正されていない可能性が高いです。各プラットフォームのIPデータベース認識速度が一致しないため、時々あるプラットフォームではロック解除地域が正常、あるプラットフォームではルート上のある位置に飛ぶ、あるプラットフォームではIPがあなたによって使用される前にいた位置に飛ぶことがあります。 | ||||
|  | ||||
| ### **メールポート検出** | ||||
| | DNS タイプ | ロック解除方式判断の必要性 | DNSのロック解除への影響 | 説明 | | ||||
| | --------- | ------------------------- | ---------------------- | ---- | | ||||
| | 公式主流DNS | 不要 | 小 | ストリーミングメディアのロック解除は主にノードIPに依存し、DNS解析は基本的にロック解除を干渉しない | | ||||
| | 非主流/自建DNS | 必要 | 大 | ストリーミングメディアのロック解除結果はDNS解析の影響を大きく受け、ネイティブロック解除かDNSロック解除かを判断する必要がある | | ||||
|  | ||||
| - **SMTP(25)**:メールサーバー間でメールを転送するために使用されます(メール送信)。 | ||||
| - **SMTPS(465)**:暗号化されたSMTPメール送信(SSL/TLS方式)に使用されます。 | ||||
| - **SMTP(587)**:クライアントからメールサーバーへのメール送信に使用され、STARTTLS暗号化をサポートします。 | ||||
| - **POP3(110)**:メールクライアントがサーバーからメールをダウンロードするために使用され、暗号化されていません。 | ||||
| - **POP3S(995)**:暗号化されたPOP3用で、安全にメールをダウンロードします(SSL/TLS方式)。 | ||||
| - **IMAP(143)**:メールクライアントがオンラインでメールを管理するために使用されます(メールの閲覧、同期)、暗号化されていません。 | ||||
| - **IMAPS(993)**:暗号化されたIMAP用で、安全にメールを管理します(SSL/TLS方式)。 | ||||
| そのため、テスト過程で、ホストが現在使用しているのが公式主流のDNSの場合、ネイティブロック解除かどうかの判断は行われません。 | ||||
|  | ||||
| 具体的に現在のホストマシンがメールサーバーとして使用されていない、または電子メールを受信しない場合、この項目の指標は気にする必要はありません。 | ||||
| ### IP品質検出 | ||||
|  | ||||
| --- | ||||
| 依存プロジェクト:[https://github.com/oneclickvirt/securityCheck](https://github.com/oneclickvirt/securityCheck) | ||||
|  | ||||
| 14のデータベースのIP関連情報を検出します。一般的に、使用タイプと会社タイプ、そしてセキュリティ情報のその他識別を見れば十分で、セキュリティスコアは本当にお遊びです。複数のプラットフォームで対応する検出項目がすべて対応する値になっている場合、現在のIPが確実にそうであることを証明します。一つのデータベースソースの情報のみを信じてはいけません。 | ||||
|  | ||||
| * **使用タイプ & 会社タイプ**: IP帰属と使用シナリオを表示し、例えば家庭ユーザー、企業オフィス、ホスティングサービス、またはクラウド/データセンターに属するかどうか。 | ||||
| * **クラウドプロバイダー / データセンター / モバイルデバイス**: IPがクラウドサービス、データセンター、またはモバイルネットワークから来ているかを判断し、共有または高リスクIPの識別に役立つ。 | ||||
| * **プロキシ / VPN / Tor / Tor出口**: IPが真の身元や位置を隠すために使用されているかを検出し、匿名アクセスや悪用行為に関与している可能性がある。 | ||||
| * **ネットワーククローラー / ロボット**: 自動化されたアクセスまたは収集プログラムを識別し、セキュリティリスク評価に参考価値がある。 | ||||
| * **匿名 / 悪用者 / 脅威 / 中継 / Bogon**: IP履歴行動特徴と予約/未割り当てIPに属するかどうかを表示し、IP信頼度判断を補助。 | ||||
| * **セキュリティスコア、評判、信頼スコア、脅威スコア、詐欺スコア、悪用スコア**: 各データベースのIPに対する定量化されたセキュリティ評価、参考のみ。 | ||||
| * **コミュニティ投票 & ブラックリスト記録**: ユーザーフィードバックと公共ブラックリスト情報を展示し、潜在的リスクを迅速に識別可能。 | ||||
| * **Google検索実行可能性**: IPがGoogle検索サービスにアクセスする実行可能性を検出し、ネットワーク制限やブロック状況を間接的に反映。 | ||||
|  | ||||
| マルチプラットフォーム比較がより信頼性が高く、異なるデータベースのアルゴリズムと更新頻度が異なるため、単一ソースは誤判断の可能性があります。複数のデータベースが類似の結果を示す場合、その結果はより信頼性が高いことを説明します。 | ||||
|  | ||||
| ### メールポート検出 | ||||
|  | ||||
| 依存プロジェクト:[https://github.com/oneclickvirt/portchecker](https://github.com/oneclickvirt/portchecker) | ||||
|  | ||||
| - **SMTP (25)**: メールサーバー間でのメール転送(メール送信)に使用。 | ||||
| - **SMTPS (465)**: 暗号化されたSMTPメール送信(SSL/TLS方式)に使用。 | ||||
| - **SMTP (587)**: クライアントからメールサーバーへのメール送信、STARTTLS暗号化をサポート。 | ||||
| - **POP3 (110)**: メールクライアントがサーバーからメールをダウンロードするために使用、暗号化なし。 | ||||
| - **POP3S (995)**: 暗号化されたPOP3、安全にメールをダウンロード(SSL/TLS方式)に使用。 | ||||
| - **IMAP (143)**: メールクライアントがオンラインでメール管理(メール閲覧、同期)、暗号化なし。 | ||||
| - **IMAPS (993)**: 暗号化されたIMAP、安全にメール管理(SSL/TLS方式)に使用。 | ||||
|  | ||||
| 現在のホストがメール局として機能せず、電子メールの送受信を行わない場合、この項目指標は無視して構いません。 | ||||
|  | ||||
| ### 近隣スピードテスト | ||||
|  | ||||
| 依存プロジェクト:[https://github.com/oneclickvirt/speedtest](https://github.com/oneclickvirt/speedtest) | ||||
|  | ||||
| まず公式推奨の測定ポイントをテストし、次に代表的な国際測定ポイントをテストします。 | ||||
|  | ||||
| 公式測定ポイントは、テスト対象のホストマシンのローカル帯域幅ベースラインを表すことができます。 | ||||
|  | ||||
| 日常的には1Gbps帯域幅のサーバーを使用することを好みます。少なくとも依存関係のダウンロードなどの速度が十分に速いからです。 | ||||
|   | ||||
| @@ -8,18 +8,10 @@ import ( | ||||
| ) | ||||
|  | ||||
| func DiskTest(language, testMethod, testPath string, isMultiCheck bool, autoChange bool) (realTestMethod, res string) { | ||||
| 	if runtime.GOOS == "windows" { | ||||
| 		if testMethod != "winsat" && testMethod != "" { | ||||
| 			// res = "Detected host is Windows, using Winsat for testing.\n" | ||||
| 			realTestMethod = "winsat" | ||||
| 		} | ||||
| 		res = disk.WinsatTest(language, isMultiCheck, testPath) | ||||
| 	} else { | ||||
| 	switch testMethod { | ||||
| 	case "fio": | ||||
| 		res = disk.FioTest(language, isMultiCheck, testPath) | ||||
| 		if res == "" && autoChange { | ||||
| 				// res = "Fio test failed, switching to DD for testing.\n" | ||||
| 			res += disk.DDTest(language, isMultiCheck, testPath) | ||||
| 			realTestMethod = "dd" | ||||
| 		} else { | ||||
| @@ -28,15 +20,17 @@ func DiskTest(language, testMethod, testPath string, isMultiCheck bool, autoChan | ||||
| 	case "dd": | ||||
| 		res = disk.DDTest(language, isMultiCheck, testPath) | ||||
| 		if res == "" && autoChange { | ||||
| 				// res = "DD test failed, switching to Fio for testing.\n" | ||||
| 			res += disk.FioTest(language, isMultiCheck, testPath) | ||||
| 			realTestMethod = "fio" | ||||
| 		} else { | ||||
| 			realTestMethod = "dd" | ||||
| 		} | ||||
| 	default: | ||||
| 			// res = "Unsupported test method specified, switching to DD for testing.\n" | ||||
| 			res += disk.DDTest(language, isMultiCheck, testPath) | ||||
| 		if runtime.GOOS == "windows" { | ||||
| 			realTestMethod = "winsat" | ||||
| 			res = disk.WinsatTest(language, isMultiCheck, testPath) | ||||
| 		} else { | ||||
| 			res = disk.DDTest(language, isMultiCheck, testPath) | ||||
| 			realTestMethod = "dd" | ||||
| 		} | ||||
| 	} | ||||
|   | ||||
							
								
								
									
										85
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										85
									
								
								go.mod
									
									
									
									
									
								
							| @@ -3,48 +3,46 @@ module github.com/oneclickvirt/ecs | ||||
| go 1.24.5 | ||||
|  | ||||
| require ( | ||||
| 	github.com/imroc/req/v3 v3.50.0 | ||||
| 	github.com/imroc/req/v3 v3.54.0 | ||||
| 	github.com/oneclickvirt/CommonMediaTests v0.0.4-20250329123841 | ||||
| 	github.com/oneclickvirt/UnlockTests v0.0.27-20250628125053 | ||||
| 	github.com/oneclickvirt/backtrace v0.0.5-20250629024536 | ||||
| 	github.com/oneclickvirt/basics v0.0.15-20250714163009 | ||||
| 	github.com/oneclickvirt/cputest v0.0.11-20250717145400 | ||||
| 	github.com/oneclickvirt/UnlockTests v0.0.28-20250924054500 | ||||
| 	github.com/oneclickvirt/backtrace v0.0.7-20250811023541 | ||||
| 	github.com/oneclickvirt/basics v0.0.15-20250812130523 | ||||
| 	github.com/oneclickvirt/cputest v0.0.12-20250720122317 | ||||
| 	github.com/oneclickvirt/defaultset v0.0.2-20240624082446 | ||||
| 	github.com/oneclickvirt/disktest v0.0.8-20250701092629 | ||||
| 	github.com/oneclickvirt/gostun v0.0.3-20250329105202 | ||||
| 	github.com/oneclickvirt/memorytest v0.0.8-20250717152547 | ||||
| 	github.com/oneclickvirt/nt3 v0.0.5-20250416131047 | ||||
| 	github.com/oneclickvirt/pingtest v0.0.8-20250701125637 | ||||
| 	github.com/oneclickvirt/portchecker v0.0.3-20250329125750 | ||||
| 	github.com/oneclickvirt/security v0.0.6-20250715102027 | ||||
| 	github.com/oneclickvirt/speedtest v0.0.10-20250701123931 | ||||
| 	github.com/oneclickvirt/disktest v0.0.10-20250924030424 | ||||
| 	github.com/oneclickvirt/gostun v0.0.5-20250727155022 | ||||
| 	github.com/oneclickvirt/memorytest v0.0.10-20250924154648 | ||||
| 	github.com/oneclickvirt/nt3 v0.0.8-20250811123903 | ||||
| 	github.com/oneclickvirt/pingtest v0.0.8-20250728015259 | ||||
| 	github.com/oneclickvirt/portchecker v0.0.3-20250728015900 | ||||
| 	github.com/oneclickvirt/speedtest v0.0.10-20250728015734 | ||||
| ) | ||||
|  | ||||
| require ( | ||||
| 	github.com/PuerkitoBio/goquery v1.9.2 // indirect | ||||
| 	github.com/StackExchange/wmi v1.2.1 // indirect | ||||
| 	github.com/andybalholm/brotli v1.1.1 // indirect | ||||
| 	github.com/andybalholm/brotli v1.2.0 // indirect | ||||
| 	github.com/andybalholm/cascadia v1.3.2 // indirect | ||||
| 	github.com/cloudflare/circl v1.5.0 // indirect | ||||
| 	github.com/cloudflare/circl v1.6.1 // indirect | ||||
| 	github.com/ebitengine/purego v0.8.4 // indirect | ||||
| 	github.com/fatih/color v1.18.0 // indirect | ||||
| 	github.com/fsnotify/fsnotify v1.9.0 // indirect | ||||
| 	github.com/ghodss/yaml v1.0.0 // indirect | ||||
| 	github.com/go-ole/go-ole v1.2.6 // indirect | ||||
| 	github.com/go-task/slim-sprig/v3 v3.0.0 // indirect | ||||
| 	github.com/go-viper/mapstructure/v2 v2.2.1 // indirect | ||||
| 	github.com/go-viper/mapstructure/v2 v2.4.0 // indirect | ||||
| 	github.com/gofrs/uuid/v5 v5.2.0 // indirect | ||||
| 	github.com/google/gopacket v1.1.19 // indirect | ||||
| 	github.com/google/pprof v0.0.0-20241210010833-40e02aabc2ad // indirect | ||||
| 	github.com/google/uuid v1.6.0 // indirect | ||||
| 	github.com/gorilla/websocket v1.5.3 // indirect | ||||
| 	github.com/hashicorp/errwrap v1.1.0 // indirect | ||||
| 	github.com/hashicorp/go-multierror v1.1.1 // indirect | ||||
| 	github.com/huin/goupnp v1.2.0 // indirect | ||||
| 	github.com/icholy/digest v1.1.0 // indirect | ||||
| 	github.com/jackpal/go-nat-pmp v1.0.2 // indirect | ||||
| 	github.com/jaypipes/ghw v0.12.0 // indirect | ||||
| 	github.com/jaypipes/pcidb v1.0.0 // indirect | ||||
| 	github.com/jaypipes/ghw v0.17.0 // indirect | ||||
| 	github.com/jaypipes/pcidb v1.0.1 // indirect | ||||
| 	github.com/json-iterator/go v1.1.12 // indirect | ||||
| 	github.com/klauspost/compress v1.17.11 // indirect | ||||
| 	github.com/klauspost/compress v1.18.0 // indirect | ||||
| 	github.com/koron/go-ssdp v0.0.4 // indirect | ||||
| 	github.com/libp2p/go-nat v0.2.0 // indirect | ||||
| 	github.com/libp2p/go-netroute v0.2.1 // indirect | ||||
| @@ -58,11 +56,11 @@ require ( | ||||
| 	github.com/mitchellh/go-homedir v1.1.0 // indirect | ||||
| 	github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd // indirect | ||||
| 	github.com/modern-go/reflect2 v1.0.2 // indirect | ||||
| 	github.com/nxtrace/NTrace-core v1.4.0 // indirect | ||||
| 	github.com/oneclickvirt/dd v0.0.2-20250701085922 // indirect | ||||
| 	github.com/oneclickvirt/fio v0.0.2-20250701085933 // indirect | ||||
| 	github.com/oneclickvirt/mbw v0.0.1-20250630140849 // indirect | ||||
| 	github.com/onsi/ginkgo/v2 v2.22.1 // indirect | ||||
| 	github.com/nxtrace/NTrace-core v1.4.2 // indirect | ||||
| 	github.com/oneclickvirt/dd v0.0.2-20250808062818 // indirect | ||||
| 	github.com/oneclickvirt/fio v0.0.2-20250808045755 // indirect | ||||
| 	github.com/oneclickvirt/mbw v0.0.1-20250808061222 // indirect | ||||
| 	github.com/oneclickvirt/stream v0.0.2-20250924154001 // indirect | ||||
| 	github.com/oschwald/maxminddb-golang v1.13.1 // indirect | ||||
| 	github.com/pelletier/go-toml/v2 v2.2.4 // indirect | ||||
| 	github.com/pion/dtls/v2 v2.2.7 // indirect | ||||
| @@ -74,20 +72,19 @@ require ( | ||||
| 	github.com/power-devops/perfstat v0.0.0-20210106213030-5aafc221ea8c // indirect | ||||
| 	github.com/prometheus-community/pro-bing v0.4.1 // indirect | ||||
| 	github.com/quic-go/qpack v0.5.1 // indirect | ||||
| 	github.com/quic-go/quic-go v0.48.2 // indirect | ||||
| 	github.com/refraction-networking/utls v1.6.7 // indirect | ||||
| 	github.com/quic-go/quic-go v0.53.0 // indirect | ||||
| 	github.com/refraction-networking/utls v1.7.3 // indirect | ||||
| 	github.com/rivo/uniseg v0.4.7 // indirect | ||||
| 	github.com/rodaine/table v1.3.0 // indirect | ||||
| 	github.com/sagikazarmark/locafero v0.9.0 // indirect | ||||
| 	github.com/schollz/progressbar/v3 v3.14.4 // indirect | ||||
| 	github.com/shirou/gopsutil v3.21.11+incompatible // indirect | ||||
| 	github.com/shirou/gopsutil/v4 v4.24.5 // indirect | ||||
| 	github.com/shoenig/go-m1cpu v0.1.6 // indirect | ||||
| 	github.com/showwin/speedtest-go v1.7.7 // indirect | ||||
| 	github.com/shirou/gopsutil/v4 v4.25.6 // indirect | ||||
| 	github.com/showwin/speedtest-go v1.7.10 // indirect | ||||
| 	github.com/sourcegraph/conc v0.3.0 // indirect | ||||
| 	github.com/spf13/afero v1.14.0 // indirect | ||||
| 	github.com/spf13/cast v1.7.1 // indirect | ||||
| 	github.com/spf13/pflag v1.0.6 // indirect | ||||
| 	github.com/spf13/cast v1.9.2 // indirect | ||||
| 	github.com/spf13/pflag v1.0.7 // indirect | ||||
| 	github.com/spf13/viper v1.20.1 // indirect | ||||
| 	github.com/subosito/gotenv v1.6.0 // indirect | ||||
| 	github.com/tidwall/gjson v1.18.0 // indirect | ||||
| @@ -97,19 +94,17 @@ require ( | ||||
| 	github.com/tklauser/numcpus v0.8.0 // indirect | ||||
| 	github.com/tsosunchia/powclient v0.1.5 // indirect | ||||
| 	github.com/yusufpapurcu/wmi v1.2.4 // indirect | ||||
| 	go.uber.org/mock v0.5.0 // indirect | ||||
| 	go.uber.org/mock v0.5.2 // indirect | ||||
| 	go.uber.org/multierr v1.11.0 // indirect | ||||
| 	go.uber.org/zap v1.27.0 // indirect | ||||
| 	golang.org/x/crypto v0.37.0 // indirect | ||||
| 	golang.org/x/exp v0.0.0-20250106191152-7588d65b2ba8 // indirect | ||||
| 	golang.org/x/mod v0.22.0 // indirect | ||||
| 	golang.org/x/net v0.39.0 // indirect | ||||
| 	golang.org/x/sync v0.13.0 // indirect | ||||
| 	golang.org/x/sys v0.33.0 // indirect | ||||
| 	golang.org/x/term v0.31.0 // indirect | ||||
| 	golang.org/x/text v0.24.0 // indirect | ||||
| 	golang.org/x/tools v0.29.0 // indirect | ||||
| 	gopkg.in/yaml.v2 v2.4.0 // indirect | ||||
| 	golang.org/x/crypto v0.40.0 // indirect | ||||
| 	golang.org/x/mod v0.25.0 // indirect | ||||
| 	golang.org/x/net v0.42.0 // indirect | ||||
| 	golang.org/x/sync v0.16.0 // indirect | ||||
| 	golang.org/x/sys v0.34.0 // indirect | ||||
| 	golang.org/x/term v0.33.0 // indirect | ||||
| 	golang.org/x/text v0.27.0 // indirect | ||||
| 	golang.org/x/tools v0.34.0 // indirect | ||||
| 	gopkg.in/yaml.v3 v3.0.1 // indirect | ||||
| 	howett.net/plist v1.0.0 // indirect | ||||
| ) | ||||
|   | ||||
							
								
								
									
										185
									
								
								go.sum
									
									
									
									
									
								
							
							
						
						
									
										185
									
								
								go.sum
									
									
									
									
									
								
							| @@ -2,43 +2,38 @@ github.com/PuerkitoBio/goquery v1.9.2 h1:4/wZksC3KgkQw7SQgkKotmKljk0M6V8TUvA8Wb4 | ||||
| github.com/PuerkitoBio/goquery v1.9.2/go.mod h1:GHPCaP0ODyyxqcNoFGYlAprUFH81NuRPd0GX3Zu2Mvk= | ||||
| github.com/StackExchange/wmi v1.2.1 h1:VIkavFPXSjcnS+O8yTq7NI32k0R5Aj+v39y29VYDOSA= | ||||
| github.com/StackExchange/wmi v1.2.1/go.mod h1:rcmrprowKIVzvc+NUiLncP2uuArMWLCbu9SBzvHz7e8= | ||||
| github.com/andybalholm/brotli v1.1.1 h1:PR2pgnyFznKEugtsUo0xLdDop5SKXd5Qf5ysW+7XdTA= | ||||
| github.com/andybalholm/brotli v1.1.1/go.mod h1:05ib4cKhjx3OQYUY22hTVd34Bc8upXjOLL2rKwwZBoA= | ||||
| github.com/andybalholm/brotli v1.2.0 h1:ukwgCxwYrmACq68yiUqwIWnGY0cTPox/M94sVwToPjQ= | ||||
| github.com/andybalholm/brotli v1.2.0/go.mod h1:rzTDkvFWvIrjDXZHkuS16NPggd91W3kUSvPlQ1pLaKY= | ||||
| github.com/andybalholm/cascadia v1.3.2 h1:3Xi6Dw5lHF15JtdcmAHD3i1+T8plmv7BQ/nsViSLyss= | ||||
| github.com/andybalholm/cascadia v1.3.2/go.mod h1:7gtRlve5FxPPgIgX36uWBX58OdBsSS6lUvCFb+h7KvU= | ||||
| github.com/cloudflare/circl v1.5.0 h1:hxIWksrX6XN5a1L2TI/h53AGPhNHoUBo+TD1ms9+pys= | ||||
| github.com/cloudflare/circl v1.5.0/go.mod h1:uddAzsPgqdMAYatqJ0lsjX1oECcQLIlRpzZh3pJrofs= | ||||
| github.com/cloudflare/circl v1.6.1 h1:zqIqSPIndyBh1bjLVVDHMPpVKqp8Su/V+6MeDzzQBQ0= | ||||
| github.com/cloudflare/circl v1.6.1/go.mod h1:uddAzsPgqdMAYatqJ0lsjX1oECcQLIlRpzZh3pJrofs= | ||||
| github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | ||||
| github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | ||||
| github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc h1:U9qPSI2PIWSS1VwoXQT9A3Wy9MM3WgvqSxFWenqJduM= | ||||
| github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | ||||
| github.com/ebitengine/purego v0.8.4 h1:CF7LEKg5FFOsASUj0+QwaXf8Ht6TlFxg09+S9wz0omw= | ||||
| github.com/ebitengine/purego v0.8.4/go.mod h1:iIjxzd6CiRiOG0UyXP+V1+jWqUXVjPKLAI0mRfJZTmQ= | ||||
| github.com/fatih/color v1.18.0 h1:S8gINlzdQ840/4pfAwic/ZE0djQEH3wM94VfqLTZcOM= | ||||
| github.com/fatih/color v1.18.0/go.mod h1:4FelSpRwEGDpQ12mAdzqdOukCy4u8WUtOY6lkT/6HfU= | ||||
| github.com/frankban/quicktest v1.14.6 h1:7Xjx+VpznH+oBnejlPUj8oUpdxnVs4f8XU8WnHkI4W8= | ||||
| github.com/frankban/quicktest v1.14.6/go.mod h1:4ptaffx2x8+WTWXmUCuVU6aPUX1/Mz7zb5vbUoiM6w0= | ||||
| github.com/fsnotify/fsnotify v1.9.0 h1:2Ml+OJNzbYCTzsxtv8vKSFD9PbJjmhYF14k/jKC7S9k= | ||||
| github.com/fsnotify/fsnotify v1.9.0/go.mod h1:8jBTzvmWwFyi3Pb8djgCCO5IBqzKJ/Jwo8TRcHyHii0= | ||||
| github.com/ghodss/yaml v1.0.0 h1:wQHKEahhL6wmXdzwWG11gIVCkOv05bNOh+Rxn0yngAk= | ||||
| github.com/ghodss/yaml v1.0.0/go.mod h1:4dBDuWmgqj2HViK6kFavaiC9ZROes6MMH2rRYeMEF04= | ||||
| github.com/go-logr/logr v1.4.2 h1:6pFjapn8bFcIbiKo3XT4j/BhANplGihG6tvd+8rYgrY= | ||||
| github.com/go-logr/logr v1.4.2/go.mod h1:9T104GzyrTigFIr8wt5mBrctHMim0Nb2HLGrmQ40KvY= | ||||
| github.com/go-ole/go-ole v1.2.5/go.mod h1:pprOEPIfldk/42T2oK7lQ4v4JSDwmV0As9GaiUsvbm0= | ||||
| github.com/go-ole/go-ole v1.2.6 h1:/Fpf6oFPoeFik9ty7siob0G6Ke8QvQEuVcuChpwXzpY= | ||||
| github.com/go-ole/go-ole v1.2.6/go.mod h1:pprOEPIfldk/42T2oK7lQ4v4JSDwmV0As9GaiUsvbm0= | ||||
| github.com/go-task/slim-sprig/v3 v3.0.0 h1:sUs3vkvUymDpBKi3qH1YSqBQk9+9D/8M2mN1vB6EwHI= | ||||
| github.com/go-task/slim-sprig/v3 v3.0.0/go.mod h1:W848ghGpv3Qj3dhTPRyJypKRiqCdHZiAzKg9hl15HA8= | ||||
| github.com/go-viper/mapstructure/v2 v2.2.1 h1:ZAaOCxANMuZx5RCeg0mBdEZk7DZasvvZIxtHqx8aGss= | ||||
| github.com/go-viper/mapstructure/v2 v2.2.1/go.mod h1:oJDH3BJKyqBA2TXFhDsKDGDTlndYOZ6rGS0BRZIxGhM= | ||||
| github.com/go-viper/mapstructure/v2 v2.4.0 h1:EBsztssimR/CONLSZZ04E8qAkxNYq4Qp9LvH92wZUgs= | ||||
| github.com/go-viper/mapstructure/v2 v2.4.0/go.mod h1:oJDH3BJKyqBA2TXFhDsKDGDTlndYOZ6rGS0BRZIxGhM= | ||||
| github.com/gofrs/uuid/v5 v5.2.0 h1:qw1GMx6/y8vhVsx626ImfKMuS5CvJmhIKKtuyvfajMM= | ||||
| github.com/gofrs/uuid/v5 v5.2.0/go.mod h1:CDOjlDMVAtN56jqyRUZh58JT31Tiw7/oQyEXZV+9bD8= | ||||
| github.com/google/go-cmp v0.5.6/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | ||||
| github.com/google/go-cmp v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI= | ||||
| github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= | ||||
| github.com/google/go-cmp v0.7.0 h1:wk8382ETsv4JYUZwIsn6YpYiWiBsYLSJiTsyBybVuN8= | ||||
| github.com/google/go-cmp v0.7.0/go.mod h1:pXiqmnSA92OHEEa9HXL2W4E7lf9JzCmGVUdgjX3N/iU= | ||||
| github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg= | ||||
| github.com/google/gopacket v1.1.19 h1:ves8RnFZPGiFnTS0uPQStjwru6uO6h+nlr9j6fL7kF8= | ||||
| github.com/google/gopacket v1.1.19/go.mod h1:iJ8V8n6KS+z2U1A8pUwu8bW5SyEMkXJB8Yo/Vo+TKTo= | ||||
| github.com/google/pprof v0.0.0-20241210010833-40e02aabc2ad h1:a6HEuzUHeKH6hwfN/ZoQgRgVIWFJljSWa/zetS2WTvg= | ||||
| github.com/google/pprof v0.0.0-20241210010833-40e02aabc2ad/go.mod h1:vavhavw2zAxS5dIdcRluK6cSGGPlZynqzFM8NdvU144= | ||||
| github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0= | ||||
| github.com/google/uuid v1.6.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= | ||||
| github.com/gorilla/websocket v1.5.3 h1:saDtZ6Pbx/0u+bgYQ3q96pZgCzfhKXGPqt7kZ72aNNg= | ||||
| @@ -50,20 +45,22 @@ github.com/hashicorp/go-multierror v1.1.1 h1:H5DkEtf6CXdFp0N0Em5UCwQpXMWke8IA0+l | ||||
| github.com/hashicorp/go-multierror v1.1.1/go.mod h1:iw975J/qwKPdAO1clOe2L8331t/9/fmwbPZ6JB6eMoM= | ||||
| github.com/huin/goupnp v1.2.0 h1:uOKW26NG1hsSSbXIZ1IR7XP9Gjd1U8pnLaCMgntmkmY= | ||||
| github.com/huin/goupnp v1.2.0/go.mod h1:gnGPsThkYa7bFi/KWmEysQRf48l2dvR5bxr2OFckNX8= | ||||
| github.com/imroc/req/v3 v3.50.0 h1:n3BVnZiTRpvkN5T1IB79LC/THhFU9iXksNRMH4ZNVaY= | ||||
| github.com/imroc/req/v3 v3.50.0/go.mod h1:tsOk8K7zI6cU4xu/VWCZVtq9Djw9IWm4MslKzme5woU= | ||||
| github.com/icholy/digest v1.1.0 h1:HfGg9Irj7i+IX1o1QAmPfIBNu/Q5A5Tu3n/MED9k9H4= | ||||
| github.com/icholy/digest v1.1.0/go.mod h1:QNrsSGQ5v7v9cReDI0+eyjsXGUoRSUZQHeQ5C4XLa0Y= | ||||
| github.com/imroc/req/v3 v3.54.0 h1:kwWJSpT7OvjJ/Q8ykp+69Ye5H486RKDcgEoepw1Ren4= | ||||
| github.com/imroc/req/v3 v3.54.0/go.mod h1:P8gCJjG/XNUFeP6WOi40VAXfYwT+uPM00xvoBWiwzUQ= | ||||
| github.com/jackpal/go-nat-pmp v1.0.2 h1:KzKSgb7qkJvOUTqYl9/Hg/me3pWgBmERKrTGD7BdWus= | ||||
| github.com/jackpal/go-nat-pmp v1.0.2/go.mod h1:QPH045xvCAeXUZOxsnwmrtiCoxIr9eob+4orBN1SBKc= | ||||
| github.com/jaypipes/ghw v0.12.0 h1:xU2/MDJfWmBhJnujHY9qwXQLs3DBsf0/Xa9vECY0Tho= | ||||
| github.com/jaypipes/ghw v0.12.0/go.mod h1:jeJGbkRB2lL3/gxYzNYzEDETV1ZJ56OKr+CSeSEym+g= | ||||
| github.com/jaypipes/pcidb v1.0.0 h1:vtZIfkiCUE42oYbJS0TAq9XSfSmcsgo9IdxSm9qzYU8= | ||||
| github.com/jaypipes/pcidb v1.0.0/go.mod h1:TnYUvqhPBzCKnH34KrIX22kAeEbDCSRJ9cqLRCuNDfk= | ||||
| github.com/jaypipes/ghw v0.17.0 h1:EVLJeNcy5z6GK/Lqby0EhBpynZo+ayl8iJWY0kbEUJA= | ||||
| github.com/jaypipes/ghw v0.17.0/go.mod h1:In8SsaDqlb1oTyrbmTC14uy+fbBMvp+xdqX51MidlD8= | ||||
| github.com/jaypipes/pcidb v1.0.1 h1:WB2zh27T3nwg8AE8ei81sNRb9yWBii3JGNJtT7K9Oic= | ||||
| github.com/jaypipes/pcidb v1.0.1/go.mod h1:6xYUz/yYEyOkIkUt2t2J2folIuZ4Yg6uByCGFXMCeE4= | ||||
| github.com/jessevdk/go-flags v1.4.0/go.mod h1:4FA24M0QyGHXBuZZK/XkWh8h0e1EYbRYJSGM75WSRxI= | ||||
| github.com/json-iterator/go v1.1.12 h1:PV8peI4a0ysnczrg+LtxykD8LfKY9ML6u2jnxaEnrnM= | ||||
| github.com/json-iterator/go v1.1.12/go.mod h1:e30LSqwooZae/UwlEbR2852Gd8hjQvJoHmT4TnhNGBo= | ||||
| github.com/k0kubun/go-ansi v0.0.0-20180517002512-3bf9e2903213/go.mod h1:vNUNkEQ1e29fT/6vq2aBdFsgNPmy8qMdSay1npru+Sw= | ||||
| github.com/klauspost/compress v1.17.11 h1:In6xLpyWOi1+C7tXUUWv2ot1QvBjxevKAaI6IXrJmUc= | ||||
| github.com/klauspost/compress v1.17.11/go.mod h1:pMDklpSncoRMuLFrf1W9Ss9KT+0rH90U12bZKk7uwG0= | ||||
| github.com/klauspost/compress v1.18.0 h1:c/Cqfb0r+Yi+JtIEq73FWXVkRonBlf0CRNYc8Zttxdo= | ||||
| github.com/klauspost/compress v1.18.0/go.mod h1:2Pp+KzxcywXVXMr50+X0Q/Lsb43OQHYWRCY2AiWywWQ= | ||||
| github.com/koron/go-ssdp v0.0.4 h1:1IDwrghSKYM7yLf7XCzbByg2sJ/JcNOZRXS2jczTwz0= | ||||
| github.com/koron/go-ssdp v0.0.4/go.mod h1:oDXq+E5IL5q0U8uSBcoAXzTzInwy5lEgC91HoKtbmZk= | ||||
| github.com/kr/pretty v0.3.1 h1:flRD4NNwYAUpkphVc1HcthR4KEIFJ65n8Mw5qdRn3LE= | ||||
| @@ -95,46 +92,42 @@ github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd h1:TRLaZ9cD/w | ||||
| github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q= | ||||
| github.com/modern-go/reflect2 v1.0.2 h1:xBagoLtFs94CBntxluKeaWgTMpvLxC4ur3nMaC9Gz0M= | ||||
| github.com/modern-go/reflect2 v1.0.2/go.mod h1:yWuevngMOJpCy52FWWMvUC8ws7m/LJsjYzDa0/r8luk= | ||||
| github.com/nxtrace/NTrace-core v1.4.0 h1:pDN2BqxIYjedDKCDDOFBcDNOBnavGcx+4wbiG65Xqiw= | ||||
| github.com/nxtrace/NTrace-core v1.4.0/go.mod h1:0AWqbqiIJbpbFG6W48vtJ6pWM8PPF+lQ1fi2371y+zA= | ||||
| github.com/nxtrace/NTrace-core v1.4.2 h1:dSRP18Bn3VGf5CZBzKt8gQWW9mDkq62Np9TCF9RAtp0= | ||||
| github.com/nxtrace/NTrace-core v1.4.2/go.mod h1:wIDOlccuYzY3wBqU89pv2KGHT41i3JA0eRqJU/x9eX4= | ||||
| github.com/oneclickvirt/CommonMediaTests v0.0.4-20250329123841 h1:Zef93z9UiZQwRAKnnZYALmpBKvvuVaq34MEsuWwk6nc= | ||||
| github.com/oneclickvirt/CommonMediaTests v0.0.4-20250329123841/go.mod h1:DAmFPRjFV5p9fEzUUSml5jJGn2f1NZJQCzTxITHDjc4= | ||||
| github.com/oneclickvirt/UnlockTests v0.0.27-20250628125053 h1:Ug8kySZR1weRUcsnGOv+f3HAl791AfkA7EWV3JmiMQA= | ||||
| github.com/oneclickvirt/UnlockTests v0.0.27-20250628125053/go.mod h1:yXWIZB6iLS88pEd9m4QJi1GENn+7I91zA72y5ONz2Oc= | ||||
| github.com/oneclickvirt/backtrace v0.0.5-20250629024536 h1:caHCa0DHmbYWBFN1bqKxpvPnN0wOxDEqJv1VDvDdLWs= | ||||
| github.com/oneclickvirt/backtrace v0.0.5-20250629024536/go.mod h1:5AH00bo41hH3d2/JVuCTlBkZUs3AXX4nlKVXb6piZcI= | ||||
| github.com/oneclickvirt/basics v0.0.15-20250714163009 h1:7I1lU7N91kClw6Cb6o+vGfScc/HngrjhsaaW10AbBFs= | ||||
| github.com/oneclickvirt/basics v0.0.15-20250714163009/go.mod h1:yN1IEOXN6v/GJqJSA70Pooo6nXBI/6rq72vTY72wJMQ= | ||||
| github.com/oneclickvirt/cputest v0.0.11-20250717145400 h1:yExJjBdHQNil37/Ai0SKQB1Dlrq9qmkvoh7Zn68U+t0= | ||||
| github.com/oneclickvirt/cputest v0.0.11-20250717145400/go.mod h1:vjlH8tkPFft1tlLOpeNskXVvurxkHaJ3+dgFxQGLXY4= | ||||
| github.com/oneclickvirt/dd v0.0.2-20250701085922 h1:WiWZwcnCPhRc8hLZdvkjD2kOEpnqn1S31z1j0x3V4l0= | ||||
| github.com/oneclickvirt/dd v0.0.2-20250701085922/go.mod h1:tImu9sPTkLWo2tf1dEN1xQzrylWKauj9hbU8PHfyAeU= | ||||
| github.com/oneclickvirt/UnlockTests v0.0.28-20250924054500 h1:ERFoRBYhTPWJBYhEVFWr3hm6KtSTUHuWD21jK7DhKZw= | ||||
| github.com/oneclickvirt/UnlockTests v0.0.28-20250924054500/go.mod h1:oOa6wj/qECtRMxwBO6D7o0L0F0Q/5sQ747OCnFQqoGE= | ||||
| github.com/oneclickvirt/backtrace v0.0.7-20250811023541 h1:GzkzvUC6U9b6Dkz/Bl4JRPeQ7XBGoW7Qw1aWqzhF+MQ= | ||||
| github.com/oneclickvirt/backtrace v0.0.7-20250811023541/go.mod h1:/+KUtOWz48TyiTTbhVTsp3D6b5WY+4pCgvFBYtUGtns= | ||||
| github.com/oneclickvirt/basics v0.0.15-20250812130523 h1:nPNTVq8d9N1rdshkMAbnhZxTb7L2Yt7NlIg6rY10YRQ= | ||||
| github.com/oneclickvirt/basics v0.0.15-20250812130523/go.mod h1:2PV+1ge01zb0Sqzj2V2I7P0wAdFSLF1XgAiumchJJbg= | ||||
| github.com/oneclickvirt/cputest v0.0.12-20250720122317 h1:toiwAK1hZE5b8klu2mOQ7J4sv5yV9lpPKwgPahfRYBQ= | ||||
| github.com/oneclickvirt/cputest v0.0.12-20250720122317/go.mod h1:vjlH8tkPFft1tlLOpeNskXVvurxkHaJ3+dgFxQGLXY4= | ||||
| github.com/oneclickvirt/dd v0.0.2-20250808062818 h1:0KHrKkdpL5oBE1OHsrRd2siRw4/2k6f9LBaP7T4JpOc= | ||||
| github.com/oneclickvirt/dd v0.0.2-20250808062818/go.mod h1:tImu9sPTkLWo2tf1dEN1xQzrylWKauj9hbU8PHfyAeU= | ||||
| github.com/oneclickvirt/defaultset v0.0.2-20240624082446 h1:5Pg3mK/u/vQvSz7anu0nxzrNdELi/AcDAU1mMsmPzyc= | ||||
| github.com/oneclickvirt/defaultset v0.0.2-20240624082446/go.mod h1:e9Jt4tf2sbemCtc84/XgKcHy9EZ2jkc5x2sW1NiJS+E= | ||||
| github.com/oneclickvirt/disktest v0.0.8-20250701092629 h1:B/gA6SOr4qL5pQmVpHl9m5bn3paDcL7wJ1SZ7aY66M8= | ||||
| github.com/oneclickvirt/disktest v0.0.8-20250701092629/go.mod h1:6YCvGr+Z0tvcP4Ue8bezZqm/GqS/dSyEnSUhvS3Q03o= | ||||
| github.com/oneclickvirt/fio v0.0.2-20250701085933 h1:4P7QcOTxbqyx5DhHdFvyeRSsdNajSo9l/H2XK0vICIc= | ||||
| github.com/oneclickvirt/fio v0.0.2-20250701085933/go.mod h1:NIq+XYTey68KNERGIy/oRDlzpwLzBVoHOCiqX8didsE= | ||||
| github.com/oneclickvirt/gostun v0.0.3-20250329105202 h1:aJ6E91Lp94lq8iWRcCaxpXTjqOOaWvufr5oras6cFtM= | ||||
| github.com/oneclickvirt/gostun v0.0.3-20250329105202/go.mod h1:f7DPEXAxbmwXSW33dbxtb0/KzqvOBWhTs2Or5xBerQA= | ||||
| github.com/oneclickvirt/mbw v0.0.1-20250630140849 h1:p6RMhOPBnQKAm9+VEQ2axAFsidrdSdrhXMyheIyv2a8= | ||||
| github.com/oneclickvirt/mbw v0.0.1-20250630140849/go.mod h1:0Vq6NRpyLmGUdfHfL3uDcFsuZhi7KlG+OCs5ky2757Y= | ||||
| github.com/oneclickvirt/memorytest v0.0.8-20250717152547 h1:V4n4kkIizrHPpCwU2OECv8wUw4VcwRxZpATDDg9j8dI= | ||||
| github.com/oneclickvirt/memorytest v0.0.8-20250717152547/go.mod h1:7xMacjQobvFAtODht2hxTsB9hM2IFS7vZk3gxx+bsjo= | ||||
| github.com/oneclickvirt/nt3 v0.0.5-20250416131047 h1:KL0xowq19cW+FMBGMJxdqpRNoeyR+eEmb+jYSubmlTk= | ||||
| github.com/oneclickvirt/nt3 v0.0.5-20250416131047/go.mod h1:CVsDJEaIdyyZHn3WKbhU8Wn6GOfmBNvJlC/dDLRqcSQ= | ||||
| github.com/oneclickvirt/pingtest v0.0.8-20250701125637 h1:J28Ai5miTq1J0I4gdT8rewJSd3LwzD90L/bNiiaKfHM= | ||||
| github.com/oneclickvirt/pingtest v0.0.8-20250701125637/go.mod h1:d3Ntx5m9lMll3a/k3+2B+5emj//vgDh4/NHTxs2qQE8= | ||||
| github.com/oneclickvirt/portchecker v0.0.3-20250329125750 h1:TTNL0pnQlRsn046kW59I/9UWRpihttFHWnU7Ixycggk= | ||||
| github.com/oneclickvirt/portchecker v0.0.3-20250329125750/go.mod h1:HQxSTrqM8/QFqHMTBZ7S8H9eEO5FkUXU1eb7ZX5Mk+k= | ||||
| github.com/oneclickvirt/security v0.0.6-20250715102027 h1:lOaFxORBT/9nBlof7EU36YP+ZIbqkhCLGyOpYQTY1qs= | ||||
| github.com/oneclickvirt/security v0.0.6-20250715102027/go.mod h1:SDFBXV0sDo8pSIcGaaJ2gfCCW+NKy4pO1q9i4SIX2jc= | ||||
| github.com/oneclickvirt/speedtest v0.0.10-20250701123931 h1:IMUM0F3trrlCdl9JTO+FBIJ9zc8mbi+oyd66IkO/8mI= | ||||
| github.com/oneclickvirt/speedtest v0.0.10-20250701123931/go.mod h1:zd5ZgIGslmtQLQehEfRjyumlvgDHTpCSMchKfKXoASI= | ||||
| github.com/onsi/ginkgo/v2 v2.22.1 h1:QW7tbJAUDyVDVOM5dFa7qaybo+CRfR7bemlQUN6Z8aM= | ||||
| github.com/onsi/ginkgo/v2 v2.22.1/go.mod h1:S6aTpoRsSq2cZOd+pssHAlKW/Q/jZt6cPrPlnj4a1xM= | ||||
| github.com/onsi/gomega v1.36.1 h1:bJDPBO7ibjxcbHMgSCoo4Yj18UWbKDlLwX1x9sybDcw= | ||||
| github.com/onsi/gomega v1.36.1/go.mod h1:PvZbdDc8J6XJEpDK4HCuRBm8a6Fzp9/DmhC9C7yFlog= | ||||
| github.com/oneclickvirt/disktest v0.0.10-20250924030424 h1:56Aq2xygO/vA/co5vJ7/MQTNijIDl8eYbVk8uCWN4mI= | ||||
| github.com/oneclickvirt/disktest v0.0.10-20250924030424/go.mod h1:Vp3iMVBD4ccReDJz5n5SlzUdq0kDuVhpRklQk21KT+8= | ||||
| github.com/oneclickvirt/fio v0.0.2-20250808045755 h1:eWihCRWcalJjPIdrF8dMe68ZiPnMkSfHC8ENvElp/xE= | ||||
| github.com/oneclickvirt/fio v0.0.2-20250808045755/go.mod h1:NIq+XYTey68KNERGIy/oRDlzpwLzBVoHOCiqX8didsE= | ||||
| github.com/oneclickvirt/gostun v0.0.5-20250727155022 h1:/e3gSUrOp1tg/1NTRx+P8B51OGcP26Q6//5EoSIjOvk= | ||||
| github.com/oneclickvirt/gostun v0.0.5-20250727155022/go.mod h1:pfp7MFZJK9n/KTLAVqqFcCAns4xqMykmjI+1UeF/vdE= | ||||
| github.com/oneclickvirt/mbw v0.0.1-20250808061222 h1:WGXOe6QvHiDRhPVMI0VcctjzW08kGvJf50yq5YeZCtw= | ||||
| github.com/oneclickvirt/mbw v0.0.1-20250808061222/go.mod h1:0Vq6NRpyLmGUdfHfL3uDcFsuZhi7KlG+OCs5ky2757Y= | ||||
| github.com/oneclickvirt/memorytest v0.0.10-20250924154648 h1:trk6oZ7xs1eVtr+6oIv5IX8LDVtEMG+E6GVzQ810BtU= | ||||
| github.com/oneclickvirt/memorytest v0.0.10-20250924154648/go.mod h1:4kiHsEWkW9r3/1ZcV5xIweU0smiKP0IRfQj74AUIiVI= | ||||
| github.com/oneclickvirt/nt3 v0.0.8-20250811123903 h1:ubSPLh/DSrXj+tOgmRABgi2vrVmbmjjSne+NrVFNmNc= | ||||
| github.com/oneclickvirt/nt3 v0.0.8-20250811123903/go.mod h1:F1v+6xInBKnbUa8gV1M40R1HOzxg+obtduNhx3CTnmA= | ||||
| github.com/oneclickvirt/pingtest v0.0.8-20250728015259 h1:egoxZRZBOWN3JqBwqEsULDyRo2/dpGMeWcmV3U87zig= | ||||
| github.com/oneclickvirt/pingtest v0.0.8-20250728015259/go.mod h1:gxwsxxwitNQiGq2OI0ZogYoOLwc8DtuOdSRe6/EvRqs= | ||||
| github.com/oneclickvirt/portchecker v0.0.3-20250728015900 h1:AomzdppSOFB70AJESQhlp0IPbsHTTJGimAWDk2TzCWM= | ||||
| github.com/oneclickvirt/portchecker v0.0.3-20250728015900/go.mod h1:9sjMDPCd4Z40wkYB0S9gQPGH8YPtnNE1ZJthVIuHUzA= | ||||
| github.com/oneclickvirt/speedtest v0.0.10-20250728015734 h1:HKO7/JQ74ueXA8Wo8NIvcK9DphbEG/YTfAAVz/akSiY= | ||||
| github.com/oneclickvirt/speedtest v0.0.10-20250728015734/go.mod h1:0W8vnMbA3iucXLXFdGfe9Ia6RPS0izRO7jvu/SnH1P8= | ||||
| github.com/oneclickvirt/stream v0.0.2-20250924154001 h1:GuJWdiPkoK84+y/+oHKr2Ghl3c/MzS9Z5m1nM+lMmy4= | ||||
| github.com/oneclickvirt/stream v0.0.2-20250924154001/go.mod h1:oWaizaHTC2VQciBC9RfaLbAOf8qeR6n20/gY7QxriDE= | ||||
| github.com/oschwald/maxminddb-golang v1.13.1 h1:G3wwjdN9JmIK2o/ermkHM+98oX5fS+k5MbwsmL4MRQE= | ||||
| github.com/oschwald/maxminddb-golang v1.13.1/go.mod h1:K4pgV9N/GcK694KSTmVSDTODk4IsCNThNdTmnaBZ/F8= | ||||
| github.com/pelletier/go-toml/v2 v2.2.4 h1:mye9XuhQ6gvn5h28+VilKrrPoQVanw5PMw/TB0t5Ec4= | ||||
| @@ -160,10 +153,10 @@ github.com/prometheus-community/pro-bing v0.4.1 h1:aMaJwyifHZO0y+h8+icUz0xbToHbi | ||||
| github.com/prometheus-community/pro-bing v0.4.1/go.mod h1:aLsw+zqCaDoa2RLVVSX3+UiCkBBXTMtZC3c7EkfWnAE= | ||||
| github.com/quic-go/qpack v0.5.1 h1:giqksBPnT/HDtZ6VhtFKgoLOWmlyo9Ei6u9PqzIMbhI= | ||||
| github.com/quic-go/qpack v0.5.1/go.mod h1:+PC4XFrEskIVkcLzpEkbLqq1uCoxPhQuvK5rH1ZgaEg= | ||||
| github.com/quic-go/quic-go v0.48.2 h1:wsKXZPeGWpMpCGSWqOcqpW2wZYic/8T3aqiOID0/KWE= | ||||
| github.com/quic-go/quic-go v0.48.2/go.mod h1:yBgs3rWBOADpga7F+jJsb6Ybg1LSYiQvwWlLX+/6HMs= | ||||
| github.com/refraction-networking/utls v1.6.7 h1:zVJ7sP1dJx/WtVuITug3qYUq034cDq9B2MR1K67ULZM= | ||||
| github.com/refraction-networking/utls v1.6.7/go.mod h1:BC3O4vQzye5hqpmDTWUqi4P5DDhzJfkV1tdqtawQIH0= | ||||
| github.com/quic-go/quic-go v0.53.0 h1:QHX46sISpG2S03dPeZBgVIZp8dGagIaiu2FiVYvpCZI= | ||||
| github.com/quic-go/quic-go v0.53.0/go.mod h1:e68ZEaCdyviluZmy44P6Iey98v/Wfz6HCjQEm+l8zTY= | ||||
| github.com/refraction-networking/utls v1.7.3 h1:L0WRhHY7Oq1T0zkdzVZMR6zWZv+sXbHB9zcuvsAEqCo= | ||||
| github.com/refraction-networking/utls v1.7.3/go.mod h1:TUhh27RHMGtQvjQq+RyO11P6ZNQNBb3N0v7wsEjKAIQ= | ||||
| github.com/rivo/uniseg v0.2.0/go.mod h1:J6wj4VEh+S6ZtnVlnTBMWIodfgj8LQOQFoIToxlJtxc= | ||||
| github.com/rivo/uniseg v0.4.7 h1:WUdvkW8uEhrYfLC4ZzdpI2ztxP1I582+49Oc5Mq64VQ= | ||||
| github.com/rivo/uniseg v0.4.7/go.mod h1:FN3SvrM+Zdj16jyLfmOkMNblXMcoc8DfTHruCPUcx88= | ||||
| @@ -177,22 +170,18 @@ github.com/schollz/progressbar/v3 v3.14.4 h1:W9ZrDSJk7eqmQhd3uxFNNcTr0QL+xuGNI9d | ||||
| github.com/schollz/progressbar/v3 v3.14.4/go.mod h1:aT3UQ7yGm+2ZjeXPqsjTenwL3ddUiuZ0kfQ/2tHlyNI= | ||||
| github.com/shirou/gopsutil v3.21.11+incompatible h1:+1+c1VGhc88SSonWP6foOcLhvnKlUeu/erjjvaPEYiI= | ||||
| github.com/shirou/gopsutil v3.21.11+incompatible/go.mod h1:5b4v6he4MtMOwMlS0TUMTu2PcXUg8+E1lC7eC3UO/RA= | ||||
| github.com/shirou/gopsutil/v4 v4.24.5 h1:gGsArG5K6vmsh5hcFOHaPm87UD003CaDMkAOweSQjhM= | ||||
| github.com/shirou/gopsutil/v4 v4.24.5/go.mod h1:aoebb2vxetJ/yIDZISmduFvVNPHqXQ9SEJwRXxkf0RA= | ||||
| github.com/shoenig/go-m1cpu v0.1.6 h1:nxdKQNcEB6vzgA2E2bvzKIYRuNj7XNJ4S/aRSwKzFtM= | ||||
| github.com/shoenig/go-m1cpu v0.1.6/go.mod h1:1JJMcUBvfNwpq05QDQVAnx3gUHr9IYF7GNg9SUEw2VQ= | ||||
| github.com/shoenig/test v0.6.4 h1:kVTaSd7WLz5WZ2IaoM0RSzRsUD+m8wRR+5qvntpn4LU= | ||||
| github.com/shoenig/test v0.6.4/go.mod h1:byHiCGXqrVaflBLAMq/srcZIHynQPQgeyvkvXnjqq0k= | ||||
| github.com/showwin/speedtest-go v1.7.7 h1:VmK75SZOTKiuWjIVrs+mo7ZoKEw0utiGCvpnurS0olU= | ||||
| github.com/showwin/speedtest-go v1.7.7/go.mod h1:uLgdWCNarXxlYsL2E5TOZpCIwpgSWnEANZp7gfHXHu0= | ||||
| github.com/shirou/gopsutil/v4 v4.25.6 h1:kLysI2JsKorfaFPcYmcJqbzROzsBWEOAtw6A7dIfqXs= | ||||
| github.com/shirou/gopsutil/v4 v4.25.6/go.mod h1:PfybzyydfZcN+JMMjkF6Zb8Mq1A/VcogFFg7hj50W9c= | ||||
| github.com/showwin/speedtest-go v1.7.10 h1:9o5zb7KsuzZKn+IE2//z5btLKJ870JwO6ETayUkqRFw= | ||||
| github.com/showwin/speedtest-go v1.7.10/go.mod h1:Ei7OCTmNPdWofMadzcfgq1rUO7mvJy9Jycj//G7vyfA= | ||||
| github.com/sourcegraph/conc v0.3.0 h1:OQTbbt6P72L20UqAkXXuLOj79LfEanQ+YQFNpLA9ySo= | ||||
| github.com/sourcegraph/conc v0.3.0/go.mod h1:Sdozi7LEKbFPqYX2/J+iBAM6HpqSLTASQIKqDmF7Mt0= | ||||
| github.com/spf13/afero v1.14.0 h1:9tH6MapGnn/j0eb0yIXiLjERO8RB6xIVZRDCX7PtqWA= | ||||
| github.com/spf13/afero v1.14.0/go.mod h1:acJQ8t0ohCGuMN3O+Pv0V0hgMxNYDlvdk+VTfyZmbYo= | ||||
| github.com/spf13/cast v1.7.1 h1:cuNEagBQEHWN1FnbGEjCXL2szYEXqfJPbP2HNUaca9Y= | ||||
| github.com/spf13/cast v1.7.1/go.mod h1:ancEpBxwJDODSW/UG4rDrAqiKolqNNh2DX3mk86cAdo= | ||||
| github.com/spf13/pflag v1.0.6 h1:jFzHGLGAlb3ruxLB8MhbI6A8+AQX/2eW4qeyNZXNp2o= | ||||
| github.com/spf13/pflag v1.0.6/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg= | ||||
| github.com/spf13/cast v1.9.2 h1:SsGfm7M8QOFtEzumm7UZrZdLLquNdzFYfIbEXntcFbE= | ||||
| github.com/spf13/cast v1.9.2/go.mod h1:jNfB8QC9IA6ZuY2ZjDp0KtFO2LZZlg4S/7bzP6qqeHo= | ||||
| github.com/spf13/pflag v1.0.7 h1:vN6T9TfwStFPFM5XzjsvmzZkLuaLX+HS+0SeFLRgU6M= | ||||
| github.com/spf13/pflag v1.0.7/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg= | ||||
| github.com/spf13/viper v1.20.1 h1:ZMi+z/lvLyPSCoNtFCpqjy0S4kPbirhpTMwl8BkW9X4= | ||||
| github.com/spf13/viper v1.20.1/go.mod h1:P9Mdzt1zoHIG8m2eZQinpiBjo6kCmZSKBClNNqjJvu4= | ||||
| github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= | ||||
| @@ -229,8 +218,8 @@ github.com/yusufpapurcu/wmi v1.2.4 h1:zFUKzehAFReQwLys1b/iSMl+JQGSCSjtVqQn9bBrPo | ||||
| github.com/yusufpapurcu/wmi v1.2.4/go.mod h1:SBZ9tNy3G9/m5Oi98Zks0QjeHVDvuK0qfxQmPyzfmi0= | ||||
| go.uber.org/goleak v1.3.0 h1:2K3zAYmnTNqV73imy9J1T3WC+gmCePx2hEGkimedGto= | ||||
| go.uber.org/goleak v1.3.0/go.mod h1:CoHD4mav9JJNrW/WLlf7HGZPjdw8EucARQHekz1X6bE= | ||||
| go.uber.org/mock v0.5.0 h1:KAMbZvZPyBPWgD14IrIQ38QCyjwpvVVV6K/bHl1IwQU= | ||||
| go.uber.org/mock v0.5.0/go.mod h1:ge71pBPLYDk7QIi1LupWxdAykm7KIEFchiOqd6z7qMM= | ||||
| go.uber.org/mock v0.5.2 h1:LbtPTcP8A5k9WPXj54PPPbjcI4Y6lhyOZXn+VS7wNko= | ||||
| go.uber.org/mock v0.5.2/go.mod h1:wLlUxC2vVTPTaE3UD51E0BGOAElKrILxhVSDYQLld5o= | ||||
| go.uber.org/multierr v1.11.0 h1:blXXJkSxSSfBVBlC76pxqeO+LN3aDfLQo+309xJstO0= | ||||
| go.uber.org/multierr v1.11.0/go.mod h1:20+QtiLqy0Nd6FdQB9TLXag12DsQkrbs3htMFfDN80Y= | ||||
| go.uber.org/zap v1.27.0 h1:aJMhYGrd5QSmlpLMr2MftRKl7t8J8PTZPA732ud/XR8= | ||||
| @@ -240,16 +229,14 @@ golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8U | ||||
| golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= | ||||
| golang.org/x/crypto v0.8.0/go.mod h1:mRqEX+O9/h5TFCrQhkgjo2yKi0yYA+9ecGkdQoHrywE= | ||||
| golang.org/x/crypto v0.12.0/go.mod h1:NF0Gs7EO5K4qLn+Ylc+fih8BSTeIjAP05siRnAh98yw= | ||||
| golang.org/x/crypto v0.37.0 h1:kJNSjF/Xp7kU0iB2Z+9viTPMW4EqqsrywMXLJOOsXSE= | ||||
| golang.org/x/crypto v0.37.0/go.mod h1:vg+k43peMZ0pUMhYmVAWysMK35e6ioLh3wB8ZCAfbVc= | ||||
| golang.org/x/exp v0.0.0-20250106191152-7588d65b2ba8 h1:yqrTHse8TCMW1M1ZCP+VAR/l0kKxwaAIqN/il7x4voA= | ||||
| golang.org/x/exp v0.0.0-20250106191152-7588d65b2ba8/go.mod h1:tujkw807nyEEAamNbDrEGzRav+ilXA7PCRAd6xsmwiU= | ||||
| golang.org/x/crypto v0.40.0 h1:r4x+VvoG5Fm+eJcxMaY8CQM7Lb0l1lsmjGBQ6s8BfKM= | ||||
| golang.org/x/crypto v0.40.0/go.mod h1:Qr1vMER5WyS2dfPHAlsOj01wgLbsyWtFn/aY+5+ZdxY= | ||||
| golang.org/x/lint v0.0.0-20200302205851-738671d3881b/go.mod h1:3xt1FjdF8hUf6vQPIChWIBhFzV8gjjsPE/fR3IyQdNY= | ||||
| golang.org/x/mod v0.1.1-0.20191105210325-c90efee705ee/go.mod h1:QqPTAvyqsEbceGzBzNggFXnrqF1CaUcvgkdR5Ot7KZg= | ||||
| golang.org/x/mod v0.6.0-dev.0.20220419223038-86c51ed26bb4/go.mod h1:jJ57K6gSWd91VN4djpZkiMVwK6gcyfeH4XE8wZrZaV4= | ||||
| golang.org/x/mod v0.8.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs= | ||||
| golang.org/x/mod v0.22.0 h1:D4nJWe9zXqHOmWqj4VMOJhvzj7bEZg4wEYa759z1pH4= | ||||
| golang.org/x/mod v0.22.0/go.mod h1:6SkKJ3Xj0I0BrPOZoBy3bdMptDDU9oJrpohJ3eWZ1fY= | ||||
| golang.org/x/mod v0.25.0 h1:n7a+ZbQKQA/Ysbyb0/6IbB1H/X41mKgbhfv7AfG/44w= | ||||
| golang.org/x/mod v0.25.0/go.mod h1:IXM97Txy2VM4PJ3gI61r1YEk/gAj6zAHN3AdZt6S9Ww= | ||||
| golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= | ||||
| golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= | ||||
| golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= | ||||
| @@ -258,14 +245,14 @@ golang.org/x/net v0.6.0/go.mod h1:2Tu9+aMcznHK/AK1HMvgo6xiTLG5rD5rZLDS+rp2Bjs= | ||||
| golang.org/x/net v0.9.0/go.mod h1:d48xBJpPfHeWQsugry2m+kC02ZBRGRgulfHnEXEuWns= | ||||
| golang.org/x/net v0.10.0/go.mod h1:0qNGK6F8kojg2nk9dLZ2mShWaEBan6FAoqfSigmmuDg= | ||||
| golang.org/x/net v0.14.0/go.mod h1:PpSgVXXLK0OxS0F31C1/tv6XNguvCrnXIDrFMspZIUI= | ||||
| golang.org/x/net v0.39.0 h1:ZCu7HMWDxpXpaiKdhzIfaltL9Lp31x/3fCP11bc6/fY= | ||||
| golang.org/x/net v0.39.0/go.mod h1:X7NRbYVEA+ewNkCNyJ513WmMdQ3BineSwVtN2zD/d+E= | ||||
| golang.org/x/net v0.42.0 h1:jzkYrhi3YQWD6MLBJcsklgQsoAcw89EcZbJw8Z614hs= | ||||
| golang.org/x/net v0.42.0/go.mod h1:FF1RA5d3u7nAYA4z2TkclSCKh68eSXtiFwcWQpPXdt8= | ||||
| golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sync v0.0.0-20210220032951-036812b2e83c/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sync v0.0.0-20220722155255-886fb9371eb4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sync v0.1.0/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sync v0.13.0 h1:AauUjRAJ9OSnvULf/ARrrVywoJDy0YS2AwQ98I37610= | ||||
| golang.org/x/sync v0.13.0/go.mod h1:1dzgHSNfp02xaA81J2MS99Qcpr2w7fw1gpm99rleRqA= | ||||
| golang.org/x/sync v0.16.0 h1:ycBJEhp9p4vXvUZNszeOq0kGTPghopOL8q0fq3vstxw= | ||||
| golang.org/x/sync v0.16.0/go.mod h1:1dzgHSNfp02xaA81J2MS99Qcpr2w7fw1gpm99rleRqA= | ||||
| golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
| golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20190916202348-b4ddaad3f8a3/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| @@ -280,8 +267,8 @@ golang.org/x/sys v0.7.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.8.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.11.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.20.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= | ||||
| golang.org/x/sys v0.33.0 h1:q3i8TbbEz+JRD9ywIRlyRAQbM0qF7hu24q3teo2hbuw= | ||||
| golang.org/x/sys v0.33.0/go.mod h1:BJP2sWEmIv4KK5OTEluFJCKSidICx8ciO85XgH3Ak8k= | ||||
| golang.org/x/sys v0.34.0 h1:H5Y5sJ2L2JRdyv7ROF1he/lPdvFsd0mJHFw2ThKHxLA= | ||||
| golang.org/x/sys v0.34.0/go.mod h1:BJP2sWEmIv4KK5OTEluFJCKSidICx8ciO85XgH3Ak8k= | ||||
| golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= | ||||
| golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= | ||||
| golang.org/x/term v0.5.0/go.mod h1:jMB1sMXY+tzblOD4FWmEbocvup2/aLOaQEp7JmGp78k= | ||||
| @@ -289,38 +276,34 @@ golang.org/x/term v0.7.0/go.mod h1:P32HKFT3hSsZrRxla30E9HqToFYAQPCMs/zFMBUFqPY= | ||||
| golang.org/x/term v0.8.0/go.mod h1:xPskH00ivmX89bAKVGSKKtLOWNx2+17Eiy94tnKShWo= | ||||
| golang.org/x/term v0.11.0/go.mod h1:zC9APTIj3jG3FdV/Ons+XE1riIZXG4aZ4GTHiPZJPIU= | ||||
| golang.org/x/term v0.20.0/go.mod h1:8UkIAJTvZgivsXaD6/pH6U9ecQzZ45awqEOzuCvwpFY= | ||||
| golang.org/x/term v0.31.0 h1:erwDkOK1Msy6offm1mOgvspSkslFnIGsFnxOKoufg3o= | ||||
| golang.org/x/term v0.31.0/go.mod h1:R4BeIy7D95HzImkxGkTW1UQTtP54tio2RyHz7PwK0aw= | ||||
| golang.org/x/term v0.33.0 h1:NuFncQrRcaRvVmgRkvM3j/F00gWIAlcmlB8ACEKmGIg= | ||||
| golang.org/x/term v0.33.0/go.mod h1:s18+ql9tYWp1IfpV9DmCtQDDSRBUjKaw9M1eAv5UeF0= | ||||
| golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= | ||||
| golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||
| golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ= | ||||
| golang.org/x/text v0.7.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8= | ||||
| golang.org/x/text v0.9.0/go.mod h1:e1OnstbJyHTd6l/uOt8jFFHp6TRDWZR/bV3emEE/zU8= | ||||
| golang.org/x/text v0.12.0/go.mod h1:TvPlkZtksWOMsz7fbANvkp4WM8x/WCo/om8BMLbz+aE= | ||||
| golang.org/x/text v0.24.0 h1:dd5Bzh4yt5KYA8f9CJHCP4FB4D51c2c6JvN37xJJkJ0= | ||||
| golang.org/x/text v0.24.0/go.mod h1:L8rBsPeo2pSS+xqN0d5u2ikmjtmoJbDBT1b7nHvFCdU= | ||||
| golang.org/x/time v0.5.0 h1:o7cqy6amK/52YcAKIPlM3a+Fpj35zvRj2TP+e1xFSfk= | ||||
| golang.org/x/time v0.5.0/go.mod h1:3BpzKBy/shNhVucY/MWOyx10tF3SFh9QdLuxbVysPQM= | ||||
| golang.org/x/text v0.27.0 h1:4fGWRpyh641NLlecmyl4LOe6yDdfaYNrGb2zdfo4JV4= | ||||
| golang.org/x/text v0.27.0/go.mod h1:1D28KMCvyooCX9hBiosv5Tz/+YLxj0j7XhWjpSUF7CU= | ||||
| golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= | ||||
| golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= | ||||
| golang.org/x/tools v0.0.0-20200130002326-2f3ba24bd6e7/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= | ||||
| golang.org/x/tools v0.1.12/go.mod h1:hNGJHUnrk76NpqgfD5Aqm5Crs+Hm0VOH/i9J2+nxYbc= | ||||
| golang.org/x/tools v0.6.0/go.mod h1:Xwgl3UAJ/d3gWutnCtw505GrjyAbvKui8lOU390QaIU= | ||||
| golang.org/x/tools v0.29.0 h1:Xx0h3TtM9rzQpQuR4dKLrdglAmCEN5Oi+P74JdhdzXE= | ||||
| golang.org/x/tools v0.29.0/go.mod h1:KMQVMRsVxU6nHCFXrBPhDB8XncLNLM0lIy/F14RP588= | ||||
| golang.org/x/tools v0.34.0 h1:qIpSLOxeCYGg9TrcJokLBG4KFA6d795g0xkBkiESGlo= | ||||
| golang.org/x/tools v0.34.0/go.mod h1:pAP9OwEaY1CAW3HOmg3hLZC5Z0CCmzjAF2UQMSqNARg= | ||||
| golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| google.golang.org/protobuf v1.35.1 h1:m3LfL6/Ca+fqnjnlqQXNpFPABW1UD7mjh8KO2mKFytA= | ||||
| google.golang.org/protobuf v1.35.1/go.mod h1:9fA7Ob0pmnwhb644+1+CVWFRbNajQ6iRojtC/QF5bRE= | ||||
| gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||
| gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c h1:Hei/4ADfdWqJk1ZMxUNpqntNwaWcugrBjAiHlqqRiVk= | ||||
| gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c/go.mod h1:JHkPIbrfpd72SG/EVd6muEfDQjcINNoR0C8j2r3qZ4Q= | ||||
| gopkg.in/yaml.v1 v1.0.0-20140924161607-9f9df34309c0/go.mod h1:WDnlLJ4WF5VGsH/HVa3CI79GS0ol3YnhVnKP89i0kNg= | ||||
| gopkg.in/yaml.v2 v2.4.0 h1:D8xgwECY7CYvx+Y2n4sBz93Jn9JRvxdiyyo8CTfuKaY= | ||||
| gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ= | ||||
| gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= | ||||
| gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA= | ||||
| gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= | ||||
| gotest.tools/v3 v3.5.1 h1:EENdUnS3pdur5nybKYIh2Vfgc8IUNBjxDPSjtiJcOzU= | ||||
| gotest.tools/v3 v3.5.1/go.mod h1:isy3WKz7GK6uNw/sbHzfKBLvlvXwUyV06n6brMxxopU= | ||||
| howett.net/plist v1.0.0 h1:7CrbWYbPPO/PyNy38b2EB/+gYbjCe2DXBxgtOOZbSQM= | ||||
| howett.net/plist v1.0.0/go.mod h1:lqaXoTrLY4hg8tnEzNru53gicrbv7rrk+2xJA/7hw9g= | ||||
|   | ||||
							
								
								
									
										185
									
								
								goecs.go
									
									
									
									
									
								
							
							
						
						
									
										185
									
								
								goecs.go
									
									
									
									
									
								
							| @@ -17,7 +17,6 @@ import ( | ||||
|  | ||||
| 	"github.com/oneclickvirt/CommonMediaTests/commediatests" | ||||
| 	unlocktestmodel "github.com/oneclickvirt/UnlockTests/model" | ||||
| 	backtrace "github.com/oneclickvirt/backtrace/bk" | ||||
| 	backtracemodel "github.com/oneclickvirt/backtrace/model" | ||||
| 	basicmodel "github.com/oneclickvirt/basics/model" | ||||
| 	cputestmodel "github.com/oneclickvirt/cputest/model" | ||||
| @@ -25,13 +24,14 @@ import ( | ||||
| 	"github.com/oneclickvirt/ecs/cputest" | ||||
| 	"github.com/oneclickvirt/ecs/disktest" | ||||
| 	"github.com/oneclickvirt/ecs/memorytest" | ||||
| 	"github.com/oneclickvirt/ecs/nexttrace" | ||||
| 	"github.com/oneclickvirt/ecs/speedtest" | ||||
| 	"github.com/oneclickvirt/ecs/unlocktest" | ||||
| 	"github.com/oneclickvirt/ecs/upstreams" | ||||
| 	"github.com/oneclickvirt/ecs/utils" | ||||
| 	gostunmodel "github.com/oneclickvirt/gostun/model" | ||||
| 	memorytestmodel "github.com/oneclickvirt/memorytest/memory" | ||||
| 	nt3model "github.com/oneclickvirt/nt3/model" | ||||
| 	"github.com/oneclickvirt/nt3/nt" | ||||
| 	ptmodel "github.com/oneclickvirt/pingtest/model" | ||||
| 	"github.com/oneclickvirt/pingtest/pt" | ||||
| 	"github.com/oneclickvirt/portchecker/email" | ||||
| @@ -39,7 +39,7 @@ import ( | ||||
| ) | ||||
|  | ||||
| var ( | ||||
| 	ecsVersion                                                        = "v0.1.62" | ||||
| 	ecsVersion                                                        = "v0.1.89" | ||||
| 	menuMode                                                          bool | ||||
| 	onlyChinaTest                                                     bool | ||||
| 	input, choice                                                     string | ||||
| @@ -59,7 +59,7 @@ var ( | ||||
| 	autoChangeDiskTestMethod                                          = true | ||||
| 	filePath                                                          = "goecs.txt" | ||||
| 	enabelUpload                                                      = true | ||||
| 	help                                                              bool | ||||
| 	onlyIpInfoCheckStatus, help                                       bool | ||||
| 	goecsFlag                                                         = flag.NewFlagSet("goecs", flag.ContinueOnError) | ||||
| 	finish                                                            bool | ||||
| ) | ||||
| @@ -99,7 +99,7 @@ func getMenuChoice(language string) string { | ||||
| 			if re.MatchString(input) { | ||||
| 				inChoice := input | ||||
| 				switch inChoice { | ||||
| 				case "1", "2", "3", "4", "5", "6", "7", "8", "9", "10": | ||||
| 				case "0", "1", "2", "3", "4", "5", "6", "7", "8", "9", "10": | ||||
| 					return inChoice | ||||
| 				default: | ||||
| 					if language == "zh" { | ||||
| @@ -139,11 +139,11 @@ func parseFlags() { | ||||
| 	goecsFlag.BoolVar(&speedTestStatus, "speed", true, "Enable/Disable speed test") | ||||
| 	goecsFlag.StringVar(&cpuTestMethod, "cpum", "sysbench", "Set CPU test method (supported: sysbench, geekbench, winsat)") | ||||
| 	goecsFlag.StringVar(&cpuTestThreadMode, "cput", "multi", "Set CPU test thread mode (supported: single, multi)") | ||||
| 	goecsFlag.StringVar(&memoryTestMethod, "memorym", "sysbench", "Set memory test method (supported: sysbench, dd, winsat)") | ||||
| 	goecsFlag.StringVar(&memoryTestMethod, "memorym", "stream", "Set memory test method (supported: stream, sysbench, dd, winsat, auto)") | ||||
| 	goecsFlag.StringVar(&diskTestMethod, "diskm", "fio", "Set disk test method (supported: fio, dd, winsat)") | ||||
| 	goecsFlag.StringVar(&diskTestPath, "diskp", "", "Set disk test path, e.g., -diskp /root") | ||||
| 	goecsFlag.BoolVar(&diskMultiCheck, "diskmc", false, "Enable/Disable multiple disk checks, e.g., -diskmc=false") | ||||
| 	goecsFlag.StringVar(&nt3Location, "nt3loc", "GZ", "Specify NT3 test location (supported: GZ, SH, BJ, CD for Guangzhou, Shanghai, Beijing, Chengdu)") | ||||
| 	goecsFlag.StringVar(&nt3Location, "nt3loc", "GZ", "Specify NT3 test location (supported: GZ, SH, BJ, CD, ALL for Guangzhou, Shanghai, Beijing, Chengdu and all)") | ||||
| 	goecsFlag.StringVar(&nt3CheckType, "nt3t", "ipv4", "Set NT3 test type (supported: both, ipv4, ipv6)") | ||||
| 	goecsFlag.IntVar(&spNum, "spnum", 2, "Set the number of servers per operator for speed test") | ||||
| 	goecsFlag.BoolVar(&enableLogger, "log", false, "Enable/Disable logging in the current path") | ||||
| @@ -185,11 +185,13 @@ func handleMenuMode(preCheck utils.NetCheckResult) { | ||||
| 	commTestStatus, utTestStatus, securityTestStatus, emailTestStatus = false, false, false, false | ||||
| 	backtraceStatus, nt3Status, speedTestStatus = false, false, false | ||||
| 	autoChangeDiskTestMethod = true | ||||
| 	printMenuOptions() | ||||
| 	printMenuOptions(preCheck) | ||||
| Loop: | ||||
| 	for { | ||||
| 		choice = getMenuChoice(language) | ||||
| 		switch choice { | ||||
| 		case "0": | ||||
| 			os.Exit(0) | ||||
| 		case "1": | ||||
| 			setFullTestStatus(preCheck) | ||||
| 			onlyChinaTest = utils.CheckChina(enableLogger) | ||||
| @@ -235,6 +237,7 @@ Loop: | ||||
| 				fmt.Println("Can not test without network connection!") | ||||
| 				return | ||||
| 			} | ||||
| 			nt3Location = "ALL" | ||||
| 			setRouteTestStatus() | ||||
| 			break Loop | ||||
| 		default: | ||||
| @@ -243,23 +246,83 @@ Loop: | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func printMenuOptions() { | ||||
| func printMenuOptions(preCheck utils.NetCheckResult) { | ||||
| 	var stats *utils.StatsResponse | ||||
| 	var statsErr error | ||||
| 	var githubInfo *utils.GitHubRelease | ||||
| 	var githubErr error | ||||
| 	// 只有在网络连接正常时才获取统计信息和版本信息 | ||||
| 	if preCheck.Connected { | ||||
| 		var pwg sync.WaitGroup | ||||
| 		pwg.Add(2) | ||||
| 		go func() { | ||||
| 			defer pwg.Done() | ||||
| 			stats, statsErr = utils.GetGoescStats() | ||||
| 		}() | ||||
| 		go func() { | ||||
| 			defer pwg.Done() | ||||
| 			githubInfo, githubErr = utils.GetLatestEcsRelease() | ||||
| 		}() | ||||
| 		pwg.Wait() | ||||
| 	} else { | ||||
| 		statsErr = fmt.Errorf("network not connected") | ||||
| 		githubErr = fmt.Errorf("network not connected") | ||||
| 	} | ||||
| 	var statsInfo string | ||||
| 	var cmp int | ||||
| 	if preCheck.Connected { | ||||
| 		// 网络连接正常时处理统计信息和版本比较 | ||||
| 		if statsErr != nil { | ||||
| 			statsInfo = "NULL" | ||||
| 		} else { | ||||
| 			switch language { | ||||
| 			case "zh": | ||||
| 		fmt.Println("VPS融合怪版本: ", ecsVersion) | ||||
| 		fmt.Println("1. 融合怪完全体") | ||||
| 				statsInfo = fmt.Sprintf("总使用量: %s | 今日使用: %s", | ||||
| 					utils.FormatGoecsNumber(stats.Total), | ||||
| 					utils.FormatGoecsNumber(stats.Daily)) | ||||
| 			case "en": | ||||
| 				statsInfo = fmt.Sprintf("Total Usage: %s | Daily Usage: %s", | ||||
| 					utils.FormatGoecsNumber(stats.Total), | ||||
| 					utils.FormatGoecsNumber(stats.Daily)) | ||||
| 			} | ||||
| 		} | ||||
| 		if githubErr == nil { | ||||
| 			cmp = utils.CompareVersions(ecsVersion, githubInfo.TagName) | ||||
| 		} else { | ||||
| 			cmp = 0 | ||||
| 		} | ||||
| 	} | ||||
| 	switch language { | ||||
| 	case "zh": | ||||
| 		fmt.Printf("VPS融合怪版本: %s\n", ecsVersion) | ||||
| 		if preCheck.Connected { | ||||
| 			switch cmp { | ||||
| 			case -1: | ||||
| 				fmt.Printf("检测到新版本 %s 如有必要请更新!\n", githubInfo.TagName) | ||||
| 			} | ||||
| 			fmt.Printf("使用统计: %s\n", statsInfo) | ||||
| 		} | ||||
| 		fmt.Println("1. 融合怪完全体(能测全测)") | ||||
| 		fmt.Println("2. 极简版(系统信息+CPU+内存+磁盘+测速节点5个)") | ||||
| 		fmt.Println("3. 精简版(系统信息+CPU+内存+磁盘+常用流媒体+路由+测速节点5个)") | ||||
| 		fmt.Println("4. 精简网络版(系统信息+CPU+内存+磁盘+回程+路由+测速节点5个)") | ||||
| 		fmt.Println("5. 精简解锁版(系统信息+CPU+内存+磁盘IO+御三家+常用流媒体+测速节点5个)") | ||||
| 		fmt.Println("6. 网络单项(IP质量检测+三网回程+三网路由与延迟+测速节点11个)") | ||||
| 		fmt.Println("6. 网络单项(IP质量检测+上游及三网回程+广州三网回程详细路由+全国延迟+测速节点11个)") | ||||
| 		fmt.Println("7. 解锁单项(御三家解锁+常用流媒体解锁)") | ||||
| 		fmt.Println("8. 硬件单项(系统信息+CPU+内存+dd磁盘测试+fio磁盘测试)") | ||||
| 		fmt.Println("9. IP质量检测(15个数据库的IP检测+邮件端口检测)") | ||||
| 		fmt.Println("10. 三网回程线路+广州三网路由+全国三网延迟") | ||||
| 		fmt.Println("8. 硬件单项(系统信息+CPU+dd磁盘测试+fio磁盘测试)") | ||||
| 		fmt.Println("9. IP质量检测(15个数据库的IP质量检测+邮件端口检测)") | ||||
| 		fmt.Println("10. 三网回程线路检测+三网回程详细路由(北京上海广州成都)+三网延迟测试(全国)") | ||||
| 		fmt.Println("0. 退出程序") | ||||
| 	case "en": | ||||
| 		fmt.Println("VPS Fusion Monster Test Version: ", ecsVersion) | ||||
| 		fmt.Println("1. VPS Fusion Monster Test Comprehensive Test Suite") | ||||
| 		fmt.Printf("VPS Fusion Monster Test Version: %s\n", ecsVersion) | ||||
| 		if preCheck.Connected { | ||||
| 			switch cmp { | ||||
| 			case -1: | ||||
| 				fmt.Printf("New version detected %s update if necessary!\n", githubInfo.TagName) | ||||
| 			} | ||||
| 			fmt.Printf("%s\n", statsInfo) | ||||
| 		} | ||||
| 		fmt.Println("1. VPS Fusion Monster Test (Full Test)") | ||||
| 		fmt.Println("2. Minimal Test Suite (System Info + CPU + Memory + Disk + 5 Speed Test Nodes)") | ||||
| 		fmt.Println("3. Standard Test Suite (System Info + CPU + Memory + Disk + Basic Unlock Tests + 5 Speed Test Nodes)") | ||||
| 		fmt.Println("4. Network-Focused Test Suite (System Info + CPU + Memory + Disk + 5 Speed Test Nodes)") | ||||
| @@ -268,6 +331,7 @@ func printMenuOptions() { | ||||
| 		fmt.Println("7. Unlock-Only Test (Basic Unlock Tests + Common Streaming Services Unlock)") | ||||
| 		fmt.Println("8. Hardware-Only Test (System Info + CPU + Memory + dd Disk Test + fio Disk Test)") | ||||
| 		fmt.Println("9. IP Quality Test (IP Test with 15 Databases + Email Port Test)") | ||||
| 		fmt.Println("0. Exit Program") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @@ -334,16 +398,18 @@ func setUnlockFocusedTestStatus(preCheck utils.NetCheckResult) { | ||||
| } | ||||
|  | ||||
| func setNetworkOnlyTestStatus() { | ||||
| 	onlyIpInfoCheckStatus = true | ||||
| 	securityTestStatus = true | ||||
| 	speedTestStatus = true | ||||
| 	backtraceStatus = true | ||||
| 	nt3Status = true | ||||
| 	pingTestStatus = true | ||||
| } | ||||
|  | ||||
| func setUnlockOnlyTestStatus() { | ||||
| 	onlyIpInfoCheckStatus = true | ||||
| 	commTestStatus = true | ||||
| 	utTestStatus = true | ||||
| 	enabelUpload = false | ||||
| } | ||||
|  | ||||
| func setHardwareOnlyTestStatus(preCheck utils.NetCheckResult) { | ||||
| @@ -357,15 +423,16 @@ func setHardwareOnlyTestStatus(preCheck utils.NetCheckResult) { | ||||
| } | ||||
|  | ||||
| func setIPQualityTestStatus() { | ||||
| 	onlyIpInfoCheckStatus = true | ||||
| 	securityTestStatus = true | ||||
| 	emailTestStatus = true | ||||
| } | ||||
|  | ||||
| func setRouteTestStatus() { | ||||
| 	onlyIpInfoCheckStatus = true | ||||
| 	backtraceStatus = true | ||||
| 	nt3Status = true | ||||
| 	pingTestStatus = true | ||||
| 	enabelUpload = false | ||||
| } | ||||
|  | ||||
| func printInvalidChoice() { | ||||
| @@ -386,7 +453,7 @@ func handleLanguageSpecificSettings() { | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func handleSignalInterrupt(sig chan os.Signal, startTime *time.Time, output *string, tempOutput string, uploadDone chan bool, outputMutex *sync.Mutex) { | ||||
| func handleSignalInterrupt(sig chan os.Signal, startTime *time.Time, output *string, _ string, uploadDone chan bool, outputMutex *sync.Mutex) { | ||||
| 	select { | ||||
| 	case <-sig: | ||||
| 		if !finish { | ||||
| @@ -467,11 +534,14 @@ func runChineseTests(preCheck utils.NetCheckResult, wg1, wg2, wg3 *sync.WaitGrou | ||||
| 	*output = runCPUTest(*output, tempOutput, outputMutex) | ||||
| 	*output = runMemoryTest(*output, tempOutput, outputMutex) | ||||
| 	*output = runDiskTest(*output, tempOutput, outputMutex) | ||||
| 	if (onlyChinaTest || pingTestStatus) && preCheck.Connected && preCheck.StackType != "" && preCheck.StackType != "None" { | ||||
| 		wg3.Add(1) | ||||
| 	if onlyIpInfoCheckStatus && !basicStatus && preCheck.Connected && preCheck.StackType != "" && preCheck.StackType != "None" { | ||||
| 		*output = runIpInfoCheck(*output, tempOutput, outputMutex) | ||||
| 	} | ||||
| 	if utTestStatus && preCheck.Connected && preCheck.StackType != "" && preCheck.StackType != "None" && !onlyChinaTest { | ||||
| 		wg1.Add(1) | ||||
| 		go func() { | ||||
| 			defer wg3.Done() | ||||
| 			*ptInfo = pt.PingTest() | ||||
| 			defer wg1.Done() | ||||
| 			*mediaInfo = unlocktest.MediaTest(language) | ||||
| 		}() | ||||
| 	} | ||||
| 	if emailTestStatus && preCheck.Connected && preCheck.StackType != "" && preCheck.StackType != "None" { | ||||
| @@ -481,11 +551,11 @@ func runChineseTests(preCheck utils.NetCheckResult, wg1, wg2, wg3 *sync.WaitGrou | ||||
| 			*emailInfo = email.EmailCheck() | ||||
| 		}() | ||||
| 	} | ||||
| 	if utTestStatus && preCheck.Connected && preCheck.StackType != "" && preCheck.StackType != "None" && !onlyChinaTest { | ||||
| 		wg1.Add(1) | ||||
| 	if (onlyChinaTest || pingTestStatus) && preCheck.Connected && preCheck.StackType != "" && preCheck.StackType != "None" { | ||||
| 		wg3.Add(1) | ||||
| 		go func() { | ||||
| 			defer wg1.Done() | ||||
| 			*mediaInfo = unlocktest.MediaTest(language) | ||||
| 			defer wg3.Done() | ||||
| 			*ptInfo = pt.PingTest() | ||||
| 		}() | ||||
| 	} | ||||
| 	if preCheck.Connected && preCheck.StackType != "" && preCheck.StackType != "None" { | ||||
| @@ -507,6 +577,9 @@ func runEnglishTests(preCheck utils.NetCheckResult, wg1, wg2 *sync.WaitGroup, ba | ||||
| 	*output = runCPUTest(*output, tempOutput, outputMutex) | ||||
| 	*output = runMemoryTest(*output, tempOutput, outputMutex) | ||||
| 	*output = runDiskTest(*output, tempOutput, outputMutex) | ||||
| 	if onlyIpInfoCheckStatus && !basicStatus && preCheck.Connected && preCheck.StackType != "" && preCheck.StackType != "None" { | ||||
| 		*output = runIpInfoCheck(*output, tempOutput, outputMutex) | ||||
| 	} | ||||
| 	if preCheck.Connected && preCheck.StackType != "" && preCheck.StackType != "None" { | ||||
| 		if utTestStatus { | ||||
| 			wg1.Add(1) | ||||
| @@ -530,6 +603,24 @@ func runEnglishTests(preCheck utils.NetCheckResult, wg1, wg2 *sync.WaitGroup, ba | ||||
| 	*output = appendTimeInfo(*output, tempOutput, startTime, outputMutex) | ||||
| } | ||||
|  | ||||
| // runIpInfoCheck 系统和网络基础信息检测不进行测试的时候,该函数检测取得本机IP信息并显示(单项测试中输出) | ||||
| func runIpInfoCheck(output, tempOutput string, outputMutex *sync.Mutex) string { | ||||
| 	outputMutex.Lock() | ||||
| 	defer outputMutex.Unlock() | ||||
| 	return utils.PrintAndCapture(func() { | ||||
| 		var ipinfo string | ||||
| 		upstreams.IPV4, upstreams.IPV6, ipinfo = utils.OnlyBasicsIpInfo(language) | ||||
| 		if ipinfo != "" { | ||||
| 			if language == "zh" { | ||||
| 				utils.PrintCenteredTitle("IP信息", width) | ||||
| 			} else { | ||||
| 				utils.PrintCenteredTitle("IP-Information", width) | ||||
| 			} | ||||
| 			fmt.Printf("%s", ipinfo) | ||||
| 		} | ||||
| 	}, tempOutput, output) | ||||
| } | ||||
|  | ||||
| func runBasicTests(preCheck utils.NetCheckResult, basicInfo, securityInfo *string, output, tempOutput string, outputMutex *sync.Mutex) string { | ||||
| 	outputMutex.Lock() | ||||
| 	defer outputMutex.Unlock() | ||||
| @@ -544,13 +635,13 @@ func runBasicTests(preCheck utils.NetCheckResult, basicInfo, securityInfo *strin | ||||
| 				} | ||||
| 			} | ||||
| 			if preCheck.Connected && preCheck.StackType == "DualStack" { | ||||
| 				*basicInfo, *securityInfo, nt3CheckType = utils.BasicsAndSecurityCheck(language, nt3CheckType, securityTestStatus) | ||||
| 				upstreams.IPV4, upstreams.IPV6, *basicInfo, *securityInfo, nt3CheckType = utils.BasicsAndSecurityCheck(language, nt3CheckType, securityTestStatus) | ||||
| 			} else if preCheck.Connected && preCheck.StackType == "IPv4" { | ||||
| 				*basicInfo, *securityInfo, nt3CheckType = utils.BasicsAndSecurityCheck(language, "ipv4", securityTestStatus) | ||||
| 				upstreams.IPV4, upstreams.IPV6, *basicInfo, *securityInfo, nt3CheckType = utils.BasicsAndSecurityCheck(language, "ipv4", securityTestStatus) | ||||
| 			} else if preCheck.Connected && preCheck.StackType == "IPv6" { | ||||
| 				*basicInfo, *securityInfo, nt3CheckType = utils.BasicsAndSecurityCheck(language, "ipv6", securityTestStatus) | ||||
| 				upstreams.IPV4, upstreams.IPV6, *basicInfo, *securityInfo, nt3CheckType = utils.BasicsAndSecurityCheck(language, "ipv6", securityTestStatus) | ||||
| 			} else { | ||||
| 				*basicInfo, *securityInfo, nt3CheckType = utils.BasicsAndSecurityCheck(language, "", false) | ||||
| 				upstreams.IPV4, upstreams.IPV6, *basicInfo, *securityInfo, nt3CheckType = utils.BasicsAndSecurityCheck(language, "", false) | ||||
| 				securityTestStatus = false | ||||
| 			} | ||||
| 			if basicStatus { | ||||
| @@ -688,24 +779,16 @@ func runEmailTests(wg2 *sync.WaitGroup, emailInfo *string, output, tempOutput st | ||||
| func runNetworkTests(wg3 *sync.WaitGroup, ptInfo *string, output, tempOutput string, outputMutex *sync.Mutex) string { | ||||
| 	outputMutex.Lock() | ||||
| 	defer outputMutex.Unlock() | ||||
| 	output = utils.PrintAndCapture(func() { | ||||
| 	return utils.PrintAndCapture(func() { | ||||
| 		if backtraceStatus && !onlyChinaTest { | ||||
| 			utils.PrintCenteredTitle("三网回程线路检测", width) | ||||
| 			if strings.Contains(output, "IPV6") { | ||||
| 				backtrace.BackTrace(true) | ||||
| 			} else { | ||||
| 				backtrace.BackTrace(false) | ||||
| 			utils.PrintCenteredTitle("上游及回程线路检测", width) | ||||
| 			upstreams.UpstreamsCheck() // 不能在重定向的同时外部并发,此处仅可以顺序执行 | ||||
| 		} | ||||
| 		} | ||||
| 	}, tempOutput, output) | ||||
| 	output = utils.PrintAndCapture(func() { | ||||
| 		if nt3Status && !onlyChinaTest { | ||||
| 			utils.PrintCenteredTitle("三网回程路由检测", width) | ||||
| 			nt.TraceRoute(language, nt3Location, nt3CheckType) | ||||
| 			nexttrace.NextTrace3Check(language, nt3Location, nt3CheckType) // 不能在重定向的同时外部并发,此处仅可以顺序执行 | ||||
| 		} | ||||
| 	}, tempOutput, output) | ||||
| 	return utils.PrintAndCapture(func() { | ||||
| 		if onlyChinaTest || pingTestStatus { | ||||
| 		if (onlyChinaTest || pingTestStatus) && *ptInfo != "" { | ||||
| 			wg3.Wait() | ||||
| 			utils.PrintCenteredTitle("三网ICMP的PING值检测", width) | ||||
| 			fmt.Println(*ptInfo) | ||||
| @@ -774,8 +857,10 @@ func handleUploadResults(output string) { | ||||
| 	if httpURL != "" || httpsURL != "" { | ||||
| 		if language == "en" { | ||||
| 			fmt.Printf("Upload successfully!\nHttp URL:  %s\nHttps URL: %s\n", httpURL, httpsURL) | ||||
| 			fmt.Println("Each Test Benchmark: https://bash.spiritlhl.net/ecsguide") | ||||
| 		} else { | ||||
| 			fmt.Printf("上传成功!\nHttp URL:  %s\nHttps URL: %s\n", httpURL, httpsURL) | ||||
| 			fmt.Println("每项测试基准见: https://bash.spiritlhl.net/ecsguide") | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| @@ -786,12 +871,16 @@ func main() { | ||||
| 		return | ||||
| 	} | ||||
| 	initLogger() | ||||
| 	go func() { | ||||
| 		http.Get("https://hits.spiritlhl.net/goecs.svg?action=hit&title=Hits&title_bg=%23555555&count_bg=%230eecf8&edge_flat=false") | ||||
| 	}() | ||||
| 	preCheck := utils.CheckPublicAccess(3 * time.Second) | ||||
| 	go func() { | ||||
| 		if preCheck.Connected { | ||||
| 			http.Get("https://hits.spiritlhl.net/goecs.svg?action=hit&title=Hits&title_bg=%23555555&count_bg=%230eecf8&edge_flat=false") | ||||
| 		} | ||||
| 	}() | ||||
| 	if menuMode { | ||||
| 		handleMenuMode(preCheck) | ||||
| 	} else { | ||||
| 		onlyIpInfoCheckStatus = true | ||||
| 	} | ||||
| 	handleLanguageSpecificSettings() | ||||
| 	if !preCheck.Connected { | ||||
|   | ||||
							
								
								
									
										333
									
								
								goecs.sh
									
									
									
									
									
								
							
							
						
						
									
										333
									
								
								goecs.sh
									
									
									
									
									
								
							| @@ -1,6 +1,6 @@ | ||||
| #!/bin/bash | ||||
| #!/bin/sh | ||||
| # From https://github.com/oneclickvirt/ecs | ||||
| # 2025.06.29 | ||||
| # 2025.08.26 | ||||
|  | ||||
| # curl -L https://raw.githubusercontent.com/oneclickvirt/ecs/master/goecs.sh -o goecs.sh && chmod +x goecs.sh | ||||
| # 或 | ||||
| @@ -20,22 +20,27 @@ cd /root >/dev/null 2>&1 | ||||
| if [ ! -d "/usr/bin/" ]; then | ||||
|     mkdir -p "/usr/bin/" | ||||
| fi | ||||
| _red() { echo -e "\033[31m\033[01m$@\033[0m"; } | ||||
| _green() { echo -e "\033[32m\033[01m$@\033[0m"; } | ||||
| _yellow() { echo -e "\033[33m\033[01m$@\033[0m"; } | ||||
| _blue() { echo -e "\033[36m\033[01m$@\033[0m"; } | ||||
| reading() { read -rp "$(_green "$1")" "$2"; } | ||||
| _red() { printf "\033[31m\033[01m%s\033[0m\n" "$*"; } | ||||
| _green() { printf "\033[32m\033[01m%s\033[0m\n" "$*"; } | ||||
| _yellow() { printf "\033[33m\033[01m%s\033[0m\n" "$*"; } | ||||
| _blue() { printf "\033[36m\033[01m%s\033[0m\n" "$*"; } | ||||
| reading() {  | ||||
|     printf "\033[32m\033[01m%s\033[0m" "$1" | ||||
|     read "$2" | ||||
| } | ||||
|  | ||||
| check_cdn() { | ||||
|     local o_url=$1 | ||||
|     for cdn_url in "${cdn_urls[@]}"; do | ||||
|         if curl -sL -k "$cdn_url$o_url" --max-time 6 | grep -q "success" >/dev/null 2>&1; then | ||||
|             export cdn_success_url="$cdn_url" | ||||
|             return | ||||
|     local o_url="$1" | ||||
|     local cdn_url | ||||
|     for cdn_url in $cdn_urls; do | ||||
|         if curl -4 -sL -k "$cdn_url$o_url" --max-time 6 | grep -q "success" >/dev/null 2>&1; then | ||||
|             cdn_success_url="$cdn_url" | ||||
|             return 0 | ||||
|         fi | ||||
|         sleep 0.5 | ||||
|     done | ||||
|     export cdn_success_url="" | ||||
|     cdn_success_url="" | ||||
|     return 1 | ||||
| } | ||||
|  | ||||
| check_cdn_file() { | ||||
| @@ -50,9 +55,9 @@ check_cdn_file() { | ||||
| download_file() { | ||||
|     local url="$1" | ||||
|     local output="$2" | ||||
|     if ! wget -O "$output" "$url"; then | ||||
|     if ! wget -O "$output" "$url" 2>/dev/null; then | ||||
|         _yellow "wget failed, trying curl..." | ||||
|         if ! curl -L -o "$output" "$url"; then | ||||
|         if ! curl -L -o "$output" "$url" 2>/dev/null; then | ||||
|             _red "Both wget and curl failed. Unable to download the file." | ||||
|             return 1 | ||||
|         fi | ||||
| @@ -62,8 +67,7 @@ download_file() { | ||||
|  | ||||
| check_china() { | ||||
|     _yellow "正在检测IP所在区域......" | ||||
|     if [[ -z "${CN}" ]]; then | ||||
|         # 首先尝试通过 ipapi.co 检测 | ||||
|     if [ -z "${CN}" ]; then | ||||
|         if curl -m 6 -s https://ipapi.co/json | grep -q 'China'; then | ||||
|             _yellow "根据ipapi.co提供的信息,当前IP可能在中国" | ||||
|             if [ "$noninteractive" != "true" ]; then | ||||
| @@ -94,28 +98,33 @@ check_china() { | ||||
|  | ||||
| get_memory_size() { | ||||
|     if [ -f /proc/meminfo ]; then | ||||
|         local mem_kb=$(grep MemTotal /proc/meminfo | awk '{print $2}') | ||||
|         local mem_kb | ||||
|         mem_kb=$(grep MemTotal /proc/meminfo | awk '{print $2}') | ||||
|         echo $((mem_kb / 1024)) # Convert to MB | ||||
|         return | ||||
|         return 0 | ||||
|     fi | ||||
|     if command -v free >/dev/null 2>&1; then | ||||
|         local mem_kb=$(free -m | awk '/^Mem:/ {print $2}') | ||||
|         local mem_kb | ||||
|         mem_kb=$(free -m | awk '/^Mem:/ {print $2}') | ||||
|         echo "$mem_kb" # Already in MB | ||||
|         return | ||||
|         return 0 | ||||
|     fi | ||||
|     if command -v sysctl >/dev/null 2>&1; then | ||||
|         local mem_bytes=$(sysctl -n hw.memsize 2>/dev/null || sysctl -n hw.physmem 2>/dev/null) | ||||
|         local mem_bytes | ||||
|         mem_bytes=$(sysctl -n hw.memsize 2>/dev/null || sysctl -n hw.physmem 2>/dev/null) | ||||
|         if [ -n "$mem_bytes" ]; then | ||||
|             echo $((mem_bytes / 1024 / 1024)) # Convert to MB | ||||
|             return | ||||
|             return 0 | ||||
|         fi | ||||
|     fi | ||||
|     echo 0 | ||||
|     return 1 | ||||
| } | ||||
|  | ||||
| cleanup_epel() { | ||||
|     _yellow "Cleaning up EPEL repositories..." | ||||
|     rm -f /etc/yum.repos.d/*epel* | ||||
|     yum clean all | ||||
|     yum clean all >/dev/null 2>&1 | ||||
| } | ||||
|  | ||||
| goecs_check() { | ||||
| @@ -143,7 +152,7 @@ goecs_check() { | ||||
|     os=$(uname -s 2>/dev/null || echo "Unknown") | ||||
|     arch=$(uname -m 2>/dev/null || echo "Unknown") | ||||
|     check_china | ||||
|     ECS_VERSION="0.1.55" | ||||
|     ECS_VERSION="0.1.89" | ||||
|     for api in \ | ||||
|         "https://api.github.com/repos/oneclickvirt/ecs/releases/latest" \ | ||||
|         "https://githubapi.spiritlhl.workers.dev/repos/oneclickvirt/ecs/releases/latest" \ | ||||
| @@ -155,23 +164,24 @@ goecs_check() { | ||||
|         sleep 1 | ||||
|     done | ||||
|     if [ -z "$ECS_VERSION" ]; then | ||||
|         _yellow "Unable to get version info, using default version 0.1.55" | ||||
|         ECS_VERSION="0.1.55" | ||||
|         _yellow "Unable to get version info, using default version 0.1.89" | ||||
|         ECS_VERSION="0.1.89" | ||||
|     fi | ||||
|     version_output="" | ||||
|     for cmd_path in "goecs" "./goecs" "/usr/bin/goecs" "/usr/local/bin/goecs"; do | ||||
|         if [ -x "$(command -v $cmd_path 2>/dev/null)" ]; then | ||||
|         if command -v "$cmd_path" >/dev/null 2>&1; then | ||||
|             version_output=$($cmd_path -v command 2>/dev/null) | ||||
|             break | ||||
|         fi | ||||
|     done | ||||
|     if [ -n "$version_output" ]; then | ||||
|         extracted_version=${version_output//v/} | ||||
|         extracted_version=${version_output#*v} | ||||
|         extracted_version=${extracted_version#v} | ||||
|         if [ -n "$extracted_version" ]; then | ||||
|             ecs_version=$ECS_VERSION | ||||
|             if [[ "$(echo -e "$extracted_version\n$ecs_version" | sort -V | tail -n 1)" == "$extracted_version" ]]; then | ||||
|             if [ "$(printf '%s\n%s\n' "$extracted_version" "$ecs_version" | sort -V | tail -n 1)" = "$extracted_version" ]; then | ||||
|                 _green "goecs version ($extracted_version) is up to date, no upgrade needed" | ||||
|                 return | ||||
|                 return 0 | ||||
|             else | ||||
|                 _yellow "goecs version ($extracted_version) < $ecs_version, upgrade needed, starting in 5 seconds" | ||||
|                 rm -rf /usr/bin/goecs /usr/local/bin/goecs ./goecs | ||||
| @@ -181,11 +191,11 @@ goecs_check() { | ||||
|         _green "goecs not found, installation needed, starting in 5 seconds" | ||||
|     fi | ||||
|     sleep 5 | ||||
|     if [[ "$CN" == true ]]; then | ||||
|     if [ "$CN" = "true" ]; then | ||||
|         _yellow "Using China mirror for download..." | ||||
|         base_url="https://cnb.cool/oneclickvirt/ecs/-/git/raw/main" | ||||
|     else | ||||
|         cdn_urls=("https://cdn0.spiritlhl.top/" "http://cdn3.spiritlhl.net/" "http://cdn1.spiritlhl.net/" "http://cdn2.spiritlhl.net/") | ||||
|         cdn_urls="https://cdn0.spiritlhl.top/ http://cdn3.spiritlhl.net/ http://cdn1.spiritlhl.net/ http://cdn2.spiritlhl.net/" | ||||
|         check_cdn_file | ||||
|         if [ -n "$cdn_success_url" ]; then | ||||
|             base_url="${cdn_success_url}https://github.com/oneclickvirt/ecs/releases/download/v${ECS_VERSION}" | ||||
| @@ -301,169 +311,179 @@ InstallSysbench() { | ||||
|     else | ||||
|         Var_OSRelease="unknown" # 未知系统分支 | ||||
|     fi | ||||
|     local mem_size=$(get_memory_size) | ||||
|     local mem_size | ||||
|     mem_size=$(get_memory_size) | ||||
|     if [ -z "$mem_size" ] || [ "$mem_size" -eq 0 ]; then | ||||
|         echo "Error: Unable to determine memory size or memory size is zero." | ||||
|     elif [ $mem_size -lt 1024 ]; then | ||||
|     elif [ "$mem_size" -lt 1024 ]; then | ||||
|         _red "Warning: Your system has less than 1GB RAM (${mem_size}MB)" | ||||
|         if [ "$noninteractive" != "true" ]; then | ||||
|             reading "Do you want to continue with EPEL installation? (y/N): " confirm | ||||
|             if [[ ! $confirm =~ ^[Yy]$ ]]; then | ||||
|             case "$confirm" in | ||||
|                 [Yy]*) | ||||
|                     ;; | ||||
|                 *) | ||||
|                     _yellow "Skipping EPEL installation" | ||||
|                     return 1 | ||||
|             fi | ||||
|                     ;; | ||||
|             esac | ||||
|         fi | ||||
|         case "$Var_OSRelease" in | ||||
|         ubuntu | debian | astra) | ||||
|             ! apt-get install -y sysbench && apt-get --fix-broken install -y && apt-get install --no-install-recommends -y sysbench ;; | ||||
|             if ! apt-get install -y sysbench; then | ||||
|                 apt-get --fix-broken install -y | ||||
|                 apt-get install --no-install-recommends -y sysbench | ||||
|             fi | ||||
|             ;; | ||||
|         centos | rhel | almalinux | redhat | opencloudos) | ||||
|             (yum -y install epel-release && yum -y install sysbench) || (dnf install epel-release -y && dnf install sysbench -y) ;; | ||||
|             if ! yum -y install epel-release || ! yum -y install sysbench; then | ||||
|                 if command -v dnf >/dev/null 2>&1; then | ||||
|                     dnf install epel-release -y | ||||
|                     dnf install sysbench -y | ||||
|                 fi | ||||
|             fi | ||||
|             ;; | ||||
|         fedora) | ||||
|             dnf -y install sysbench ;; | ||||
|         arch) | ||||
|             pacman -S --needed --noconfirm sysbench && pacman -S --needed --noconfirm libaio && ldconfig ;; | ||||
|             pacman -S --needed --noconfirm sysbench | ||||
|             pacman -S --needed --noconfirm libaio | ||||
|             ldconfig | ||||
|             ;; | ||||
|         freebsd) | ||||
|             pkg install -y sysbench ;; | ||||
|         alpinelinux) | ||||
|             if [ "$noninteractive" != "true" ]; then | ||||
|                 reading "Do you want to continue with sysbench installation? (y/N): " confirm | ||||
|                 if [[ ! $confirm =~ ^[Yy]$ ]]; then | ||||
|                 case "$confirm" in | ||||
|                     [Yy]*) | ||||
|                         ;; | ||||
|                     *) | ||||
|                         _yellow "Skipping sysbench installation" | ||||
|                         return 1 | ||||
|                 fi | ||||
|                         ;; | ||||
|                 esac | ||||
|             fi | ||||
|             ALPINE_VERSION=$(grep -o '^[0-9]\+\.[0-9]\+' /etc/alpine-release) | ||||
|             COMMUNITY_REPO="http://dl-cdn.alpinelinux.org/alpine/v${ALPINE_VERSION}/community" | ||||
|             if grep -q "^${COMMUNITY_REPO}" /etc/apk/repositories; then | ||||
|                 echo "Community repository is already enabled." | ||||
|             else | ||||
|             if ! grep -q "^${COMMUNITY_REPO}" /etc/apk/repositories; then | ||||
|                 echo "Enabling community repository..." | ||||
|                 echo "${COMMUNITY_REPO}" >> /etc/apk/repositories | ||||
|                 echo "Community repository has been added." | ||||
|                 echo "Updating apk package index..." | ||||
|                 apk update && echo "Package index updated successfully." | ||||
|             else | ||||
|                 echo "Community repository is already enabled." | ||||
|             fi | ||||
|             if apk info sysbench >/dev/null 2>&1; then | ||||
|                 echo -e "${Msg_Info}Sysbench already installed." | ||||
|                 echo "Sysbench already installed." | ||||
|             else | ||||
|                 apk add --no-cache sysbench | ||||
|                 if [ $? -ne 0 ]; then | ||||
|                     echo -e "${Msg_Warning}Sysbench Module not found, installing ..." && echo -e "${Msg_Warning}SysBench Current not support Alpine Linux, Skipping..." && Var_Skip_SysBench="1" | ||||
|                 if ! apk add --no-cache sysbench; then | ||||
|                     echo "Sysbench Module not found, installing ..." | ||||
|                     echo "SysBench Current not support Alpine Linux, Skipping..." | ||||
|                     Var_Skip_SysBench="1" | ||||
|                 else | ||||
|                     echo -e "${Msg_Success}Sysbench installed successfully." | ||||
|                     echo "Sysbench installed successfully." | ||||
|                 fi | ||||
|             fi ;; | ||||
|             fi | ||||
|             ;; | ||||
|         *) | ||||
|             _red "Sysbench Install Error: Unknown OS release: $Var_OSRelease" ;; | ||||
|         esac | ||||
|         if [[ $SYSTEM =~ ^(CentOS|RHEL|AlmaLinux)$ ]]; then | ||||
|         case "$SYSTEM" in | ||||
|             CentOS|RHEL|AlmaLinux) | ||||
|                 _yellow "Installing EPEL repository..." | ||||
|                 if ! yum -y install epel-release; then | ||||
|                     _red "EPEL installation failed!" | ||||
|                     cleanup_epel | ||||
|                     _yellow "Attempting to continue without EPEL..." | ||||
|                 fi | ||||
|         fi | ||||
|     fi | ||||
| } | ||||
|  | ||||
| Check_SysBench() { | ||||
|     if [ ! -f "/usr/bin/sysbench" ] && [ ! -f "/usr/local/bin/sysbench" ]; then | ||||
|         InstallSysbench | ||||
|     fi | ||||
|     # 尝试编译安装 | ||||
|     if [ ! -f "/usr/bin/sysbench" ] && [ ! -f "/usr/local/bin/sysbench" ]; then | ||||
|         echo -e "${Msg_Warning}Sysbench Module install Failure, trying compile modules ..." | ||||
|         Check_Sysbench_InstantBuild | ||||
|     fi | ||||
|     source ~/.bashrc | ||||
|     # 最终检测 | ||||
|     if [ "$(command -v sysbench)" ] || [ -f "/usr/bin/sysbench" ] || [ -f "/usr/local/bin/sysbench" ]; then | ||||
|         _yellow "Install sysbench successfully!" | ||||
|     else | ||||
|         _red "SysBench Moudle install Failure! Try Restart Bench or Manually install it! (/usr/bin/sysbench)" | ||||
|         _blue "Will try to test with geekbench5 instead later." | ||||
|     fi | ||||
|     sleep 3 | ||||
| } | ||||
|  | ||||
| Check_Sysbench_InstantBuild() { | ||||
|     if [ "${Var_OSRelease}" = "centos" ] || [ "${Var_OSRelease}" = "rhel" ] || [ "${Var_OSRelease}" = "almalinux" ] || [ "${Var_OSRelease}" = "ubuntu" ] || [ "${Var_OSRelease}" = "debian" ] || [ "${Var_OSRelease}" = "fedora" ] || [ "${Var_OSRelease}" = "arch" ] || [ "${Var_OSRelease}" = "astra" ]; then | ||||
|         local os_sysbench=${Var_OSRelease} | ||||
|         if [ "$os_sysbench" = "astra" ]; then | ||||
|             os_sysbench="debian" | ||||
|         fi | ||||
|         if [ "$os_sysbench" = "opencloudos" ]; then | ||||
|             os_sysbench="centos" | ||||
|         fi | ||||
|         echo -e "${Msg_Info}Release Detected: ${os_sysbench}" | ||||
|         echo -e "${Msg_Info}Preparing compile enviorment ..." | ||||
|         prepare_compile_env "${os_sysbench}" | ||||
|         echo -e "${Msg_Info}Downloading Source code (Version 1.0.20)..." | ||||
|         mkdir -p /tmp/sysbench_install/src/ | ||||
|         mv /tmp/sysbench-1.0.20 /tmp/sysbench_install/src/ | ||||
|         echo -e "${Msg_Info}Compiling Sysbench Module ..." | ||||
|         cd /tmp/sysbench_install/src/sysbench-1.0.20 | ||||
|         ./autogen.sh && ./configure --without-mysql && make -j8 && make install | ||||
|         echo -e "${Msg_Info}Cleaning up ..." | ||||
|         cd /tmp && rm -rf /tmp/sysbench_install/src/sysbench* | ||||
|     else | ||||
|         echo -e "${Msg_Warning}Unsupported operating system: ${Var_OSRelease}" | ||||
|     fi | ||||
| } | ||||
|  | ||||
| prepare_compile_env() { | ||||
|     local system="$1" | ||||
|     if [ "${system}" = "centos" ] || [ "${system}" = "rhel" ] || [ "${system}" = "almalinux" ]; then | ||||
|         yum install -y epel-release | ||||
|         yum install -y wget curl make gcc gcc-c++ make automake libtool pkgconfig libaio-devel | ||||
|     elif [ "${system}" = "ubuntu" ] || [ "${system}" = "debian" ]; then | ||||
|         ! apt-get update && apt-get --fix-broken install -y && apt-get update | ||||
|         ! apt-get -y install --no-install-recommends curl wget make automake libtool pkg-config libaio-dev unzip && apt-get --fix-broken install -y && apt-get -y install --no-install-recommends curl wget make automake libtool pkg-config libaio-dev unzip | ||||
|     elif [ "${system}" = "fedora" ]; then | ||||
|         dnf install -y wget curl gcc gcc-c++ make automake libtool pkgconfig libaio-devel | ||||
|     elif [ "${system}" = "arch" ]; then | ||||
|         pacman -S --needed --noconfirm wget curl gcc gcc make automake libtool pkgconfig libaio lib32-libaio | ||||
|     else | ||||
|         echo -e "${Msg_Warning}Unsupported operating system: ${system}" | ||||
|                 ;; | ||||
|         esac | ||||
|     fi | ||||
| } | ||||
|  | ||||
| env_check() { | ||||
|     REGEX=("debian|astra" "ubuntu" "centos|red hat|kernel|oracle linux|alma|rocky" "'amazon linux'" "fedora" "arch" "freebsd" "alpine" "openbsd" "opencloudos") | ||||
|     RELEASE=("Debian" "Ubuntu" "CentOS" "CentOS" "Fedora" "Arch" "FreeBSD" "Alpine" "OpenBSD" "OpenCloudOS") | ||||
|     PACKAGE_UPDATE=("apt-get update" "apt-get update" "yum -y update" "yum -y update" "yum -y update" "pacman -Sy" "pkg update" "apk update" "pkg_add -qu" "yum -y update") | ||||
|     PACKAGE_INSTALL=("apt-get -y install" "apt-get -y install" "yum -y install" "yum -y install" "yum -y install" "pacman -Sy --noconfirm --needed" "pkg install -y" "apk add --no-cache" "pkg_add -I" "yum -y install") | ||||
|     PACKAGE_REMOVE=("apt-get -y remove" "apt-get -y remove" "yum -y remove" "yum -y remove" "yum -y remove" "pacman -Rsc --noconfirm" "pkg delete" "apk del" "pkg_delete -I" "yum -y remove") | ||||
|     PACKAGE_UNINSTALL=("apt-get -y autoremove" "apt-get -y autoremove" "yum -y autoremove" "yum -y autoremove" "yum -y autoremove" "pacman -Rns --noconfirm" "pkg autoremove" "apk autoremove" "pkg_delete -a" "yum -y autoremove") | ||||
|     if [ -f /etc/opencloudos-release ]; then | ||||
|         SYS="opencloudos" | ||||
|     elif [ -s /etc/os-release ]; then | ||||
|         SYS="$(grep -i pretty_name /etc/os-release | cut -d \" -f2)" | ||||
|     elif [ -x "$(type -p hostnamectl)" ]; then | ||||
|         SYS="$(hostnamectl | grep -i system | cut -d : -f2 | xargs)" | ||||
|     elif [ -x "$(type -p lsb_release)" ]; then | ||||
|     elif command -v hostnamectl >/dev/null 2>&1; then | ||||
|         SYS="$(hostnamectl | grep -i system | cut -d : -f2 | sed 's/^ *//')" | ||||
|     elif command -v lsb_release >/dev/null 2>&1; then | ||||
|         SYS="$(lsb_release -sd)" | ||||
|     elif [ -s /etc/lsb-release ]; then | ||||
|         SYS="$(grep -i description /etc/lsb-release | cut -d \" -f2)" | ||||
|     elif [ -s /etc/redhat-release ]; then | ||||
|         SYS="$(grep . /etc/redhat-release)" | ||||
|         SYS="$(cat /etc/redhat-release)" | ||||
|     elif [ -s /etc/issue ]; then | ||||
|         SYS="$(grep . /etc/issue | cut -d '\' -f1 | sed '/^[ ]*$/d')" | ||||
|         SYS="$(head -n1 /etc/issue | cut -d '\' -f1 | sed '/^[ ]*$/d')" | ||||
|     else | ||||
|         SYS="$(uname -s)" | ||||
|     fi | ||||
|     SYSTEM="" | ||||
|     for ((int = 0; int < ${#REGEX[@]}; int++)); do | ||||
|         if [[ $(echo "$SYS" | tr '[:upper:]' '[:lower:]') =~ ${REGEX[int]} ]]; then | ||||
|             SYSTEM="${RELEASE[int]}" | ||||
|             UPDATE_CMD=${PACKAGE_UPDATE[int]} | ||||
|             INSTALL_CMD=${PACKAGE_INSTALL[int]} | ||||
|             REMOVE_CMD=${PACKAGE_REMOVE[int]} | ||||
|             UNINSTALL_CMD=${PACKAGE_UNINSTALL[int]} | ||||
|             break | ||||
|     sys_lower=$(echo "$SYS" | tr '[:upper:]' '[:lower:]') | ||||
|     if echo "$sys_lower" | grep -E "debian|astra" >/dev/null 2>&1; then | ||||
|         SYSTEM="Debian" | ||||
|         UPDATE_CMD="apt-get update" | ||||
|         INSTALL_CMD="apt-get -y install" | ||||
|         REMOVE_CMD="apt-get -y remove" | ||||
|         UNINSTALL_CMD="apt-get -y autoremove" | ||||
|     elif echo "$sys_lower" | grep -E "ubuntu" >/dev/null 2>&1; then | ||||
|         SYSTEM="Ubuntu" | ||||
|         UPDATE_CMD="apt-get update" | ||||
|         INSTALL_CMD="apt-get -y install" | ||||
|         REMOVE_CMD="apt-get -y remove" | ||||
|         UNINSTALL_CMD="apt-get -y autoremove" | ||||
|     elif echo "$sys_lower" | grep -E "centos|red hat|kernel|oracle linux|alma|rocky" >/dev/null 2>&1; then | ||||
|         SYSTEM="CentOS" | ||||
|         UPDATE_CMD="yum -y update" | ||||
|         INSTALL_CMD="yum -y install" | ||||
|         REMOVE_CMD="yum -y remove" | ||||
|         UNINSTALL_CMD="yum -y autoremove" | ||||
|     elif echo "$sys_lower" | grep -E "amazon linux" >/dev/null 2>&1; then | ||||
|         SYSTEM="CentOS" | ||||
|         UPDATE_CMD="yum -y update" | ||||
|         INSTALL_CMD="yum -y install" | ||||
|         REMOVE_CMD="yum -y remove" | ||||
|         UNINSTALL_CMD="yum -y autoremove" | ||||
|     elif echo "$sys_lower" | grep -E "fedora" >/dev/null 2>&1; then | ||||
|         SYSTEM="Fedora" | ||||
|         UPDATE_CMD="yum -y update" | ||||
|         INSTALL_CMD="yum -y install" | ||||
|         REMOVE_CMD="yum -y remove" | ||||
|         UNINSTALL_CMD="yum -y autoremove" | ||||
|     elif echo "$sys_lower" | grep -E "arch" >/dev/null 2>&1; then | ||||
|         SYSTEM="Arch" | ||||
|         UPDATE_CMD="pacman -Sy" | ||||
|         INSTALL_CMD="pacman -Sy --noconfirm --needed" | ||||
|         REMOVE_CMD="pacman -Rsc --noconfirm" | ||||
|         UNINSTALL_CMD="pacman -Rns --noconfirm" | ||||
|     elif echo "$sys_lower" | grep -E "freebsd" >/dev/null 2>&1; then | ||||
|         SYSTEM="FreeBSD" | ||||
|         UPDATE_CMD="pkg update" | ||||
|         INSTALL_CMD="pkg install -y" | ||||
|         REMOVE_CMD="pkg delete" | ||||
|         UNINSTALL_CMD="pkg autoremove" | ||||
|     elif echo "$sys_lower" | grep -E "alpine" >/dev/null 2>&1; then | ||||
|         SYSTEM="Alpine" | ||||
|         UPDATE_CMD="apk update" | ||||
|         INSTALL_CMD="apk add --no-cache" | ||||
|         REMOVE_CMD="apk del" | ||||
|         UNINSTALL_CMD="apk autoremove" | ||||
|     elif echo "$sys_lower" | grep -E "openbsd" >/dev/null 2>&1; then | ||||
|         SYSTEM="OpenBSD" | ||||
|         UPDATE_CMD="pkg_add -qu" | ||||
|         INSTALL_CMD="pkg_add -I" | ||||
|         REMOVE_CMD="pkg_delete -I" | ||||
|         UNINSTALL_CMD="pkg_delete -a" | ||||
|     elif echo "$sys_lower" | grep -E "opencloudos" >/dev/null 2>&1; then | ||||
|         SYSTEM="OpenCloudOS" | ||||
|         UPDATE_CMD="yum -y update" | ||||
|         INSTALL_CMD="yum -y install" | ||||
|         REMOVE_CMD="yum -y remove" | ||||
|         UNINSTALL_CMD="yum -y autoremove" | ||||
|     fi | ||||
|     done | ||||
|     if [ -z "$SYSTEM" ]; then | ||||
|         _yellow "Unable to recognize system, trying common package managers..." | ||||
|         if command -v apt-get >/dev/null 2>&1; then | ||||
| @@ -510,7 +530,7 @@ env_check() { | ||||
|     _green "System information: $SYSTEM" | ||||
|     _green "Update command: $UPDATE_CMD" | ||||
|     _green "Install command: $INSTALL_CMD" | ||||
|     cdn_urls=("https://cdn0.spiritlhl.top/" "http://cdn3.spiritlhl.net/" "http://cdn1.spiritlhl.net/" "http://cdn2.spiritlhl.net/") | ||||
|     cdn_urls="https://cdn0.spiritlhl.top/ http://cdn3.spiritlhl.net/ http://cdn1.spiritlhl.net/ http://cdn2.spiritlhl.net/" | ||||
|     check_cdn_file | ||||
|     _yellow "Warning: System update will be performed" | ||||
|     _yellow "This operation may:" | ||||
| @@ -520,15 +540,18 @@ env_check() { | ||||
|     _yellow "4. Affect subsequent system startups" | ||||
|     if [ "$noninteractive" != "true" ]; then | ||||
|         reading "Continue with system update? (y/N): " update_confirm | ||||
|         if [[ ! $update_confirm =~ ^[Yy]$ ]]; then | ||||
|             _yellow "Skipping system update" | ||||
|             _yellow "Note: Some packages may fail to install" | ||||
|         else | ||||
|         case "$update_confirm" in | ||||
|             [Yy]*) | ||||
|                 _green "Updating system package manager..." | ||||
|                 if ! ${UPDATE_CMD} 2>/dev/null; then | ||||
|                     _red "System update failed!" | ||||
|                 fi | ||||
|         fi | ||||
|                 ;; | ||||
|             *) | ||||
|                 _yellow "Skipping system update" | ||||
|                 _yellow "Note: Some packages may fail to install" | ||||
|                 ;; | ||||
|         esac | ||||
|     fi | ||||
|     for cmd in sudo wget tar unzip iproute2 systemd-detect-virt dd fio; do | ||||
|         if ! command -v "$cmd" >/dev/null 2>&1; then | ||||
| @@ -538,43 +561,37 @@ env_check() { | ||||
|     done | ||||
|     if ! command -v sysbench >/dev/null 2>&1; then | ||||
|         _green "Installing sysbench" | ||||
|         ${INSTALL_CMD} sysbench | ||||
|         if [ $? -ne 0 ]; then | ||||
|             echo "Unable to download sysbench through package manager, attempting compilation..." | ||||
|             wget -O /tmp/sysbench.zip "${cdn_success_url}https://github.com/akopytov/sysbench/archive/1.0.20.zip" || curl -Lk -o /tmp/sysbench.zip "${cdn_success_url}https://github.com/akopytov/sysbench/archive/1.0.20.zip" | ||||
|             if [ ! -f /tmp/sysbench.zip ]; then | ||||
|                 wget -q -O /tmp/sysbench.zip "https://hub.fgit.cf/akopytov/sysbench/archive/1.0.20.zip" | ||||
|             fi | ||||
|             chmod +x /tmp/sysbench.zip | ||||
|             unzip /tmp/sysbench.zip -d /tmp | ||||
|             Check_SysBench | ||||
|         if ! ${INSTALL_CMD} sysbench; then | ||||
|             _red "Unable to install sysbench through package manager" | ||||
|             _yellow "Sysbench installation skipped" | ||||
|         fi | ||||
|     fi | ||||
|     if ! command -v geekbench >/dev/null 2>&1; then | ||||
|         _green "Installing geekbench" | ||||
|         curl -L "${cdn_success_url}https://raw.githubusercontent.com/oneclickvirt/cputest/main/dgb.sh" -o dgb.sh && chmod +x dgb.sh | ||||
|         bash dgb.sh -v gb5 | ||||
|         sh dgb.sh -v gb5 | ||||
|         rm -rf dgb.sh | ||||
|     fi | ||||
|     if ! command -v speedtest >/dev/null 2>&1; then | ||||
|         _green "Installing speedtest" | ||||
|         curl -L "${cdn_success_url}https://raw.githubusercontent.com/oneclickvirt/speedtest/main/dspt.sh" -o dspt.sh && chmod +x dspt.sh | ||||
|         bash dspt.sh | ||||
|         sh dspt.sh | ||||
|         rm -rf dspt.sh | ||||
|         rm -rf speedtest.tar.gz | ||||
|     fi | ||||
|     if ! command -v ping >/dev/null 2>&1; then | ||||
|         _green "Installing ping" | ||||
|         ${INSTALL_CMD} iputils-ping >/dev/null 2>&1 | ||||
|         ${INSTALL_CMD} ping >/dev/null 2>&1 | ||||
|         ${INSTALL_CMD} iputils-ping >/dev/null 2>&1 || ${INSTALL_CMD} ping >/dev/null 2>&1 | ||||
|     fi | ||||
|     if [ "$(uname -s)" = "Darwin" ]; then | ||||
|         echo "Detected MacOS, installing sysbench iproute2mac..." | ||||
|         if command -v brew >/dev/null 2>&1; then | ||||
|             brew install --force sysbench iproute2mac | ||||
|         fi | ||||
|     else | ||||
|         if ! grep -q "^net.ipv4.ping_group_range = 0 2147483647$" /etc/sysctl.conf; then | ||||
|             echo "net.ipv4.ping_group_range = 0 2147483647" >> /etc/sysctl.conf | ||||
|             sysctl -p | ||||
|         if ! grep -q "^net.ipv4.ping_group_range = 0 2147483647$" /etc/sysctl.conf 2>/dev/null; then | ||||
|             echo "net.ipv4.ping_group_range = 0 2147483647" >> /etc/sysctl.conf 2>/dev/null | ||||
|             sysctl -p >/dev/null 2>&1 | ||||
|         fi | ||||
|     fi | ||||
|     _green "Environment preparation complete." | ||||
|   | ||||
							
								
								
									
										113
									
								
								goecs.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										113
									
								
								goecs.txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,113 @@ | ||||
| -----------------------VPS Fusion Monster Test (Unofficial)----------------------- | ||||
| Version: v0.1.89 | ||||
| Review Channel: https://t.me/+UHVoo2U4VyA5NTQ1 | ||||
| Go Project: https://github.com/oneclickvirt/ecs | ||||
| Shell Project: https://github.com/spiritLHLS/ecs | ||||
| -----------------------------System-Basic-Information----------------------------- | ||||
|  CPU Model           : AMD EPYC 7763 64-Core Processor @ 3243.593 MHz | ||||
|  CPU Cores           : 4 Physical CPU(s) | ||||
|  CPU Cache           : L1: 128 KB / L2: 1 MB / L3: 32 MB | ||||
|  AES-NI              : ✔️ Enabled | ||||
|  VM-x/AMD-V/Hyper-V  : ✔️ Enabled | ||||
|  RAM                 : 896.88 MB / 15.62 GB | ||||
|  Virtio Balloon      : ❌ Undetected | ||||
|  KSM                 : ❌ Undetected | ||||
|  Swap                : 0.00 MB / 4.00 GB | ||||
|  Disk 1              : 4.00 GB / 73.27 GB [5.5%%] /dev/sda1 - /mnt | ||||
|  Disk 2              : 47.23 GB / 71.61 GB [66.0%%] /dev/sdb1 - / | ||||
|  Boot Path           : /dev/sdb1 | ||||
|  OS Release          : ubuntu 24.04 [x86_64]  | ||||
|  Kernel              : 6.11.0-1018-azure | ||||
|  Uptime              : 0 days, 00 hours, 01 minutes | ||||
|  Current Time Zone   : UTC | ||||
|  Load                : 3.40 / 1.04 / 0.36 | ||||
|  VM Type             : Microsoft Hyper-V | ||||
|  NAT Type            : Port Restricted Cone | ||||
|  Tcp Accelerate      : cubic | ||||
|  IPV4 ASN            : AS8075 MICROSOFT-CORP-MSN-AS-BLOCK | ||||
|  IPV4 Location       : Phoenix / Arizona / United States | ||||
|  IPV4 Active IPs     : 31424/2097152 (prefix /11) | ||||
| ----------------------------CPU-Test--sysbench-Method----------------------------- | ||||
| 1 Thread(s) Test: 3384.91 | ||||
| 4 Thread(s) Test: 7482.71 | ||||
| ----------------------------Memory-Test--stream-Method---------------------------- | ||||
| Function    Best Rate MB/s  Avg time     Min time     Max time | ||||
| Copy:           38742.0     0.004222     0.004130     0.004312 | ||||
| Scale:          28674.1     0.005744     0.005580     0.005969 | ||||
| Add:            29906.8     0.008481     0.008025     0.008803 | ||||
| Triad:          30317.5     0.008315     0.007916     0.008671 | ||||
| ------------------------------Disk-Test--fio-Method------------------------------- | ||||
| Test Path    Block     Read(IOPS)           Write(IOPS)          Total(IOPS)          | ||||
| /tmp              4k        39.16 MB/s(9790)        39.24 MB/s(9810)        78.40 MB/s(19.6k)       | ||||
| /tmp              64k       205.99 MB/s(3218)       207.08 MB/s(3235)       413.07 MB/s(6453)       | ||||
| /tmp              512k      202.25 MB/s(395)        212.99 MB/s(415)        415.24 MB/s(810)        | ||||
| /tmp              1m        200.65 MB/s(195)        214.01 MB/s(208)        414.66 MB/s(403)        | ||||
| -----------------------Cross-Border-Streaming-Media-Unlock------------------------ | ||||
| IPV4: | ||||
| ===============[ Global ]=============== | ||||
| Apple                     YES (Region: USA) [Native] | ||||
| BingSearch                YES (Region: US) | ||||
| Claude                    YES [Native] | ||||
| Dazn                      YES (Region: US) [Native] | ||||
| Disney+                   NO (forbidden-location) | ||||
| Gemini                    YES (Region: US) [Native] | ||||
| GoogleSearch              YES | ||||
| Google Play Store         YES (Region: US) [Native] | ||||
| IQiYi                     YES (Region: US) [Native] | ||||
| Instagram Licensed Audio  YES [Native] | ||||
| KOCOWA                    YES [Native] | ||||
| MetaAI                    YES (Region: US) [Native] | ||||
| Netflix                   YES (Region: US) [Native] | ||||
| Netflix CDN               CO | ||||
| OneTrust                  YES (Region: US ARIZONA) [Via DNS] | ||||
| ChatGPT                   YES (Region: US) [Native] | ||||
| Paramount+                YES [Native] | ||||
| Amazon Prime Video        YES (Region: US) [Native] | ||||
| Reddit                    NO | ||||
| SonyLiv                   YES (Region: IN) [Native] | ||||
| Sora                      YES (Region: US) | ||||
| Spotify Registration      NO | ||||
| Steam Store               YES (Community Available) (Region: US) | ||||
| TVBAnywhere+              YES (Region: US) [Native] | ||||
| TikTok                    YES (Region: US) [Native] | ||||
| Viu.com                   YES [Native] | ||||
| Wikipedia Editability     NO | ||||
| YouTube Region            YES (Region: US) [Native] | ||||
| YouTube CDN               PHX | ||||
| ---------------------------------Email-Port-Check--------------------------------- | ||||
| Platform  SMTP  SMTPS POP3  POP3S IMAP  IMAPS | ||||
| LocalPort ✘     ✘     ✘     ✘     ✘     ✘     | ||||
| QQ        ✔     ✔     ✔     ✘     ✔     ✘     | ||||
| 163       ✔     ✔     ✔     ✘     ✔     ✘     | ||||
| Sohu      ✔     ✔     ✔     ✘     ✔     ✘     | ||||
| Yandex    ✔     ✔     ✔     ✘     ✔     ✘     | ||||
| Gmail     ✔     ✔     ✘     ✘     ✘     ✘     | ||||
| Outlook   ✔     ✘     ✔     ✘     ✔     ✘     | ||||
| Office365 ✔     ✘     ✔     ✘     ✔     ✘     | ||||
| Yahoo     ✔     ✔     ✘     ✘     ✘     ✘     | ||||
| MailCOM   ✔     ✔     ✔     ✘     ✔     ✘     | ||||
| MailRU    ✔     ✔     ✘     ✘     ✔     ✘     | ||||
| AOL       ✔     ✔     ✘     ✘     ✘     ✘     | ||||
| GMX       ✔     ✔     ✔     ✘     ✔     ✘     | ||||
| Sina      ✔     ✔     ✔     ✘     ✔     ✘     | ||||
| Apple     ✘     ✔     ✘     ✘     ✘     ✘     | ||||
| FastMail  ✘     ✔     ✘     ✘     ✘     ✘     | ||||
| ProtonMail✘     ✘     ✘     ✘     ✘     ✘     | ||||
| MXRoute   ✔     ✘     ✔     ✘     ✔     ✘     | ||||
| Namecrane ✔     ✔     ✔     ✘     ✔     ✘     | ||||
| XYAMail   ✘     ✘     ✘     ✘     ✘     ✘     | ||||
| ZohoMail  ✘     ✔     ✘     ✘     ✘     ✘     | ||||
| Inbox_eu  ✔     ✔     ✔     ✘     ✘     ✘     | ||||
| Free_fr   ✘     ✔     ✔     ✘     ✔     ✘     | ||||
| ------------------------------------Speed-Test------------------------------------ | ||||
| Location        Upload Speed    Download Speed  Latency         PacketLoss       | ||||
| Speedtest.net   8236.20 Mbps    8604.96 Mbps    1.79558ms       0.00% (Sent: 402/Dup: 0/Max: 401) | ||||
| LosAngeles      736.78 Mbps     876.41 Mbps     8.850929ms      N/A              | ||||
| Tokyo,Japan     484.27 Mbps     757.20 Mbps     108.75945ms     N/A              | ||||
| Frankfurt       413.76 Mbps     623.82 Mbps     144.130972ms    N/A              | ||||
| Singapore       353.62 Mbps     459.36 Mbps     174.702269ms    N/A              | ||||
| HongKong        6.31 Mbps       7.47 Mbps       186.860599ms    N/A              | ||||
| ---------------------------------------------------------------------------------- | ||||
| Cost    Time          : 5 min 14 sec | ||||
| Current Time          : Wed Sep 24 16:40:15 UTC 2025 | ||||
| ---------------------------------------------------------------------------------- | ||||
| @@ -8,30 +8,78 @@ import ( | ||||
| ) | ||||
|  | ||||
| func MemoryTest(language, testMethod string) (realTestMethod, res string) { | ||||
| 	testMethod = strings.ToLower(testMethod) | ||||
| 	if testMethod == "" { | ||||
| 		testMethod = "auto" | ||||
| 	} | ||||
| 	if runtime.GOOS == "windows" { | ||||
| 		if testMethod != "winsat" && testMethod != "" { | ||||
| 			// res = "Detected host is Windows, using Winsat for testing.\n" | ||||
| 		switch testMethod { | ||||
| 		case "stream": | ||||
| 			res = memory.WinsatTest(language) | ||||
| 			realTestMethod = "winsat" | ||||
| 		case "dd": | ||||
| 			res = memory.WindowsDDTest(language) | ||||
| 			if res == "" || strings.TrimSpace(res) == "" { | ||||
| 				res += memory.WinsatTest(language) | ||||
| 				realTestMethod = "winsat" | ||||
| 			} else { | ||||
| 				realTestMethod = "dd" | ||||
| 			} | ||||
| 		case "sysbench": | ||||
| 			res = memory.WinsatTest(language) | ||||
| 			realTestMethod = "winsat" | ||||
| 		case "auto", "winsat": | ||||
| 			res = memory.WinsatTest(language) | ||||
| 			realTestMethod = "winsat" | ||||
| 		default: | ||||
| 			res = memory.WinsatTest(language) | ||||
| 			realTestMethod = "winsat" | ||||
| 		} | ||||
| 		res += memory.WinsatTest(language) | ||||
| 	} else { | ||||
| 		switch testMethod { | ||||
| 		case "stream": | ||||
| 			res = memory.StreamTest(language) | ||||
| 			if res == "" || strings.TrimSpace(res) == "" { | ||||
| 				res += memory.DDTest(language) | ||||
| 				realTestMethod = "dd" | ||||
| 			} else { | ||||
| 				realTestMethod = "stream" | ||||
| 			} | ||||
| 		case "dd": | ||||
| 			res = memory.DDTest(language) | ||||
| 			realTestMethod = "dd" | ||||
| 		case "sysbench": | ||||
| 			res = memory.SysBenchTest(language) | ||||
| 			if res == "" { | ||||
| 				// res = "sysbench test failed, switch to use dd test.\n" | ||||
| 			if res == "" || strings.TrimSpace(res) == "" { | ||||
| 				res += memory.DDTest(language) | ||||
| 				realTestMethod = "dd" | ||||
| 			} else { | ||||
| 				realTestMethod = "sysbench" | ||||
| 			} | ||||
| 		case "dd": | ||||
| 		case "auto": | ||||
| 			res = memory.StreamTest(language) | ||||
| 			if res == "" || strings.TrimSpace(res) == "" { | ||||
| 				res = memory.DDTest(language) | ||||
| 				if res == "" || strings.TrimSpace(res) == "" { | ||||
| 					res = memory.SysBenchTest(language) | ||||
| 					if res == "" || strings.TrimSpace(res) == "" { | ||||
| 						realTestMethod = "" | ||||
| 					} else { | ||||
| 						realTestMethod = "sysbench" | ||||
| 					} | ||||
| 				} else { | ||||
| 					realTestMethod = "dd" | ||||
| 				} | ||||
| 			} else { | ||||
| 				realTestMethod = "stream" | ||||
| 			} | ||||
| 		case "winsat": | ||||
| 			// winsat 仅 Windows 支持,非 Windows fallback 到 dd | ||||
| 			res = memory.DDTest(language) | ||||
| 			realTestMethod = "dd" | ||||
| 		default: | ||||
| 			// res = "Unsupported test method, switch to use dd test.\n" | ||||
| 			res += memory.DDTest(language) | ||||
| 			realTestMethod = "dd" | ||||
| 			res = "Unsupported test method" | ||||
| 			realTestMethod = "" | ||||
| 		} | ||||
| 	} | ||||
| 	if !strings.Contains(res, "\n") && res != "" { | ||||
|   | ||||
| @@ -6,6 +6,6 @@ import ( | ||||
| ) | ||||
|  | ||||
| func Test(t *testing.T) { | ||||
| 	_, res := MemoryTest("zh", "sysbench") | ||||
| 	_, res := MemoryTest("zh", "stream") | ||||
| 	fmt.Print(res) | ||||
| } | ||||
|   | ||||
							
								
								
									
										44
									
								
								nexttrace/nexttrace.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										44
									
								
								nexttrace/nexttrace.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,44 @@ | ||||
| package nexttrace | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"strings" | ||||
|  | ||||
| 	"github.com/oneclickvirt/nt3/nt" | ||||
| ) | ||||
|  | ||||
| func NextTrace3Check(language, nt3Location, nt3CheckType string) { | ||||
| 	resultChan := make(chan nt.TraceResult, 100) | ||||
| 	go nt.TraceRoute(language, nt3Location, nt3CheckType, resultChan) | ||||
| 	for result := range resultChan { | ||||
| 		if result.Index == -1 { | ||||
| 			for index, res := range result.Output { | ||||
| 				res = strings.TrimSpace(res) | ||||
| 				if res != "" && index == 0 { | ||||
| 					fmt.Println(res) | ||||
| 				} | ||||
| 			} | ||||
| 			continue | ||||
| 		} | ||||
| 		if result.ISPName == "Error" { | ||||
| 			for _, res := range result.Output { | ||||
| 				res = strings.TrimSpace(res) | ||||
| 				if res != "" { | ||||
| 					fmt.Println(res) | ||||
| 				} | ||||
| 			} | ||||
| 			return | ||||
| 		} | ||||
| 		for _, res := range result.Output { | ||||
| 			res = strings.TrimSpace(res) | ||||
| 			if res == "" { | ||||
| 				continue | ||||
| 			} | ||||
| 			if strings.Contains(res, "ICMP") { | ||||
| 				fmt.Print(res) | ||||
| 			} else { | ||||
| 				fmt.Println(res) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										13
									
								
								nexttrace/nexttrace_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								nexttrace/nexttrace_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | ||||
| package nexttrace | ||||
|  | ||||
| import ( | ||||
| 	"testing" | ||||
| 	"time" | ||||
| ) | ||||
|  | ||||
| func TestNextTrace3Check(t *testing.T) { | ||||
| 	start := time.Now() | ||||
| 	NextTrace3Check("zh", "ALL", "ipv4") | ||||
| 	duration := time.Since(start) | ||||
| 	t.Logf("执行耗时: %s", duration) | ||||
| } | ||||
							
								
								
									
										66
									
								
								upstreams/upstreams.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										66
									
								
								upstreams/upstreams.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,66 @@ | ||||
| package upstreams | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"sync" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/oneclickvirt/UnlockTests/uts" | ||||
| 	bgptools "github.com/oneclickvirt/backtrace/bgptools" | ||||
| 	backtrace "github.com/oneclickvirt/backtrace/bk" | ||||
| 	. "github.com/oneclickvirt/defaultset" | ||||
| ) | ||||
|  | ||||
| type IpInfo struct { | ||||
| 	Ip      string `json:"ip"` | ||||
| 	City    string `json:"city"` | ||||
| 	Region  string `json:"region"` | ||||
| 	Country string `json:"country"` | ||||
| 	Org     string `json:"org"` | ||||
| } | ||||
|  | ||||
| type ConcurrentResults struct { | ||||
| 	bgpResult       string | ||||
| 	backtraceResult string | ||||
| 	bgpError        error | ||||
| 	// backtraceError  error | ||||
| } | ||||
|  | ||||
| var IPV4, IPV6 string | ||||
|  | ||||
| func UpstreamsCheck() { | ||||
| 	results := ConcurrentResults{} | ||||
| 	var wg sync.WaitGroup | ||||
| 	if IPV4 != "" { | ||||
| 		wg.Add(1) | ||||
| 		go func() { | ||||
| 			defer wg.Done() | ||||
| 			for i := 0; i < 2; i++ { | ||||
| 				result, err := bgptools.GetPoPInfo(IPV4) | ||||
| 				results.bgpError = err | ||||
| 				if err == nil && result.Result != "" { | ||||
| 					results.bgpResult = result.Result | ||||
| 					return | ||||
| 				} | ||||
| 				if i == 0 { | ||||
| 					time.Sleep(3 * time.Second) | ||||
| 				} | ||||
| 			} | ||||
| 		}() | ||||
| 	} | ||||
| 	wg.Add(1) | ||||
| 	go func() { | ||||
| 		defer wg.Done() | ||||
| 		result := backtrace.BackTrace(uts.IPV6) | ||||
| 		results.backtraceResult = result | ||||
| 	}() | ||||
| 	wg.Wait() | ||||
| 	if results.bgpResult != "" { | ||||
| 		fmt.Print(results.bgpResult) | ||||
| 	} | ||||
| 	if results.backtraceResult != "" { | ||||
| 		fmt.Printf("%s\n", results.backtraceResult) | ||||
| 	} | ||||
| 	fmt.Println(Yellow("准确线路自行查看详细路由,本测试结果仅作参考")) | ||||
| 	fmt.Println(Yellow("同一目标地址多个线路时,检测可能已越过汇聚层,除第一个线路外,后续信息可能无效")) | ||||
| } | ||||
							
								
								
									
										8
									
								
								upstreams/uptreams_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								upstreams/uptreams_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | ||||
| package upstreams | ||||
|  | ||||
| import "testing" | ||||
|  | ||||
| func TestUpstreamsCheck(t *testing.T) { | ||||
| 	IPV4 = "148.100.85.25" | ||||
| 	UpstreamsCheck() | ||||
| } | ||||
							
								
								
									
										150
									
								
								utils/utils.go
									
									
									
									
									
								
							
							
						
						
									
										150
									
								
								utils/utils.go
									
									
									
									
									
								
							| @@ -11,6 +11,7 @@ import ( | ||||
| 	"os" | ||||
| 	"path/filepath" | ||||
| 	"regexp" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| 	"sync" | ||||
| 	"time" | ||||
| @@ -18,10 +19,28 @@ import ( | ||||
|  | ||||
| 	"github.com/imroc/req/v3" | ||||
| 	"github.com/oneclickvirt/UnlockTests/uts" | ||||
| 	bnetwork "github.com/oneclickvirt/basics/network" | ||||
| 	"github.com/oneclickvirt/basics/system" | ||||
| 	butils "github.com/oneclickvirt/basics/utils" | ||||
| 	. "github.com/oneclickvirt/defaultset" | ||||
| 	"github.com/oneclickvirt/security/network" | ||||
| 	"github.com/oneclickvirt/basics/network" | ||||
| ) | ||||
| const token = "OvwKx5qgJtf7PZgCKbtyojSU.MTcwMTUxNzY1MTgwMw" | ||||
|  | ||||
| // 获取本程序本日及总执行的统计信息 | ||||
| type StatsResponse struct { | ||||
| 	Counter   string `json:"counter"` | ||||
| 	Action    string `json:"action"` | ||||
| 	Total     int    `json:"total"` | ||||
| 	Daily     int    `json:"daily"` | ||||
| 	Date      string `json:"date"` | ||||
| 	Timestamp string `json:"timestamp"` | ||||
| } | ||||
|  | ||||
| // 获取最新的Github的仓库中的版本 | ||||
| type GitHubRelease struct { | ||||
| 	TagName string `json:"tag_name"` | ||||
| } | ||||
|  | ||||
| // PrintCenteredTitle 根据指定的宽度打印居中标题 | ||||
| func PrintCenteredTitle(title string, width int) { | ||||
| @@ -36,15 +55,15 @@ func PrintCenteredTitle(title string, width int) { | ||||
| // PrintHead 根据语言打印头部信息 | ||||
| func PrintHead(language string, width int, ecsVersion string) { | ||||
| 	if language == "zh" { | ||||
| 		PrintCenteredTitle("VPS融合怪测试", width) | ||||
| 		PrintCenteredTitle("VPS融合怪测试(非官方编译)", width) | ||||
| 		fmt.Printf("版本:%s\n", ecsVersion) | ||||
| 		fmt.Println("测评频道: https://t.me/vps_reviews\n" + | ||||
| 		fmt.Println("测评频道: https://t.me/+UHVoo2U4VyA5NTQ1\n" + | ||||
| 			"Go项目地址:https://github.com/oneclickvirt/ecs\n" + | ||||
| 			"Shell项目地址:https://github.com/spiritLHLS/ecs") | ||||
| 	} else { | ||||
| 		PrintCenteredTitle("VPS Fusion Monster Test", width) | ||||
| 		PrintCenteredTitle("VPS Fusion Monster Test (Unofficial)", width) | ||||
| 		fmt.Printf("Version: %s\n", ecsVersion) | ||||
| 		fmt.Println("Review Channel: https://t.me/vps_reviews\n" + | ||||
| 		fmt.Println("Review Channel: https://t.me/+UHVoo2U4VyA5NTQ1\n" + | ||||
| 			"Go Project: https://github.com/oneclickvirt/ecs\n" + | ||||
| 			"Shell Project: https://github.com/spiritLHLS/ecs") | ||||
| 	} | ||||
| @@ -98,18 +117,38 @@ func CheckChina(enableLogger bool) bool { | ||||
| 	return selectChina | ||||
| } | ||||
|  | ||||
| // OnlyBasicsIpInfo 仅检查和输出IP信息 | ||||
| func OnlyBasicsIpInfo(language string) (string, string, string) { | ||||
| 	ipv4, ipv6, ipInfo, _, err := bnetwork.NetworkCheck("both", false, language) | ||||
| 	if err != nil { | ||||
| 		return "", "", "" | ||||
| 	} | ||||
| 	basicInfo := ipInfo | ||||
| 	if strings.Contains(ipInfo, "IPV4") && strings.Contains(ipInfo, "IPV6") && ipv4 != "" && ipv6 != "" { | ||||
| 		uts.IPV4 = true | ||||
| 		uts.IPV6 = true | ||||
| 	} else if strings.Contains(ipInfo, "IPV4") && ipv4 != "" { | ||||
| 		uts.IPV4 = true | ||||
| 		uts.IPV6 = false | ||||
| 	} else if strings.Contains(ipInfo, "IPV6") && ipv6 != "" { | ||||
| 		uts.IPV6 = true | ||||
| 		uts.IPV4 = false | ||||
| 	} | ||||
| 	basicInfo = strings.ReplaceAll(basicInfo, "\n\n", "\n") | ||||
| 	return ipv4, ipv6, basicInfo | ||||
| } | ||||
|  | ||||
| // BasicsAndSecurityCheck 执行安全检查 | ||||
| func BasicsAndSecurityCheck(language, nt3CheckType string, securityCheckStatus bool) (string, string, string) { | ||||
| func BasicsAndSecurityCheck(language, nt3CheckType string, securityCheckStatus bool) (string, string, string, string, string) { | ||||
| 	var wgt sync.WaitGroup | ||||
| 	var ipInfo, securityInfo, systemInfo string | ||||
| 	var err error | ||||
| 	var ipv4, ipv6, ipInfo, securityInfo, systemInfo string | ||||
| 	wgt.Add(1) | ||||
| 	go func() { | ||||
| 		defer wgt.Done() | ||||
| 		ipInfo, securityInfo, err = network.NetworkCheck("both", securityCheckStatus, language) | ||||
| 		if err != nil { | ||||
| 			fmt.Println(err.Error()) | ||||
| 		} | ||||
| 		ipv4, ipv6, ipInfo, securityInfo, _ = network.NetworkCheck("both", securityCheckStatus, language) | ||||
| 		// if err != nil { | ||||
| 		// 	fmt.Println(err.Error()) | ||||
| 		// } | ||||
| 	}() | ||||
| 	wgt.Add(1) | ||||
| 	go func() { | ||||
| @@ -118,19 +157,19 @@ func BasicsAndSecurityCheck(language, nt3CheckType string, securityCheckStatus b | ||||
| 	}() | ||||
| 	wgt.Wait() | ||||
| 	basicInfo := systemInfo + ipInfo | ||||
| 	if strings.Contains(ipInfo, "IPV4") && strings.Contains(ipInfo, "IPV6") { | ||||
| 	if strings.Contains(ipInfo, "IPV4") && strings.Contains(ipInfo, "IPV6") && ipv4 != "" && ipv6 != "" { | ||||
| 		uts.IPV4 = true | ||||
| 		uts.IPV6 = true | ||||
| 		if nt3CheckType == "" { | ||||
| 			nt3CheckType = "ipv4" | ||||
| 		} | ||||
| 	} else if strings.Contains(ipInfo, "IPV4") { | ||||
| 	} else if strings.Contains(ipInfo, "IPV4") && ipv4 != "" { | ||||
| 		uts.IPV4 = true | ||||
| 		uts.IPV6 = false | ||||
| 		if nt3CheckType == "" { | ||||
| 			nt3CheckType = "ipv4" | ||||
| 		} | ||||
| 	} else if strings.Contains(ipInfo, "IPV6") { | ||||
| 	} else if strings.Contains(ipInfo, "IPV6") && ipv6 != "" { | ||||
| 		uts.IPV6 = true | ||||
| 		uts.IPV4 = false | ||||
| 		if nt3CheckType == "" { | ||||
| @@ -143,7 +182,7 @@ func BasicsAndSecurityCheck(language, nt3CheckType string, securityCheckStatus b | ||||
| 		nt3CheckType = "ipv4" | ||||
| 	} | ||||
| 	basicInfo = strings.ReplaceAll(basicInfo, "\n\n", "\n") | ||||
| 	return basicInfo, securityInfo, nt3CheckType | ||||
| 	return ipv4, ipv6, basicInfo, securityInfo, nt3CheckType | ||||
| } | ||||
|  | ||||
| // CaptureOutput 捕获函数输出和错误输出,实时输出,并返回字符串 | ||||
| @@ -212,7 +251,6 @@ func PrintAndCapture(f func(), tempOutput, output string) string { | ||||
| func UploadText(absPath string) (string, string, error) { | ||||
| 	primaryURL := "http://hpaste.spiritlhl.net/api/UL/upload" | ||||
| 	backupURL := "https://paste.spiritlhl.net/api/UL/upload" | ||||
| 	token := network.SecurityUploadToken | ||||
| 	client := req.C().SetTimeout(6 * time.Second) | ||||
| 	client.R(). | ||||
| 		SetRetryCount(2). | ||||
| @@ -335,8 +373,6 @@ func ProcessAndUpload(output string, filePath string, enableUplaod bool) (string | ||||
| 	return "", "" | ||||
| } | ||||
|  | ||||
| // ============================= 前置联网能力检测 ============================= | ||||
|  | ||||
| var StackType string | ||||
|  | ||||
| type NetCheckResult struct { | ||||
| @@ -358,6 +394,7 @@ func makeResolver(proto, dnsAddr string) *net.Resolver { | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // 前置联网能力检测 | ||||
| func CheckPublicAccess(timeout time.Duration) NetCheckResult { | ||||
| 	if timeout < 2*time.Second { | ||||
| 		timeout = 2 * time.Second | ||||
| @@ -483,6 +520,7 @@ result: | ||||
| 		stack = "IPv6" | ||||
| 	} | ||||
| 	StackType = stack | ||||
| 	butils.CheckPublicAccess(3 * time.Second) // 设置basics检测,避免部分测试未启用 | ||||
| 	return NetCheckResult{ | ||||
| 		HasIPv4:   hasV4, | ||||
| 		HasIPv6:   hasV6, | ||||
| @@ -490,3 +528,77 @@ result: | ||||
| 		StackType: stack, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // 获取每日/总的程序执行统计信息 | ||||
| func GetGoescStats() (*StatsResponse, error) { | ||||
| 	client := req.C().SetTimeout(5 * time.Second) | ||||
| 	var stats StatsResponse | ||||
| 	resp, err := client.R(). | ||||
| 		SetSuccessResult(&stats). | ||||
| 		Get("https://hits.spiritlhl.net/goecs") | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	if !resp.IsSuccessState() { | ||||
| 		return nil, fmt.Errorf("unexpected status code: %d", resp.StatusCode) | ||||
| 	} | ||||
| 	return &stats, nil | ||||
| } | ||||
|  | ||||
| // 统计结果单位转换 | ||||
| func FormatGoecsNumber(num int) string { | ||||
| 	if num >= 1000000 { | ||||
| 		return fmt.Sprintf("%.1fM", float64(num)/1000000) | ||||
| 	} else if num >= 1000 { | ||||
| 		return fmt.Sprintf("%.1fK", float64(num)/1000) | ||||
| 	} | ||||
| 	return fmt.Sprintf("%d", num) | ||||
| } | ||||
|  | ||||
| // 通过Github的API检索仓库最新TAG的版本 | ||||
| func GetLatestEcsRelease() (*GitHubRelease, error) { | ||||
| 	urls := []string{ | ||||
| 		"https://api.github.com/repos/oneclickvirt/ecs/releases/latest", | ||||
| 		"https://fd.spiritlhl.top/https://api.github.com/repos/oneclickvirt/ecs/releases/latest", | ||||
| 		"https://githubapi.spiritlhl.top/repos/oneclickvirt/ecs/releases/latest", | ||||
| 		"https://githubapi.spiritlhl.workers.dev/repos/oneclickvirt/ecs/releases/latest", | ||||
| 	} | ||||
| 	client := req.C().SetTimeout(3 * time.Second) | ||||
| 	for _, url := range urls { | ||||
| 		var release GitHubRelease | ||||
| 		resp, err := client.R(). | ||||
| 			SetSuccessResult(&release). | ||||
| 			Get(url) | ||||
| 		if err != nil { | ||||
| 			continue | ||||
| 		} | ||||
| 		if resp.IsSuccessState() && release.TagName != "" { | ||||
| 			return &release, nil | ||||
| 		} | ||||
| 	} | ||||
| 	return nil, fmt.Errorf("failed to fetch release from all sources") | ||||
| } | ||||
|  | ||||
| // 比较程序版本是否需要升级 | ||||
| func CompareVersions(v1, v2 string) int { | ||||
| 	normalize := func(s string) []int { | ||||
| 		s = strings.TrimPrefix(strings.ToLower(s), "v") | ||||
| 		parts := strings.Split(s, ".") | ||||
| 		result := make([]int, 3) | ||||
| 		for i := 0; i < 3 && i < len(parts); i++ { | ||||
| 			n, _ := strconv.Atoi(parts[i]) | ||||
| 			result[i] = n | ||||
| 		} | ||||
| 		return result | ||||
| 	} | ||||
| 	a := normalize(v1) | ||||
| 	b := normalize(v2) | ||||
| 	for i := 0; i < 3; i++ { | ||||
| 		if a[i] < b[i] { | ||||
| 			return -1 | ||||
| 		} else if a[i] > b[i] { | ||||
| 			return 1 | ||||
| 		} | ||||
| 	} | ||||
| 	return 0 | ||||
| } | ||||
|   | ||||
| @@ -6,7 +6,17 @@ import ( | ||||
| 	"time" | ||||
| ) | ||||
|  | ||||
| func TestCheckPublicAccess(t *testing.T) { | ||||
| // func TestCheckPublicAccess(t *testing.T) { | ||||
| // 	timeout := 3 * time.Second | ||||
| // 	result := CheckPublicAccess(timeout) | ||||
| // 	if result.Connected { | ||||
| // 		fmt.Print("✅ 本机有公网连接,类型: %s\n", result.StackType) | ||||
| // 	} else { | ||||
| // 		fmt.Println("❌ 本机未检测到公网连接") | ||||
| // 	} | ||||
| // } | ||||
|  | ||||
| func TestBasicsAndSecurityCheck(t *testing.T) { | ||||
| 	timeout := 3 * time.Second | ||||
| 	result := CheckPublicAccess(timeout) | ||||
| 	if result.Connected { | ||||
| @@ -14,4 +24,8 @@ func TestCheckPublicAccess(t *testing.T) { | ||||
| 	} else { | ||||
| 		fmt.Println("❌ 本机未检测到公网连接") | ||||
| 	} | ||||
| 	_, _, basicInfo, securityInfo, nt3CheckType := BasicsAndSecurityCheck("zh", "ipv4", false) | ||||
| 	fmt.Println(basicInfo) | ||||
| 	fmt.Println(securityInfo) | ||||
| 	fmt.Println(nt3CheckType) | ||||
| } | ||||
|   | ||||
		Reference in New Issue
	
	Block a user