Compare commits
118 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 81b42946f8 | |||
| 975428a00c | |||
|
|
12abf383e9 | ||
|
|
5bef02bd6d | ||
|
|
592f1260b5 | ||
|
|
18303688d7 | ||
|
|
5c4b7f6636 | ||
|
|
37cce2fa35 | ||
|
|
6f8b65c75b | ||
|
|
83c63b914a | ||
|
|
1ca2485d2a | ||
|
|
cc4154bb0d | ||
|
|
d7f77f220c | ||
|
|
86f45d103d | ||
|
|
0077751f75 | ||
|
|
fa2b4b3dc9 | ||
|
|
23cacb8339 | ||
|
|
9ffa6a4eb6 | ||
|
|
386209b835 | ||
|
|
830dc89c32 | ||
|
|
693afe3560 | ||
|
|
95361e8b65 | ||
|
|
3ff7299aca | ||
|
|
34fc4de0c2 | ||
|
|
91536d3923 | ||
|
|
6b87c09a96 | ||
|
|
ecaac2ac61 | ||
|
|
ad74b1584d | ||
|
|
514d76953a | ||
|
|
5b82f17995 | ||
|
|
20ce35bc30 | ||
|
|
c0fca0dddd | ||
|
|
2ba896e17e | ||
|
|
f61e6d8c63 | ||
|
|
d3e2e55ecf | ||
|
|
30e663cd4f | ||
|
|
054efeb32c | ||
|
|
2ebd2b28a8 | ||
|
|
84f812c8ee | ||
|
|
b6ee40ab8d | ||
|
|
7f24f4a15f | ||
|
|
0d307671d1 | ||
|
|
8ea5a57988 | ||
|
|
4fb41aeca1 | ||
|
|
3f0bcf7b83 | ||
|
|
7e712fcdeb | ||
|
|
e634e6dae3 | ||
|
|
24f8d767b1 | ||
|
|
31a8ddef23 | ||
|
|
30e9e64fd5 | ||
|
|
f677934257 | ||
|
|
df7ca81837 | ||
|
|
53bd03dea2 | ||
|
|
1f8dd1a52d | ||
| 2f0299a32d | |||
| 65a8bedb45 | |||
|
|
d5460d758b | ||
|
|
6e38357b7d | ||
|
|
1990850d9a | ||
|
|
e6cb146671 | ||
|
|
4da59cd767 | ||
| d6e40fb7f5 | |||
| 4db65c2132 | |||
|
|
e20c11c1a7 | ||
|
|
a6af95e083 | ||
|
|
6f06b16c76 | ||
|
|
70ddf4ecfc | ||
|
|
187356cb9e | ||
|
|
32583ea8b3 | ||
|
|
69797c10f2 | ||
|
|
ddc8c9b1cd | ||
| 7f9ac74b86 | |||
|
|
753e7b81b6 | ||
|
|
725b094fb1 | ||
|
|
6de5a5215d | ||
|
|
8d86aa2b72 | ||
|
|
1aee3950f4 | ||
|
|
091b79f7cf | ||
|
|
ed2c77062e | ||
|
|
8b1105c7e2 | ||
|
|
11c203ad19 | ||
|
|
ab6a6b879e | ||
|
|
b218f0b501 | ||
|
|
7b5686cd8f | ||
|
|
d727ff40bb | ||
|
|
1b5069a933 | ||
|
|
18ea6fdc00 | ||
|
|
67494108ad | ||
|
|
38b2a7d2ca | ||
|
|
bf3703bca1 | ||
|
|
554632cc07 | ||
|
|
12fc3e9566 | ||
|
|
c2ef3a4a8c | ||
|
|
86eb8297dd | ||
|
|
c63d4e83f9 | ||
|
|
bf1fb0f92e | ||
|
|
3c4865982b | ||
| 1cb0ef2f72 | |||
| 8e8035af36 | |||
| 77aa28f46a | |||
|
|
22c233f0cd | ||
|
|
b2d6282755 | ||
|
|
c8d89e3dce | ||
| 62a2558174 | |||
|
|
d3b1810eab | ||
| 45b6fe4d5a | |||
|
|
51409a3e28 | ||
| 6b8b2d0b1b | |||
|
|
1a0f50a41e | ||
|
|
83d4a9c18e | ||
|
|
b4c20e7b81 | ||
|
|
7c76308c93 | ||
|
|
f28fa31c14 | ||
|
|
bbedc4dbb1 | ||
|
|
ecf42cb85d | ||
|
|
e4701d6703 | ||
|
|
54a47d00a3 | ||
|
|
964572817b |
@@ -9,6 +9,10 @@ end_of_line = crlf
|
|||||||
trim_trailing_whitespace = true
|
trim_trailing_whitespace = true
|
||||||
insert_final_newline = true
|
insert_final_newline = true
|
||||||
|
|
||||||
|
[*.sh]
|
||||||
|
end_of_line = lf
|
||||||
|
indent_size = 2
|
||||||
|
|
||||||
[*.{yml,yaml}]
|
[*.{yml,yaml}]
|
||||||
indent_style = space
|
indent_style = space
|
||||||
indent_size = 2
|
indent_size = 2
|
||||||
@@ -157,14 +161,14 @@ dotnet_naming_rule.non_field_members_should_be_pascal.symbols = non_field_member
|
|||||||
dotnet_naming_rule.non_field_members_should_be_pascal.style = pascal
|
dotnet_naming_rule.non_field_members_should_be_pascal.style = pascal
|
||||||
dotnet_naming_symbols.interface.applicable_kinds = interface
|
dotnet_naming_symbols.interface.applicable_kinds = interface
|
||||||
dotnet_naming_symbols.interface.applicable_accessibilities = *
|
dotnet_naming_symbols.interface.applicable_accessibilities = *
|
||||||
dotnet_naming_symbols.interface.required_modifiers =
|
dotnet_naming_symbols.interface.required_modifiers =
|
||||||
dotnet_naming_symbols.types.applicable_kinds = class, struct, interface, enum
|
dotnet_naming_symbols.types.applicable_kinds = class, struct, interface, enum
|
||||||
dotnet_naming_symbols.types.applicable_accessibilities = *
|
dotnet_naming_symbols.types.applicable_accessibilities = *
|
||||||
dotnet_naming_symbols.types.required_modifiers =
|
dotnet_naming_symbols.types.required_modifiers =
|
||||||
dotnet_naming_symbols.non_field_members.applicable_kinds = property, event, method
|
dotnet_naming_symbols.non_field_members.applicable_kinds = property, event, method
|
||||||
dotnet_naming_symbols.non_field_members.applicable_accessibilities = *
|
dotnet_naming_symbols.non_field_members.applicable_accessibilities = *
|
||||||
dotnet_naming_symbols.non_field_members.required_modifiers =
|
dotnet_naming_symbols.non_field_members.required_modifiers =
|
||||||
dotnet_naming_style.pascal.required_prefix =
|
dotnet_naming_style.pascal.required_prefix =
|
||||||
dotnet_naming_style.pascal.required_suffix =
|
dotnet_naming_style.pascal.required_suffix =
|
||||||
dotnet_naming_style.pascal.word_separator =
|
dotnet_naming_style.pascal.word_separator =
|
||||||
dotnet_naming_style.pascal.capitalization = pascal_case
|
dotnet_naming_style.pascal.capitalization = pascal_case
|
||||||
|
|||||||
112
.github/workflows/build-linux.yml
vendored
112
.github/workflows/build-linux.yml
vendored
@@ -9,6 +9,12 @@ on:
|
|||||||
push:
|
push:
|
||||||
branches:
|
branches:
|
||||||
- master
|
- master
|
||||||
|
tags:
|
||||||
|
- 'v*'
|
||||||
|
- 'V*'
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
contents: write
|
||||||
|
|
||||||
env:
|
env:
|
||||||
OutputArch: "linux-64"
|
OutputArch: "linux-64"
|
||||||
@@ -21,31 +27,30 @@ jobs:
|
|||||||
strategy:
|
strategy:
|
||||||
matrix:
|
matrix:
|
||||||
configuration: [Release]
|
configuration: [Release]
|
||||||
|
|
||||||
runs-on: ubuntu-24.04
|
runs-on: ubuntu-24.04
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout
|
- name: Checkout
|
||||||
uses: actions/checkout@v5.0.0
|
uses: actions/checkout@v6.0.1
|
||||||
with:
|
with:
|
||||||
submodules: 'recursive'
|
submodules: 'recursive'
|
||||||
fetch-depth: '0'
|
fetch-depth: '0'
|
||||||
|
|
||||||
- name: Setup
|
- name: Setup .NET
|
||||||
uses: actions/setup-dotnet@v5.0.0
|
uses: actions/setup-dotnet@v5.0.1
|
||||||
with:
|
with:
|
||||||
dotnet-version: '8.0.x'
|
dotnet-version: '8.0.x'
|
||||||
|
|
||||||
- name: Build
|
- name: Build
|
||||||
run: |
|
run: |
|
||||||
cd v2rayN
|
cd v2rayN
|
||||||
dotnet publish ./v2rayN.Desktop/v2rayN.Desktop.csproj -c Release -r linux-x64 --self-contained=true -o $OutputPath64
|
dotnet publish ./v2rayN.Desktop/v2rayN.Desktop.csproj -c Release -r linux-x64 -p:SelfContained=true -o "$OutputPath64"
|
||||||
dotnet publish ./v2rayN.Desktop/v2rayN.Desktop.csproj -c Release -r linux-arm64 --self-contained=true -o $OutputPathArm64
|
dotnet publish ./v2rayN.Desktop/v2rayN.Desktop.csproj -c Release -r linux-arm64 -p:SelfContained=true -o "$OutputPathArm64"
|
||||||
dotnet publish ./AmazTool/AmazTool.csproj -c Release -r linux-x64 --self-contained=true -p:PublishTrimmed=true -o $OutputPath64
|
dotnet publish ./AmazTool/AmazTool.csproj -c Release -r linux-x64 -p:SelfContained=true -p:PublishTrimmed=true -o "$OutputPath64"
|
||||||
dotnet publish ./AmazTool/AmazTool.csproj -c Release -r linux-arm64 --self-contained=true -p:PublishTrimmed=true -o $OutputPathArm64
|
dotnet publish ./AmazTool/AmazTool.csproj -c Release -r linux-arm64 -p:SelfContained=true -p:PublishTrimmed=true -o "$OutputPathArm64"
|
||||||
|
|
||||||
- name: Upload build artifacts
|
- name: Upload build artifacts
|
||||||
uses: actions/upload-artifact@v4.6.2
|
uses: actions/upload-artifact@v5.0.0
|
||||||
with:
|
with:
|
||||||
name: v2rayN-linux
|
name: v2rayN-linux
|
||||||
path: |
|
path: |
|
||||||
@@ -56,8 +61,8 @@ jobs:
|
|||||||
if: github.event.inputs.release_tag != ''
|
if: github.event.inputs.release_tag != ''
|
||||||
run: |
|
run: |
|
||||||
chmod 755 package-debian.sh
|
chmod 755 package-debian.sh
|
||||||
./package-debian.sh $OutputArch $OutputPath64 ${{ github.event.inputs.release_tag }}
|
./package-debian.sh "$OutputArch" "$OutputPath64" "${{ github.event.inputs.release_tag }}"
|
||||||
./package-debian.sh $OutputArchArm $OutputPathArm64 ${{ github.event.inputs.release_tag }}
|
./package-debian.sh "$OutputArchArm" "$OutputPathArm64" "${{ github.event.inputs.release_tag }}"
|
||||||
|
|
||||||
- name: Upload deb to release
|
- name: Upload deb to release
|
||||||
uses: svenstaro/upload-release-action@v2
|
uses: svenstaro/upload-release-action@v2
|
||||||
@@ -68,28 +73,13 @@ jobs:
|
|||||||
file_glob: true
|
file_glob: true
|
||||||
prerelease: true
|
prerelease: true
|
||||||
|
|
||||||
- name: Package AppImage
|
|
||||||
if: github.event.inputs.release_tag != ''
|
|
||||||
run: |
|
|
||||||
chmod a+x package-appimage.sh
|
|
||||||
./package-appimage.sh
|
|
||||||
|
|
||||||
- name: Upload AppImage to release
|
|
||||||
uses: svenstaro/upload-release-action@v2
|
|
||||||
if: github.event.inputs.release_tag != ''
|
|
||||||
with:
|
|
||||||
file: ${{ github.workspace }}/v2rayN*.AppImage
|
|
||||||
tag: ${{ github.event.inputs.release_tag }}
|
|
||||||
file_glob: true
|
|
||||||
prerelease: true
|
|
||||||
|
|
||||||
# release zip archive
|
# release zip archive
|
||||||
- name: Package release zip archive
|
- name: Package release zip archive
|
||||||
if: github.event.inputs.release_tag != ''
|
if: github.event.inputs.release_tag != ''
|
||||||
run: |
|
run: |
|
||||||
chmod 755 package-release-zip.sh
|
chmod 755 package-release-zip.sh
|
||||||
./package-release-zip.sh $OutputArch $OutputPath64
|
./package-release-zip.sh "$OutputArch" "$OutputPath64"
|
||||||
./package-release-zip.sh $OutputArchArm $OutputPathArm64
|
./package-release-zip.sh "$OutputArchArm" "$OutputPathArm64"
|
||||||
|
|
||||||
- name: Upload zip archive to release
|
- name: Upload zip archive to release
|
||||||
uses: svenstaro/upload-release-action@v2
|
uses: svenstaro/upload-release-action@v2
|
||||||
@@ -100,36 +90,62 @@ jobs:
|
|||||||
file_glob: true
|
file_glob: true
|
||||||
prerelease: true
|
prerelease: true
|
||||||
|
|
||||||
# release RHEL package
|
rpm:
|
||||||
- name: Package RPM (RHEL-family)
|
needs: build
|
||||||
if: github.event.inputs.release_tag != ''
|
if: |
|
||||||
|
(github.event_name == 'workflow_dispatch' && github.event.inputs.release_tag != '') ||
|
||||||
|
(github.event_name == 'push' && startsWith(github.ref, 'refs/tags/'))
|
||||||
|
runs-on: ubuntu-24.04
|
||||||
|
container:
|
||||||
|
image: registry.access.redhat.com/ubi10/ubi
|
||||||
|
env:
|
||||||
|
RELEASE_TAG: ${{ github.event.inputs.release_tag != '' && github.event.inputs.release_tag || github.ref_name }}
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Prepare tools (Red Hat)
|
||||||
run: |
|
run: |
|
||||||
chmod 755 package-rhel.sh
|
dnf repolist all
|
||||||
# Build for both x86_64 and aarch64 in one go (explicit version passed; no --buildfrom)
|
dnf -y makecache
|
||||||
./package-rhel.sh "${{ github.event.inputs.release_tag }}" --arch all
|
dnf -y install https://dl.fedoraproject.org/pub/epel/epel-release-latest-10.noarch.rpm
|
||||||
|
dnf -y install sudo git rpm-build rpmdevtools dnf-plugins-core rsync findutils tar gzip unzip which
|
||||||
|
|
||||||
|
- name: Checkout repo (for scripts)
|
||||||
|
uses: actions/checkout@v6.0.1
|
||||||
|
with:
|
||||||
|
submodules: 'recursive'
|
||||||
|
fetch-depth: '0'
|
||||||
|
|
||||||
|
- name: Restore build artifacts
|
||||||
|
uses: actions/download-artifact@v6
|
||||||
|
with:
|
||||||
|
name: v2rayN-linux
|
||||||
|
path: ${{ github.workspace }}/v2rayN/Release
|
||||||
|
|
||||||
|
- name: Ensure script permissions
|
||||||
|
run: chmod 755 package-rhel.sh
|
||||||
|
|
||||||
|
- name: Package RPM (RHEL-family)
|
||||||
|
run: ./package-rhel.sh "${RELEASE_TAG}" --arch all
|
||||||
|
|
||||||
- name: Collect RPMs into workspace
|
- name: Collect RPMs into workspace
|
||||||
if: github.event.inputs.release_tag != ''
|
|
||||||
run: |
|
run: |
|
||||||
mkdir -p "${{ github.workspace }}/dist/rpm"
|
mkdir -p "$GITHUB_WORKSPACE/dist/rpm"
|
||||||
rsync -av "$HOME/rpmbuild/RPMS/" "${{ github.workspace }}/dist/rpm/"
|
rsync -av "$HOME/rpmbuild/RPMS/" "$GITHUB_WORKSPACE/dist/rpm/" || true
|
||||||
# Rename to requested filenames
|
find "$GITHUB_WORKSPACE/dist/rpm" -name "v2rayN-*-1*.x86_64.rpm" -exec mv {} "$GITHUB_WORKSPACE/dist/rpm/v2rayN-linux-rhel-64.rpm" \; || true
|
||||||
find "${{ github.workspace }}/dist/rpm" -name "v2rayN-*-1.x86_64.rpm" -exec mv {} "${{ github.workspace }}/dist/rpm/v2rayN-linux-rhel-x64.rpm" \; || true
|
find "$GITHUB_WORKSPACE/dist/rpm" -name "v2rayN-*-1*.aarch64.rpm" -exec mv {} "$GITHUB_WORKSPACE/dist/rpm/v2rayN-linux-rhel-arm64.rpm" \; || true
|
||||||
find "${{ github.workspace }}/dist/rpm" -name "v2rayN-*-1.aarch64.rpm" -exec mv {} "${{ github.workspace }}/dist/rpm/v2rayN-linux-rhel-arm64.rpm" \; || true
|
echo "==== Dist tree ===="
|
||||||
|
ls -R "$GITHUB_WORKSPACE/dist/rpm" || true
|
||||||
|
|
||||||
- name: Upload RPM artifacts
|
- name: Upload RPM artifacts
|
||||||
if: github.event.inputs.release_tag != ''
|
uses: actions/upload-artifact@v5.0.0
|
||||||
uses: actions/upload-artifact@v4.6.2
|
|
||||||
with:
|
with:
|
||||||
name: v2rayN-rpm
|
name: v2rayN-rpm
|
||||||
path: |
|
path: dist/rpm/**/*.rpm
|
||||||
${{ github.workspace }}/dist/rpm/**/*.rpm
|
|
||||||
|
|
||||||
- name: Upload RPMs to release
|
- name: Upload RPMs to release
|
||||||
uses: svenstaro/upload-release-action@v2
|
uses: svenstaro/upload-release-action@v2
|
||||||
if: github.event.inputs.release_tag != ''
|
|
||||||
with:
|
with:
|
||||||
file: ${{ github.workspace }}/dist/rpm/**/*.rpm
|
file: dist/rpm/**/*.rpm
|
||||||
tag: ${{ github.event.inputs.release_tag }}
|
tag: ${{ env.RELEASE_TAG }}
|
||||||
file_glob: true
|
file_glob: true
|
||||||
prerelease: true
|
prerelease: true
|
||||||
|
|||||||
14
.github/workflows/build-osx.yml
vendored
14
.github/workflows/build-osx.yml
vendored
@@ -26,26 +26,26 @@ jobs:
|
|||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout
|
- name: Checkout
|
||||||
uses: actions/checkout@v5.0.0
|
uses: actions/checkout@v6.0.1
|
||||||
with:
|
with:
|
||||||
submodules: 'recursive'
|
submodules: 'recursive'
|
||||||
fetch-depth: '0'
|
fetch-depth: '0'
|
||||||
|
|
||||||
- name: Setup
|
- name: Setup
|
||||||
uses: actions/setup-dotnet@v5.0.0
|
uses: actions/setup-dotnet@v5.0.1
|
||||||
with:
|
with:
|
||||||
dotnet-version: '8.0.x'
|
dotnet-version: '8.0.x'
|
||||||
|
|
||||||
- name: Build
|
- name: Build
|
||||||
run: |
|
run: |
|
||||||
cd v2rayN
|
cd v2rayN
|
||||||
dotnet publish ./v2rayN.Desktop/v2rayN.Desktop.csproj -c Release -r osx-x64 --self-contained=true -o $OutputPath64
|
dotnet publish ./v2rayN.Desktop/v2rayN.Desktop.csproj -c Release -r osx-x64 -p:SelfContained=true -o $OutputPath64
|
||||||
dotnet publish ./v2rayN.Desktop/v2rayN.Desktop.csproj -c Release -r osx-arm64 --self-contained=true -o $OutputPathArm64
|
dotnet publish ./v2rayN.Desktop/v2rayN.Desktop.csproj -c Release -r osx-arm64 -p:SelfContained=true -o $OutputPathArm64
|
||||||
dotnet publish ./AmazTool/AmazTool.csproj -c Release -r osx-x64 --self-contained=true -p:PublishTrimmed=true -o $OutputPath64
|
dotnet publish ./AmazTool/AmazTool.csproj -c Release -r osx-x64 -p:SelfContained=true -p:PublishTrimmed=true -o $OutputPath64
|
||||||
dotnet publish ./AmazTool/AmazTool.csproj -c Release -r osx-arm64 --self-contained=true -p:PublishTrimmed=true -o $OutputPathArm64
|
dotnet publish ./AmazTool/AmazTool.csproj -c Release -r osx-arm64 -p:SelfContained=true -p:PublishTrimmed=true -o $OutputPathArm64
|
||||||
|
|
||||||
- name: Upload build artifacts
|
- name: Upload build artifacts
|
||||||
uses: actions/upload-artifact@v4.6.2
|
uses: actions/upload-artifact@v5.0.0
|
||||||
with:
|
with:
|
||||||
name: v2rayN-macos
|
name: v2rayN-macos
|
||||||
path: |
|
path: |
|
||||||
|
|||||||
14
.github/workflows/build-windows-desktop.yml
vendored
14
.github/workflows/build-windows-desktop.yml
vendored
@@ -26,26 +26,26 @@ jobs:
|
|||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout
|
- name: Checkout
|
||||||
uses: actions/checkout@v5.0.0
|
uses: actions/checkout@v6.0.1
|
||||||
with:
|
with:
|
||||||
submodules: 'recursive'
|
submodules: 'recursive'
|
||||||
fetch-depth: '0'
|
fetch-depth: '0'
|
||||||
|
|
||||||
- name: Setup
|
- name: Setup
|
||||||
uses: actions/setup-dotnet@v5.0.0
|
uses: actions/setup-dotnet@v5.0.1
|
||||||
with:
|
with:
|
||||||
dotnet-version: '8.0.x'
|
dotnet-version: '8.0.x'
|
||||||
|
|
||||||
- name: Build
|
- name: Build
|
||||||
run: |
|
run: |
|
||||||
cd v2rayN
|
cd v2rayN
|
||||||
dotnet publish ./v2rayN.Desktop/v2rayN.Desktop.csproj -c Release -r win-x64 --self-contained=true -p:EnableWindowsTargeting=true -o $OutputPath64
|
dotnet publish ./v2rayN.Desktop/v2rayN.Desktop.csproj -c Release -r win-x64 -p:SelfContained=true -p:EnableWindowsTargeting=true -o $OutputPath64
|
||||||
dotnet publish ./v2rayN.Desktop/v2rayN.Desktop.csproj -c Release -r win-arm64 --self-contained=true -p:EnableWindowsTargeting=true -o $OutputPathArm64
|
dotnet publish ./v2rayN.Desktop/v2rayN.Desktop.csproj -c Release -r win-arm64 -p:SelfContained=true -p:EnableWindowsTargeting=true -o $OutputPathArm64
|
||||||
dotnet publish ./AmazTool/AmazTool.csproj -c Release -r win-x64 --self-contained=true -p:EnableWindowsTargeting=true -p:PublishTrimmed=true -o $OutputPath64
|
dotnet publish ./AmazTool/AmazTool.csproj -c Release -r win-x64 -p:SelfContained=true -p:EnableWindowsTargeting=true -p:PublishTrimmed=true -o $OutputPath64
|
||||||
dotnet publish ./AmazTool/AmazTool.csproj -c Release -r win-arm64 --self-contained=true -p:EnableWindowsTargeting=true -p:PublishTrimmed=true -o $OutputPathArm64
|
dotnet publish ./AmazTool/AmazTool.csproj -c Release -r win-arm64 -p:SelfContained=true -p:EnableWindowsTargeting=true -p:PublishTrimmed=true -o $OutputPathArm64
|
||||||
|
|
||||||
- name: Upload build artifacts
|
- name: Upload build artifacts
|
||||||
uses: actions/upload-artifact@v4.6.2
|
uses: actions/upload-artifact@v5.0.0
|
||||||
with:
|
with:
|
||||||
name: v2rayN-windows-desktop
|
name: v2rayN-windows-desktop
|
||||||
path: |
|
path: |
|
||||||
|
|||||||
20
.github/workflows/build-windows.yml
vendored
20
.github/workflows/build-windows.yml
vendored
@@ -27,26 +27,26 @@ jobs:
|
|||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout
|
- name: Checkout
|
||||||
uses: actions/checkout@v5.0.0
|
uses: actions/checkout@v6.0.1
|
||||||
|
|
||||||
- name: Setup
|
- name: Setup
|
||||||
uses: actions/setup-dotnet@v5.0.0
|
uses: actions/setup-dotnet@v5.0.1
|
||||||
with:
|
with:
|
||||||
dotnet-version: '8.0.x'
|
dotnet-version: '8.0.x'
|
||||||
|
|
||||||
- name: Build
|
- name: Build
|
||||||
run: |
|
run: |
|
||||||
cd v2rayN
|
cd v2rayN
|
||||||
dotnet publish ./v2rayN/v2rayN.csproj -c Release -r win-x64 --self-contained=false -p:EnableWindowsTargeting=true -o $OutputPath64
|
dotnet publish ./v2rayN/v2rayN.csproj -c Release -r win-x64 -p:SelfContained=false -p:EnableWindowsTargeting=true -o $OutputPath64
|
||||||
dotnet publish ./v2rayN/v2rayN.csproj -c Release -r win-arm64 --self-contained=false -p:EnableWindowsTargeting=true -o $OutputPathArm64
|
dotnet publish ./v2rayN/v2rayN.csproj -c Release -r win-arm64 -p:SelfContained=false -p:EnableWindowsTargeting=true -o $OutputPathArm64
|
||||||
dotnet publish ./v2rayN/v2rayN.csproj -c Release -r win-x64 --self-contained=true -p:EnableWindowsTargeting=true -o $OutputPath64Sc
|
dotnet publish ./v2rayN/v2rayN.csproj -c Release -r win-x64 -p:SelfContained=true -p:EnableWindowsTargeting=true -o $OutputPath64Sc
|
||||||
dotnet publish ./AmazTool/AmazTool.csproj -c Release -r win-x64 --self-contained=false -p:EnableWindowsTargeting=true -o $OutputPath64
|
dotnet publish ./AmazTool/AmazTool.csproj -c Release -r win-x64 -p:SelfContained=false -p:EnableWindowsTargeting=true -o $OutputPath64
|
||||||
dotnet publish ./AmazTool/AmazTool.csproj -c Release -r win-arm64 --self-contained=false -p:EnableWindowsTargeting=true -o $OutputPathArm64
|
dotnet publish ./AmazTool/AmazTool.csproj -c Release -r win-arm64 -p:SelfContained=false -p:EnableWindowsTargeting=true -o $OutputPathArm64
|
||||||
dotnet publish ./AmazTool/AmazTool.csproj -c Release -r win-x64 --self-contained=true -p:EnableWindowsTargeting=true -p:PublishTrimmed=true -o $OutputPath64Sc
|
dotnet publish ./AmazTool/AmazTool.csproj -c Release -r win-x64 -p:SelfContained=true -p:EnableWindowsTargeting=true -p:PublishTrimmed=true -o $OutputPath64Sc
|
||||||
|
|
||||||
|
|
||||||
- name: Upload build artifacts
|
- name: Upload build artifacts
|
||||||
uses: actions/upload-artifact@v4.6.2
|
uses: actions/upload-artifact@v5.0.0
|
||||||
with:
|
with:
|
||||||
name: v2rayN-windows
|
name: v2rayN-windows
|
||||||
path: |
|
path: |
|
||||||
@@ -68,4 +68,4 @@ jobs:
|
|||||||
file: ${{ github.workspace }}/v2rayN*.zip
|
file: ${{ github.workspace }}/v2rayN*.zip
|
||||||
tag: ${{ github.event.inputs.release_tag }}
|
tag: ${{ github.event.inputs.release_tag }}
|
||||||
file_glob: true
|
file_glob: true
|
||||||
prerelease: true
|
prerelease: true
|
||||||
|
|||||||
@@ -1,67 +0,0 @@
|
|||||||
#!/bin/bash
|
|
||||||
set -euo pipefail
|
|
||||||
|
|
||||||
# Install deps
|
|
||||||
sudo apt update -y
|
|
||||||
sudo apt install -y libfuse2 wget file
|
|
||||||
|
|
||||||
# Get tools
|
|
||||||
wget -qO appimagetool https://github.com/AppImage/AppImageKit/releases/download/continuous/appimagetool-x86_64.AppImage
|
|
||||||
chmod +x appimagetool
|
|
||||||
|
|
||||||
# x86_64 AppDir
|
|
||||||
APPDIR_X64="AppDir-x86_64"
|
|
||||||
rm -rf "$APPDIR_X64"
|
|
||||||
mkdir -p "$APPDIR_X64/usr/lib/v2rayN" "$APPDIR_X64/usr/bin" "$APPDIR_X64/usr/share/applications" "$APPDIR_X64/usr/share/pixmaps"
|
|
||||||
cp -rf "$OutputPath64"/* "$APPDIR_X64/usr/lib/v2rayN" || true
|
|
||||||
[ -f "$APPDIR_X64/usr/lib/v2rayN/v2rayN.png" ] && cp "$APPDIR_X64/usr/lib/v2rayN/v2rayN.png" "$APPDIR_X64/usr/share/pixmaps/v2rayN.png" || true
|
|
||||||
[ -f "$APPDIR_X64/usr/lib/v2rayN/v2rayN.png" ] && cp "$APPDIR_X64/usr/lib/v2rayN/v2rayN.png" "$APPDIR_X64/v2rayN.png" || true
|
|
||||||
|
|
||||||
printf '%s\n' '#!/bin/sh' 'HERE="$(dirname "$(readlink -f "$0")")"' 'cd "$HERE/usr/lib/v2rayN"' 'exec "$HERE/usr/lib/v2rayN/v2rayN" "$@"' > "$APPDIR_X64/AppRun"
|
|
||||||
chmod +x "$APPDIR_X64/AppRun"
|
|
||||||
ln -sf usr/lib/v2rayN/v2rayN "$APPDIR_X64/usr/bin/v2rayN"
|
|
||||||
cat > "$APPDIR_X64/v2rayN.desktop" <<EOF
|
|
||||||
[Desktop Entry]
|
|
||||||
Name=v2rayN
|
|
||||||
Comment=A GUI client for Windows and Linux, support Xray core and sing-box-core and others
|
|
||||||
Exec=v2rayN
|
|
||||||
Icon=v2rayN
|
|
||||||
Terminal=false
|
|
||||||
Type=Application
|
|
||||||
Categories=Network;
|
|
||||||
EOF
|
|
||||||
install -Dm644 "$APPDIR_X64/v2rayN.desktop" "$APPDIR_X64/usr/share/applications/v2rayN.desktop"
|
|
||||||
|
|
||||||
ARCH=x86_64 ./appimagetool "$APPDIR_X64" "v2rayN-${OutputArch}.AppImage"
|
|
||||||
file "v2rayN-${OutputArch}.AppImage" | grep -q 'x86-64'
|
|
||||||
|
|
||||||
# aarch64 AppDir
|
|
||||||
APPDIR_ARM64="AppDir-aarch64"
|
|
||||||
rm -rf "$APPDIR_ARM64"
|
|
||||||
mkdir -p "$APPDIR_ARM64/usr/lib/v2rayN" "$APPDIR_ARM64/usr/bin" "$APPDIR_ARM64/usr/share/applications" "$APPDIR_ARM64/usr/share/pixmaps"
|
|
||||||
cp -rf "$OutputPathArm64"/* "$APPDIR_ARM64/usr/lib/v2rayN" || true
|
|
||||||
[ -f "$APPDIR_ARM64/usr/lib/v2rayN/v2rayN.png" ] && cp "$APPDIR_ARM64/usr/lib/v2rayN/v2rayN.png" "$APPDIR_ARM64/usr/share/pixmaps/v2rayN.png" || true
|
|
||||||
[ -f "$APPDIR_ARM64/usr/lib/v2rayN/v2rayN.png" ] && cp "$APPDIR_ARM64/usr/lib/v2rayN/v2rayN.png" "$APPDIR_ARM64/v2rayN.png" || true
|
|
||||||
|
|
||||||
printf '%s\n' '#!/bin/sh' 'HERE="$(dirname "$(readlink -f "$0")")"' 'cd "$HERE/usr/lib/v2rayN"' 'exec "$HERE/usr/lib/v2rayN/v2rayN" "$@"' > "$APPDIR_ARM64/AppRun"
|
|
||||||
chmod +x "$APPDIR_ARM64/AppRun"
|
|
||||||
ln -sf usr/lib/v2rayN/v2rayN "$APPDIR_ARM64/usr/bin/v2rayN"
|
|
||||||
cat > "$APPDIR_ARM64/v2rayN.desktop" <<EOF
|
|
||||||
[Desktop Entry]
|
|
||||||
Name=v2rayN
|
|
||||||
Comment=A GUI client for Windows and Linux, support Xray core and sing-box-core and others
|
|
||||||
Exec=v2rayN
|
|
||||||
Icon=v2rayN
|
|
||||||
Terminal=false
|
|
||||||
Type=Application
|
|
||||||
Categories=Network;
|
|
||||||
EOF
|
|
||||||
install -Dm644 "$APPDIR_ARM64/v2rayN.desktop" "$APPDIR_ARM64/usr/share/applications/v2rayN.desktop"
|
|
||||||
|
|
||||||
# aarch64 runtime
|
|
||||||
wget -qO runtime-aarch64 https://github.com/AppImage/AppImageKit/releases/download/continuous/runtime-aarch64
|
|
||||||
chmod +x runtime-aarch64
|
|
||||||
|
|
||||||
# build aarch64 AppImage
|
|
||||||
ARCH=aarch64 ./appimagetool --runtime-file ./runtime-aarch64 "$APPDIR_ARM64" "v2rayN-${OutputArchArm}.AppImage"
|
|
||||||
file "v2rayN-${OutputArchArm}.AppImage" | grep -q 'ARM aarch64'
|
|
||||||
47
package-debian-repo.sh
Executable file
47
package-debian-repo.sh
Executable file
@@ -0,0 +1,47 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
# Root directory = the script's location
|
||||||
|
SCRIPT_DIR="$(cd -- "$(dirname -- "${BASH_SOURCE[0]}")" && pwd)"
|
||||||
|
cd "$SCRIPT_DIR"
|
||||||
|
|
||||||
|
Version="$1"
|
||||||
|
|
||||||
|
PackagePath="v2rayn-unofficial-repo"
|
||||||
|
mkdir -p "${PackagePath}/DEBIAN"
|
||||||
|
mkdir -p "${PackagePath}"/etc/apt/{keyrings,sources.list.d}
|
||||||
|
|
||||||
|
curl -fsSLo "${PackagePath}/etc/apt/keyrings/v2rayn-unofficial.asc" "https://git.vlyaii.ru/api/packages/voronin9032/debian/repository.key"
|
||||||
|
|
||||||
|
# basic
|
||||||
|
cat >"${PackagePath}/DEBIAN/control" <<-EOF
|
||||||
|
Package: v2rayn-unofficial-repo
|
||||||
|
Version: $Version
|
||||||
|
Maintainer: Vlyaii <voronin9032n3@gmail.com>
|
||||||
|
Homepage: https://git.vlyaii.ru/voronin9032/v2rayN
|
||||||
|
Architecture: all
|
||||||
|
Depends: ca-certificates
|
||||||
|
Description: v2rayn-unofficial repository configuration
|
||||||
|
EOF
|
||||||
|
|
||||||
|
cat >"${PackagePath}/etc/apt/sources.list.d/v2rayn-unofficial.sources" <<-EOF
|
||||||
|
Types: deb
|
||||||
|
URIs: https://git.vlyaii.ru/api/packages/voronin9032/debian
|
||||||
|
Suites: debian
|
||||||
|
Components: stable
|
||||||
|
Architectures: amd64 all
|
||||||
|
Signed-By: /etc/apt/keyrings/v2rayn-unofficial.asc
|
||||||
|
EOF
|
||||||
|
|
||||||
|
# Patch
|
||||||
|
# set owner to root:root
|
||||||
|
sudo chown -R root:root "${PackagePath}"
|
||||||
|
# set all directories to 755 (readable & traversable by all users)
|
||||||
|
sudo find "${PackagePath}/etc" -type d -exec chmod 755 {} +
|
||||||
|
# set all regular files to 644 (readable by all users)
|
||||||
|
sudo find "${PackagePath}/etc" -type f -exec chmod 644 {} +
|
||||||
|
# ensure main binaries are 755 (executable by all users)
|
||||||
|
|
||||||
|
# build deb package
|
||||||
|
sudo dpkg-deb -Zzstd -z19 --build "$PackagePath"
|
||||||
|
sudo mv "${PackagePath}.deb" "v2rayn-unofficial-repo_${Version}_all.deb"
|
||||||
68
package-debian.sh
Normal file → Executable file
68
package-debian.sh
Normal file → Executable file
@@ -1,38 +1,63 @@
|
|||||||
#!/bin/bash
|
#!/usr/bin/env bash
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
Arch="$1"
|
# Root directory = the script's location
|
||||||
OutputPath="$2"
|
SCRIPT_DIR="$(cd -- "$(dirname -- "${BASH_SOURCE[0]}")" && pwd)"
|
||||||
Version="$3"
|
cd "$SCRIPT_DIR"
|
||||||
|
source ./utils.sh
|
||||||
|
|
||||||
FileName="v2rayN-${Arch}.zip"
|
Arch="linux-64"
|
||||||
wget -nv -O $FileName "https://github.com/2dust/v2rayN-core-bin/raw/refs/heads/master/$FileName"
|
OutputPath="$(mktemp -d)"
|
||||||
7z x $FileName
|
Version="$1"
|
||||||
cp -rf v2rayN-${Arch}/* $OutputPath
|
|
||||||
|
|
||||||
PackagePath="v2rayN-Package-${Arch}"
|
PROJ="./v2rayN/v2rayN.Desktop/v2rayN.Desktop.csproj"
|
||||||
|
dotnet restore "$PROJ"
|
||||||
|
sudo rm -rf "$(dirname "$PROJ")/bin/Release/net8.0"
|
||||||
|
dotnet publish "${PROJ}" \
|
||||||
|
-c Release \
|
||||||
|
-r "linux-x64" \
|
||||||
|
-p:SelfContained=true \
|
||||||
|
-p:PublishSingleFile=false \
|
||||||
|
-p:PublishTrimmed=false \
|
||||||
|
-p:DebugType=none \
|
||||||
|
-p:DebugSymbols=false \
|
||||||
|
-p:IncludeNativeLibrariesForSelfExtract=true \
|
||||||
|
-o "$OutputPath"
|
||||||
|
|
||||||
|
export RID_DIR="linux-x64"
|
||||||
|
download_xray "$OutputPath/bin/xray"
|
||||||
|
download_singbox "$OutputPath/bin/sing_box"
|
||||||
|
download_geo_assets "$OutputPath"
|
||||||
|
|
||||||
|
PackagePath="v2rayn-unofficial"
|
||||||
mkdir -p "${PackagePath}/DEBIAN"
|
mkdir -p "${PackagePath}/DEBIAN"
|
||||||
mkdir -p "${PackagePath}/opt"
|
mkdir -p "${PackagePath}/opt"
|
||||||
cp -rf $OutputPath "${PackagePath}/opt/v2rayN"
|
cp -rf "$OutputPath" "${PackagePath}/opt/v2rayN"
|
||||||
echo "When this file exists, app will not store configs under this folder" > "${PackagePath}/opt/v2rayN/NotStoreConfigHere.txt"
|
echo "When this file exists, app will not store configs under this folder" >"${PackagePath}/opt/v2rayN/NotStoreConfigHere.txt"
|
||||||
|
sudo find "${PackagePath}/opt/v2rayN" -type f -name "*.so" -exec strip {} +
|
||||||
|
|
||||||
if [ $Arch = "linux-64" ]; then
|
if [ "$Arch" = "linux-64" ]; then
|
||||||
Arch2="amd64"
|
Arch2="amd64"
|
||||||
else
|
else
|
||||||
Arch2="arm64"
|
Arch2="arm64"
|
||||||
fi
|
fi
|
||||||
echo $Arch2
|
|
||||||
|
|
||||||
# basic
|
# basic
|
||||||
cat >"${PackagePath}/DEBIAN/control" <<-EOF
|
cat >"${PackagePath}/DEBIAN/control" <<-EOF
|
||||||
Package: v2rayN
|
Package: v2rayn-unofficial
|
||||||
Version: $Version
|
Version: $Version
|
||||||
|
Maintainer: Vlyaii <voronin9032n3@gmail.com>
|
||||||
|
Homepage: https://git.vlyaii.ru/voronin9032/v2rayN
|
||||||
Architecture: $Arch2
|
Architecture: $Arch2
|
||||||
Maintainer: https://github.com/2dust/v2rayN
|
Replaces: v2rayn
|
||||||
Depends: desktop-file-utils, xdg-utils
|
Depends: libc6 (>= 2.34), fontconfig (>= 2.13.1), desktop-file-utils (>= 0.26), xdg-utils (>= 1.1.3), coreutils (>= 8.32), bash (>= 5.1), libfreetype6 (>= 2.11)
|
||||||
|
Breaks: v2rayn
|
||||||
|
Conflicts: v2rayn
|
||||||
Description: A GUI client for Windows and Linux, support Xray core and sing-box-core and others
|
Description: A GUI client for Windows and Linux, support Xray core and sing-box-core and others
|
||||||
EOF
|
EOF
|
||||||
|
|
||||||
cat >"${PackagePath}/DEBIAN/postinst" <<-EOF
|
cat >"${PackagePath}/DEBIAN/postinst" <<-EOF
|
||||||
|
#!/bin/sh
|
||||||
if [ ! -s /usr/share/applications/v2rayN.desktop ]; then
|
if [ ! -s /usr/share/applications/v2rayN.desktop ]; then
|
||||||
cat >/usr/share/applications/v2rayN.desktop<<-END
|
cat >/usr/share/applications/v2rayN.desktop<<-END
|
||||||
[Desktop Entry]
|
[Desktop Entry]
|
||||||
@@ -51,7 +76,6 @@ EOF
|
|||||||
|
|
||||||
sudo chmod 0755 "${PackagePath}/DEBIAN/postinst"
|
sudo chmod 0755 "${PackagePath}/DEBIAN/postinst"
|
||||||
sudo chmod 0755 "${PackagePath}/opt/v2rayN/v2rayN"
|
sudo chmod 0755 "${PackagePath}/opt/v2rayN/v2rayN"
|
||||||
sudo chmod 0755 "${PackagePath}/opt/v2rayN/AmazTool"
|
|
||||||
|
|
||||||
# Patch
|
# Patch
|
||||||
# set owner to root:root
|
# set owner to root:root
|
||||||
@@ -62,8 +86,8 @@ sudo find "${PackagePath}/opt/v2rayN" -type d -exec chmod 755 {} +
|
|||||||
sudo find "${PackagePath}/opt/v2rayN" -type f -exec chmod 644 {} +
|
sudo find "${PackagePath}/opt/v2rayN" -type f -exec chmod 644 {} +
|
||||||
# ensure main binaries are 755 (executable by all users)
|
# ensure main binaries are 755 (executable by all users)
|
||||||
sudo chmod 755 "${PackagePath}/opt/v2rayN/v2rayN" 2>/dev/null || true
|
sudo chmod 755 "${PackagePath}/opt/v2rayN/v2rayN" 2>/dev/null || true
|
||||||
sudo chmod 755 "${PackagePath}/opt/v2rayN/AmazTool" 2>/dev/null || true
|
|
||||||
|
|
||||||
# build deb package
|
# build deb package
|
||||||
sudo dpkg-deb -Zxz --build $PackagePath
|
sudo dpkg-deb -Zzstd -z19 --build "$PackagePath"
|
||||||
sudo mv "${PackagePath}.deb" "v2rayN-${Arch}.deb"
|
sudo mv "${PackagePath}.deb" "v2rayn-unofficial_${Version}_${Arch2}.deb"
|
||||||
|
sudo rm -rf "$OutputPath"
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
#!/bin/bash
|
#!/usr/bin/env bash
|
||||||
|
|
||||||
Arch="$1"
|
Arch="$1"
|
||||||
OutputPath="$2"
|
OutputPath="$2"
|
||||||
@@ -13,7 +13,7 @@ PackagePath="v2rayN-Package-${Arch}"
|
|||||||
mkdir -p "$PackagePath/v2rayN.app/Contents/Resources"
|
mkdir -p "$PackagePath/v2rayN.app/Contents/Resources"
|
||||||
cp -rf "$OutputPath" "$PackagePath/v2rayN.app/Contents/MacOS"
|
cp -rf "$OutputPath" "$PackagePath/v2rayN.app/Contents/MacOS"
|
||||||
cp -f "$PackagePath/v2rayN.app/Contents/MacOS/v2rayN.icns" "$PackagePath/v2rayN.app/Contents/Resources/AppIcon.icns"
|
cp -f "$PackagePath/v2rayN.app/Contents/MacOS/v2rayN.icns" "$PackagePath/v2rayN.app/Contents/Resources/AppIcon.icns"
|
||||||
echo "When this file exists, app will not store configs under this folder" > "$PackagePath/v2rayN.app/Contents/MacOS/NotStoreConfigHere.txt"
|
echo "When this file exists, app will not store configs under this folder" >"$PackagePath/v2rayN.app/Contents/MacOS/NotStoreConfigHere.txt"
|
||||||
chmod +x "$PackagePath/v2rayN.app/Contents/MacOS/v2rayN"
|
chmod +x "$PackagePath/v2rayN.app/Contents/MacOS/v2rayN"
|
||||||
|
|
||||||
cat >"$PackagePath/v2rayN.app/Contents/Info.plist" <<-EOF
|
cat >"$PackagePath/v2rayN.app/Contents/Info.plist" <<-EOF
|
||||||
@@ -43,6 +43,8 @@ cat >"$PackagePath/v2rayN.app/Contents/Info.plist" <<-EOF
|
|||||||
<true/>
|
<true/>
|
||||||
<key>NSHighResolutionCapable</key>
|
<key>NSHighResolutionCapable</key>
|
||||||
<true/>
|
<true/>
|
||||||
|
<key>LSMinimumSystemVersion</key>
|
||||||
|
<string>12.7</string>
|
||||||
</dict>
|
</dict>
|
||||||
</plist>
|
</plist>
|
||||||
EOF
|
EOF
|
||||||
@@ -55,4 +57,4 @@ create-dmg \
|
|||||||
--hide-extension "v2rayN.app" \
|
--hide-extension "v2rayN.app" \
|
||||||
--app-drop-link 500 185 \
|
--app-drop-link 500 185 \
|
||||||
"v2rayN-${Arch}.dmg" \
|
"v2rayN-${Arch}.dmg" \
|
||||||
"$PackagePath/v2rayN.app"
|
"$PackagePath/v2rayN.app"
|
||||||
|
|||||||
4
package-release-zip.sh
Normal file → Executable file
4
package-release-zip.sh
Normal file → Executable file
@@ -1,4 +1,4 @@
|
|||||||
#!/bin/bash
|
#!/usr/bin/env bash
|
||||||
|
|
||||||
Arch="$1"
|
Arch="$1"
|
||||||
OutputPath="$2"
|
OutputPath="$2"
|
||||||
@@ -12,4 +12,4 @@ ZipPath64="./$OutputArch"
|
|||||||
mkdir $ZipPath64
|
mkdir $ZipPath64
|
||||||
|
|
||||||
cp -rf $OutputPath "$ZipPath64/$OutputArch"
|
cp -rf $OutputPath "$ZipPath64/$OutputArch"
|
||||||
7z a -tZip $FileName "$ZipPath64/$OutputArch" -mx1
|
7z a -tZip $FileName "$ZipPath64/$OutputArch" -mx1
|
||||||
|
|||||||
693
package-rhel.sh
Normal file → Executable file
693
package-rhel.sh
Normal file → Executable file
@@ -1,31 +1,45 @@
|
|||||||
#!/usr/bin/env bash
|
#!/usr/bin/env bash
|
||||||
set -euo pipefail
|
set -euo pipefail
|
||||||
|
|
||||||
# == Require Red Hat Enterprise Linux/FedoraLinux/RockyLinux/AlmaLinux/CentOS OR Ubuntu/Debian ==
|
# == Require Red Hat Enterprise Linux/FedoraLinux/RockyLinux/AlmaLinux/CentOS OR Ubuntu ==
|
||||||
if [[ -r /etc/os-release ]]; then
|
if [[ -r /etc/os-release ]]; then
|
||||||
. /etc/os-release
|
source /etc/os-release
|
||||||
case "$ID" in
|
case "$ID" in
|
||||||
rhel|rocky|almalinux|fedora|centos|ubuntu|debian)
|
rhel | rocky | almalinux | fedora | centos | ubuntu)
|
||||||
echo "[OK] Detected supported system: $NAME $VERSION_ID"
|
echo "[OK] Detected supported system: $NAME $VERSION_ID"
|
||||||
;;
|
;;
|
||||||
*)
|
*)
|
||||||
echo "[ERROR] Unsupported system: $NAME ($ID)."
|
echo "[ERROR] Unsupported system: $NAME ($ID)."
|
||||||
echo "This script only supports Red Hat Enterprise Linux/RockyLinux/AlmaLinux/CentOS or Ubuntu/Debian."
|
echo "This script only supports Red Hat Enterprise Linux/RockyLinux/AlmaLinux/CentOS or Ubuntu."
|
||||||
exit 1
|
exit 1
|
||||||
;;
|
;;
|
||||||
esac
|
esac
|
||||||
else
|
else
|
||||||
echo "[ERROR] Cannot detect system (missing /etc/os-release)."
|
echo "[ERROR] Cannot detect system (missing /etc/os-release)."
|
||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
# ======================== Kernel version check (require >= 6.11) =======================
|
||||||
|
MIN_KERNEL_MAJOR=6
|
||||||
|
MIN_KERNEL_MINOR=11
|
||||||
|
KERNEL_FULL=$(uname -r)
|
||||||
|
KERNEL_MAJOR=$(echo "$KERNEL_FULL" | cut -d. -f1)
|
||||||
|
KERNEL_MINOR=$(echo "$KERNEL_FULL" | cut -d. -f2)
|
||||||
|
|
||||||
|
echo "[INFO] Detected kernel version: $KERNEL_FULL"
|
||||||
|
|
||||||
|
if (( KERNEL_MAJOR < MIN_KERNEL_MAJOR )) || { (( KERNEL_MAJOR == MIN_KERNEL_MAJOR )) && (( KERNEL_MINOR < MIN_KERNEL_MINOR )); }; then
|
||||||
|
echo "[ERROR] Kernel $KERNEL_FULL is too old. Requires Linux >= ${MIN_KERNEL_MAJOR}.${MIN_KERNEL_MINOR}."
|
||||||
|
echo "Please upgrade your system or use a newer container (e.g. Fedora 42+, RHEL 10+, Debian 13+)."
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "[OK] Kernel version >= ${MIN_KERNEL_MAJOR}.${MIN_KERNEL_MINOR}."
|
||||||
|
|
||||||
# ===== Config & Parse arguments =========================================================
|
# ===== Config & Parse arguments =========================================================
|
||||||
VERSION_ARG="${1:-}" # Pass version number like 7.13.8, or leave empty
|
VERSION_ARG="${1:-}" # Pass version number like 7.13.8, or leave empty
|
||||||
WITH_CORE="both" # Default: bundle both xray+sing-box
|
WITH_CORE="both" # Default: bundle both xray+sing-box
|
||||||
AUTOSTART=0 # 1 = enable system-wide autostart (/etc/xdg/autostart)
|
ARCH_OVERRIDE="" # --arch x64|arm64|all (optional compile target)
|
||||||
FORCE_NETCORE=0 # --netcore => skip archive bundle, use separate downloads
|
|
||||||
ARCH_OVERRIDE="" # --arch x64|arm64|all (optional compile target)
|
|
||||||
BUILD_FROM="" # --buildfrom 1|2|3 to select channel non-interactively
|
|
||||||
|
|
||||||
# If the first argument starts with --, do not treat it as a version number
|
# If the first argument starts with --, do not treat it as a version number
|
||||||
if [[ "${VERSION_ARG:-}" == --* ]]; then
|
if [[ "${VERSION_ARG:-}" == --* ]]; then
|
||||||
@@ -37,97 +51,86 @@ if [[ -n "${VERSION_ARG:-}" ]]; then shift || true; fi
|
|||||||
# Parse remaining optional arguments
|
# Parse remaining optional arguments
|
||||||
while [[ $# -gt 0 ]]; do
|
while [[ $# -gt 0 ]]; do
|
||||||
case "$1" in
|
case "$1" in
|
||||||
--with-core) WITH_CORE="${2:-both}"; shift 2;;
|
--with-core)
|
||||||
--autostart) AUTOSTART=1; shift;;
|
WITH_CORE="${2:-both}"
|
||||||
--xray-ver) XRAY_VER="${2:-}"; shift 2;;
|
shift 2
|
||||||
--singbox-ver) SING_VER="${2:-}"; shift 2;;
|
;;
|
||||||
--netcore) FORCE_NETCORE=1; shift;;
|
--xray-ver)
|
||||||
--arch) ARCH_OVERRIDE="${2:-}"; shift 2;;
|
XRAY_VER="${2:-}"
|
||||||
--buildfrom) BUILD_FROM="${2:-}"; shift 2;;
|
shift 2
|
||||||
*)
|
;;
|
||||||
if [[ -z "${VERSION_ARG:-}" ]]; then VERSION_ARG="$1"; fi
|
--singbox-ver)
|
||||||
shift;;
|
SING_VER="${2:-}"
|
||||||
|
shift 2
|
||||||
|
;;
|
||||||
|
--arch)
|
||||||
|
ARCH_OVERRIDE="${2:-}"
|
||||||
|
shift 2
|
||||||
|
;;
|
||||||
|
--release)
|
||||||
|
RPM_RELEASE="${2:-}"
|
||||||
|
shift 2
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
if [[ -z "${VERSION_ARG:-}" ]]; then VERSION_ARG="$1"; fi
|
||||||
|
shift
|
||||||
|
;;
|
||||||
esac
|
esac
|
||||||
done
|
done
|
||||||
|
|
||||||
# Conflict: version number AND --buildfrom cannot be used together
|
if [[ -z "${RPM_RELEASE:-}" ]]; then
|
||||||
if [[ -n "${VERSION_ARG:-}" && -n "${BUILD_FROM:-}" ]]; then
|
echo "--release is required"
|
||||||
echo "[ERROR] You cannot specify both an explicit version and --buildfrom at the same time."
|
|
||||||
echo " Provide either a version (e.g. 7.14.0) OR --buildfrom 1|2|3."
|
|
||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# ===== Environment check + Dependencies ========================================
|
# ===== Environment check + Dependencies ========================================
|
||||||
host_arch="$(uname -m)"
|
host_arch="$(uname -m)"
|
||||||
[[ "$host_arch" == "aarch64" || "$host_arch" == "x86_64" ]] || { echo "Only supports aarch64 / x86_64"; exit 1; }
|
if ! [[ "$host_arch" == "aarch64" || "$host_arch" == "x86_64" ]]; then
|
||||||
|
echo "Only supports aarch64 / x86_64"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
install_ok=0
|
install_ok=0
|
||||||
case "$ID" in
|
case "$ID" in
|
||||||
# ------------------------------ RHEL family (UNCHANGED) ------------------------------
|
# ------------------------------ RHEL family (UNCHANGED) ------------------------------
|
||||||
rhel|rocky|almalinux|centos)
|
rhel | rocky | almalinux | centos)
|
||||||
if command -v dnf >/dev/null 2>&1; then
|
if command -v dnf >/dev/null 2>&1; then
|
||||||
sudo dnf -y install dotnet-sdk-8.0 rpm-build rpmdevtools curl unzip tar rsync || \
|
sudo dnf -y install dotnet-sdk-8.0 rpm-build rpmdevtools curl unzip tar rsync ||
|
||||||
sudo dnf -y install dotnet-sdk rpm-build rpmdevtools curl unzip tar rsync
|
sudo dnf -y install dotnet-sdk rpm-build rpmdevtools curl unzip tar rsync
|
||||||
install_ok=1
|
install_ok=1
|
||||||
elif command -v yum >/dev/null 2>&1; then
|
elif command -v yum >/dev/null 2>&1; then
|
||||||
sudo yum -y install dotnet-sdk-8.0 rpm-build rpmdevtools curl unzip tar rsync || \
|
sudo yum -y install dotnet-sdk-8.0 rpm-build rpmdevtools curl unzip tar rsync ||
|
||||||
sudo yum -y install dotnet-sdk rpm-build rpmdevtools curl unzip tar rsync
|
sudo yum -y install dotnet-sdk rpm-build rpmdevtools curl unzip tar rsync
|
||||||
install_ok=1
|
|
||||||
fi
|
|
||||||
;;
|
|
||||||
# ------------------------------ Ubuntu ----------------------------------------------
|
|
||||||
ubuntu)
|
|
||||||
sudo apt-get update
|
|
||||||
# Ensure 'universe' (Ubuntu) to get 'rpm'
|
|
||||||
if ! apt-cache policy | grep -q '^500 .*ubuntu.com/ubuntu.* universe'; then
|
|
||||||
sudo apt-get -y install software-properties-common || true
|
|
||||||
sudo add-apt-repository -y universe || true
|
|
||||||
sudo apt-get update
|
|
||||||
fi
|
|
||||||
# Base tools + rpm (provides rpmbuild)
|
|
||||||
sudo apt-get -y install curl unzip tar rsync rpm || true
|
|
||||||
# Cross-arch binutils so strip matches target arch + objdump for brp scripts
|
|
||||||
sudo apt-get -y install binutils binutils-x86-64-linux-gnu binutils-aarch64-linux-gnu || true
|
|
||||||
# rpmbuild presence check
|
|
||||||
if ! command -v rpmbuild >/dev/null 2>&1; then
|
|
||||||
echo "[ERROR] 'rpmbuild' not found after installing 'rpm'."
|
|
||||||
echo " Please ensure the 'rpm' package is available from your repos (universe on Ubuntu)."
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
# .NET SDK 8 (best effort via apt)
|
|
||||||
if ! command -v dotnet >/dev/null 2>&1; then
|
|
||||||
sudo apt-get -y install dotnet-sdk-8.0 || true
|
|
||||||
sudo apt-get -y install dotnet-sdk-8 || true
|
|
||||||
sudo apt-get -y install dotnet-sdk || true
|
|
||||||
fi
|
|
||||||
install_ok=1
|
install_ok=1
|
||||||
;;
|
fi
|
||||||
# ------------------------------ Debian (KEEP, with local dotnet install) ------------
|
;;
|
||||||
debian)
|
# ------------------------------ Ubuntu ----------------------------------------------
|
||||||
|
ubuntu)
|
||||||
|
sudo apt-get update
|
||||||
|
# Ensure 'universe' (Ubuntu) to get 'rpm'
|
||||||
|
if ! apt-cache policy | grep -q '^500 .*ubuntu.com/ubuntu.* universe'; then
|
||||||
|
sudo apt-get -y install software-properties-common || true
|
||||||
|
sudo add-apt-repository -y universe || true
|
||||||
sudo apt-get update
|
sudo apt-get update
|
||||||
# Base tools + rpm (provides rpmbuild on Debian) + objdump/strip
|
fi
|
||||||
sudo apt-get -y install curl unzip tar rsync rpm binutils || true
|
# Base tools + rpm (provides rpmbuild)
|
||||||
# rpmbuild presence check
|
sudo apt-get -y install curl unzip tar rsync rpm || true
|
||||||
if ! command -v rpmbuild >/dev/null 2>&1; then
|
# Cross-arch binutils so strip matches target arch + objdump for brp scripts
|
||||||
echo "[ERROR] 'rpmbuild' not found after installing 'rpm'."
|
sudo apt-get -y install binutils binutils-x86-64-linux-gnu binutils-aarch64-linux-gnu || true
|
||||||
echo " Please ensure 'rpm' is available from Debian repos."
|
# rpmbuild presence check
|
||||||
exit 1
|
if ! command -v rpmbuild >/dev/null 2>&1; then
|
||||||
fi
|
echo "[ERROR] 'rpmbuild' not found after installing 'rpm'."
|
||||||
# Try apt for dotnet; fallback to official installer into $HOME/.dotnet
|
echo " Please ensure the 'rpm' package is available from your repos (universe on Ubuntu)."
|
||||||
if ! command -v dotnet >/dev/null 2>&1; then
|
exit 1
|
||||||
echo "[INFO] 'dotnet' not found. Installing .NET 8 SDK locally to \$HOME/.dotnet ..."
|
fi
|
||||||
tmp="$(mktemp -d)"; trap '[[ -n "${tmp:-}" ]] && rm -rf "$tmp"' RETURN
|
# .NET SDK 8 (best effort via apt)
|
||||||
curl -fsSL https://dot.net/v1/dotnet-install.sh -o "$tmp/dotnet-install.sh"
|
if ! command -v dotnet >/dev/null 2>&1; then
|
||||||
bash "$tmp/dotnet-install.sh" --channel 8.0 --install-dir "$HOME/.dotnet"
|
sudo apt-get -y install dotnet-sdk-8.0 || true
|
||||||
export PATH="$HOME/.dotnet:$HOME/.dotnet/tools:$PATH"
|
sudo apt-get -y install dotnet-sdk-8 || true
|
||||||
export DOTNET_ROOT="$HOME/.dotnet"
|
sudo apt-get -y install dotnet-sdk || true
|
||||||
if ! command -v dotnet >/dev/null 2>&1; then
|
fi
|
||||||
echo "[ERROR] dotnet installation failed."
|
install_ok=1
|
||||||
exit 1
|
;;
|
||||||
fi
|
|
||||||
fi
|
|
||||||
install_ok=1
|
|
||||||
;;
|
|
||||||
esac
|
esac
|
||||||
|
|
||||||
if [[ "$install_ok" -ne 1 ]]; then
|
if [[ "$install_ok" -ne 1 ]]; then
|
||||||
@@ -141,6 +144,8 @@ command -v curl >/dev/null
|
|||||||
SCRIPT_DIR="$(cd -- "$(dirname -- "${BASH_SOURCE[0]}")" && pwd)"
|
SCRIPT_DIR="$(cd -- "$(dirname -- "${BASH_SOURCE[0]}")" && pwd)"
|
||||||
cd "$SCRIPT_DIR"
|
cd "$SCRIPT_DIR"
|
||||||
|
|
||||||
|
source ./utils.sh
|
||||||
|
|
||||||
# Git submodules (best effort)
|
# Git submodules (best effort)
|
||||||
if [[ -f .gitmodules ]]; then
|
if [[ -f .gitmodules ]]; then
|
||||||
git submodule sync --recursive || true
|
git submodule sync --recursive || true
|
||||||
@@ -152,348 +157,16 @@ PROJECT="v2rayN.Desktop/v2rayN.Desktop.csproj"
|
|||||||
if [[ ! -f "$PROJECT" ]]; then
|
if [[ ! -f "$PROJECT" ]]; then
|
||||||
PROJECT="$(find . -maxdepth 3 -name 'v2rayN.Desktop.csproj' | head -n1 || true)"
|
PROJECT="$(find . -maxdepth 3 -name 'v2rayN.Desktop.csproj' | head -n1 || true)"
|
||||||
fi
|
fi
|
||||||
[[ -f "$PROJECT" ]] || { echo "v2rayN.Desktop.csproj not found"; exit 1; }
|
[[ -f "$PROJECT" ]] || {
|
||||||
|
echo "v2rayN.Desktop.csproj not found"
|
||||||
# ===== Resolve GUI version & auto checkout ============================================
|
exit 1
|
||||||
VERSION=""
|
|
||||||
|
|
||||||
choose_channel() {
|
|
||||||
# If --buildfrom provided, map it directly and skip interaction.
|
|
||||||
if [[ -n "${BUILD_FROM:-}" ]]; then
|
|
||||||
case "$BUILD_FROM" in
|
|
||||||
1) echo "latest"; return 0;;
|
|
||||||
2) echo "prerelease"; return 0;;
|
|
||||||
3) echo "keep"; return 0;;
|
|
||||||
*) echo "[ERROR] Invalid --buildfrom value: ${BUILD_FROM}. Use 1|2|3." >&2; exit 1;;
|
|
||||||
esac
|
|
||||||
fi
|
|
||||||
|
|
||||||
# Print menu to stderr and read from /dev/tty so stdout only carries the token.
|
|
||||||
local ch="latest" sel=""
|
|
||||||
if [[ -t 0 ]]; then
|
|
||||||
echo "[?] Choose v2rayN release channel:" >&2
|
|
||||||
echo " 1) Latest (stable) [default]" >&2
|
|
||||||
echo " 2) Pre-release (preview)" >&2
|
|
||||||
echo " 3) Keep current (do nothing)" >&2
|
|
||||||
printf "Enter 1, 2 or 3 [default 1]: " >&2
|
|
||||||
if read -r sel </dev/tty; then
|
|
||||||
case "${sel:-}" in
|
|
||||||
2) ch="prerelease" ;;
|
|
||||||
3) ch="keep" ;;
|
|
||||||
*) ch="latest" ;;
|
|
||||||
esac
|
|
||||||
else
|
|
||||||
ch="latest"
|
|
||||||
fi
|
|
||||||
else
|
|
||||||
ch="latest"
|
|
||||||
fi
|
|
||||||
echo "$ch"
|
|
||||||
}
|
}
|
||||||
|
|
||||||
get_latest_tag_latest() {
|
VERSION="$VERSION_ARG"
|
||||||
# Resolve /releases/latest → tag_name
|
|
||||||
curl -fsSL "https://api.github.com/repos/2dust/v2rayN/releases/latest" \
|
|
||||||
| grep -Eo '"tag_name":\s*"v?[^"]+"' \
|
|
||||||
| head -n1 \
|
|
||||||
| sed -E 's/.*"tag_name":\s*"v?([^"]+)".*/\1/'
|
|
||||||
}
|
|
||||||
|
|
||||||
get_latest_tag_prerelease() {
|
|
||||||
# Resolve newest prerelease=true tag; prefer jq, fallback to sed/grep (no awk)
|
|
||||||
local json tag
|
|
||||||
json="$(curl -fsSL "https://api.github.com/repos/2dust/v2rayN/releases?per_page=20")" || return 1
|
|
||||||
|
|
||||||
# 1) Use jq if present
|
|
||||||
if command -v jq >/dev/null 2>&1; then
|
|
||||||
tag="$(printf '%s' "$json" \
|
|
||||||
| jq -r '[.[] | select(.prerelease==true)][0].tag_name' 2>/dev/null \
|
|
||||||
| sed 's/^v//')" || true
|
|
||||||
fi
|
|
||||||
|
|
||||||
# 2) Fallback to sed/grep only
|
|
||||||
if [[ -z "${tag:-}" || "${tag:-}" == "null" ]]; then
|
|
||||||
tag="$(printf '%s' "$json" \
|
|
||||||
| tr '\n' ' ' \
|
|
||||||
| sed 's/},[[:space:]]*{/\n/g' \
|
|
||||||
| grep -m1 -E '"prerelease"[[:space:]]*:[[:space:]]*true' \
|
|
||||||
| grep -Eo '"tag_name"[[:space:]]*:[[:space:]]*"v?[^"]+"' \
|
|
||||||
| head -n1 \
|
|
||||||
| sed -E 's/.*"tag_name"[[:space:]]*:[[:space:]]*"v?([^"]+)".*/\1/')" || true
|
|
||||||
fi
|
|
||||||
|
|
||||||
[[ -n "${tag:-}" && "${tag:-}" != "null" ]] || return 1
|
|
||||||
printf '%s\n' "$tag"
|
|
||||||
}
|
|
||||||
|
|
||||||
git_try_checkout() {
|
|
||||||
# Try a series of refs and checkout when found.
|
|
||||||
local want="$1" ref=""
|
|
||||||
if git rev-parse --git-dir >/dev/null 2>&1; then
|
|
||||||
git fetch --tags --force --prune --depth=1 || true
|
|
||||||
if git rev-parse "refs/tags/v${want}" >/dev/null 2>&1; then
|
|
||||||
ref="v${want}"
|
|
||||||
elif git rev-parse "refs/tags/${want}" >/dev/null 2>&1; then
|
|
||||||
ref="${want}"
|
|
||||||
elif git rev-parse --verify "${want}" >/dev/null 2>&1; then
|
|
||||||
ref="${want}"
|
|
||||||
fi
|
|
||||||
if [[ -n "$ref" ]]; then
|
|
||||||
echo "[OK] Found ref '${ref}', checking out..."
|
|
||||||
git checkout -f "${ref}"
|
|
||||||
if [[ -f .gitmodules ]]; then
|
|
||||||
git submodule sync --recursive || true
|
|
||||||
git submodule update --init --recursive || true
|
|
||||||
fi
|
|
||||||
return 0
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
return 1
|
|
||||||
}
|
|
||||||
|
|
||||||
if git rev-parse --git-dir >/dev/null 2>&1; then
|
|
||||||
if [[ -n "${VERSION_ARG:-}" ]]; then
|
|
||||||
echo "[*] Trying to switch v2rayN repo to version: ${VERSION_ARG}"
|
|
||||||
if git_try_checkout "${VERSION_ARG#v}"; then
|
|
||||||
VERSION="${VERSION_ARG#v}"
|
|
||||||
else
|
|
||||||
echo "[WARN] Tag '${VERSION_ARG}' not found."
|
|
||||||
ch="$(choose_channel)"
|
|
||||||
if [[ "$ch" == "keep" ]]; then
|
|
||||||
echo "[*] Keep current repository state (no checkout)."
|
|
||||||
if git describe --tags --abbrev=0 >/dev/null 2>&1; then
|
|
||||||
VERSION="$(git describe --tags --abbrev=0)"
|
|
||||||
else
|
|
||||||
VERSION="0.0.0+git"
|
|
||||||
fi
|
|
||||||
VERSION="${VERSION#v}"
|
|
||||||
else
|
|
||||||
echo "[*] Resolving ${ch} tag from GitHub releases..."
|
|
||||||
tag=""
|
|
||||||
if [[ "$ch" == "prerelease" ]]; then
|
|
||||||
tag="$(get_latest_tag_prerelease || true)"
|
|
||||||
if [[ -z "$tag" ]]; then
|
|
||||||
echo "[WARN] Failed to resolve prerelease tag, falling back to latest."
|
|
||||||
tag="$(get_latest_tag_latest || true)"
|
|
||||||
fi
|
|
||||||
else
|
|
||||||
tag="$(get_latest_tag_latest || true)"
|
|
||||||
fi
|
|
||||||
[[ -n "$tag" ]] || { echo "[ERROR] Failed to resolve latest tag for channel '${ch}'."; exit 1; }
|
|
||||||
echo "[*] Latest tag for '${ch}': ${tag}"
|
|
||||||
git_try_checkout "$tag" || { echo "[ERROR] Failed to checkout '${tag}'."; exit 1; }
|
|
||||||
VERSION="${tag#v}"
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
else
|
|
||||||
ch="$(choose_channel)"
|
|
||||||
if [[ "$ch" == "keep" ]]; then
|
|
||||||
echo "[*] Keep current repository state (no checkout)."
|
|
||||||
if git describe --tags --abbrev=0 >/dev/null 2>&1; then
|
|
||||||
VERSION="$(git describe --tags --abbrev=0)"
|
|
||||||
else
|
|
||||||
VERSION="0.0.0+git"
|
|
||||||
fi
|
|
||||||
VERSION="${VERSION#v}"
|
|
||||||
else
|
|
||||||
echo "[*] Resolving ${ch} tag from GitHub releases..."
|
|
||||||
tag=""
|
|
||||||
if [[ "$ch" == "prerelease" ]]; then
|
|
||||||
tag="$(get_latest_tag_prerelease || true)"
|
|
||||||
if [[ -z "$tag" ]]; then
|
|
||||||
echo "[WARN] Failed to resolve prerelease tag, falling back to latest."
|
|
||||||
tag="$(get_latest_tag_latest || true)"
|
|
||||||
fi
|
|
||||||
else
|
|
||||||
tag="$(get_latest_tag_latest || true)"
|
|
||||||
fi
|
|
||||||
[[ -n "$tag" ]] || { echo "[ERROR] Failed to resolve latest tag for channel '${ch}'."; exit 1; }
|
|
||||||
echo "[*] Latest tag for '${ch}': ${tag}"
|
|
||||||
git_try_checkout "$tag" || { echo "[ERROR] Failed to checkout '${tag}'."; exit 1; }
|
|
||||||
VERSION="${tag#v}"
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
else
|
|
||||||
echo "[WARN] Current directory is not a git repo; cannot checkout version. Proceeding on current tree."
|
|
||||||
VERSION="${VERSION_ARG:-}"
|
|
||||||
if [[ -z "$VERSION" ]]; then
|
|
||||||
if git describe --tags --abbrev=0 >/dev/null 2>&1; then
|
|
||||||
VERSION="$(git describe --tags --abbrev=0)"
|
|
||||||
else
|
|
||||||
VERSION="0.0.0+git"
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
VERSION="${VERSION#v}"
|
|
||||||
fi
|
|
||||||
echo "[*] GUI version resolved as: ${VERSION}"
|
|
||||||
|
|
||||||
# ===== Helpers for core/rules download (use RID_DIR for arch sync) =====================
|
|
||||||
download_xray() {
|
|
||||||
# Download Xray core and install to outdir/xray
|
|
||||||
local outdir="$1" ver="${XRAY_VER:-}" url tmp zipname="xray.zip"
|
|
||||||
mkdir -p "$outdir"
|
|
||||||
if [[ -n "${XRAY_VER:-}" ]]; then ver="${XRAY_VER}"; fi
|
|
||||||
if [[ -z "$ver" ]]; then
|
|
||||||
ver="$(curl -fsSL https://api.github.com/repos/XTLS/Xray-core/releases/latest \
|
|
||||||
| grep -Eo '"tag_name":\s*"v[^"]+"' | sed -E 's/.*"v([^"]+)".*/\1/' | head -n1)" || true
|
|
||||||
fi
|
|
||||||
[[ -n "$ver" ]] || { echo "[xray] Failed to get version"; return 1; }
|
|
||||||
if [[ "$RID_DIR" == "linux-arm64" ]]; then
|
|
||||||
url="https://github.com/XTLS/Xray-core/releases/download/v${ver}/Xray-linux-arm64-v8a.zip"
|
|
||||||
else
|
|
||||||
url="https://github.com/XTLS/Xray-core/releases/download/v${ver}/Xray-linux-64.zip"
|
|
||||||
fi
|
|
||||||
echo "[+] Download xray: $url"
|
|
||||||
tmp="$(mktemp -d)"; trap '[[ -n "${tmp:-}" ]] && rm -rf "$tmp"' RETURN
|
|
||||||
curl -fL "$url" -o "$tmp/$zipname"
|
|
||||||
unzip -q "$tmp/$zipname" -d "$tmp"
|
|
||||||
install -Dm755 "$tmp/xray" "$outdir/xray"
|
|
||||||
}
|
|
||||||
|
|
||||||
download_singbox() {
|
|
||||||
# Download sing-box core and install to outdir/sing-box
|
|
||||||
local outdir="$1" ver="${SING_VER:-}" url tmp tarname="singbox.tar.gz" bin
|
|
||||||
mkdir -p "$outdir"
|
|
||||||
if [[ -n "${SING_VER:-}" ]]; then ver="${SING_VER}"; fi
|
|
||||||
if [[ -z "$ver" ]]; then
|
|
||||||
ver="$(curl -fsSL https://api.github.com/repos/SagerNet/sing-box/releases/latest \
|
|
||||||
| grep -Eo '"tag_name":\s*"v[^"]+"' | sed -E 's/.*"v([^"]+)".*/\1/' | head -n1)" || true
|
|
||||||
fi
|
|
||||||
[[ -n "$ver" ]] || { echo "[sing-box] Failed to get version"; return 1; }
|
|
||||||
if [[ "$RID_DIR" == "linux-arm64" ]]; then
|
|
||||||
url="https://github.com/SagerNet/sing-box/releases/download/v${ver}/sing-box-${ver}-linux-arm64.tar.gz"
|
|
||||||
else
|
|
||||||
url="https://github.com/SagerNet/sing-box/releases/download/v${ver}/sing-box-${ver}-linux-amd64.tar.gz"
|
|
||||||
fi
|
|
||||||
echo "[+] Download sing-box: $url"
|
|
||||||
tmp="$(mktemp -d)"; trap '[[ -n "${tmp:-}" ]] && rm -rf "$tmp"' RETURN
|
|
||||||
curl -fL "$url" -o "$tmp/$tarname"
|
|
||||||
tar -C "$tmp" -xzf "$tmp/$tarname"
|
|
||||||
bin="$(find "$tmp" -type f -name 'sing-box' | head -n1 || true)"
|
|
||||||
[[ -n "$bin" ]] || { echo "[!] sing-box unpack failed"; return 1; }
|
|
||||||
install -Dm755 "$bin" "$outdir/sing-box"
|
|
||||||
}
|
|
||||||
|
|
||||||
# ---- NEW: download_mihomo (REQUIRED in --netcore mode) ----
|
|
||||||
download_mihomo() {
|
|
||||||
# Download mihomo into outroot/bin/mihomo/mihomo
|
|
||||||
local outroot="$1"
|
|
||||||
local url=""
|
|
||||||
if [[ "$RID_DIR" == "linux-arm64" ]]; then
|
|
||||||
url="https://raw.githubusercontent.com/2dust/v2rayN-core-bin/refs/heads/master/v2rayN-linux-arm64/bin/mihomo/mihomo"
|
|
||||||
else
|
|
||||||
url="https://raw.githubusercontent.com/2dust/v2rayN-core-bin/refs/heads/master/v2rayN-linux-64/bin/mihomo/mihomo"
|
|
||||||
fi
|
|
||||||
echo "[+] Download mihomo: $url"
|
|
||||||
mkdir -p "$outroot/bin/mihomo"
|
|
||||||
curl -fL "$url" -o "$outroot/bin/mihomo/mihomo"
|
|
||||||
chmod +x "$outroot/bin/mihomo/mihomo" || true
|
|
||||||
}
|
|
||||||
|
|
||||||
# Move geo files to a unified path: outroot/bin
|
|
||||||
unify_geo_layout() {
|
|
||||||
local outroot="$1"
|
|
||||||
mkdir -p "$outroot/bin"
|
|
||||||
local names=( \
|
|
||||||
"geosite.dat" \
|
|
||||||
"geoip.dat" \
|
|
||||||
"geoip-only-cn-private.dat" \
|
|
||||||
"Country.mmdb" \
|
|
||||||
"geoip.metadb" \
|
|
||||||
)
|
|
||||||
for n in "${names[@]}"; do
|
|
||||||
# If file exists under bin/xray/, move it up to bin/
|
|
||||||
if [[ -f "$outroot/bin/xray/$n" ]]; then
|
|
||||||
mv -f "$outroot/bin/xray/$n" "$outroot/bin/$n"
|
|
||||||
fi
|
|
||||||
# If file already in bin/, leave it as-is
|
|
||||||
if [[ -f "$outroot/bin/$n" ]]; then
|
|
||||||
:
|
|
||||||
fi
|
|
||||||
done
|
|
||||||
}
|
|
||||||
|
|
||||||
# Download geo/rule assets; then unify to bin/
|
|
||||||
download_geo_assets() {
|
|
||||||
local outroot="$1"
|
|
||||||
local bin_dir="$outroot/bin"
|
|
||||||
local srss_dir="$bin_dir/srss"
|
|
||||||
mkdir -p "$bin_dir" "$srss_dir"
|
|
||||||
|
|
||||||
echo "[+] Download Xray Geo to ${bin_dir}"
|
|
||||||
curl -fsSL -o "$bin_dir/geosite.dat" \
|
|
||||||
"https://github.com/Loyalsoldier/V2ray-rules-dat/releases/latest/download/geosite.dat"
|
|
||||||
curl -fsSL -o "$bin_dir/geoip.dat" \
|
|
||||||
"https://github.com/Loyalsoldier/V2ray-rules-dat/releases/latest/download/geoip.dat"
|
|
||||||
curl -fsSL -o "$bin_dir/geoip-only-cn-private.dat" \
|
|
||||||
"https://raw.githubusercontent.com/Loyalsoldier/geoip/release/geoip-only-cn-private.dat"
|
|
||||||
curl -fsSL -o "$bin_dir/Country.mmdb" \
|
|
||||||
"https://raw.githubusercontent.com/Loyalsoldier/geoip/release/Country.mmdb"
|
|
||||||
|
|
||||||
echo "[+] Download sing-box rule DB & rule-sets"
|
|
||||||
curl -fsSL -o "$bin_dir/geoip.metadb" \
|
|
||||||
"https://github.com/MetaCubeX/meta-rules-dat/releases/latest/download/geoip.metadb" || true
|
|
||||||
|
|
||||||
for f in \
|
|
||||||
geoip-private.srs geoip-cn.srs geoip-facebook.srs geoip-fastly.srs \
|
|
||||||
geoip-google.srs geoip-netflix.srs geoip-telegram.srs geoip-twitter.srs; do
|
|
||||||
curl -fsSL -o "$srss_dir/$f" \
|
|
||||||
"https://raw.githubusercontent.com/2dust/sing-box-rules/rule-set-geoip/$f" || true
|
|
||||||
done
|
|
||||||
for f in \
|
|
||||||
geosite-cn.srs geosite-gfw.srs geosite-greatfire.srs \
|
|
||||||
geosite-geolocation-cn.srs geosite-category-ads-all.srs geosite-private.srs; do
|
|
||||||
curl -fsSL -o "$srss_dir/$f" \
|
|
||||||
"https://raw.githubusercontent.com/2dust/sing-box-rules/rule-set-geosite/$f" || true
|
|
||||||
done
|
|
||||||
|
|
||||||
# Unify to bin/
|
|
||||||
unify_geo_layout "$outroot"
|
|
||||||
}
|
|
||||||
|
|
||||||
# Prefer the prebuilt v2rayN core bundle; then unify geo layout
|
|
||||||
download_v2rayn_bundle() {
|
|
||||||
local outroot="$1"
|
|
||||||
local url=""
|
|
||||||
if [[ "$RID_DIR" == "linux-arm64" ]]; then
|
|
||||||
url="https://raw.githubusercontent.com/2dust/v2rayN-core-bin/refs/heads/master/v2rayN-linux-arm64.zip"
|
|
||||||
else
|
|
||||||
url="https://raw.githubusercontent.com/2dust/v2rayN-core-bin/refs/heads/master/v2rayN-linux-64.zip"
|
|
||||||
fi
|
|
||||||
echo "[+] Try v2rayN bundle archive: $url"
|
|
||||||
local tmp zipname
|
|
||||||
tmp="$(mktemp -d)"; zipname="$tmp/v2rayn.zip"
|
|
||||||
curl -fL "$url" -o "$zipname" || { echo "[!] Bundle download failed"; return 1; }
|
|
||||||
unzip -q "$zipname" -d "$tmp" || { echo "[!] Bundle unzip failed"; return 1; }
|
|
||||||
|
|
||||||
if [[ -d "$tmp/bin" ]]; then
|
|
||||||
mkdir -p "$outroot/bin"
|
|
||||||
rsync -a "$tmp/bin/" "$outroot/bin/"
|
|
||||||
else
|
|
||||||
rsync -a "$tmp/" "$outroot/"
|
|
||||||
fi
|
|
||||||
|
|
||||||
rm -f "$outroot/v2rayn.zip" 2>/dev/null || true
|
|
||||||
# keep mihomo
|
|
||||||
# find "$outroot" -type d -name "mihomo" -prune -exec rm -rf {} + 2>/dev/null || true
|
|
||||||
|
|
||||||
local nested_dir
|
|
||||||
nested_dir="$(find "$outroot" -maxdepth 1 -type d -name 'v2rayN-linux-*' | head -n1 || true)"
|
|
||||||
if [[ -n "${nested_dir:-}" && -d "$nested_dir/bin" ]]; then
|
|
||||||
mkdir -p "$outroot/bin"
|
|
||||||
rsync -a "$nested_dir/bin/" "$outroot/bin/"
|
|
||||||
rm -rf "$nested_dir"
|
|
||||||
fi
|
|
||||||
|
|
||||||
# Unify to bin/
|
|
||||||
unify_geo_layout "$outroot"
|
|
||||||
|
|
||||||
echo "[+] Bundle extracted to $outroot"
|
|
||||||
}
|
|
||||||
|
|
||||||
# ===== Build results collection for --arch all ========================================
|
# ===== Build results collection for --arch all ========================================
|
||||||
BUILT_RPMS=() # Will collect absolute paths of built RPMs
|
BUILT_RPMS=() # Will collect absolute paths of built RPMs
|
||||||
BUILT_ALL=0 # Flag to know if we should print the final summary
|
BUILT_ALL=0 # Flag to know if we should print the final summary
|
||||||
|
|
||||||
# ===== Build (single-arch) function ====================================================
|
# ===== Build (single-arch) function ====================================================
|
||||||
build_for_arch() {
|
build_for_arch() {
|
||||||
@@ -501,22 +174,34 @@ build_for_arch() {
|
|||||||
local short="$1"
|
local short="$1"
|
||||||
local rid rpm_target archdir
|
local rid rpm_target archdir
|
||||||
case "$short" in
|
case "$short" in
|
||||||
x64) rid="linux-x64"; rpm_target="x86_64"; archdir="x86_64" ;;
|
x64)
|
||||||
arm64) rid="linux-arm64"; rpm_target="aarch64"; archdir="aarch64" ;;
|
rid="linux-x64"
|
||||||
*) echo "[ERROR] Unknown arch '$short' (use x64|arm64)"; return 1;;
|
rpm_target="x86_64"
|
||||||
|
archdir="x86_64"
|
||||||
|
;;
|
||||||
|
arm64)
|
||||||
|
rid="linux-arm64"
|
||||||
|
rpm_target="aarch64"
|
||||||
|
archdir="aarch64"
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
echo "[ERROR] Unknown arch '$short' (use x64|arm64)"
|
||||||
|
return 1
|
||||||
|
;;
|
||||||
esac
|
esac
|
||||||
|
|
||||||
echo "[*] Building for target: $short (RID=$rid, RPM --target $rpm_target)"
|
echo "[*] Building for target: $short (RID=$rid, RPM --target $rpm_target)"
|
||||||
|
|
||||||
# .NET publish (self-contained) for this RID
|
# .NET publish (self-contained) for this RID
|
||||||
dotnet clean "$PROJECT" -c Release
|
|
||||||
rm -rf "$(dirname "$PROJECT")/bin/Release/net8.0" || true
|
|
||||||
|
|
||||||
dotnet restore "$PROJECT"
|
dotnet restore "$PROJECT"
|
||||||
|
rm -rf "$(dirname "$PROJECT")/bin/Release/net8.0" || true
|
||||||
dotnet publish "$PROJECT" \
|
dotnet publish "$PROJECT" \
|
||||||
-c Release -r "$rid" \
|
-c Release -r "$rid" \
|
||||||
-p:PublishSingleFile=false \
|
|
||||||
-p:SelfContained=true \
|
-p:SelfContained=true \
|
||||||
|
-p:PublishSingleFile=false \
|
||||||
|
-p:PublishTrimmed=false \
|
||||||
|
-p:DebugType=none \
|
||||||
|
-p:DebugSymbols=false \
|
||||||
-p:IncludeNativeLibrariesForSelfExtract=true
|
-p:IncludeNativeLibrariesForSelfExtract=true
|
||||||
|
|
||||||
# Per-arch variables (scoped)
|
# Per-arch variables (scoped)
|
||||||
@@ -524,6 +209,7 @@ build_for_arch() {
|
|||||||
local PUBDIR
|
local PUBDIR
|
||||||
PUBDIR="$(dirname "$PROJECT")/bin/Release/net8.0/${RID_DIR}/publish"
|
PUBDIR="$(dirname "$PROJECT")/bin/Release/net8.0/${RID_DIR}/publish"
|
||||||
[[ -d "$PUBDIR" ]]
|
[[ -d "$PUBDIR" ]]
|
||||||
|
sudo find "$PUBDIR" -type f -name "*.so" -exec strip {} +
|
||||||
|
|
||||||
# Make RID_DIR visible to download helpers (they read this var)
|
# Make RID_DIR visible to download helpers (they read this var)
|
||||||
export RID_DIR
|
export RID_DIR
|
||||||
@@ -563,31 +249,13 @@ build_for_arch() {
|
|||||||
mkdir -p "$WORKDIR/$PKGROOT/bin/xray" "$WORKDIR/$PKGROOT/bin/sing_box"
|
mkdir -p "$WORKDIR/$PKGROOT/bin/xray" "$WORKDIR/$PKGROOT/bin/sing_box"
|
||||||
|
|
||||||
# Bundle / cores per-arch
|
# Bundle / cores per-arch
|
||||||
if [[ "$FORCE_NETCORE" -eq 0 ]]; then
|
if [[ "$WITH_CORE" == "xray" || "$WITH_CORE" == "both" ]]; then
|
||||||
if download_v2rayn_bundle "$WORKDIR/$PKGROOT"; then
|
download_xray "$WORKDIR/$PKGROOT/bin/xray" || echo "[!] xray download failed (skipped)"
|
||||||
echo "[*] Using v2rayN bundle archive."
|
|
||||||
else
|
|
||||||
echo "[*] Bundle failed, fallback to separate core + rules."
|
|
||||||
if [[ "$WITH_CORE" == "xray" || "$WITH_CORE" == "both" ]]; then
|
|
||||||
download_xray "$WORKDIR/$PKGROOT/bin/xray" || echo "[!] xray download failed (skipped)"
|
|
||||||
fi
|
|
||||||
if [[ "$WITH_CORE" == "sing-box" || "$WITH_CORE" == "both" ]]; then
|
|
||||||
download_singbox "$WORKDIR/$PKGROOT/bin/sing_box" || echo "[!] sing-box download failed (skipped)"
|
|
||||||
fi
|
|
||||||
download_geo_assets "$WORKDIR/$PKGROOT" || echo "[!] Geo rules download failed (skipped)"
|
|
||||||
fi
|
|
||||||
else
|
|
||||||
echo "[*] --netcore specified: use separate core + rules."
|
|
||||||
if [[ "$WITH_CORE" == "xray" || "$WITH_CORE" == "both" ]]; then
|
|
||||||
download_xray "$WORKDIR/$PKGROOT/bin/xray" || echo "[!] xray download failed (skipped)"
|
|
||||||
fi
|
|
||||||
if [[ "$WITH_CORE" == "sing-box" || "$WITH_CORE" == "both" ]]; then
|
|
||||||
download_singbox "$WORKDIR/$PKGROOT/bin/sing_box" || echo "[!] sing-box download failed (skipped)"
|
|
||||||
fi
|
|
||||||
download_geo_assets "$WORKDIR/$PKGROOT" || echo "[!] Geo rules download failed (skipped)"
|
|
||||||
# ---- REQUIRED: always fetch mihomo in netcore mode, per-arch ----
|
|
||||||
download_mihomo "$WORKDIR/$PKGROOT" || echo "[!] mihomo download failed (skipped)"
|
|
||||||
fi
|
fi
|
||||||
|
if [[ "$WITH_CORE" == "sing-box" || "$WITH_CORE" == "both" ]]; then
|
||||||
|
download_singbox "$WORKDIR/$PKGROOT/bin/sing_box" || echo "[!] sing-box download failed (skipped)"
|
||||||
|
fi
|
||||||
|
download_geo_assets "$WORKDIR/$PKGROOT" || echo "[!] Geo rules download failed (skipped)"
|
||||||
|
|
||||||
# Tarball
|
# Tarball
|
||||||
mkdir -p "$SOURCEDIR"
|
mkdir -p "$SOURCEDIR"
|
||||||
@@ -596,26 +264,35 @@ build_for_arch() {
|
|||||||
# SPEC
|
# SPEC
|
||||||
local SPECFILE="$SPECDIR/v2rayN.spec"
|
local SPECFILE="$SPECDIR/v2rayN.spec"
|
||||||
mkdir -p "$SPECDIR"
|
mkdir -p "$SPECDIR"
|
||||||
cat > "$SPECFILE" <<'SPEC'
|
cat >"$SPECFILE" <<'SPEC'
|
||||||
%global debug_package %{nil}
|
%global debug_package %{nil}
|
||||||
%undefine _debuginfo_subpackages
|
%undefine _debuginfo_subpackages
|
||||||
%undefine _debugsource_packages
|
%undefine _debugsource_packages
|
||||||
# Ignore outdated LTTng dependencies incorrectly reported by the .NET runtime (to avoid installation failures)
|
# Ignore outdated LTTng dependencies incorrectly reported by the .NET runtime (to avoid installation failures)
|
||||||
%global __requires_exclude ^liblttng-ust\.so\..*$
|
%global __requires_exclude ^liblttng-ust\.so\..*$
|
||||||
|
|
||||||
Name: v2rayN
|
Name: v2rayn-unofficial
|
||||||
Version: __VERSION__
|
Version: __VERSION__
|
||||||
Release: 1%{?dist}
|
Release: __RELEASE__
|
||||||
Summary: v2rayN (Avalonia) GUI client for Linux (x86_64/aarch64)
|
Summary: v2rayN (Avalonia) GUI client for Linux (x86_64/aarch64)
|
||||||
License: GPL-3.0-only
|
License: GPL-3.0-only
|
||||||
URL: https://github.com/2dust/v2rayN
|
URL: https://git.vlyaii.ru/voronin9032/v2rayN
|
||||||
BugURL: https://github.com/2dust/v2rayN/issues
|
BugURL: https://github.com/2dust/v2rayN/issues
|
||||||
ExclusiveArch: aarch64 x86_64
|
ExclusiveArch: aarch64 x86_64
|
||||||
Source0: __PKGROOT__.tar.gz
|
Source0: __PKGROOT__.tar.gz
|
||||||
|
|
||||||
# Runtime dependencies (Avalonia / X11 / Fonts / GL)
|
# Runtime dependencies (Avalonia / X11 / Fonts / GL)
|
||||||
Requires: libX11, libXrandr, libXcursor, libXi, libXext, libxcb, libXrender, libXfixes, libXinerama, libxkbcommon
|
Requires: cairo, pango, openssl, mesa-libEGL, mesa-libGL
|
||||||
Requires: fontconfig, freetype, cairo, pango, mesa-libEGL, mesa-libGL, xdg-utils
|
Requires: glibc >= 2.34
|
||||||
|
Requires: fontconfig >= 2.13.1
|
||||||
|
Requires: desktop-file-utils >= 0.26
|
||||||
|
Requires: xdg-utils >= 1.1.3
|
||||||
|
Requires: coreutils >= 8.32
|
||||||
|
Requires: bash >= 5.1
|
||||||
|
Requires: freetype >= 2.10
|
||||||
|
|
||||||
|
Conflicts: v2rayN
|
||||||
|
Obsoletes: v2rayN
|
||||||
|
|
||||||
%description
|
%description
|
||||||
v2rayN Linux for Red Hat Enterprise Linux
|
v2rayN Linux for Red Hat Enterprise Linux
|
||||||
@@ -689,41 +366,9 @@ fi
|
|||||||
%{_datadir}/icons/hicolor/256x256/apps/v2rayn.png
|
%{_datadir}/icons/hicolor/256x256/apps/v2rayn.png
|
||||||
SPEC
|
SPEC
|
||||||
|
|
||||||
# Autostart injection (inside %install) and %files entry
|
|
||||||
if [[ "$AUTOSTART" -eq 1 ]]; then
|
|
||||||
awk '
|
|
||||||
BEGIN{ins=0}
|
|
||||||
/^%post$/ && !ins {
|
|
||||||
print "# --- Autostart (.desktop) ---"
|
|
||||||
print "install -dm0755 %{buildroot}/etc/xdg/autostart"
|
|
||||||
print "cat > %{buildroot}/etc/xdg/autostart/v2rayn.desktop << '\''EOF'\''"
|
|
||||||
print "[Desktop Entry]"
|
|
||||||
print "Type=Application"
|
|
||||||
print "Name=v2rayN (Autostart)"
|
|
||||||
print "Exec=v2rayn"
|
|
||||||
print "X-GNOME-Autostart-enabled=true"
|
|
||||||
print "NoDisplay=false"
|
|
||||||
print "EOF"
|
|
||||||
ins=1
|
|
||||||
}
|
|
||||||
{print}
|
|
||||||
' "$SPECFILE" > "${SPECFILE}.tmp" && mv "${SPECFILE}.tmp" "$SPECFILE"
|
|
||||||
|
|
||||||
awk '
|
|
||||||
BEGIN{infiles=0; done=0}
|
|
||||||
/^%files$/ {infiles=1}
|
|
||||||
infiles && done==0 && $0 ~ /%{_datadir}\/icons\/hicolor\/256x256\/apps\/v2rayn\.png/ {
|
|
||||||
print
|
|
||||||
print "%config(noreplace) /etc/xdg/autostart/v2rayn.desktop"
|
|
||||||
done=1
|
|
||||||
next
|
|
||||||
}
|
|
||||||
{print}
|
|
||||||
' "$SPECFILE" > "${SPECFILE}.tmp" && mv "${SPECFILE}.tmp" "$SPECFILE"
|
|
||||||
fi
|
|
||||||
|
|
||||||
# Replace placeholders
|
# Replace placeholders
|
||||||
sed -i "s/__VERSION__/${VERSION}/g" "$SPECFILE"
|
sed -i "s/__VERSION__/${VERSION}/g" "$SPECFILE"
|
||||||
|
sed -i "s/__RELEASE__/${RPM_RELEASE}/g" "$SPECFILE"
|
||||||
sed -i "s/__PKGROOT__/${PKGROOT}/g" "$SPECFILE"
|
sed -i "s/__PKGROOT__/${PKGROOT}/g" "$SPECFILE"
|
||||||
|
|
||||||
# ----- Select proper 'strip' per target arch on Ubuntu only (cross-binutils) -----
|
# ----- Select proper 'strip' per target arch on Ubuntu only (cross-binutils) -----
|
||||||
@@ -738,7 +383,7 @@ SPEC
|
|||||||
STRIP_BIN="/usr/bin/aarch64-linux-gnu-strip"
|
STRIP_BIN="/usr/bin/aarch64-linux-gnu-strip"
|
||||||
fi
|
fi
|
||||||
if [[ -x "$STRIP_BIN" ]]; then
|
if [[ -x "$STRIP_BIN" ]]; then
|
||||||
STRIP_ARGS=( --define "__strip $STRIP_BIN" )
|
STRIP_ARGS=(--define "__strip $STRIP_BIN")
|
||||||
fi
|
fi
|
||||||
fi
|
fi
|
||||||
|
|
||||||
@@ -758,7 +403,7 @@ SPEC
|
|||||||
|
|
||||||
echo "Build done for $short. RPM at:"
|
echo "Build done for $short. RPM at:"
|
||||||
local f
|
local f
|
||||||
for f in "${TOPDIR}/RPMS/${archdir}/v2rayN-${VERSION}-1"*.rpm; do
|
for f in "${TOPDIR}/RPMS/${archdir}/v2rayN-${VERSION}-${RPM_RELEASE}"*.rpm; do
|
||||||
[[ -e "$f" ]] || continue
|
[[ -e "$f" ]] || continue
|
||||||
echo " $f"
|
echo " $f"
|
||||||
BUILT_RPMS+=("$f")
|
BUILT_RPMS+=("$f")
|
||||||
@@ -767,30 +412,30 @@ SPEC
|
|||||||
|
|
||||||
# ===== Arch selection and build orchestration =========================================
|
# ===== Arch selection and build orchestration =========================================
|
||||||
case "${ARCH_OVERRIDE:-}" in
|
case "${ARCH_OVERRIDE:-}" in
|
||||||
"")
|
"")
|
||||||
# No --arch: use host architecture
|
# No --arch: use host architecture
|
||||||
if [[ "$host_arch" == "aarch64" ]]; then
|
if [[ "$host_arch" == "aarch64" ]]; then
|
||||||
build_for_arch arm64
|
|
||||||
else
|
|
||||||
build_for_arch x64
|
|
||||||
fi
|
|
||||||
;;
|
|
||||||
x64|amd64)
|
|
||||||
build_for_arch x64
|
|
||||||
;;
|
|
||||||
arm64|aarch64)
|
|
||||||
build_for_arch arm64
|
build_for_arch arm64
|
||||||
;;
|
else
|
||||||
all)
|
|
||||||
BUILT_ALL=1
|
|
||||||
# Build x64 and arm64 separately; each package contains its own arch-only binaries.
|
|
||||||
build_for_arch x64
|
build_for_arch x64
|
||||||
build_for_arch arm64
|
fi
|
||||||
;;
|
;;
|
||||||
*)
|
x64 | amd64)
|
||||||
echo "[ERROR] Unknown --arch '${ARCH_OVERRIDE}'. Use x64|arm64|all."
|
build_for_arch x64
|
||||||
exit 1
|
;;
|
||||||
;;
|
arm64 | aarch64)
|
||||||
|
build_for_arch arm64
|
||||||
|
;;
|
||||||
|
all)
|
||||||
|
BUILT_ALL=1
|
||||||
|
# Build x64 and arm64 separately; each package contains its own arch-only binaries.
|
||||||
|
build_for_arch x64
|
||||||
|
build_for_arch arm64
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
echo "[ERROR] Unknown --arch '${ARCH_OVERRIDE}'. Use x64|arm64|all."
|
||||||
|
exit 1
|
||||||
|
;;
|
||||||
esac
|
esac
|
||||||
|
|
||||||
# ===== Final summary if building both arches ==========================================
|
# ===== Final summary if building both arches ==========================================
|
||||||
|
|||||||
120
utils.sh
Normal file
120
utils.sh
Normal file
@@ -0,0 +1,120 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
|
||||||
|
download_xray() {
|
||||||
|
# Download Xray core and install to outdir/xray
|
||||||
|
local outdir="$1" ver="${XRAY_VER:-}" url tmp zipname="xray.zip"
|
||||||
|
mkdir -p "$outdir"
|
||||||
|
if [[ -n "${XRAY_VER:-}" ]]; then ver="${XRAY_VER}"; fi
|
||||||
|
if [[ -z "$ver" ]]; then
|
||||||
|
echo "Downloading latest xray"
|
||||||
|
ver="$(curl -fsSL https://api.github.com/repos/XTLS/Xray-core/releases/latest |
|
||||||
|
grep -Eo '"tag_name":\s*"v[^"]+"' | sed -E 's/.*"v([^"]+)".*/\1/' | head -n1)" || true
|
||||||
|
fi
|
||||||
|
if [[ -z "$ver" ]]; then
|
||||||
|
echo "[xray] Failed to get version"
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
if [[ "$RID_DIR" == "linux-arm64" ]]; then
|
||||||
|
url="https://github.com/XTLS/Xray-core/releases/download/v${ver}/Xray-linux-arm64-v8a.zip"
|
||||||
|
else
|
||||||
|
url="https://github.com/XTLS/Xray-core/releases/download/v${ver}/Xray-linux-64.zip"
|
||||||
|
fi
|
||||||
|
echo "[+] Download xray: $url"
|
||||||
|
tmp="$(mktemp -d)"
|
||||||
|
trap '[[ -n "${tmp:-}" ]] && rm -rf "$tmp"' RETURN
|
||||||
|
curl -fL "$url" -o "$tmp/$zipname"
|
||||||
|
unzip -q "$tmp/$zipname" -d "$tmp"
|
||||||
|
install -Dm755 "$tmp/xray" "$outdir/xray"
|
||||||
|
}
|
||||||
|
|
||||||
|
download_singbox() {
|
||||||
|
# Download sing-box core and install to outdir/sing-box
|
||||||
|
local outdir="$1" ver="${SING_VER:-}" url tmp tarname="singbox.tar.gz" bin
|
||||||
|
mkdir -p "$outdir"
|
||||||
|
if [[ -n "${SING_VER:-}" ]]; then ver="${SING_VER}"; fi
|
||||||
|
if [[ -z "$ver" ]]; then
|
||||||
|
echo "Downloading latest sing-box"
|
||||||
|
ver="$(curl -fsSL https://api.github.com/repos/SagerNet/sing-box/releases/latest |
|
||||||
|
grep -Eo '"tag_name":\s*"v[^"]+"' | sed -E 's/.*"v([^"]+)".*/\1/' | head -n1)" || true
|
||||||
|
fi
|
||||||
|
if [[ -z "$ver" ]]; then
|
||||||
|
echo "[sing-box] Failed to get version"
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
if [[ "$RID_DIR" == "linux-arm64" ]]; then
|
||||||
|
url="https://github.com/SagerNet/sing-box/releases/download/v${ver}/sing-box-${ver}-linux-arm64.tar.gz"
|
||||||
|
else
|
||||||
|
url="https://github.com/SagerNet/sing-box/releases/download/v${ver}/sing-box-${ver}-linux-amd64.tar.gz"
|
||||||
|
fi
|
||||||
|
echo "[+] Download sing-box: $url"
|
||||||
|
tmp="$(mktemp -d)"
|
||||||
|
trap '[[ -n "${tmp:-}" ]] && rm -rf "$tmp"' RETURN
|
||||||
|
curl -fL "$url" -o "$tmp/$tarname"
|
||||||
|
tar -C "$tmp" -xzf "$tmp/$tarname"
|
||||||
|
bin="$(find "$tmp" -type f -name 'sing-box' | head -n1 || true)"
|
||||||
|
[[ -n "$bin" ]] || {
|
||||||
|
echo "[!] sing-box unpack failed"
|
||||||
|
return 1
|
||||||
|
}
|
||||||
|
install -Dm755 "$bin" "$outdir/sing-box"
|
||||||
|
}
|
||||||
|
|
||||||
|
# Move geo files to a unified path: outroot/bin
|
||||||
|
unify_geo_layout() {
|
||||||
|
local outroot="$1"
|
||||||
|
mkdir -p "$outroot/bin"
|
||||||
|
local names=(
|
||||||
|
"geosite.dat"
|
||||||
|
"geoip.dat"
|
||||||
|
"geoip-only-cn-private.dat"
|
||||||
|
"Country.mmdb"
|
||||||
|
"geoip.metadb"
|
||||||
|
)
|
||||||
|
for n in "${names[@]}"; do
|
||||||
|
# If file exists under bin/xray/, move it up to bin/
|
||||||
|
if [[ -f "$outroot/bin/xray/$n" ]]; then
|
||||||
|
mv -f "$outroot/bin/xray/$n" "$outroot/bin/$n"
|
||||||
|
fi
|
||||||
|
# If file already in bin/, leave it as-is
|
||||||
|
if [[ -f "$outroot/bin/$n" ]]; then
|
||||||
|
:
|
||||||
|
fi
|
||||||
|
done
|
||||||
|
}
|
||||||
|
|
||||||
|
# Download geo/rule assets; then unify to bin/
|
||||||
|
download_geo_assets() {
|
||||||
|
local outroot="$1"
|
||||||
|
local bin_dir="$outroot/bin"
|
||||||
|
local srss_dir="$bin_dir/srss"
|
||||||
|
mkdir -p "$bin_dir" "$srss_dir"
|
||||||
|
echo "[+] Download Xray Geo to ${bin_dir}"
|
||||||
|
curl -fsSL -o "$bin_dir/geosite.dat" \
|
||||||
|
"https://github.com/Loyalsoldier/V2ray-rules-dat/releases/latest/download/geosite.dat"
|
||||||
|
curl -fsSL -o "$bin_dir/geoip.dat" \
|
||||||
|
"https://github.com/Loyalsoldier/V2ray-rules-dat/releases/latest/download/geoip.dat"
|
||||||
|
curl -fsSL -o "$bin_dir/geoip-only-cn-private.dat" \
|
||||||
|
"https://raw.githubusercontent.com/Loyalsoldier/geoip/release/geoip-only-cn-private.dat"
|
||||||
|
curl -fsSL -o "$bin_dir/Country.mmdb" \
|
||||||
|
"https://raw.githubusercontent.com/Loyalsoldier/geoip/release/Country.mmdb"
|
||||||
|
|
||||||
|
echo "[+] Download sing-box rule DB & rule-sets"
|
||||||
|
curl -fsSL -o "$bin_dir/geoip.metadb" \
|
||||||
|
"https://github.com/MetaCubeX/meta-rules-dat/releases/latest/download/geoip.metadb" || true
|
||||||
|
|
||||||
|
for f in \
|
||||||
|
geoip-private.srs geoip-cn.srs geoip-facebook.srs geoip-fastly.srs \
|
||||||
|
geoip-google.srs geoip-netflix.srs geoip-telegram.srs geoip-twitter.srs; do
|
||||||
|
curl -fsSL -o "$srss_dir/$f" \
|
||||||
|
"https://raw.githubusercontent.com/2dust/sing-box-rules/rule-set-geoip/$f" || true
|
||||||
|
done
|
||||||
|
for f in \
|
||||||
|
geosite-cn.srs geosite-gfw.srs geosite-greatfire.srs \
|
||||||
|
geosite-geolocation-cn.srs geosite-category-ads-all.srs geosite-private.srs; do
|
||||||
|
curl -fsSL -o "$srss_dir/$f" \
|
||||||
|
"https://raw.githubusercontent.com/2dust/sing-box-rules/rule-set-geosite/$f" || true
|
||||||
|
done
|
||||||
|
|
||||||
|
# Unify to bin/
|
||||||
|
unify_geo_layout "$outroot"
|
||||||
|
}
|
||||||
@@ -1,14 +1,14 @@
|
|||||||
<Project>
|
<Project>
|
||||||
|
|
||||||
<PropertyGroup>
|
<PropertyGroup>
|
||||||
<Version>7.15.5</Version>
|
<Version>7.16.6</Version>
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
|
|
||||||
<PropertyGroup>
|
<PropertyGroup>
|
||||||
<TargetFramework>net8.0</TargetFramework>
|
<TargetFramework>net8.0</TargetFramework>
|
||||||
<TargetLatestRuntimePatch>true</TargetLatestRuntimePatch>
|
<TargetLatestRuntimePatch>true</TargetLatestRuntimePatch>
|
||||||
<CheckForOverflowUnderflow>true</CheckForOverflowUnderflow>
|
<CheckForOverflowUnderflow>true</CheckForOverflowUnderflow>
|
||||||
<NoWarn>CA1031;CS1591;NU1507;CA1416;IDE0058</NoWarn>
|
<NoWarn>CA1031;CS1591;NU1507;CA1416;IDE0058;IDE0053;IDE0200</NoWarn>
|
||||||
<Nullable>annotations</Nullable>
|
<Nullable>annotations</Nullable>
|
||||||
<ImplicitUsings>enable</ImplicitUsings>
|
<ImplicitUsings>enable</ImplicitUsings>
|
||||||
<Authors>2dust</Authors>
|
<Authors>2dust</Authors>
|
||||||
|
|||||||
@@ -6,23 +6,23 @@
|
|||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<PackageVersion Include="Avalonia.AvaloniaEdit" Version="11.3.0" />
|
<PackageVersion Include="Avalonia.AvaloniaEdit" Version="11.3.0" />
|
||||||
<PackageVersion Include="Avalonia.Controls.DataGrid" Version="11.3.7" />
|
<PackageVersion Include="Avalonia.Controls.DataGrid" Version="11.3.9" />
|
||||||
<PackageVersion Include="Avalonia.Desktop" Version="11.3.7" />
|
<PackageVersion Include="Avalonia.Desktop" Version="11.3.9" />
|
||||||
<PackageVersion Include="Avalonia.Diagnostics" Version="11.3.7" />
|
<PackageVersion Include="Avalonia.Diagnostics" Version="11.3.9" />
|
||||||
<PackageVersion Include="Avalonia.ReactiveUI" Version="11.3.7" />
|
<PackageVersion Include="ReactiveUI.Avalonia" Version="11.3.8" />
|
||||||
<PackageVersion Include="CliWrap" Version="3.9.0" />
|
<PackageVersion Include="CliWrap" Version="3.10.0" />
|
||||||
<PackageVersion Include="Downloader" Version="4.0.3" />
|
<PackageVersion Include="Downloader" Version="4.0.3" />
|
||||||
<PackageVersion Include="H.NotifyIcon.Wpf" Version="2.3.1" />
|
<PackageVersion Include="H.NotifyIcon.Wpf" Version="2.4.1" />
|
||||||
<PackageVersion Include="MaterialDesignThemes" Version="5.3.0" />
|
<PackageVersion Include="MaterialDesignThemes" Version="5.3.0" />
|
||||||
<PackageVersion Include="MessageBox.Avalonia" Version="3.2.0" />
|
<PackageVersion Include="MessageBox.Avalonia" Version="3.3.0" />
|
||||||
<PackageVersion Include="QRCoder" Version="1.7.0" />
|
<PackageVersion Include="QRCoder" Version="1.7.0" />
|
||||||
<PackageVersion Include="ReactiveUI" Version="20.4.1" />
|
<PackageVersion Include="ReactiveUI" Version="22.3.1" />
|
||||||
<PackageVersion Include="ReactiveUI.Fody" Version="19.5.41" />
|
<PackageVersion Include="ReactiveUI.Fody" Version="19.5.41" />
|
||||||
<PackageVersion Include="ReactiveUI.WPF" Version="20.4.1" />
|
<PackageVersion Include="ReactiveUI.WPF" Version="22.3.1" />
|
||||||
<PackageVersion Include="Semi.Avalonia" Version="11.3.7" />
|
<PackageVersion Include="Semi.Avalonia" Version="11.3.7.1" />
|
||||||
<PackageVersion Include="Semi.Avalonia.AvaloniaEdit" Version="11.2.0.1" />
|
<PackageVersion Include="Semi.Avalonia.AvaloniaEdit" Version="11.2.0.1" />
|
||||||
<PackageVersion Include="Semi.Avalonia.DataGrid" Version="11.3.7" />
|
<PackageVersion Include="Semi.Avalonia.DataGrid" Version="11.3.7.1" />
|
||||||
<PackageVersion Include="NLog" Version="6.0.5" />
|
<PackageVersion Include="NLog" Version="6.0.6" />
|
||||||
<PackageVersion Include="sqlite-net-pcl" Version="1.9.172" />
|
<PackageVersion Include="sqlite-net-pcl" Version="1.9.172" />
|
||||||
<PackageVersion Include="TaskScheduler" Version="2.12.2" />
|
<PackageVersion Include="TaskScheduler" Version="2.12.2" />
|
||||||
<PackageVersion Include="WebDav.Client" Version="2.9.0" />
|
<PackageVersion Include="WebDav.Client" Version="2.9.0" />
|
||||||
|
|||||||
@@ -3,7 +3,7 @@ using System.IO.Compression;
|
|||||||
|
|
||||||
namespace ServiceLib.Common;
|
namespace ServiceLib.Common;
|
||||||
|
|
||||||
public static class FileManager
|
public static class FileUtils
|
||||||
{
|
{
|
||||||
private static readonly string _tag = "FileManager";
|
private static readonly string _tag = "FileManager";
|
||||||
|
|
||||||
@@ -35,9 +35,13 @@ public class JsonUtils
|
|||||||
/// <typeparam name="T"></typeparam>
|
/// <typeparam name="T"></typeparam>
|
||||||
/// <param name="obj"></param>
|
/// <param name="obj"></param>
|
||||||
/// <returns></returns>
|
/// <returns></returns>
|
||||||
public static T DeepCopy<T>(T obj)
|
public static T? DeepCopy<T>(T? obj)
|
||||||
{
|
{
|
||||||
return Deserialize<T>(Serialize(obj, false))!;
|
if (obj is null)
|
||||||
|
{
|
||||||
|
return default;
|
||||||
|
}
|
||||||
|
return Deserialize<T>(Serialize(obj, false));
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
@@ -67,7 +71,7 @@ public class JsonUtils
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
/// <param name="strJson"></param>
|
/// <param name="strJson"></param>
|
||||||
/// <returns></returns>
|
/// <returns></returns>
|
||||||
public static JsonNode? ParseJson(string strJson)
|
public static JsonNode? ParseJson(string? strJson)
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
@@ -116,7 +120,7 @@ public class JsonUtils
|
|||||||
/// <param name="obj"></param>
|
/// <param name="obj"></param>
|
||||||
/// <param name="options"></param>
|
/// <param name="options"></param>
|
||||||
/// <returns></returns>
|
/// <returns></returns>
|
||||||
public static string Serialize(object? obj, JsonSerializerOptions options)
|
public static string Serialize(object? obj, JsonSerializerOptions? options)
|
||||||
{
|
{
|
||||||
var result = string.Empty;
|
var result = string.Empty;
|
||||||
try
|
try
|
||||||
@@ -125,7 +129,7 @@ public class JsonUtils
|
|||||||
{
|
{
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
result = JsonSerializer.Serialize(obj, options);
|
result = JsonSerializer.Serialize(obj, options ?? _defaultSerializeOptions);
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -9,7 +9,7 @@ public class Utils
|
|||||||
{
|
{
|
||||||
private static readonly string _tag = "Utils";
|
private static readonly string _tag = "Utils";
|
||||||
|
|
||||||
#region 转换函数
|
#region Conversion Functions
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Convert to comma-separated string
|
/// Convert to comma-separated string
|
||||||
@@ -306,7 +306,10 @@ public class Utils
|
|||||||
public static bool IsBase64String(string? plainText)
|
public static bool IsBase64String(string? plainText)
|
||||||
{
|
{
|
||||||
if (plainText.IsNullOrEmpty())
|
if (plainText.IsNullOrEmpty())
|
||||||
|
{
|
||||||
return false;
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
var buffer = new Span<byte>(new byte[plainText.Length]);
|
var buffer = new Span<byte>(new byte[plainText.Length]);
|
||||||
return Convert.TryFromBase64String(plainText, buffer, out var _);
|
return Convert.TryFromBase64String(plainText, buffer, out var _);
|
||||||
}
|
}
|
||||||
@@ -422,9 +425,9 @@ public class Utils
|
|||||||
var domain = authority;
|
var domain = authority;
|
||||||
|
|
||||||
// Handle IPv6 addresses, e.g., "[2001:db8::1]:443"
|
// Handle IPv6 addresses, e.g., "[2001:db8::1]:443"
|
||||||
if (authority.StartsWith("[") && authority.Contains("]"))
|
if (authority.StartsWith('[') && authority.Contains(']'))
|
||||||
{
|
{
|
||||||
int closingBracketIndex = authority.LastIndexOf(']');
|
var closingBracketIndex = authority.LastIndexOf(']');
|
||||||
if (closingBracketIndex < authority.Length - 1 && authority[closingBracketIndex + 1] == ':')
|
if (closingBracketIndex < authority.Length - 1 && authority[closingBracketIndex + 1] == ':')
|
||||||
{
|
{
|
||||||
// Port exists
|
// Port exists
|
||||||
@@ -459,9 +462,9 @@ public class Utils
|
|||||||
return (domain, port);
|
return (domain, port);
|
||||||
}
|
}
|
||||||
|
|
||||||
#endregion 转换函数
|
#endregion Conversion Functions
|
||||||
|
|
||||||
#region 数据检查
|
#region Data Checks
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Determine if the input is a number
|
/// Determine if the input is a number
|
||||||
@@ -520,40 +523,62 @@ public class Utils
|
|||||||
{
|
{
|
||||||
// Loopback address check (127.0.0.1 for IPv4, ::1 for IPv6)
|
// Loopback address check (127.0.0.1 for IPv4, ::1 for IPv6)
|
||||||
if (IPAddress.IsLoopback(address))
|
if (IPAddress.IsLoopback(address))
|
||||||
|
{
|
||||||
return true;
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
var ipBytes = address.GetAddressBytes();
|
var ipBytes = address.GetAddressBytes();
|
||||||
if (address.AddressFamily == AddressFamily.InterNetwork)
|
if (address.AddressFamily == AddressFamily.InterNetwork)
|
||||||
{
|
{
|
||||||
// IPv4 private address check
|
// IPv4 private address check
|
||||||
if (ipBytes[0] == 10)
|
if (ipBytes[0] == 10)
|
||||||
|
{
|
||||||
return true;
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
if (ipBytes[0] == 172 && ipBytes[1] >= 16 && ipBytes[1] <= 31)
|
if (ipBytes[0] == 172 && ipBytes[1] >= 16 && ipBytes[1] <= 31)
|
||||||
|
{
|
||||||
return true;
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
if (ipBytes[0] == 192 && ipBytes[1] == 168)
|
if (ipBytes[0] == 192 && ipBytes[1] == 168)
|
||||||
|
{
|
||||||
return true;
|
return true;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
else if (address.AddressFamily == AddressFamily.InterNetworkV6)
|
else if (address.AddressFamily == AddressFamily.InterNetworkV6)
|
||||||
{
|
{
|
||||||
// IPv6 private address check
|
// IPv6 private address check
|
||||||
// Link-local address fe80::/10
|
// Link-local address fe80::/10
|
||||||
if (ipBytes[0] == 0xfe && (ipBytes[1] & 0xc0) == 0x80)
|
if (ipBytes[0] == 0xfe && (ipBytes[1] & 0xc0) == 0x80)
|
||||||
|
{
|
||||||
return true;
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
// Unique local address fc00::/7 (typically fd00::/8)
|
// Unique local address fc00::/7 (typically fd00::/8)
|
||||||
if ((ipBytes[0] & 0xfe) == 0xfc)
|
if ((ipBytes[0] & 0xfe) == 0xfc)
|
||||||
|
{
|
||||||
return true;
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
// Private portion in IPv4-mapped addresses ::ffff:0:0/96
|
// Private portion in IPv4-mapped addresses ::ffff:0:0/96
|
||||||
if (address.IsIPv4MappedToIPv6)
|
if (address.IsIPv4MappedToIPv6)
|
||||||
{
|
{
|
||||||
var ipv4Bytes = ipBytes.Skip(12).ToArray();
|
var ipv4Bytes = ipBytes.Skip(12).ToArray();
|
||||||
if (ipv4Bytes[0] == 10)
|
if (ipv4Bytes[0] == 10)
|
||||||
|
{
|
||||||
return true;
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
if (ipv4Bytes[0] == 172 && ipv4Bytes[1] >= 16 && ipv4Bytes[1] <= 31)
|
if (ipv4Bytes[0] == 172 && ipv4Bytes[1] >= 16 && ipv4Bytes[1] <= 31)
|
||||||
|
{
|
||||||
return true;
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
if (ipv4Bytes[0] == 192 && ipv4Bytes[1] == 168)
|
if (ipv4Bytes[0] == 192 && ipv4Bytes[1] == 168)
|
||||||
|
{
|
||||||
return true;
|
return true;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -561,9 +586,9 @@ public class Utils
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
#endregion 数据检查
|
#endregion Data Checks
|
||||||
|
|
||||||
#region 测速
|
#region Speed Test
|
||||||
|
|
||||||
private static bool PortInUse(int port)
|
private static bool PortInUse(int port)
|
||||||
{
|
{
|
||||||
@@ -616,9 +641,9 @@ public class Utils
|
|||||||
return 59090;
|
return 59090;
|
||||||
}
|
}
|
||||||
|
|
||||||
#endregion 测速
|
#endregion Speed Test
|
||||||
|
|
||||||
#region 杂项
|
#region Miscellaneous
|
||||||
|
|
||||||
public static bool UpgradeAppExists(out string upgradeFileName)
|
public static bool UpgradeAppExists(out string upgradeFileName)
|
||||||
{
|
{
|
||||||
@@ -708,10 +733,16 @@ public class Utils
|
|||||||
foreach (var host in hostsList)
|
foreach (var host in hostsList)
|
||||||
{
|
{
|
||||||
if (host.StartsWith("#"))
|
if (host.StartsWith("#"))
|
||||||
|
{
|
||||||
continue;
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
var hostItem = host.Split(new[] { ' ', '\t' }, StringSplitOptions.RemoveEmptyEntries);
|
var hostItem = host.Split(new[] { ' ', '\t' }, StringSplitOptions.RemoveEmptyEntries);
|
||||||
if (hostItem.Length < 2)
|
if (hostItem.Length < 2)
|
||||||
|
{
|
||||||
continue;
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
systemHosts.Add(hostItem[1], hostItem[0]);
|
systemHosts.Add(hostItem[1], hostItem[0]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -762,7 +793,7 @@ public class Utils
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
#endregion 杂项
|
#endregion Miscellaneous
|
||||||
|
|
||||||
#region TempPath
|
#region TempPath
|
||||||
|
|
||||||
@@ -963,13 +994,13 @@ public class Utils
|
|||||||
|
|
||||||
#region Platform
|
#region Platform
|
||||||
|
|
||||||
public static bool IsWindows() => RuntimeInformation.IsOSPlatform(OSPlatform.Windows);
|
public static bool IsWindows() => OperatingSystem.IsWindows();
|
||||||
|
|
||||||
public static bool IsLinux() => RuntimeInformation.IsOSPlatform(OSPlatform.Linux);
|
public static bool IsLinux() => OperatingSystem.IsLinux();
|
||||||
|
|
||||||
public static bool IsOSX() => RuntimeInformation.IsOSPlatform(OSPlatform.OSX);
|
public static bool IsMacOS() => OperatingSystem.IsMacOS();
|
||||||
|
|
||||||
public static bool IsNonWindows() => !RuntimeInformation.IsOSPlatform(OSPlatform.Windows);
|
public static bool IsNonWindows() => !OperatingSystem.IsWindows();
|
||||||
|
|
||||||
public static string GetExeName(string name)
|
public static string GetExeName(string name)
|
||||||
{
|
{
|
||||||
@@ -989,16 +1020,11 @@ public class Utils
|
|||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
if (IsWindows() || IsOSX())
|
if (IsWindows() || IsMacOS())
|
||||||
{
|
{
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!string.IsNullOrEmpty(Environment.GetEnvironmentVariable("APPIMAGE")))
|
|
||||||
{
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
var exePath = GetExePath();
|
var exePath = GetExePath();
|
||||||
var baseDir = string.IsNullOrEmpty(exePath) ? StartupPath() : Path.GetDirectoryName(exePath) ?? "";
|
var baseDir = string.IsNullOrEmpty(exePath) ? StartupPath() : Path.GetDirectoryName(exePath) ?? "";
|
||||||
var p = baseDir.Replace('\\', '/');
|
var p = baseDir.Replace('\\', '/');
|
||||||
@@ -1008,11 +1034,6 @@ public class Utils
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (p.Contains("/.mount_", StringComparison.Ordinal))
|
|
||||||
{
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (p.StartsWith("/opt/v2rayN", StringComparison.OrdinalIgnoreCase))
|
if (p.StartsWith("/opt/v2rayN", StringComparison.OrdinalIgnoreCase))
|
||||||
{
|
{
|
||||||
return true;
|
return true;
|
||||||
|
|||||||
@@ -1,180 +0,0 @@
|
|||||||
using System.Diagnostics;
|
|
||||||
using System.Runtime.InteropServices;
|
|
||||||
|
|
||||||
namespace ServiceLib.Common;
|
|
||||||
/*
|
|
||||||
* See:
|
|
||||||
* http://stackoverflow.com/questions/6266820/working-example-of-createjobobject-setinformationjobobject-pinvoke-in-net
|
|
||||||
*/
|
|
||||||
|
|
||||||
public sealed class WindowsJob : IDisposable
|
|
||||||
{
|
|
||||||
private IntPtr handle = IntPtr.Zero;
|
|
||||||
|
|
||||||
public WindowsJob()
|
|
||||||
{
|
|
||||||
handle = CreateJobObject(IntPtr.Zero, null);
|
|
||||||
var extendedInfoPtr = IntPtr.Zero;
|
|
||||||
var info = new JOBOBJECT_BASIC_LIMIT_INFORMATION
|
|
||||||
{
|
|
||||||
LimitFlags = 0x2000
|
|
||||||
};
|
|
||||||
|
|
||||||
var extendedInfo = new JOBOBJECT_EXTENDED_LIMIT_INFORMATION
|
|
||||||
{
|
|
||||||
BasicLimitInformation = info
|
|
||||||
};
|
|
||||||
|
|
||||||
try
|
|
||||||
{
|
|
||||||
var length = Marshal.SizeOf(typeof(JOBOBJECT_EXTENDED_LIMIT_INFORMATION));
|
|
||||||
extendedInfoPtr = Marshal.AllocHGlobal(length);
|
|
||||||
Marshal.StructureToPtr(extendedInfo, extendedInfoPtr, false);
|
|
||||||
|
|
||||||
if (!SetInformationJobObject(handle, JobObjectInfoType.ExtendedLimitInformation, extendedInfoPtr,
|
|
||||||
(uint)length))
|
|
||||||
{
|
|
||||||
throw new Exception(string.Format("Unable to set information. Error: {0}",
|
|
||||||
Marshal.GetLastWin32Error()));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
finally
|
|
||||||
{
|
|
||||||
if (extendedInfoPtr != IntPtr.Zero)
|
|
||||||
{
|
|
||||||
Marshal.FreeHGlobal(extendedInfoPtr);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool AddProcess(IntPtr processHandle)
|
|
||||||
{
|
|
||||||
var succ = AssignProcessToJobObject(handle, processHandle);
|
|
||||||
|
|
||||||
if (!succ)
|
|
||||||
{
|
|
||||||
Logging.SaveLog("Failed to call AssignProcessToJobObject! GetLastError=" + Marshal.GetLastWin32Error());
|
|
||||||
}
|
|
||||||
|
|
||||||
return succ;
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool AddProcess(int processId)
|
|
||||||
{
|
|
||||||
return AddProcess(Process.GetProcessById(processId).Handle);
|
|
||||||
}
|
|
||||||
|
|
||||||
#region IDisposable
|
|
||||||
|
|
||||||
private bool disposed;
|
|
||||||
|
|
||||||
public void Dispose()
|
|
||||||
{
|
|
||||||
Dispose(true);
|
|
||||||
GC.SuppressFinalize(this);
|
|
||||||
}
|
|
||||||
|
|
||||||
private void Dispose(bool disposing)
|
|
||||||
{
|
|
||||||
if (disposed)
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
disposed = true;
|
|
||||||
|
|
||||||
if (disposing)
|
|
||||||
{
|
|
||||||
// no managed objects to free
|
|
||||||
}
|
|
||||||
|
|
||||||
if (handle != IntPtr.Zero)
|
|
||||||
{
|
|
||||||
CloseHandle(handle);
|
|
||||||
handle = IntPtr.Zero;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
~WindowsJob()
|
|
||||||
{
|
|
||||||
Dispose(false);
|
|
||||||
}
|
|
||||||
|
|
||||||
#endregion IDisposable
|
|
||||||
|
|
||||||
#region Interop
|
|
||||||
|
|
||||||
[DllImport("kernel32.dll", CharSet = CharSet.Unicode)]
|
|
||||||
private static extern IntPtr CreateJobObject(IntPtr a, string? lpName);
|
|
||||||
|
|
||||||
[DllImport("kernel32.dll", SetLastError = true)]
|
|
||||||
private static extern bool SetInformationJobObject(IntPtr hJob, JobObjectInfoType infoType, IntPtr lpJobObjectInfo, uint cbJobObjectInfoLength);
|
|
||||||
|
|
||||||
[DllImport("kernel32.dll", SetLastError = true)]
|
|
||||||
private static extern bool AssignProcessToJobObject(IntPtr job, IntPtr process);
|
|
||||||
|
|
||||||
[DllImport("kernel32.dll", SetLastError = true)]
|
|
||||||
[return: MarshalAs(UnmanagedType.Bool)]
|
|
||||||
private static extern bool CloseHandle(IntPtr hObject);
|
|
||||||
|
|
||||||
#endregion Interop
|
|
||||||
}
|
|
||||||
|
|
||||||
#region Helper classes
|
|
||||||
|
|
||||||
[StructLayout(LayoutKind.Sequential)]
|
|
||||||
internal struct IO_COUNTERS
|
|
||||||
{
|
|
||||||
public ulong ReadOperationCount;
|
|
||||||
public ulong WriteOperationCount;
|
|
||||||
public ulong OtherOperationCount;
|
|
||||||
public ulong ReadTransferCount;
|
|
||||||
public ulong WriteTransferCount;
|
|
||||||
public ulong OtherTransferCount;
|
|
||||||
}
|
|
||||||
|
|
||||||
[StructLayout(LayoutKind.Sequential)]
|
|
||||||
internal struct JOBOBJECT_BASIC_LIMIT_INFORMATION
|
|
||||||
{
|
|
||||||
public long PerProcessUserTimeLimit;
|
|
||||||
public long PerJobUserTimeLimit;
|
|
||||||
public uint LimitFlags;
|
|
||||||
public UIntPtr MinimumWorkingSetSize;
|
|
||||||
public UIntPtr MaximumWorkingSetSize;
|
|
||||||
public uint ActiveProcessLimit;
|
|
||||||
public UIntPtr Affinity;
|
|
||||||
public uint PriorityClass;
|
|
||||||
public uint SchedulingClass;
|
|
||||||
}
|
|
||||||
|
|
||||||
[StructLayout(LayoutKind.Sequential)]
|
|
||||||
public struct SECURITY_ATTRIBUTES
|
|
||||||
{
|
|
||||||
public uint nLength;
|
|
||||||
public IntPtr lpSecurityDescriptor;
|
|
||||||
public int bInheritHandle;
|
|
||||||
}
|
|
||||||
|
|
||||||
[StructLayout(LayoutKind.Sequential)]
|
|
||||||
internal struct JOBOBJECT_EXTENDED_LIMIT_INFORMATION
|
|
||||||
{
|
|
||||||
public JOBOBJECT_BASIC_LIMIT_INFORMATION BasicLimitInformation;
|
|
||||||
public IO_COUNTERS IoInfo;
|
|
||||||
public UIntPtr ProcessMemoryLimit;
|
|
||||||
public UIntPtr JobMemoryLimit;
|
|
||||||
public UIntPtr PeakProcessMemoryUsed;
|
|
||||||
public UIntPtr PeakJobMemoryUsed;
|
|
||||||
}
|
|
||||||
|
|
||||||
public enum JobObjectInfoType
|
|
||||||
{
|
|
||||||
AssociateCompletionPortInformation = 7,
|
|
||||||
BasicLimitInformation = 2,
|
|
||||||
BasicUIRestrictions = 4,
|
|
||||||
EndOfJobTimeInformation = 6,
|
|
||||||
ExtendedLimitInformation = 9,
|
|
||||||
SecurityLimitInformation = 5,
|
|
||||||
GroupInformation = 11
|
|
||||||
}
|
|
||||||
|
|
||||||
#endregion Helper classes
|
|
||||||
|
|
||||||
@@ -5,5 +5,6 @@ public enum ESpeedActionType
|
|||||||
Tcping,
|
Tcping,
|
||||||
Realping,
|
Realping,
|
||||||
Speedtest,
|
Speedtest,
|
||||||
Mixedtest
|
Mixedtest,
|
||||||
|
FastRealping
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -73,6 +73,7 @@ public class Global
|
|||||||
public const string GrpcMultiMode = "multi";
|
public const string GrpcMultiMode = "multi";
|
||||||
public const int MaxPort = 65536;
|
public const int MaxPort = 65536;
|
||||||
public const int MinFontSize = 8;
|
public const int MinFontSize = 8;
|
||||||
|
public const int MinFontSizeCount = 13;
|
||||||
public const string RebootAs = "rebootas";
|
public const string RebootAs = "rebootas";
|
||||||
public const string AvaAssets = "avares://v2rayN/Assets/";
|
public const string AvaAssets = "avares://v2rayN/Assets/";
|
||||||
public const string LocalAppData = "V2RAYN_LOCAL_APPLICATION_DATA_V2";
|
public const string LocalAppData = "V2RAYN_LOCAL_APPLICATION_DATA_V2";
|
||||||
@@ -427,6 +428,7 @@ public class Global
|
|||||||
"zh-Hant",
|
"zh-Hant",
|
||||||
"en",
|
"en",
|
||||||
"fa-Ir",
|
"fa-Ir",
|
||||||
|
"fr",
|
||||||
"ru",
|
"ru",
|
||||||
"hu"
|
"hu"
|
||||||
];
|
];
|
||||||
@@ -584,7 +586,6 @@ public class Global
|
|||||||
|
|
||||||
public static readonly List<string> IPAPIUrls =
|
public static readonly List<string> IPAPIUrls =
|
||||||
[
|
[
|
||||||
@"https://speed.cloudflare.com/meta",
|
|
||||||
@"https://api.ip.sb/geoip",
|
@"https://api.ip.sb/geoip",
|
||||||
@"https://api-ipv4.ip.sb/geoip",
|
@"https://api-ipv4.ip.sb/geoip",
|
||||||
@"https://api-ipv6.ip.sb/geoip",
|
@"https://api-ipv6.ip.sb/geoip",
|
||||||
|
|||||||
@@ -26,7 +26,7 @@ public static class AutoStartupHandler
|
|||||||
await SetTaskLinux();
|
await SetTaskLinux();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if (Utils.IsOSX())
|
else if (Utils.IsMacOS())
|
||||||
{
|
{
|
||||||
await ClearTaskOSX();
|
await ClearTaskOSX();
|
||||||
|
|
||||||
|
|||||||
@@ -97,7 +97,7 @@ public static class ConfigHandler
|
|||||||
|
|
||||||
config.UiItem ??= new UIItem()
|
config.UiItem ??= new UIItem()
|
||||||
{
|
{
|
||||||
EnableAutoAdjustMainLvColWidth = true
|
EnableUpdateSubOnlyRemarksExist = true
|
||||||
};
|
};
|
||||||
config.UiItem.MainColumnItem ??= new();
|
config.UiItem.MainColumnItem ??= new();
|
||||||
config.UiItem.WindowSizeItem ??= new();
|
config.UiItem.WindowSizeItem ??= new();
|
||||||
@@ -252,6 +252,7 @@ public static class ConfigHandler
|
|||||||
item.Mldsa65Verify = profileItem.Mldsa65Verify;
|
item.Mldsa65Verify = profileItem.Mldsa65Verify;
|
||||||
item.Extra = profileItem.Extra;
|
item.Extra = profileItem.Extra;
|
||||||
item.MuxEnabled = profileItem.MuxEnabled;
|
item.MuxEnabled = profileItem.MuxEnabled;
|
||||||
|
item.Cert = profileItem.Cert;
|
||||||
}
|
}
|
||||||
|
|
||||||
var ret = item.ConfigType switch
|
var ret = item.ConfigType switch
|
||||||
@@ -447,13 +448,13 @@ public static class ConfigHandler
|
|||||||
/// <returns>0 if successful, -1 if failed</returns>
|
/// <returns>0 if successful, -1 if failed</returns>
|
||||||
public static async Task<int> MoveServer(Config config, List<ProfileItem> lstProfile, int index, EMove eMove, int pos = -1)
|
public static async Task<int> MoveServer(Config config, List<ProfileItem> lstProfile, int index, EMove eMove, int pos = -1)
|
||||||
{
|
{
|
||||||
int count = lstProfile.Count;
|
var count = lstProfile.Count;
|
||||||
if (index < 0 || index > lstProfile.Count - 1)
|
if (index < 0 || index > lstProfile.Count - 1)
|
||||||
{
|
{
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
for (int i = 0; i < lstProfile.Count; i++)
|
for (var i = 0; i < lstProfile.Count; i++)
|
||||||
{
|
{
|
||||||
ProfileExManager.Instance.SetSort(lstProfile[i].IndexId, (i + 1) * 10);
|
ProfileExManager.Instance.SetSort(lstProfile[i].IndexId, (i + 1) * 10);
|
||||||
}
|
}
|
||||||
@@ -527,7 +528,7 @@ public static class ConfigHandler
|
|||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
var ext = Path.GetExtension(fileName);
|
var ext = Path.GetExtension(fileName);
|
||||||
string newFileName = $"{Utils.GetGuid()}{ext}";
|
var newFileName = $"{Utils.GetGuid()}{ext}";
|
||||||
//newFileName = Path.Combine(Utile.GetTempPath(), newFileName);
|
//newFileName = Path.Combine(Utile.GetTempPath(), newFileName);
|
||||||
|
|
||||||
try
|
try
|
||||||
@@ -1221,6 +1222,7 @@ public static class ConfigHandler
|
|||||||
CoreType = coreType,
|
CoreType = coreType,
|
||||||
ConfigType = EConfigType.PolicyGroup,
|
ConfigType = EConfigType.PolicyGroup,
|
||||||
Remarks = remark,
|
Remarks = remark,
|
||||||
|
IsSub = false
|
||||||
};
|
};
|
||||||
if (!subId.IsNullOrEmpty())
|
if (!subId.IsNullOrEmpty())
|
||||||
{
|
{
|
||||||
@@ -1355,7 +1357,7 @@ public static class ConfigHandler
|
|||||||
}
|
}
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
var profileItem = FmtHandler.ResolveConfig(str, out string msg);
|
var profileItem = FmtHandler.ResolveConfig(str, out var msg);
|
||||||
if (profileItem is null)
|
if (profileItem is null)
|
||||||
{
|
{
|
||||||
continue;
|
continue;
|
||||||
@@ -1439,7 +1441,7 @@ public static class ConfigHandler
|
|||||||
{
|
{
|
||||||
await RemoveServersViaSubid(config, subid, isSub);
|
await RemoveServersViaSubid(config, subid, isSub);
|
||||||
}
|
}
|
||||||
int count = 0;
|
var count = 0;
|
||||||
foreach (var it in lstProfiles)
|
foreach (var it in lstProfiles)
|
||||||
{
|
{
|
||||||
it.Subid = subid;
|
it.Subid = subid;
|
||||||
@@ -1529,7 +1531,7 @@ public static class ConfigHandler
|
|||||||
var lstSsServer = ShadowsocksFmt.ResolveSip008(strData);
|
var lstSsServer = ShadowsocksFmt.ResolveSip008(strData);
|
||||||
if (lstSsServer?.Count > 0)
|
if (lstSsServer?.Count > 0)
|
||||||
{
|
{
|
||||||
int counter = 0;
|
var counter = 0;
|
||||||
foreach (var ssItem in lstSsServer)
|
foreach (var ssItem in lstSsServer)
|
||||||
{
|
{
|
||||||
ssItem.Subid = subid;
|
ssItem.Subid = subid;
|
||||||
@@ -1649,7 +1651,9 @@ public static class ConfigHandler
|
|||||||
|
|
||||||
var uri = Utils.TryUri(url);
|
var uri = Utils.TryUri(url);
|
||||||
if (uri == null)
|
if (uri == null)
|
||||||
|
{
|
||||||
return -1;
|
return -1;
|
||||||
|
}
|
||||||
//Do not allow http protocol
|
//Do not allow http protocol
|
||||||
if (url.StartsWith(Global.HttpProtocol) && !Utils.IsPrivateNetwork(uri.IdnHost))
|
if (url.StartsWith(Global.HttpProtocol) && !Utils.IsPrivateNetwork(uri.IdnHost))
|
||||||
{
|
{
|
||||||
@@ -1704,7 +1708,7 @@ public static class ConfigHandler
|
|||||||
var maxSort = 0;
|
var maxSort = 0;
|
||||||
if (await SQLiteHelper.Instance.TableAsync<SubItem>().CountAsync() > 0)
|
if (await SQLiteHelper.Instance.TableAsync<SubItem>().CountAsync() > 0)
|
||||||
{
|
{
|
||||||
var lstSubs = (await AppManager.Instance.SubItems());
|
var lstSubs = await AppManager.Instance.SubItems();
|
||||||
maxSort = lstSubs.LastOrDefault()?.Sort ?? 0;
|
maxSort = lstSubs.LastOrDefault()?.Sort ?? 0;
|
||||||
}
|
}
|
||||||
item.Sort = maxSort + 1;
|
item.Sort = maxSort + 1;
|
||||||
@@ -1866,7 +1870,7 @@ public static class ConfigHandler
|
|||||||
/// <returns>0 if successful, -1 if failed</returns>
|
/// <returns>0 if successful, -1 if failed</returns>
|
||||||
public static async Task<int> MoveRoutingRule(List<RulesItem> rules, int index, EMove eMove, int pos = -1)
|
public static async Task<int> MoveRoutingRule(List<RulesItem> rules, int index, EMove eMove, int pos = -1)
|
||||||
{
|
{
|
||||||
int count = rules.Count;
|
var count = rules.Count;
|
||||||
if (index < 0 || index > rules.Count - 1)
|
if (index < 0 || index > rules.Count - 1)
|
||||||
{
|
{
|
||||||
return -1;
|
return -1;
|
||||||
@@ -2016,11 +2020,15 @@ public static class ConfigHandler
|
|||||||
var downloadHandle = new DownloadService();
|
var downloadHandle = new DownloadService();
|
||||||
var templateContent = await downloadHandle.TryDownloadString(config.ConstItem.RouteRulesTemplateSourceUrl, true, "");
|
var templateContent = await downloadHandle.TryDownloadString(config.ConstItem.RouteRulesTemplateSourceUrl, true, "");
|
||||||
if (templateContent.IsNullOrEmpty())
|
if (templateContent.IsNullOrEmpty())
|
||||||
|
{
|
||||||
return await InitBuiltinRouting(config, blImportAdvancedRules); // fallback
|
return await InitBuiltinRouting(config, blImportAdvancedRules); // fallback
|
||||||
|
}
|
||||||
|
|
||||||
var template = JsonUtils.Deserialize<RoutingTemplate>(templateContent);
|
var template = JsonUtils.Deserialize<RoutingTemplate>(templateContent);
|
||||||
if (template == null)
|
if (template == null)
|
||||||
|
{
|
||||||
return await InitBuiltinRouting(config, blImportAdvancedRules); // fallback
|
return await InitBuiltinRouting(config, blImportAdvancedRules); // fallback
|
||||||
|
}
|
||||||
|
|
||||||
var items = await AppManager.Instance.RoutingItems();
|
var items = await AppManager.Instance.RoutingItems();
|
||||||
var maxSort = items.Count;
|
var maxSort = items.Count;
|
||||||
@@ -2033,14 +2041,18 @@ public static class ConfigHandler
|
|||||||
var item = template.RoutingItems[i];
|
var item = template.RoutingItems[i];
|
||||||
|
|
||||||
if (item.Url.IsNullOrEmpty() && item.RuleSet.IsNullOrEmpty())
|
if (item.Url.IsNullOrEmpty() && item.RuleSet.IsNullOrEmpty())
|
||||||
|
{
|
||||||
continue;
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
var ruleSetsString = !item.RuleSet.IsNullOrEmpty()
|
var ruleSetsString = !item.RuleSet.IsNullOrEmpty()
|
||||||
? item.RuleSet
|
? item.RuleSet
|
||||||
: await downloadHandle.TryDownloadString(item.Url, true, "");
|
: await downloadHandle.TryDownloadString(item.Url, true, "");
|
||||||
|
|
||||||
if (ruleSetsString.IsNullOrEmpty())
|
if (ruleSetsString.IsNullOrEmpty())
|
||||||
|
{
|
||||||
continue;
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
item.Remarks = $"{template.Version}-{item.Remarks}";
|
item.Remarks = $"{template.Version}-{item.Remarks}";
|
||||||
item.Enabled = true;
|
item.Enabled = true;
|
||||||
@@ -2068,7 +2080,7 @@ public static class ConfigHandler
|
|||||||
/// <returns>0 if successful</returns>
|
/// <returns>0 if successful</returns>
|
||||||
public static async Task<int> InitBuiltinRouting(Config config, bool blImportAdvancedRules = false)
|
public static async Task<int> InitBuiltinRouting(Config config, bool blImportAdvancedRules = false)
|
||||||
{
|
{
|
||||||
var ver = "V3-";
|
var ver = "V4-";
|
||||||
var items = await AppManager.Instance.RoutingItems();
|
var items = await AppManager.Instance.RoutingItems();
|
||||||
|
|
||||||
//TODO Temporary code to be removed later
|
//TODO Temporary code to be removed later
|
||||||
@@ -2079,7 +2091,7 @@ public static class ConfigHandler
|
|||||||
items = await AppManager.Instance.RoutingItems();
|
items = await AppManager.Instance.RoutingItems();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!blImportAdvancedRules && items.Count > 0)
|
if (!blImportAdvancedRules && items.Count(u => u.Remarks.StartsWith(ver)) > 0)
|
||||||
{
|
{
|
||||||
//migrate
|
//migrate
|
||||||
//TODO Temporary code to be removed later
|
//TODO Temporary code to be removed later
|
||||||
@@ -2236,17 +2248,25 @@ public static class ConfigHandler
|
|||||||
var downloadHandle = new DownloadService();
|
var downloadHandle = new DownloadService();
|
||||||
var templateContent = await downloadHandle.TryDownloadString(url, true, "");
|
var templateContent = await downloadHandle.TryDownloadString(url, true, "");
|
||||||
if (templateContent.IsNullOrEmpty())
|
if (templateContent.IsNullOrEmpty())
|
||||||
|
{
|
||||||
return currentItem;
|
return currentItem;
|
||||||
|
}
|
||||||
|
|
||||||
var template = JsonUtils.Deserialize<DNSItem>(templateContent);
|
var template = JsonUtils.Deserialize<DNSItem>(templateContent);
|
||||||
if (template == null)
|
if (template == null)
|
||||||
|
{
|
||||||
return currentItem;
|
return currentItem;
|
||||||
|
}
|
||||||
|
|
||||||
if (!template.NormalDNS.IsNullOrEmpty())
|
if (!template.NormalDNS.IsNullOrEmpty())
|
||||||
|
{
|
||||||
template.NormalDNS = await downloadHandle.TryDownloadString(template.NormalDNS, true, "");
|
template.NormalDNS = await downloadHandle.TryDownloadString(template.NormalDNS, true, "");
|
||||||
|
}
|
||||||
|
|
||||||
if (!template.TunDNS.IsNullOrEmpty())
|
if (!template.TunDNS.IsNullOrEmpty())
|
||||||
|
{
|
||||||
template.TunDNS = await downloadHandle.TryDownloadString(template.TunDNS, true, "");
|
template.TunDNS = await downloadHandle.TryDownloadString(template.TunDNS, true, "");
|
||||||
|
}
|
||||||
|
|
||||||
template.Id = currentItem.Id;
|
template.Id = currentItem.Id;
|
||||||
template.Enabled = currentItem.Enabled;
|
template.Enabled = currentItem.Enabled;
|
||||||
@@ -2280,10 +2300,16 @@ public static class ConfigHandler
|
|||||||
var downloadHandle = new DownloadService();
|
var downloadHandle = new DownloadService();
|
||||||
var templateContent = await downloadHandle.TryDownloadString(url, true, "");
|
var templateContent = await downloadHandle.TryDownloadString(url, true, "");
|
||||||
if (templateContent.IsNullOrEmpty())
|
if (templateContent.IsNullOrEmpty())
|
||||||
|
{
|
||||||
return null;
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
var template = JsonUtils.Deserialize<SimpleDNSItem>(templateContent);
|
var template = JsonUtils.Deserialize<SimpleDNSItem>(templateContent);
|
||||||
if (template == null)
|
if (template == null)
|
||||||
|
{
|
||||||
return null;
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
return template;
|
return template;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -6,7 +6,7 @@ public static class ConnectionHandler
|
|||||||
|
|
||||||
public static async Task<string> RunAvailabilityCheck()
|
public static async Task<string> RunAvailabilityCheck()
|
||||||
{
|
{
|
||||||
var time = await GetRealPingTime();
|
var time = await GetRealPingTimeInfo();
|
||||||
var ip = time > 0 ? await GetIPInfo() ?? Global.None : Global.None;
|
var ip = time > 0 ? await GetIPInfo() ?? Global.None : Global.None;
|
||||||
|
|
||||||
return string.Format(ResUI.TestMeOutput, time, ip);
|
return string.Format(ResUI.TestMeOutput, time, ip);
|
||||||
@@ -39,7 +39,7 @@ public static class ConnectionHandler
|
|||||||
return $"({country ?? "unknown"}) {ip}";
|
return $"({country ?? "unknown"}) {ip}";
|
||||||
}
|
}
|
||||||
|
|
||||||
private static async Task<int> GetRealPingTime()
|
private static async Task<int> GetRealPingTimeInfo()
|
||||||
{
|
{
|
||||||
var responseTime = -1;
|
var responseTime = -1;
|
||||||
try
|
try
|
||||||
@@ -50,7 +50,7 @@ public static class ConnectionHandler
|
|||||||
|
|
||||||
for (var i = 0; i < 2; i++)
|
for (var i = 0; i < 2; i++)
|
||||||
{
|
{
|
||||||
responseTime = await HttpClientHelper.Instance.GetRealPingTime(url, webProxy, 10);
|
responseTime = await GetRealPingTime(url, webProxy, 10);
|
||||||
if (responseTime > 0)
|
if (responseTime > 0)
|
||||||
{
|
{
|
||||||
break;
|
break;
|
||||||
@@ -65,4 +65,34 @@ public static class ConnectionHandler
|
|||||||
}
|
}
|
||||||
return responseTime;
|
return responseTime;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static async Task<int> GetRealPingTime(string url, IWebProxy? webProxy, int downloadTimeout)
|
||||||
|
{
|
||||||
|
var responseTime = -1;
|
||||||
|
try
|
||||||
|
{
|
||||||
|
using var cts = new CancellationTokenSource();
|
||||||
|
cts.CancelAfter(TimeSpan.FromSeconds(downloadTimeout));
|
||||||
|
using var client = new HttpClient(new SocketsHttpHandler()
|
||||||
|
{
|
||||||
|
Proxy = webProxy,
|
||||||
|
UseProxy = webProxy != null
|
||||||
|
});
|
||||||
|
|
||||||
|
List<int> oneTime = new();
|
||||||
|
for (var i = 0; i < 2; i++)
|
||||||
|
{
|
||||||
|
var timer = Stopwatch.StartNew();
|
||||||
|
await client.GetAsync(url, cts.Token).ConfigureAwait(false);
|
||||||
|
timer.Stop();
|
||||||
|
oneTime.Add((int)timer.Elapsed.TotalMilliseconds);
|
||||||
|
await Task.Delay(100);
|
||||||
|
}
|
||||||
|
responseTime = oneTime.Where(x => x > 0).OrderBy(x => x).FirstOrDefault();
|
||||||
|
}
|
||||||
|
catch
|
||||||
|
{
|
||||||
|
}
|
||||||
|
return responseTime;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -58,7 +58,7 @@ public static class CoreConfigHandler
|
|||||||
File.Delete(fileName);
|
File.Delete(fileName);
|
||||||
}
|
}
|
||||||
|
|
||||||
string addressFileName = node.Address;
|
var addressFileName = node.Address;
|
||||||
if (!File.Exists(addressFileName))
|
if (!File.Exists(addressFileName))
|
||||||
{
|
{
|
||||||
addressFileName = Utils.GetConfigPath(addressFileName);
|
addressFileName = Utils.GetConfigPath(addressFileName);
|
||||||
|
|||||||
@@ -23,7 +23,7 @@ public class AnytlsFmt : BaseFmt
|
|||||||
item.Id = rawUserInfo;
|
item.Id = rawUserInfo;
|
||||||
|
|
||||||
var query = Utils.ParseQueryString(parsedUrl.Query);
|
var query = Utils.ParseQueryString(parsedUrl.Query);
|
||||||
_ = ResolveStdTransport(query, ref item);
|
ResolveUriQuery(query, ref item);
|
||||||
|
|
||||||
return item;
|
return item;
|
||||||
}
|
}
|
||||||
@@ -41,7 +41,7 @@ public class AnytlsFmt : BaseFmt
|
|||||||
}
|
}
|
||||||
var pw = item.Id;
|
var pw = item.Id;
|
||||||
var dicQuery = new Dictionary<string, string>();
|
var dicQuery = new Dictionary<string, string>();
|
||||||
_ = GetStdTransport(item, Global.None, ref dicQuery);
|
ToUriQuery(item, Global.None, ref dicQuery);
|
||||||
|
|
||||||
return ToUri(EConfigType.Anytls, item.Address, item.Port, pw, dicQuery, remark);
|
return ToUri(EConfigType.Anytls, item.Address, item.Port, pw, dicQuery, remark);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,6 +4,8 @@ namespace ServiceLib.Handler.Fmt;
|
|||||||
|
|
||||||
public class BaseFmt
|
public class BaseFmt
|
||||||
{
|
{
|
||||||
|
private static readonly string[] _allowInsecureArray = new[] { "insecure", "allowInsecure", "allow_insecure" };
|
||||||
|
|
||||||
protected static string GetIpv6(string address)
|
protected static string GetIpv6(string address)
|
||||||
{
|
{
|
||||||
if (Utils.IsIpv6(address))
|
if (Utils.IsIpv6(address))
|
||||||
@@ -17,7 +19,7 @@ public class BaseFmt
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
protected static int GetStdTransport(ProfileItem item, string? securityDef, ref Dictionary<string, string> dicQuery)
|
protected static int ToUriQuery(ProfileItem item, string? securityDef, ref Dictionary<string, string> dicQuery)
|
||||||
{
|
{
|
||||||
if (item.Flow.IsNotEmpty())
|
if (item.Flow.IsNotEmpty())
|
||||||
{
|
{
|
||||||
@@ -37,11 +39,7 @@ public class BaseFmt
|
|||||||
}
|
}
|
||||||
if (item.Sni.IsNotEmpty())
|
if (item.Sni.IsNotEmpty())
|
||||||
{
|
{
|
||||||
dicQuery.Add("sni", item.Sni);
|
dicQuery.Add("sni", Utils.UrlEncode(item.Sni));
|
||||||
}
|
|
||||||
if (item.Alpn.IsNotEmpty())
|
|
||||||
{
|
|
||||||
dicQuery.Add("alpn", Utils.UrlEncode(item.Alpn));
|
|
||||||
}
|
}
|
||||||
if (item.Fingerprint.IsNotEmpty())
|
if (item.Fingerprint.IsNotEmpty())
|
||||||
{
|
{
|
||||||
@@ -63,9 +61,14 @@ public class BaseFmt
|
|||||||
{
|
{
|
||||||
dicQuery.Add("pqv", Utils.UrlEncode(item.Mldsa65Verify));
|
dicQuery.Add("pqv", Utils.UrlEncode(item.Mldsa65Verify));
|
||||||
}
|
}
|
||||||
if (item.AllowInsecure.Equals("true"))
|
|
||||||
|
if (item.StreamSecurity.Equals(Global.StreamSecurity))
|
||||||
{
|
{
|
||||||
dicQuery.Add("allowInsecure", "1");
|
if (item.Alpn.IsNotEmpty())
|
||||||
|
{
|
||||||
|
dicQuery.Add("alpn", Utils.UrlEncode(item.Alpn));
|
||||||
|
}
|
||||||
|
ToUriQueryAllowInsecure(item, ref dicQuery);
|
||||||
}
|
}
|
||||||
|
|
||||||
dicQuery.Add("type", item.Network.IsNotEmpty() ? item.Network : nameof(ETransport.tcp));
|
dicQuery.Add("type", item.Network.IsNotEmpty() ? item.Network : nameof(ETransport.tcp));
|
||||||
@@ -115,7 +118,16 @@ public class BaseFmt
|
|||||||
}
|
}
|
||||||
if (item.Extra.IsNotEmpty())
|
if (item.Extra.IsNotEmpty())
|
||||||
{
|
{
|
||||||
dicQuery.Add("extra", Utils.UrlEncode(item.Extra));
|
var node = JsonUtils.ParseJson(item.Extra);
|
||||||
|
var extra = node != null
|
||||||
|
? JsonUtils.Serialize(node, new JsonSerializerOptions
|
||||||
|
{
|
||||||
|
WriteIndented = false,
|
||||||
|
DefaultIgnoreCondition = JsonIgnoreCondition.Never,
|
||||||
|
Encoder = JavaScriptEncoder.UnsafeRelaxedJsonEscaping
|
||||||
|
})
|
||||||
|
: item.Extra;
|
||||||
|
dicQuery.Add("extra", Utils.UrlEncode(extra));
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
@@ -153,7 +165,40 @@ public class BaseFmt
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected static int ResolveStdTransport(NameValueCollection query, ref ProfileItem item)
|
protected static int ToUriQueryLite(ProfileItem item, ref Dictionary<string, string> dicQuery)
|
||||||
|
{
|
||||||
|
if (item.Sni.IsNotEmpty())
|
||||||
|
{
|
||||||
|
dicQuery.Add("sni", Utils.UrlEncode(item.Sni));
|
||||||
|
}
|
||||||
|
if (item.Alpn.IsNotEmpty())
|
||||||
|
{
|
||||||
|
dicQuery.Add("alpn", Utils.UrlEncode(item.Alpn));
|
||||||
|
}
|
||||||
|
|
||||||
|
ToUriQueryAllowInsecure(item, ref dicQuery);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static int ToUriQueryAllowInsecure(ProfileItem item, ref Dictionary<string, string> dicQuery)
|
||||||
|
{
|
||||||
|
if (item.AllowInsecure.Equals(Global.AllowInsecure.First()))
|
||||||
|
{
|
||||||
|
// Add two for compatibility
|
||||||
|
dicQuery.Add("insecure", "1");
|
||||||
|
dicQuery.Add("allowInsecure", "1");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
dicQuery.Add("insecure", "0");
|
||||||
|
dicQuery.Add("allowInsecure", "0");
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected static int ResolveUriQuery(NameValueCollection query, ref ProfileItem item)
|
||||||
{
|
{
|
||||||
item.Flow = GetQueryValue(query, "flow");
|
item.Flow = GetQueryValue(query, "flow");
|
||||||
item.StreamSecurity = GetQueryValue(query, "security");
|
item.StreamSecurity = GetQueryValue(query, "security");
|
||||||
@@ -164,7 +209,19 @@ public class BaseFmt
|
|||||||
item.ShortId = GetQueryDecoded(query, "sid");
|
item.ShortId = GetQueryDecoded(query, "sid");
|
||||||
item.SpiderX = GetQueryDecoded(query, "spx");
|
item.SpiderX = GetQueryDecoded(query, "spx");
|
||||||
item.Mldsa65Verify = GetQueryDecoded(query, "pqv");
|
item.Mldsa65Verify = GetQueryDecoded(query, "pqv");
|
||||||
item.AllowInsecure = new[] { "allowInsecure", "allow_insecure", "insecure" }.Any(k => (query[k] ?? "") == "1") ? "true" : "";
|
|
||||||
|
if (_allowInsecureArray.Any(k => GetQueryDecoded(query, k) == "1"))
|
||||||
|
{
|
||||||
|
item.AllowInsecure = Global.AllowInsecure.First();
|
||||||
|
}
|
||||||
|
else if (_allowInsecureArray.Any(k => GetQueryDecoded(query, k) == "0"))
|
||||||
|
{
|
||||||
|
item.AllowInsecure = Global.AllowInsecure.Skip(1).First();
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
item.AllowInsecure = string.Empty;
|
||||||
|
}
|
||||||
|
|
||||||
item.Network = GetQueryValue(query, "type", nameof(ETransport.tcp));
|
item.Network = GetQueryValue(query, "type", nameof(ETransport.tcp));
|
||||||
switch (item.Network)
|
switch (item.Network)
|
||||||
@@ -189,7 +246,21 @@ public class BaseFmt
|
|||||||
item.RequestHost = GetQueryDecoded(query, "host");
|
item.RequestHost = GetQueryDecoded(query, "host");
|
||||||
item.Path = GetQueryDecoded(query, "path", "/");
|
item.Path = GetQueryDecoded(query, "path", "/");
|
||||||
item.HeaderType = GetQueryDecoded(query, "mode");
|
item.HeaderType = GetQueryDecoded(query, "mode");
|
||||||
item.Extra = GetQueryDecoded(query, "extra");
|
var extraDecoded = GetQueryDecoded(query, "extra");
|
||||||
|
if (extraDecoded.IsNotEmpty())
|
||||||
|
{
|
||||||
|
var node = JsonUtils.ParseJson(extraDecoded);
|
||||||
|
if (node != null)
|
||||||
|
{
|
||||||
|
extraDecoded = JsonUtils.Serialize(node, new JsonSerializerOptions
|
||||||
|
{
|
||||||
|
WriteIndented = true,
|
||||||
|
DefaultIgnoreCondition = JsonIgnoreCondition.Never,
|
||||||
|
Encoder = JavaScriptEncoder.UnsafeRelaxedJsonEscaping
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
item.Extra = extraDecoded;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case nameof(ETransport.http):
|
case nameof(ETransport.http):
|
||||||
|
|||||||
@@ -37,7 +37,7 @@ public class FmtHandler
|
|||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
string str = config.TrimEx();
|
var str = config.TrimEx();
|
||||||
if (str.IsNullOrEmpty())
|
if (str.IsNullOrEmpty())
|
||||||
{
|
{
|
||||||
msg = ResUI.FailedReadConfiguration;
|
msg = ResUI.FailedReadConfiguration;
|
||||||
|
|||||||
@@ -12,7 +12,9 @@ public class Hysteria2Fmt : BaseFmt
|
|||||||
|
|
||||||
var url = Utils.TryUri(str);
|
var url = Utils.TryUri(str);
|
||||||
if (url == null)
|
if (url == null)
|
||||||
|
{
|
||||||
return null;
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
item.Address = url.IdnHost;
|
item.Address = url.IdnHost;
|
||||||
item.Port = url.Port;
|
item.Port = url.Port;
|
||||||
@@ -20,10 +22,8 @@ public class Hysteria2Fmt : BaseFmt
|
|||||||
item.Id = Utils.UrlDecode(url.UserInfo);
|
item.Id = Utils.UrlDecode(url.UserInfo);
|
||||||
|
|
||||||
var query = Utils.ParseQueryString(url.Query);
|
var query = Utils.ParseQueryString(url.Query);
|
||||||
ResolveStdTransport(query, ref item);
|
ResolveUriQuery(query, ref item);
|
||||||
item.Path = GetQueryDecoded(query, "obfs-password");
|
item.Path = GetQueryDecoded(query, "obfs-password");
|
||||||
item.AllowInsecure = GetQueryValue(query, "insecure") == "1" ? "true" : "false";
|
|
||||||
|
|
||||||
item.Ports = GetQueryDecoded(query, "mport");
|
item.Ports = GetQueryDecoded(query, "mport");
|
||||||
|
|
||||||
return item;
|
return item;
|
||||||
@@ -32,29 +32,25 @@ public class Hysteria2Fmt : BaseFmt
|
|||||||
public static string? ToUri(ProfileItem? item)
|
public static string? ToUri(ProfileItem? item)
|
||||||
{
|
{
|
||||||
if (item == null)
|
if (item == null)
|
||||||
|
{
|
||||||
return null;
|
return null;
|
||||||
string url = string.Empty;
|
}
|
||||||
|
|
||||||
string remark = string.Empty;
|
var url = string.Empty;
|
||||||
|
|
||||||
|
var remark = string.Empty;
|
||||||
if (item.Remarks.IsNotEmpty())
|
if (item.Remarks.IsNotEmpty())
|
||||||
{
|
{
|
||||||
remark = "#" + Utils.UrlEncode(item.Remarks);
|
remark = "#" + Utils.UrlEncode(item.Remarks);
|
||||||
}
|
}
|
||||||
var dicQuery = new Dictionary<string, string>();
|
var dicQuery = new Dictionary<string, string>();
|
||||||
if (item.Sni.IsNotEmpty())
|
ToUriQueryLite(item, ref dicQuery);
|
||||||
{
|
|
||||||
dicQuery.Add("sni", item.Sni);
|
|
||||||
}
|
|
||||||
if (item.Alpn.IsNotEmpty())
|
|
||||||
{
|
|
||||||
dicQuery.Add("alpn", Utils.UrlEncode(item.Alpn));
|
|
||||||
}
|
|
||||||
if (item.Path.IsNotEmpty())
|
if (item.Path.IsNotEmpty())
|
||||||
{
|
{
|
||||||
dicQuery.Add("obfs", "salamander");
|
dicQuery.Add("obfs", "salamander");
|
||||||
dicQuery.Add("obfs-password", Utils.UrlEncode(item.Path));
|
dicQuery.Add("obfs-password", Utils.UrlEncode(item.Path));
|
||||||
}
|
}
|
||||||
dicQuery.Add("insecure", item.AllowInsecure.ToLower() == "true" ? "1" : "0");
|
|
||||||
if (item.Ports.IsNotEmpty())
|
if (item.Ports.IsNotEmpty())
|
||||||
{
|
{
|
||||||
dicQuery.Add("mport", Utils.UrlEncode(item.Ports.Replace(':', '-')));
|
dicQuery.Add("mport", Utils.UrlEncode(item.Ports.Replace(':', '-')));
|
||||||
|
|||||||
@@ -41,7 +41,66 @@ public class ShadowsocksFmt : BaseFmt
|
|||||||
//url = Utile.Base64Encode(url);
|
//url = Utile.Base64Encode(url);
|
||||||
//new Sip002
|
//new Sip002
|
||||||
var pw = Utils.Base64Encode($"{item.Security}:{item.Id}", true);
|
var pw = Utils.Base64Encode($"{item.Security}:{item.Id}", true);
|
||||||
return ToUri(EConfigType.Shadowsocks, item.Address, item.Port, pw, null, remark);
|
|
||||||
|
// plugin
|
||||||
|
var plugin = string.Empty;
|
||||||
|
var pluginArgs = string.Empty;
|
||||||
|
|
||||||
|
if (item.Network == nameof(ETransport.tcp) && item.HeaderType == Global.TcpHeaderHttp)
|
||||||
|
{
|
||||||
|
plugin = "obfs-local";
|
||||||
|
pluginArgs = $"obfs=http;obfs-host={item.RequestHost};";
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if (item.Network == nameof(ETransport.ws))
|
||||||
|
{
|
||||||
|
pluginArgs += "mode=websocket;";
|
||||||
|
pluginArgs += $"host={item.RequestHost};";
|
||||||
|
pluginArgs += $"path={item.Path};";
|
||||||
|
}
|
||||||
|
else if (item.Network == nameof(ETransport.quic))
|
||||||
|
{
|
||||||
|
pluginArgs += "mode=quic;";
|
||||||
|
}
|
||||||
|
if (item.StreamSecurity == Global.StreamSecurity)
|
||||||
|
{
|
||||||
|
pluginArgs += "tls;";
|
||||||
|
var certs = CertPemManager.ParsePemChain(item.Cert);
|
||||||
|
if (certs.Count > 0)
|
||||||
|
{
|
||||||
|
var cert = certs.First();
|
||||||
|
const string beginMarker = "-----BEGIN CERTIFICATE-----\n";
|
||||||
|
const string endMarker = "\n-----END CERTIFICATE-----";
|
||||||
|
|
||||||
|
var base64Content = cert.Replace(beginMarker, "").Replace(endMarker, "").Trim();
|
||||||
|
|
||||||
|
// https://github.com/shadowsocks/v2ray-plugin/blob/e9af1cdd2549d528deb20a4ab8d61c5fbe51f306/args.go#L172
|
||||||
|
// Equal signs and commas [and backslashes] must be escaped with a backslash.
|
||||||
|
base64Content = base64Content.Replace("=", "\\=");
|
||||||
|
|
||||||
|
pluginArgs += $"certRaw={base64Content};";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (pluginArgs.Length > 0)
|
||||||
|
{
|
||||||
|
plugin = "v2ray-plugin";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var dicQuery = new Dictionary<string, string>();
|
||||||
|
if (plugin.IsNotEmpty())
|
||||||
|
{
|
||||||
|
var pluginStr = plugin + ";" + pluginArgs;
|
||||||
|
// pluginStr remove last ';' and url encode
|
||||||
|
if (pluginStr.EndsWith(';'))
|
||||||
|
{
|
||||||
|
pluginStr = pluginStr[..^1];
|
||||||
|
}
|
||||||
|
dicQuery["plugin"] = Utils.UrlEncode(pluginStr);
|
||||||
|
}
|
||||||
|
|
||||||
|
return ToUri(EConfigType.Shadowsocks, item.Address, item.Port, pw, dicQuery, remark);
|
||||||
}
|
}
|
||||||
|
|
||||||
private static readonly Regex UrlFinder = new(@"ss://(?<base64>[A-Za-z0-9+-/=_]+)(?:#(?<tag>\S+))?", RegexOptions.IgnoreCase | RegexOptions.Compiled);
|
private static readonly Regex UrlFinder = new(@"ss://(?<base64>[A-Za-z0-9+-/=_]+)(?:#(?<tag>\S+))?", RegexOptions.IgnoreCase | RegexOptions.Compiled);
|
||||||
@@ -124,19 +183,82 @@ public class ShadowsocksFmt : BaseFmt
|
|||||||
var queryParameters = Utils.ParseQueryString(parsedUrl.Query);
|
var queryParameters = Utils.ParseQueryString(parsedUrl.Query);
|
||||||
if (queryParameters["plugin"] != null)
|
if (queryParameters["plugin"] != null)
|
||||||
{
|
{
|
||||||
//obfs-host exists
|
var pluginStr = queryParameters["plugin"];
|
||||||
var obfsHost = queryParameters["plugin"]?.Split(';').FirstOrDefault(t => t.Contains("obfs-host"));
|
var pluginParts = pluginStr.Split(new[] { ';' }, StringSplitOptions.RemoveEmptyEntries);
|
||||||
if (queryParameters["plugin"].Contains("obfs=http") && obfsHost.IsNotEmpty())
|
|
||||||
{
|
if (pluginParts.Length == 0)
|
||||||
obfsHost = obfsHost?.Replace("obfs-host=", "");
|
|
||||||
item.Network = Global.DefaultNetwork;
|
|
||||||
item.HeaderType = Global.TcpHeaderHttp;
|
|
||||||
item.RequestHost = obfsHost ?? "";
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
{
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var pluginName = pluginParts[0];
|
||||||
|
|
||||||
|
// A typo in https://github.com/shadowsocks/shadowsocks-org/blob/6b1c064db4129de99c516294960e731934841c94/docs/doc/sip002.md?plain=1#L15
|
||||||
|
// "simple-obfs" should be "obfs-local"
|
||||||
|
if (pluginName == "simple-obfs")
|
||||||
|
{
|
||||||
|
pluginName = "obfs-local";
|
||||||
|
}
|
||||||
|
|
||||||
|
// Parse obfs-local plugin
|
||||||
|
if (pluginName == "obfs-local")
|
||||||
|
{
|
||||||
|
var obfsMode = pluginParts.FirstOrDefault(t => t.StartsWith("obfs="));
|
||||||
|
var obfsHost = pluginParts.FirstOrDefault(t => t.StartsWith("obfs-host="));
|
||||||
|
|
||||||
|
if ((!obfsMode.IsNullOrEmpty()) && obfsMode.Contains("obfs=http") && obfsHost.IsNotEmpty())
|
||||||
|
{
|
||||||
|
obfsHost = obfsHost.Replace("obfs-host=", "");
|
||||||
|
item.Network = Global.DefaultNetwork;
|
||||||
|
item.HeaderType = Global.TcpHeaderHttp;
|
||||||
|
item.RequestHost = obfsHost;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Parse v2ray-plugin
|
||||||
|
else if (pluginName == "v2ray-plugin")
|
||||||
|
{
|
||||||
|
var mode = pluginParts.FirstOrDefault(t => t.StartsWith("mode="), "websocket");
|
||||||
|
var host = pluginParts.FirstOrDefault(t => t.StartsWith("host="));
|
||||||
|
var path = pluginParts.FirstOrDefault(t => t.StartsWith("path="));
|
||||||
|
var hasTls = pluginParts.Any(t => t == "tls");
|
||||||
|
var certRaw = pluginParts.FirstOrDefault(t => t.StartsWith("certRaw="));
|
||||||
|
|
||||||
|
var modeValue = mode.Replace("mode=", "");
|
||||||
|
if (modeValue == "websocket")
|
||||||
|
{
|
||||||
|
item.Network = nameof(ETransport.ws);
|
||||||
|
if (!host.IsNullOrEmpty())
|
||||||
|
{
|
||||||
|
item.RequestHost = host.Replace("host=", "");
|
||||||
|
item.Sni = item.RequestHost;
|
||||||
|
}
|
||||||
|
if (!path.IsNullOrEmpty())
|
||||||
|
{
|
||||||
|
item.Path = path.Replace("path=", "");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (modeValue == "quic")
|
||||||
|
{
|
||||||
|
item.Network = nameof(ETransport.quic);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (hasTls)
|
||||||
|
{
|
||||||
|
item.StreamSecurity = Global.StreamSecurity;
|
||||||
|
|
||||||
|
if (!certRaw.IsNullOrEmpty())
|
||||||
|
{
|
||||||
|
var certBase64 = certRaw.Replace("certRaw=", "");
|
||||||
|
|
||||||
|
certBase64 = certBase64.Replace("\\=", "=");
|
||||||
|
|
||||||
|
const string beginMarker = "-----BEGIN CERTIFICATE-----\n";
|
||||||
|
const string endMarker = "\n-----END CERTIFICATE-----";
|
||||||
|
var certPem = beginMarker + certBase64 + endMarker;
|
||||||
|
item.Cert = certPem;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return item;
|
return item;
|
||||||
|
|||||||
@@ -45,18 +45,18 @@ public class SocksFmt : BaseFmt
|
|||||||
};
|
};
|
||||||
result = result[Global.ProtocolShares[EConfigType.SOCKS].Length..];
|
result = result[Global.ProtocolShares[EConfigType.SOCKS].Length..];
|
||||||
//remark
|
//remark
|
||||||
var indexRemark = result.IndexOf("#");
|
var indexRemark = result.IndexOf('#');
|
||||||
if (indexRemark > 0)
|
if (indexRemark > 0)
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
item.Remarks = Utils.UrlDecode(result.Substring(indexRemark + 1, result.Length - indexRemark - 1));
|
item.Remarks = Utils.UrlDecode(result.Substring(indexRemark + 1));
|
||||||
}
|
}
|
||||||
catch { }
|
catch { }
|
||||||
result = result[..indexRemark];
|
result = result[..indexRemark];
|
||||||
}
|
}
|
||||||
//part decode
|
//part decode
|
||||||
var indexS = result.IndexOf("@");
|
var indexS = result.IndexOf('@');
|
||||||
if (indexS > 0)
|
if (indexS > 0)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -23,7 +23,7 @@ public class TrojanFmt : BaseFmt
|
|||||||
item.Id = Utils.UrlDecode(url.UserInfo);
|
item.Id = Utils.UrlDecode(url.UserInfo);
|
||||||
|
|
||||||
var query = Utils.ParseQueryString(url.Query);
|
var query = Utils.ParseQueryString(url.Query);
|
||||||
_ = ResolveStdTransport(query, ref item);
|
ResolveUriQuery(query, ref item);
|
||||||
|
|
||||||
return item;
|
return item;
|
||||||
}
|
}
|
||||||
@@ -40,7 +40,7 @@ public class TrojanFmt : BaseFmt
|
|||||||
remark = "#" + Utils.UrlEncode(item.Remarks);
|
remark = "#" + Utils.UrlEncode(item.Remarks);
|
||||||
}
|
}
|
||||||
var dicQuery = new Dictionary<string, string>();
|
var dicQuery = new Dictionary<string, string>();
|
||||||
_ = GetStdTransport(item, null, ref dicQuery);
|
ToUriQuery(item, null, ref dicQuery);
|
||||||
|
|
||||||
return ToUri(EConfigType.Trojan, item.Address, item.Port, item.Id, dicQuery, remark);
|
return ToUri(EConfigType.Trojan, item.Address, item.Port, item.Id, dicQuery, remark);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -29,7 +29,7 @@ public class TuicFmt : BaseFmt
|
|||||||
}
|
}
|
||||||
|
|
||||||
var query = Utils.ParseQueryString(url.Query);
|
var query = Utils.ParseQueryString(url.Query);
|
||||||
ResolveStdTransport(query, ref item);
|
ResolveUriQuery(query, ref item);
|
||||||
item.HeaderType = GetQueryValue(query, "congestion_control");
|
item.HeaderType = GetQueryValue(query, "congestion_control");
|
||||||
|
|
||||||
return item;
|
return item;
|
||||||
@@ -47,15 +47,10 @@ public class TuicFmt : BaseFmt
|
|||||||
{
|
{
|
||||||
remark = "#" + Utils.UrlEncode(item.Remarks);
|
remark = "#" + Utils.UrlEncode(item.Remarks);
|
||||||
}
|
}
|
||||||
|
|
||||||
var dicQuery = new Dictionary<string, string>();
|
var dicQuery = new Dictionary<string, string>();
|
||||||
if (item.Sni.IsNotEmpty())
|
ToUriQueryLite(item, ref dicQuery);
|
||||||
{
|
|
||||||
dicQuery.Add("sni", item.Sni);
|
|
||||||
}
|
|
||||||
if (item.Alpn.IsNotEmpty())
|
|
||||||
{
|
|
||||||
dicQuery.Add("alpn", Utils.UrlEncode(item.Alpn));
|
|
||||||
}
|
|
||||||
dicQuery.Add("congestion_control", item.HeaderType);
|
dicQuery.Add("congestion_control", item.HeaderType);
|
||||||
|
|
||||||
return ToUri(EConfigType.TUIC, item.Address, item.Port, $"{item.Id}:{item.Security}", dicQuery, remark);
|
return ToUri(EConfigType.TUIC, item.Address, item.Port, $"{item.Id}:{item.Security}", dicQuery, remark);
|
||||||
|
|||||||
@@ -26,7 +26,7 @@ public class VLESSFmt : BaseFmt
|
|||||||
var query = Utils.ParseQueryString(url.Query);
|
var query = Utils.ParseQueryString(url.Query);
|
||||||
item.Security = GetQueryValue(query, "encryption", Global.None);
|
item.Security = GetQueryValue(query, "encryption", Global.None);
|
||||||
item.StreamSecurity = GetQueryValue(query, "security");
|
item.StreamSecurity = GetQueryValue(query, "security");
|
||||||
_ = ResolveStdTransport(query, ref item);
|
ResolveUriQuery(query, ref item);
|
||||||
|
|
||||||
return item;
|
return item;
|
||||||
}
|
}
|
||||||
@@ -52,7 +52,7 @@ public class VLESSFmt : BaseFmt
|
|||||||
{
|
{
|
||||||
dicQuery.Add("encryption", Global.None);
|
dicQuery.Add("encryption", Global.None);
|
||||||
}
|
}
|
||||||
_ = GetStdTransport(item, Global.None, ref dicQuery);
|
ToUriQuery(item, Global.None, ref dicQuery);
|
||||||
|
|
||||||
return ToUri(EConfigType.VLESS, item.Address, item.Port, item.Id, dicQuery, remark);
|
return ToUri(EConfigType.VLESS, item.Address, item.Port, item.Id, dicQuery, remark);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -39,7 +39,8 @@ public class VmessFmt : BaseFmt
|
|||||||
tls = item.StreamSecurity,
|
tls = item.StreamSecurity,
|
||||||
sni = item.Sni,
|
sni = item.Sni,
|
||||||
alpn = item.Alpn,
|
alpn = item.Alpn,
|
||||||
fp = item.Fingerprint
|
fp = item.Fingerprint,
|
||||||
|
insecure = item.AllowInsecure.Equals(Global.AllowInsecure.First()) ? "1" : "0"
|
||||||
};
|
};
|
||||||
|
|
||||||
var url = JsonUtils.Serialize(vmessQRCode);
|
var url = JsonUtils.Serialize(vmessQRCode);
|
||||||
@@ -94,6 +95,7 @@ public class VmessFmt : BaseFmt
|
|||||||
item.Sni = Utils.ToString(vmessQRCode.sni);
|
item.Sni = Utils.ToString(vmessQRCode.sni);
|
||||||
item.Alpn = Utils.ToString(vmessQRCode.alpn);
|
item.Alpn = Utils.ToString(vmessQRCode.alpn);
|
||||||
item.Fingerprint = Utils.ToString(vmessQRCode.fp);
|
item.Fingerprint = Utils.ToString(vmessQRCode.fp);
|
||||||
|
item.AllowInsecure = vmessQRCode.insecure == "1" ? Global.AllowInsecure.First() : string.Empty;
|
||||||
|
|
||||||
return item;
|
return item;
|
||||||
}
|
}
|
||||||
@@ -118,7 +120,7 @@ public class VmessFmt : BaseFmt
|
|||||||
item.Id = Utils.UrlDecode(url.UserInfo);
|
item.Id = Utils.UrlDecode(url.UserInfo);
|
||||||
|
|
||||||
var query = Utils.ParseQueryString(url.Query);
|
var query = Utils.ParseQueryString(url.Query);
|
||||||
ResolveStdTransport(query, ref item);
|
ResolveUriQuery(query, ref item);
|
||||||
|
|
||||||
return item;
|
return item;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -18,7 +18,13 @@ public static class ProxySettingLinux
|
|||||||
|
|
||||||
private static async Task ExecCmd(List<string> args)
|
private static async Task ExecCmd(List<string> args)
|
||||||
{
|
{
|
||||||
var fileName = await FileManager.CreateLinuxShellFile(_proxySetFileName, EmbedUtils.GetEmbedText(Global.ProxySetLinuxShellFileName), false);
|
var customSystemProxyScriptPath = AppManager.Instance.Config.SystemProxyItem?.CustomSystemProxyScriptPath;
|
||||||
|
var fileName = (customSystemProxyScriptPath.IsNotEmpty() && File.Exists(customSystemProxyScriptPath))
|
||||||
|
? customSystemProxyScriptPath
|
||||||
|
: await FileUtils.CreateLinuxShellFile(_proxySetFileName, EmbedUtils.GetEmbedText(Global.ProxySetLinuxShellFileName), false);
|
||||||
|
|
||||||
|
// TODO: temporarily notify which script is being used
|
||||||
|
NoticeManager.Instance.SendMessage(fileName);
|
||||||
|
|
||||||
await Utils.GetCliWrapOutput(fileName, args);
|
await Utils.GetCliWrapOutput(fileName, args);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -23,7 +23,13 @@ public static class ProxySettingOSX
|
|||||||
|
|
||||||
private static async Task ExecCmd(List<string> args)
|
private static async Task ExecCmd(List<string> args)
|
||||||
{
|
{
|
||||||
var fileName = await FileManager.CreateLinuxShellFile(_proxySetFileName, EmbedUtils.GetEmbedText(Global.ProxySetOSXShellFileName), false);
|
var customSystemProxyScriptPath = AppManager.Instance.Config.SystemProxyItem?.CustomSystemProxyScriptPath;
|
||||||
|
var fileName = (customSystemProxyScriptPath.IsNotEmpty() && File.Exists(customSystemProxyScriptPath))
|
||||||
|
? customSystemProxyScriptPath
|
||||||
|
: await FileUtils.CreateLinuxShellFile(_proxySetFileName, EmbedUtils.GetEmbedText(Global.ProxySetOSXShellFileName), false);
|
||||||
|
|
||||||
|
// TODO: temporarily notify which script is being used
|
||||||
|
NoticeManager.Instance.SendMessage(fileName);
|
||||||
|
|
||||||
await Utils.GetCliWrapOutput(fileName, args);
|
await Utils.GetCliWrapOutput(fileName, args);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -33,7 +33,7 @@ public static class SysProxyHandler
|
|||||||
await ProxySettingLinux.SetProxy(Global.Loopback, port, exceptions);
|
await ProxySettingLinux.SetProxy(Global.Loopback, port, exceptions);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case ESysProxyType.ForcedChange when Utils.IsOSX():
|
case ESysProxyType.ForcedChange when Utils.IsMacOS():
|
||||||
await ProxySettingOSX.SetProxy(Global.Loopback, port, exceptions);
|
await ProxySettingOSX.SetProxy(Global.Loopback, port, exceptions);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
@@ -45,7 +45,7 @@ public static class SysProxyHandler
|
|||||||
await ProxySettingLinux.UnsetProxy();
|
await ProxySettingLinux.UnsetProxy();
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case ESysProxyType.ForcedClear when Utils.IsOSX():
|
case ESysProxyType.ForcedClear when Utils.IsMacOS():
|
||||||
await ProxySettingOSX.UnsetProxy();
|
await ProxySettingOSX.UnsetProxy();
|
||||||
break;
|
break;
|
||||||
|
|
||||||
@@ -91,7 +91,7 @@ public static class SysProxyHandler
|
|||||||
private static async Task SetWindowsProxyPac(int port)
|
private static async Task SetWindowsProxyPac(int port)
|
||||||
{
|
{
|
||||||
var portPac = AppManager.Instance.GetLocalPort(EInboundProtocol.pac);
|
var portPac = AppManager.Instance.GetLocalPort(EInboundProtocol.pac);
|
||||||
await PacManager.Instance.StartAsync(Utils.GetConfigPath(), port, portPac);
|
await PacManager.Instance.StartAsync(port, portPac);
|
||||||
var strProxy = $"{Global.HttpProtocol}{Global.Loopback}:{portPac}/pac?t={DateTime.Now.Ticks}";
|
var strProxy = $"{Global.HttpProtocol}{Global.Loopback}:{portPac}/pac?t={DateTime.Now.Ticks}";
|
||||||
ProxySettingWindows.SetProxy(strProxy, "", 4);
|
ProxySettingWindows.SetProxy(strProxy, "", 4);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -71,28 +71,25 @@ public class DownloaderHelper
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
var totalDatetime = DateTime.Now;
|
var lastUpdateTime = DateTime.Now;
|
||||||
var totalSecond = 0;
|
|
||||||
var hasValue = false;
|
var hasValue = false;
|
||||||
double maxSpeed = 0;
|
double maxSpeed = 0;
|
||||||
await using var downloader = new Downloader.DownloadService(downloadOpt);
|
await using var downloader = new Downloader.DownloadService(downloadOpt);
|
||||||
//downloader.DownloadStarted += (sender, value) =>
|
|
||||||
//{
|
|
||||||
// if (progress != null)
|
|
||||||
// {
|
|
||||||
// progress.Report("Start download data...");
|
|
||||||
// }
|
|
||||||
//};
|
|
||||||
downloader.DownloadProgressChanged += (sender, value) =>
|
downloader.DownloadProgressChanged += (sender, value) =>
|
||||||
{
|
{
|
||||||
var ts = DateTime.Now - totalDatetime;
|
if (progress != null && value.BytesPerSecondSpeed > 0)
|
||||||
if (progress != null && ts.Seconds > totalSecond)
|
|
||||||
{
|
{
|
||||||
hasValue = true;
|
hasValue = true;
|
||||||
totalSecond = ts.Seconds;
|
|
||||||
if (value.BytesPerSecondSpeed > maxSpeed)
|
if (value.BytesPerSecondSpeed > maxSpeed)
|
||||||
{
|
{
|
||||||
maxSpeed = value.BytesPerSecondSpeed;
|
maxSpeed = value.BytesPerSecondSpeed;
|
||||||
|
}
|
||||||
|
|
||||||
|
var ts = DateTime.Now - lastUpdateTime;
|
||||||
|
if (ts.TotalMilliseconds >= 1000)
|
||||||
|
{
|
||||||
|
lastUpdateTime = DateTime.Now;
|
||||||
var speed = (maxSpeed / 1000 / 1000).ToString("#0.0");
|
var speed = (maxSpeed / 1000 / 1000).ToString("#0.0");
|
||||||
progress.Report(speed);
|
progress.Report(speed);
|
||||||
}
|
}
|
||||||
@@ -102,10 +99,19 @@ public class DownloaderHelper
|
|||||||
{
|
{
|
||||||
if (progress != null)
|
if (progress != null)
|
||||||
{
|
{
|
||||||
if (!hasValue && value.Error != null)
|
if (hasValue && maxSpeed > 0)
|
||||||
|
{
|
||||||
|
var finalSpeed = (maxSpeed / 1000 / 1000).ToString("#0.0");
|
||||||
|
progress.Report(finalSpeed);
|
||||||
|
}
|
||||||
|
else if (value.Error != null)
|
||||||
{
|
{
|
||||||
progress.Report(value.Error?.Message);
|
progress.Report(value.Error?.Message);
|
||||||
}
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
progress.Report("0");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
//progress.Report("......");
|
//progress.Report("......");
|
||||||
|
|||||||
@@ -49,15 +49,6 @@ public class HttpClientHelper
|
|||||||
return await httpClient.GetStringAsync(url);
|
return await httpClient.GetStringAsync(url);
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task<string?> GetAsync(HttpClient client, string url, CancellationToken token = default)
|
|
||||||
{
|
|
||||||
if (url.IsNullOrEmpty())
|
|
||||||
{
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
return await client.GetStringAsync(url, token);
|
|
||||||
}
|
|
||||||
|
|
||||||
public async Task PutAsync(string url, Dictionary<string, string> headers)
|
public async Task PutAsync(string url, Dictionary<string, string> headers)
|
||||||
{
|
{
|
||||||
var jsonContent = JsonUtils.Serialize(headers);
|
var jsonContent = JsonUtils.Serialize(headers);
|
||||||
@@ -80,156 +71,4 @@ public class HttpClientHelper
|
|||||||
{
|
{
|
||||||
await httpClient.DeleteAsync(url);
|
await httpClient.DeleteAsync(url);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async Task DownloadFileAsync(HttpClient client, string url, string fileName, IProgress<double>? progress, CancellationToken token = default)
|
|
||||||
{
|
|
||||||
ArgumentNullException.ThrowIfNull(url);
|
|
||||||
ArgumentNullException.ThrowIfNull(fileName);
|
|
||||||
if (File.Exists(fileName))
|
|
||||||
{
|
|
||||||
File.Delete(fileName);
|
|
||||||
}
|
|
||||||
|
|
||||||
using var response = await client.GetAsync(url, HttpCompletionOption.ResponseHeadersRead, token);
|
|
||||||
|
|
||||||
if (!response.IsSuccessStatusCode)
|
|
||||||
{
|
|
||||||
throw new Exception(response.StatusCode.ToString());
|
|
||||||
}
|
|
||||||
|
|
||||||
var total = response.Content.Headers.ContentLength ?? -1L;
|
|
||||||
var canReportProgress = total != -1 && progress != null;
|
|
||||||
|
|
||||||
await using var stream = await response.Content.ReadAsStreamAsync(token);
|
|
||||||
await using var file = File.Create(fileName);
|
|
||||||
var totalRead = 0L;
|
|
||||||
var buffer = new byte[1024 * 1024];
|
|
||||||
var progressPercentage = 0;
|
|
||||||
|
|
||||||
while (true)
|
|
||||||
{
|
|
||||||
token.ThrowIfCancellationRequested();
|
|
||||||
|
|
||||||
var read = await stream.ReadAsync(buffer, token);
|
|
||||||
totalRead += read;
|
|
||||||
|
|
||||||
if (read == 0)
|
|
||||||
{
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
await file.WriteAsync(buffer.AsMemory(0, read), token);
|
|
||||||
|
|
||||||
if (canReportProgress)
|
|
||||||
{
|
|
||||||
var percent = (int)(100.0 * totalRead / total);
|
|
||||||
//if (progressPercentage != percent && percent % 10 == 0)
|
|
||||||
{
|
|
||||||
progressPercentage = percent;
|
|
||||||
progress?.Report(percent);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (canReportProgress)
|
|
||||||
{
|
|
||||||
progress?.Report(101);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public async Task DownloadDataAsync4Speed(HttpClient client, string url, IProgress<string> progress, CancellationToken token = default)
|
|
||||||
{
|
|
||||||
if (url.IsNullOrEmpty())
|
|
||||||
{
|
|
||||||
throw new ArgumentNullException(nameof(url));
|
|
||||||
}
|
|
||||||
|
|
||||||
var response = await client.GetAsync(url, HttpCompletionOption.ResponseHeadersRead, token);
|
|
||||||
|
|
||||||
if (!response.IsSuccessStatusCode)
|
|
||||||
{
|
|
||||||
throw new Exception(response.StatusCode.ToString());
|
|
||||||
}
|
|
||||||
|
|
||||||
//var total = response.Content.Headers.ContentLength.HasValue ? response.Content.Headers.ContentLength.Value : -1L;
|
|
||||||
//var canReportProgress = total != -1 && progress != null;
|
|
||||||
|
|
||||||
await using var stream = await response.Content.ReadAsStreamAsync(token);
|
|
||||||
var totalRead = 0L;
|
|
||||||
var buffer = new byte[1024 * 64];
|
|
||||||
var isMoreToRead = true;
|
|
||||||
var progressSpeed = string.Empty;
|
|
||||||
var totalDatetime = DateTime.Now;
|
|
||||||
var totalSecond = 0;
|
|
||||||
|
|
||||||
do
|
|
||||||
{
|
|
||||||
if (token.IsCancellationRequested)
|
|
||||||
{
|
|
||||||
if (totalRead > 0)
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
token.ThrowIfCancellationRequested();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
var read = await stream.ReadAsync(buffer, token);
|
|
||||||
|
|
||||||
if (read == 0)
|
|
||||||
{
|
|
||||||
isMoreToRead = false;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
var data = new byte[read];
|
|
||||||
buffer.ToList().CopyTo(0, data, 0, read);
|
|
||||||
|
|
||||||
totalRead += read;
|
|
||||||
|
|
||||||
var ts = DateTime.Now - totalDatetime;
|
|
||||||
if (progress != null && ts.Seconds > totalSecond)
|
|
||||||
{
|
|
||||||
totalSecond = ts.Seconds;
|
|
||||||
var speed = (totalRead * 1d / ts.TotalMilliseconds / 1000).ToString("#0.0");
|
|
||||||
if (progressSpeed != speed)
|
|
||||||
{
|
|
||||||
progressSpeed = speed;
|
|
||||||
progress.Report(speed);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} while (isMoreToRead);
|
|
||||||
}
|
|
||||||
|
|
||||||
public async Task<int> GetRealPingTime(string url, IWebProxy? webProxy, int downloadTimeout)
|
|
||||||
{
|
|
||||||
var responseTime = -1;
|
|
||||||
try
|
|
||||||
{
|
|
||||||
using var cts = new CancellationTokenSource();
|
|
||||||
cts.CancelAfter(TimeSpan.FromSeconds(downloadTimeout));
|
|
||||||
using var client = new HttpClient(new SocketsHttpHandler()
|
|
||||||
{
|
|
||||||
Proxy = webProxy,
|
|
||||||
UseProxy = webProxy != null
|
|
||||||
});
|
|
||||||
|
|
||||||
List<int> oneTime = new();
|
|
||||||
for (var i = 0; i < 2; i++)
|
|
||||||
{
|
|
||||||
var timer = Stopwatch.StartNew();
|
|
||||||
await client.GetAsync(url, cts.Token).ConfigureAwait(false);
|
|
||||||
timer.Stop();
|
|
||||||
oneTime.Add((int)timer.Elapsed.TotalMilliseconds);
|
|
||||||
await Task.Delay(100);
|
|
||||||
}
|
|
||||||
responseTime = oneTime.Where(x => x > 0).OrderBy(x => x).FirstOrDefault();
|
|
||||||
}
|
|
||||||
catch //(Exception ex)
|
|
||||||
{
|
|
||||||
//Utile.SaveLog(ex.Message, ex);
|
|
||||||
}
|
|
||||||
return responseTime;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -10,6 +10,15 @@ public class ActionPrecheckManager(Config config)
|
|||||||
|
|
||||||
private readonly Config _config = config;
|
private readonly Config _config = config;
|
||||||
|
|
||||||
|
// sing-box supported transports for different protocol types
|
||||||
|
private static readonly HashSet<string> SingboxUnsupportedTransports = [nameof(ETransport.kcp), nameof(ETransport.xhttp)];
|
||||||
|
|
||||||
|
private static readonly HashSet<EConfigType> SingboxTransportSupportedProtocols =
|
||||||
|
[EConfigType.VMess, EConfigType.VLESS, EConfigType.Trojan, EConfigType.Shadowsocks];
|
||||||
|
|
||||||
|
private static readonly HashSet<string> SingboxShadowsocksAllowedTransports =
|
||||||
|
[nameof(ETransport.tcp), nameof(ETransport.ws), nameof(ETransport.quic)];
|
||||||
|
|
||||||
public async Task<List<string>> Check(string? indexId)
|
public async Task<List<string>> Check(string? indexId)
|
||||||
{
|
{
|
||||||
if (indexId.IsNullOrEmpty())
|
if (indexId.IsNullOrEmpty())
|
||||||
@@ -81,21 +90,36 @@ public class ActionPrecheckManager(Config config)
|
|||||||
{
|
{
|
||||||
case EConfigType.VMess:
|
case EConfigType.VMess:
|
||||||
if (item.Id.IsNullOrEmpty() || !Utils.IsGuidByParse(item.Id))
|
if (item.Id.IsNullOrEmpty() || !Utils.IsGuidByParse(item.Id))
|
||||||
|
{
|
||||||
errors.Add(string.Format(ResUI.InvalidProperty, "Id"));
|
errors.Add(string.Format(ResUI.InvalidProperty, "Id"));
|
||||||
|
}
|
||||||
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case EConfigType.VLESS:
|
case EConfigType.VLESS:
|
||||||
if (item.Id.IsNullOrEmpty() || !Utils.IsGuidByParse(item.Id) && item.Id.Length > 30)
|
if (item.Id.IsNullOrEmpty() || (!Utils.IsGuidByParse(item.Id) && item.Id.Length > 30))
|
||||||
|
{
|
||||||
errors.Add(string.Format(ResUI.InvalidProperty, "Id"));
|
errors.Add(string.Format(ResUI.InvalidProperty, "Id"));
|
||||||
|
}
|
||||||
|
|
||||||
if (!Global.Flows.Contains(item.Flow))
|
if (!Global.Flows.Contains(item.Flow))
|
||||||
|
{
|
||||||
errors.Add(string.Format(ResUI.InvalidProperty, "Flow"));
|
errors.Add(string.Format(ResUI.InvalidProperty, "Flow"));
|
||||||
|
}
|
||||||
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case EConfigType.Shadowsocks:
|
case EConfigType.Shadowsocks:
|
||||||
if (item.Id.IsNullOrEmpty())
|
if (item.Id.IsNullOrEmpty())
|
||||||
|
{
|
||||||
errors.Add(string.Format(ResUI.InvalidProperty, "Id"));
|
errors.Add(string.Format(ResUI.InvalidProperty, "Id"));
|
||||||
|
}
|
||||||
|
|
||||||
if (string.IsNullOrEmpty(item.Security) || !Global.SsSecuritiesInSingbox.Contains(item.Security))
|
if (string.IsNullOrEmpty(item.Security) || !Global.SsSecuritiesInSingbox.Contains(item.Security))
|
||||||
|
{
|
||||||
errors.Add(string.Format(ResUI.InvalidProperty, "Security"));
|
errors.Add(string.Format(ResUI.InvalidProperty, "Security"));
|
||||||
|
}
|
||||||
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -115,7 +139,7 @@ public class ActionPrecheckManager(Config config)
|
|||||||
if (item.ConfigType.IsGroupType())
|
if (item.ConfigType.IsGroupType())
|
||||||
{
|
{
|
||||||
ProfileGroupItemManager.Instance.TryGet(item.IndexId, out var group);
|
ProfileGroupItemManager.Instance.TryGet(item.IndexId, out var group);
|
||||||
if (group is null || group.ChildItems.IsNullOrEmpty())
|
if (group is null || group.NotHasChild())
|
||||||
{
|
{
|
||||||
errors.Add(string.Format(ResUI.GroupEmpty, item.Remarks));
|
errors.Add(string.Format(ResUI.GroupEmpty, item.Remarks));
|
||||||
return errors;
|
return errors;
|
||||||
@@ -128,7 +152,11 @@ public class ActionPrecheckManager(Config config)
|
|||||||
return errors;
|
return errors;
|
||||||
}
|
}
|
||||||
|
|
||||||
foreach (var child in Utils.String2List(group.ChildItems))
|
var childIds = Utils.String2List(group.ChildItems) ?? [];
|
||||||
|
var subItems = await ProfileGroupItemManager.GetSubChildProfileItems(group);
|
||||||
|
childIds.AddRange(subItems.Select(p => p.IndexId));
|
||||||
|
|
||||||
|
foreach (var child in childIds)
|
||||||
{
|
{
|
||||||
var childErrors = new List<string>();
|
var childErrors = new List<string>();
|
||||||
if (child.IsNullOrEmpty())
|
if (child.IsNullOrEmpty())
|
||||||
@@ -155,26 +183,16 @@ public class ActionPrecheckManager(Config config)
|
|||||||
return errors;
|
return errors;
|
||||||
}
|
}
|
||||||
|
|
||||||
var net = item.GetNetwork() ?? item.Network;
|
var net = item.GetNetwork();
|
||||||
|
|
||||||
if (coreType == ECoreType.sing_box)
|
if (coreType == ECoreType.sing_box)
|
||||||
{
|
{
|
||||||
// sing-box does not support xhttp / kcp
|
var transportError = ValidateSingboxTransport(item.ConfigType, net);
|
||||||
// sing-box does not support transports like ws/http/httpupgrade/etc. when the node is not vmess/trojan/vless
|
if (transportError != null)
|
||||||
if (net is nameof(ETransport.kcp) or nameof(ETransport.xhttp))
|
|
||||||
{
|
{
|
||||||
errors.Add(string.Format(ResUI.CoreNotSupportNetwork, nameof(ECoreType.sing_box), net));
|
errors.Add(transportError);
|
||||||
return errors;
|
return errors;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (item.ConfigType is not (EConfigType.VMess or EConfigType.VLESS or EConfigType.Trojan))
|
|
||||||
{
|
|
||||||
if (net is nameof(ETransport.ws) or nameof(ETransport.http) or nameof(ETransport.h2) or nameof(ETransport.quic) or nameof(ETransport.httpupgrade))
|
|
||||||
{
|
|
||||||
errors.Add(string.Format(ResUI.CoreNotSupportProtocolTransport, nameof(ECoreType.sing_box), item.ConfigType.ToString(), net));
|
|
||||||
return errors;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
else if (coreType is ECoreType.Xray)
|
else if (coreType is ECoreType.Xray)
|
||||||
{
|
{
|
||||||
@@ -190,6 +208,31 @@ public class ActionPrecheckManager(Config config)
|
|||||||
return errors;
|
return errors;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static string? ValidateSingboxTransport(EConfigType configType, string net)
|
||||||
|
{
|
||||||
|
// sing-box does not support xhttp / kcp transports
|
||||||
|
if (SingboxUnsupportedTransports.Contains(net))
|
||||||
|
{
|
||||||
|
return string.Format(ResUI.CoreNotSupportNetwork, nameof(ECoreType.sing_box), net);
|
||||||
|
}
|
||||||
|
|
||||||
|
// sing-box does not support non-tcp transports for protocols other than vmess/trojan/vless/shadowsocks
|
||||||
|
if (!SingboxTransportSupportedProtocols.Contains(configType) && net != nameof(ETransport.tcp))
|
||||||
|
{
|
||||||
|
return string.Format(ResUI.CoreNotSupportProtocolTransport,
|
||||||
|
nameof(ECoreType.sing_box), configType.ToString(), net);
|
||||||
|
}
|
||||||
|
|
||||||
|
// sing-box shadowsocks only supports tcp/ws/quic transports
|
||||||
|
if (configType == EConfigType.Shadowsocks && !SingboxShadowsocksAllowedTransports.Contains(net))
|
||||||
|
{
|
||||||
|
return string.Format(ResUI.CoreNotSupportProtocolTransport,
|
||||||
|
nameof(ECoreType.sing_box), configType.ToString(), net);
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
private async Task<List<string>> ValidateRelatedNodesExistAndValid(ProfileItem? item)
|
private async Task<List<string>> ValidateRelatedNodesExistAndValid(ProfileItem? item)
|
||||||
{
|
{
|
||||||
var errors = new List<string>();
|
var errors = new List<string>();
|
||||||
|
|||||||
419
v2rayN/ServiceLib/Manager/CertPemManager.cs
Normal file
419
v2rayN/ServiceLib/Manager/CertPemManager.cs
Normal file
@@ -0,0 +1,419 @@
|
|||||||
|
using System.Net.Security;
|
||||||
|
using System.Security.Cryptography.X509Certificates;
|
||||||
|
|
||||||
|
namespace ServiceLib.Manager;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Manager for certificate operations with CA pinning to prevent MITM attacks
|
||||||
|
/// </summary>
|
||||||
|
public class CertPemManager
|
||||||
|
{
|
||||||
|
private static readonly string _tag = "CertPemManager";
|
||||||
|
private static readonly Lazy<CertPemManager> _instance = new(() => new());
|
||||||
|
public static CertPemManager Instance => _instance.Value;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Trusted CA certificate thumbprints (SHA256) to prevent MITM attacks
|
||||||
|
/// </summary>
|
||||||
|
private static readonly HashSet<string> TrustedCaThumbprints = new(StringComparer.OrdinalIgnoreCase)
|
||||||
|
{
|
||||||
|
"EBD41040E4BB3EC742C9E381D31EF2A41A48B6685C96E7CEF3C1DF6CD4331C99", // GlobalSign Root CA
|
||||||
|
"6DC47172E01CBCB0BF62580D895FE2B8AC9AD4F873801E0C10B9C837D21EB177", // Entrust.net Premium 2048 Secure Server CA
|
||||||
|
"73C176434F1BC6D5ADF45B0E76E727287C8DE57616C1E6E6141A2B2CBC7D8E4C", // Entrust Root Certification Authority
|
||||||
|
"D8E0FEBC1DB2E38D00940F37D27D41344D993E734B99D5656D9778D4D8143624", // Certum Root CA
|
||||||
|
"D7A7A0FB5D7E2731D771E9484EBCDEF71D5F0C3E0A2948782BC83EE0EA699EF4", // Comodo AAA Services root
|
||||||
|
"85A0DD7DD720ADB7FF05F83D542B209DC7FF4528F7D677B18389FEA5E5C49E86", // QuoVadis Root CA 2
|
||||||
|
"18F1FC7F205DF8ADDDEB7FE007DD57E3AF375A9C4D8D73546BF4F1FED1E18D35", // QuoVadis Root CA 3
|
||||||
|
"CECDDC905099D8DADFC5B1D209B737CBE2C18CFB2C10C0FF0BCF0D3286FC1AA2", // XRamp Global CA Root
|
||||||
|
"C3846BF24B9E93CA64274C0EC67C1ECC5E024FFCACD2D74019350E81FE546AE4", // Go Daddy Class 2 CA
|
||||||
|
"1465FA205397B876FAA6F0A9958E5590E40FCC7FAA4FB7C2C8677521FB5FB658", // Starfield Class 2 CA
|
||||||
|
"3E9099B5015E8F486C00BCEA9D111EE721FABA355A89BCF1DF69561E3DC6325C", // DigiCert Assured ID Root CA
|
||||||
|
"4348A0E9444C78CB265E058D5E8944B4D84F9662BD26DB257F8934A443C70161", // DigiCert Global Root CA
|
||||||
|
"7431E5F4C3C1CE4690774F0B61E05440883BA9A01ED00BA6ABD7806ED3B118CF", // DigiCert High Assurance EV Root CA
|
||||||
|
"62DD0BE9B9F50A163EA0F8E75C053B1ECA57EA55C8688F647C6881F2C8357B95", // SwissSign Gold CA - G2
|
||||||
|
"F1C1B50AE5A20DD8030EC9F6BC24823DD367B5255759B4E71B61FCE9F7375D73", // SecureTrust CA
|
||||||
|
"4200F5043AC8590EBB527D209ED1503029FBCBD41CA1B506EC27F15ADE7DAC69", // Secure Global CA
|
||||||
|
"0C2CD63DF7806FA399EDE809116B575BF87989F06518F9808C860503178BAF66", // COMODO Certification Authority
|
||||||
|
"1793927A0614549789ADCE2F8F34F7F0B66D0F3AE3A3B84D21EC15DBBA4FADC7", // COMODO ECC Certification Authority
|
||||||
|
"41C923866AB4CAD6B7AD578081582E020797A6CBDF4FFF78CE8396B38937D7F5", // OISTE WISeKey Global Root GA CA
|
||||||
|
"E3B6A2DB2ED7CE48842F7AC53241C7B71D54144BFB40C11F3F1D0B42F5EEA12D", // Certigna
|
||||||
|
"C0A6F4DC63A24BFDCF54EF2A6A082A0A72DE35803E2FF5FF527AE5D87206DFD5", // ePKI Root Certification Authority
|
||||||
|
"EAA962C4FA4A6BAFEBE415196D351CCD888D4F53F3FA8AE6D7C466A94E6042BB", // certSIGN ROOT CA
|
||||||
|
"6C61DAC3A2DEF031506BE036D2A6FE401994FBD13DF9C8D466599274C446EC98", // NetLock Arany (Class Gold) Főtanúsítvány
|
||||||
|
"3C5F81FEA5FAB82C64BFA2EAECAFCDE8E077FC8620A7CAE537163DF36EDBF378", // Microsec e-Szigno Root CA 2009
|
||||||
|
"CBB522D7B7F127AD6A0113865BDF1CD4102E7D0759AF635A7CF4720DC963C53B", // GlobalSign Root CA - R3
|
||||||
|
"2530CC8E98321502BAD96F9B1FBA1B099E2D299E0F4548BB914F363BC0D4531F", // Izenpe.com
|
||||||
|
"45140B3247EB9CC8C5B4F0D7B53091F73292089E6E5A63E2749DD3ACA9198EDA", // Go Daddy Root Certificate Authority - G2
|
||||||
|
"2CE1CB0BF9D2F9E102993FBE215152C3B2DD0CABDE1C68E5319B839154DBB7F5", // Starfield Root Certificate Authority - G2
|
||||||
|
"568D6905A2C88708A4B3025190EDCFEDB1974A606A13C6E5290FCB2AE63EDAB5", // Starfield Services Root Certificate Authority - G2
|
||||||
|
"0376AB1D54C5F9803CE4B2E201A0EE7EEF7B57B636E8A93C9B8D4860C96F5FA7", // AffirmTrust Commercial
|
||||||
|
"0A81EC5A929777F145904AF38D5D509F66B5E2C58FCDB531058B0E17F3F0B41B", // AffirmTrust Networking
|
||||||
|
"70A73F7F376B60074248904534B11482D5BF0E698ECC498DF52577EBF2E93B9A", // AffirmTrust Premium
|
||||||
|
"BD71FDF6DA97E4CF62D1647ADD2581B07D79ADF8397EB4ECBA9C5E8488821423", // AffirmTrust Premium ECC
|
||||||
|
"5C58468D55F58E497E743982D2B50010B6D165374ACF83A7D4A32DB768C4408E", // Certum Trusted Network CA
|
||||||
|
"BFD88FE1101C41AE3E801BF8BE56350EE9BAD1A6B9BD515EDC5C6D5B8711AC44", // TWCA Root Certification Authority
|
||||||
|
"513B2CECB810D4CDE5DD85391ADFC6C2DD60D87BB736D2B521484AA47A0EBEF6", // Security Communication RootCA2
|
||||||
|
"55926084EC963A64B96E2ABE01CE0BA86A64FBFEBCC7AAB5AFC155B37FD76066", // Actalis Authentication Root CA
|
||||||
|
"9A114025197C5BB95D94E63D55CD43790847B646B23CDF11ADA4A00EFF15FB48", // Buypass Class 2 Root CA
|
||||||
|
"EDF7EBBCA27A2A384D387B7D4010C666E2EDB4843E4C29B4AE1D5B9332E6B24D", // Buypass Class 3 Root CA
|
||||||
|
"FD73DAD31C644FF1B43BEF0CCDDA96710B9CD9875ECA7E31707AF3E96D522BBD", // T-TeleSec GlobalRoot Class 3
|
||||||
|
"49E7A442ACF0EA6287050054B52564B650E4F49E42E348D6AA38E039E957B1C1", // D-TRUST Root Class 3 CA 2 2009
|
||||||
|
"EEC5496B988CE98625B934092EEC2908BED0B0F316C2D4730C84EAF1F3D34881", // D-TRUST Root Class 3 CA 2 EV 2009
|
||||||
|
"E23D4A036D7B70E9F595B1422079D2B91EDFBB1FB651A0633EAA8A9DC5F80703", // CA Disig Root R2
|
||||||
|
"9A6EC012E1A7DA9DBE34194D478AD7C0DB1822FB071DF12981496ED104384113", // ACCVRAIZ1
|
||||||
|
"59769007F7685D0FCD50872F9F95D5755A5B2B457D81F3692B610A98672F0E1B", // TWCA Global Root CA
|
||||||
|
"DD6936FE21F8F077C123A1A521C12224F72255B73E03A7260693E8A24B0FA389", // TeliaSonera Root CA v1
|
||||||
|
"91E2F5788D5810EBA7BA58737DE1548A8ECACD014598BC0B143E041B17052552", // T-TeleSec GlobalRoot Class 2
|
||||||
|
"F356BEA244B7A91EB35D53CA9AD7864ACE018E2D35D5F8F96DDF68A6F41AA474", // Atos TrustedRoot 2011
|
||||||
|
"8A866FD1B276B57E578E921C65828A2BED58E9F2F288054134B7F1F4BFC9CC74", // QuoVadis Root CA 1 G3
|
||||||
|
"8FE4FB0AF93A4D0D67DB0BEBB23E37C71BF325DCBCDD240EA04DAF58B47E1840", // QuoVadis Root CA 2 G3
|
||||||
|
"88EF81DE202EB018452E43F864725CEA5FBD1FC2D9D205730709C5D8B8690F46", // QuoVadis Root CA 3 G3
|
||||||
|
"7D05EBB682339F8C9451EE094EEBFEFA7953A114EDB2F44949452FAB7D2FC185", // DigiCert Assured ID Root G2
|
||||||
|
"7E37CB8B4C47090CAB36551BA6F45DB840680FBA166A952DB100717F43053FC2", // DigiCert Assured ID Root G3
|
||||||
|
"CB3CCBB76031E5E0138F8DD39A23F9DE47FFC35E43C1144CEA27D46A5AB1CB5F", // DigiCert Global Root G2
|
||||||
|
"31AD6648F8104138C738F39EA4320133393E3A18CC02296EF97C2AC9EF6731D0", // DigiCert Global Root G3
|
||||||
|
"552F7BDCF1A7AF9E6CE672017F4F12ABF77240C78E761AC203D1D9D20AC89988", // DigiCert Trusted Root G4
|
||||||
|
"52F0E1C4E58EC629291B60317F074671B85D7EA80D5B07273463534B32B40234", // COMODO RSA Certification Authority
|
||||||
|
"E793C9B02FD8AA13E21C31228ACCB08119643B749C898964B1746D46C3D4CBD2", // USERTrust RSA Certification Authority
|
||||||
|
"4FF460D54B9C86DABFBCFC5712E0400D2BED3FBC4D4FBDAA86E06ADCD2A9AD7A", // USERTrust ECC Certification Authority
|
||||||
|
"179FBC148A3DD00FD24EA13458CC43BFA7F59C8182D783A513F6EBEC100C8924", // GlobalSign ECC Root CA - R5
|
||||||
|
"3C4FB0B95AB8B30032F432B86F535FE172C185D0FD39865837CF36187FA6F428", // Staat der Nederlanden Root CA - G3
|
||||||
|
"5D56499BE4D2E08BCFCAD08A3E38723D50503BDE706948E42F55603019E528AE", // IdenTrust Commercial Root CA 1
|
||||||
|
"30D0895A9A448A262091635522D1F52010B5867ACAE12C78EF958FD4F4389F2F", // IdenTrust Public Sector Root CA 1
|
||||||
|
"43DF5774B03E7FEF5FE40D931A7BEDF1BB2E6B42738C4E6D3841103D3AA7F339", // Entrust Root Certification Authority - G2
|
||||||
|
"02ED0EB28C14DA45165C566791700D6451D7FB56F0B2AB1D3B8EB070E56EDFF5", // Entrust Root Certification Authority - EC1
|
||||||
|
"5CC3D78E4E1D5E45547A04E6873E64F90CF9536D1CCC2EF800F355C4C5FD70FD", // CFCA EV ROOT
|
||||||
|
"6B9C08E86EB0F767CFAD65CD98B62149E5494A67F5845E7BD1ED019F27B86BD6", // OISTE WISeKey Global Root GB CA
|
||||||
|
"A1339D33281A0B56E557D3D32B1CE7F9367EB094BD5FA72A7E5004C8DED7CAFE", // SZAFIR ROOT CA2
|
||||||
|
"B676F2EDDAE8775CD36CB0F63CD1D4603961F49E6265BA013A2F0307B6D0B804", // Certum Trusted Network CA 2
|
||||||
|
"A040929A02CE53B4ACF4F2FFC6981CE4496F755E6D45FE0B2A692BCD52523F36", // Hellenic Academic and Research Institutions RootCA 2015
|
||||||
|
"44B545AA8A25E65A73CA15DC27FC36D24C1CB9953A066539B11582DC487B4833", // Hellenic Academic and Research Institutions ECC RootCA 2015
|
||||||
|
"96BCEC06264976F37460779ACF28C5A7CFE8A3C0AAE11A8FFCEE05C0BDDF08C6", // ISRG Root X1
|
||||||
|
"EBC5570C29018C4D67B1AA127BAF12F703B4611EBC17B7DAB5573894179B93FA", // AC RAIZ FNMT-RCM
|
||||||
|
"8ECDE6884F3D87B1125BA31AC3FCB13D7016DE7F57CC904FE1CB97C6AE98196E", // Amazon Root CA 1
|
||||||
|
"1BA5B2AA8C65401A82960118F80BEC4F62304D83CEC4713A19C39C011EA46DB4", // Amazon Root CA 2
|
||||||
|
"18CE6CFE7BF14E60B2E347B8DFE868CB31D02EBB3ADA271569F50343B46DB3A4", // Amazon Root CA 3
|
||||||
|
"E35D28419ED02025CFA69038CD623962458DA5C695FBDEA3C22B0BFB25897092", // Amazon Root CA 4
|
||||||
|
"A1A86D04121EB87F027C66F53303C28E5739F943FC84B38AD6AF009035DD9457", // D-TRUST Root CA 3 2013
|
||||||
|
"46EDC3689046D53A453FB3104AB80DCAEC658B2660EA1629DD7E867990648716", // TUBITAK Kamu SM SSL Kok Sertifikasi - Surum 1
|
||||||
|
"BFFF8FD04433487D6A8AA60C1A29767A9FC2BBB05E420F713A13B992891D3893", // GDCA TrustAUTH R5 ROOT
|
||||||
|
"85666A562EE0BE5CE925C1D8890A6F76A87EC16D4D7D5F29EA7419CF20123B69", // SSL.com Root Certification Authority RSA
|
||||||
|
"3417BB06CC6007DA1B961C920B8AB4CE3FAD820E4AA30B9ACBC4A74EBDCEBC65", // SSL.com Root Certification Authority ECC
|
||||||
|
"2E7BF16CC22485A7BBE2AA8696750761B0AE39BE3B2FE9D0CC6D4EF73491425C", // SSL.com EV Root Certification Authority RSA R2
|
||||||
|
"22A2C1F7BDED704CC1E701B5F408C310880FE956B5DE2A4A44F99C873A25A7C8", // SSL.com EV Root Certification Authority ECC
|
||||||
|
"2CABEAFE37D06CA22ABA7391C0033D25982952C453647349763A3AB5AD6CCF69", // GlobalSign Root CA - R6
|
||||||
|
"8560F91C3624DABA9570B5FEA0DBE36FF11A8323BE9486854FB3F34A5571198D", // OISTE WISeKey Global Root GC CA
|
||||||
|
"9BEA11C976FE014764C1BE56A6F914B5A560317ABD9988393382E5161AA0493C", // UCA Global G2 Root
|
||||||
|
"D43AF9B35473755C9684FC06D7D8CB70EE5C28E773FB294EB41EE71722924D24", // UCA Extended Validation Root
|
||||||
|
"D48D3D23EEDB50A459E55197601C27774B9D7B18C94D5A059511A10250B93168", // Certigna Root CA
|
||||||
|
"40F6AF0346A99AA1CD1D555A4E9CCE62C7F9634603EE406615833DC8C8D00367", // emSign Root CA - G1
|
||||||
|
"86A1ECBA089C4A8D3BBE2734C612BA341D813E043CF9E8A862CD5C57A36BBE6B", // emSign ECC Root CA - G3
|
||||||
|
"125609AA301DA0A249B97A8239CB6A34216F44DCAC9F3954B14292F2E8C8608F", // emSign Root CA - C1
|
||||||
|
"BC4D809B15189D78DB3E1D8CF4F9726A795DA1643CA5F1358E1DDB0EDC0D7EB3", // emSign ECC Root CA - C3
|
||||||
|
"5A2FC03F0C83B090BBFA40604B0988446C7636183DF9846E17101A447FB8EFD6", // Hongkong Post Root CA 3
|
||||||
|
"DB3517D1F6732A2D5AB97C533EC70779EE3270A62FB4AC4238372460E6F01E88", // Entrust Root Certification Authority - G4
|
||||||
|
"358DF39D764AF9E1B766E9C972DF352EE15CFAC227AF6AD1D70E8E4A6EDCBA02", // Microsoft ECC Root Certificate Authority 2017
|
||||||
|
"C741F70F4B2A8D88BF2E71C14122EF53EF10EBA0CFA5E64CFA20F418853073E0", // Microsoft RSA Root Certificate Authority 2017
|
||||||
|
"BEB00B30839B9BC32C32E4447905950641F26421B15ED089198B518AE2EA1B99", // e-Szigno Root CA 2017
|
||||||
|
"657CFE2FA73FAA38462571F332A2363A46FCE7020951710702CDFBB6EEDA3305", // certSIGN Root CA G2
|
||||||
|
"97552015F5DDFC3C8788C006944555408894450084F100867086BC1A2BB58DC8", // Trustwave Global Certification Authority
|
||||||
|
"945BBC825EA554F489D1FD51A73DDF2EA624AC7019A05205225C22A78CCFA8B4", // Trustwave Global ECC P256 Certification Authority
|
||||||
|
"55903859C8C0C3EBB8759ECE4E2557225FF5758BBD38EBD48276601E1BD58097", // Trustwave Global ECC P384 Certification Authority
|
||||||
|
"88F438DCF8FFD1FA8F429115FFE5F82AE1E06E0C70C375FAAD717B34A49E7265", // NAVER Global Root Certification Authority
|
||||||
|
"554153B13D2CF9DDB753BFBE1A4E0AE08D0AA4187058FE60A2B862B2E4B87BCB", // AC RAIZ FNMT-RCM SERVIDORES SEGUROS
|
||||||
|
"319AF0A7729E6F89269C131EA6A3A16FCD86389FDCAB3C47A4A675C161A3F974", // GlobalSign Secure Mail Root R45
|
||||||
|
"5CBF6FB81FD417EA4128CD6F8172A3C9402094F74AB2ED3A06B4405D04F30B19", // GlobalSign Secure Mail Root E45
|
||||||
|
"4FA3126D8D3A11D1C4855A4F807CBAD6CF919D3A5A88B03BEA2C6372D93C40C9", // GlobalSign Root R46
|
||||||
|
"CBB9C44D84B8043E1050EA31A69F514955D7BFD2E2C6B49301019AD61D9F5058", // GlobalSign Root E46
|
||||||
|
"9A296A5182D1D451A2E37F439B74DAAFA267523329F90F9A0D2007C334E23C9A", // GLOBALTRUST 2020
|
||||||
|
"FB8FEC759169B9106B1E511644C618C51304373F6C0643088D8BEFFD1B997599", // ANF Secure Server Root CA
|
||||||
|
"6B328085625318AA50D173C98D8BDA09D57E27413D114CF787A0F5D06C030CF6", // Certum EC-384 CA
|
||||||
|
"FE7696573855773E37A95E7AD4D9CC96C30157C15D31765BA9B15704E1AE78FD", // Certum Trusted Root CA
|
||||||
|
"2E44102AB58CB85419451C8E19D9ACF3662CAFBC614B6A53960A30F7D0E2EB41", // TunTrust Root CA
|
||||||
|
"D95D0E8EDA79525BF9BEB11B14D2100D3294985F0C62D9FABD9CD999ECCB7B1D", // HARICA TLS RSA Root CA 2021
|
||||||
|
"3F99CC474ACFCE4DFED58794665E478D1547739F2E780F1BB4CA9B133097D401", // HARICA TLS ECC Root CA 2021
|
||||||
|
"1BE7ABE30686B16348AFD1C61B6866A0EA7F4821E67D5E8AF937CF8011BC750D", // HARICA Client RSA Root CA 2021
|
||||||
|
"8DD4B5373CB0DE36769C12339280D82746B3AA6CD426E797A31BABE4279CF00B", // HARICA Client ECC Root CA 2021
|
||||||
|
"57DE0583EFD2B26E0361DA99DA9DF4648DEF7EE8441C3B728AFA9BCDE0F9B26A", // Autoridad de Certificacion Firmaprofesional CIF A62634068
|
||||||
|
"30FBBA2C32238E2A98547AF97931E550428B9B3F1C8EEB6633DCFA86C5B27DD3", // vTrus ECC Root CA
|
||||||
|
"8A71DE6559336F426C26E53880D00D88A18DA4C6A91F0DCB6194E206C5C96387", // vTrus Root CA
|
||||||
|
"69729B8E15A86EFC177A57AFB7171DFC64ADD28C2FCA8CF1507E34453CCB1470", // ISRG Root X2
|
||||||
|
"F015CE3CC239BFEF064BE9F1D2C417E1A0264A0A94BE1F0C8D121864EB6949CC", // HiPKI Root CA - G1
|
||||||
|
"B085D70B964F191A73E4AF0D54AE7A0E07AAFDAF9B71DD0862138AB7325A24A2", // GlobalSign ECC Root CA - R4
|
||||||
|
"D947432ABDE7B7FA90FC2E6B59101B1280E0E1C7E4E40FA3C6887FFF57A7F4CF", // GTS Root R1
|
||||||
|
"8D25CD97229DBF70356BDA4EB3CC734031E24CF00FAFCFD32DC76EB5841C7EA8", // GTS Root R2
|
||||||
|
"34D8A73EE208D9BCDB0D956520934B4E40E69482596E8B6F73C8426B010A6F48", // GTS Root R3
|
||||||
|
"349DFA4058C5E263123B398AE795573C4E1313C83FE68F93556CD5E8031B3C7D", // GTS Root R4
|
||||||
|
"242B69742FCB1E5B2ABF98898B94572187544E5B4D9911786573621F6A74B82C", // Telia Root CA v2
|
||||||
|
"E59AAA816009C22BFF5B25BAD37DF306F049797C1F81D85AB089E657BD8F0044", // D-TRUST BR Root CA 1 2020
|
||||||
|
"08170D1AA36453901A2F959245E347DB0C8D37ABAABC56B81AA100DC958970DB", // D-TRUST EV Root CA 1 2020
|
||||||
|
"018E13F0772532CF809BD1B17281867283FC48C6E13BE9C69812854A490C1B05", // DigiCert TLS ECC P384 Root G5
|
||||||
|
"371A00DC0533B3721A7EEB40E8419E70799D2B0A0F2C1D80693165F7CEC4AD75", // DigiCert TLS RSA4096 Root G5
|
||||||
|
"E8E8176536A60CC2C4E10187C3BEFCA20EF263497018F566D5BEA0F94D0C111B", // DigiCert SMIME ECC P384 Root G5
|
||||||
|
"90370D3EFA88BF58C30105BA25104A358460A7FA52DFC2011DF233A0F417912A", // DigiCert SMIME RSA4096 Root G5
|
||||||
|
"77B82CD8644C4305F7ACC5CB156B45675004033D51C60C6202A8E0C33467D3A0", // Certainly Root R1
|
||||||
|
"B4585F22E4AC756A4E8612A1361C5D9D031A93FD84FEBB778FA3068B0FC42DC2", // Certainly Root E1
|
||||||
|
"82BD5D851ACF7F6E1BA7BFCBC53030D0E7BC3C21DF772D858CAB41D199BDF595", // DIGITALSIGN GLOBAL ROOT RSA CA
|
||||||
|
"261D7114AE5F8FF2D8C7209A9DE4289E6AFC9D717023D85450909199F1857CFE", // DIGITALSIGN GLOBAL ROOT ECDSA CA
|
||||||
|
"E74FBDA55BD564C473A36B441AA799C8A68E077440E8288B9FA1E50E4BBACA11", // Security Communication ECC RootCA1
|
||||||
|
"F3896F88FE7C0A882766A7FA6AD2749FB57A7F3E98FB769C1FA7B09C2C44D5AE", // BJCA Global Root CA1
|
||||||
|
"574DF6931E278039667B720AFDC1600FC27EB66DD3092979FB73856487212882", // BJCA Global Root CA2
|
||||||
|
"48E1CF9E43B688A51044160F46D773B8277FE45BEAAD0E4DF90D1974382FEA99", // LAWtrust Root CA2 (4096)
|
||||||
|
"22D9599234D60F1D4BC7C7E96F43FA555B07301FD475175089DAFB8C25E477B3", // Sectigo Public Email Protection Root E46
|
||||||
|
"D5917A7791EB7CF20A2E57EB98284A67B28A57E89182DA53D546678C9FDE2B4F", // Sectigo Public Email Protection Root R46
|
||||||
|
"C90F26F0FB1B4018B22227519B5CA2B53E2CA5B3BE5CF18EFE1BEF47380C5383", // Sectigo Public Server Authentication Root E46
|
||||||
|
"7BB647A62AEEAC88BF257AA522D01FFEA395E0AB45C73F93F65654EC38F25A06", // Sectigo Public Server Authentication Root R46
|
||||||
|
"8FAF7D2E2CB4709BB8E0B33666BF75A5DD45B5DE480F8EA8D4BFE6BEBC17F2ED", // SSL.com TLS RSA Root CA 2022
|
||||||
|
"C32FFD9F46F936D16C3673990959434B9AD60AAFBB9E7CF33654F144CC1BA143", // SSL.com TLS ECC Root CA 2022
|
||||||
|
"AD7DD58D03AEDB22A30B5084394920CE12230C2D8017AD9B81AB04079BDD026B", // SSL.com Client ECC Root CA 2022
|
||||||
|
"1D4CA4A2AB21D0093659804FC0EB2175A617279B56A2475245C9517AFEB59153", // SSL.com Client RSA Root CA 2022
|
||||||
|
"E38655F4B0190C84D3B3893D840A687E190A256D98052F159E6D4A39F589A6EB", // Atos TrustedRoot Root CA ECC G2 2020
|
||||||
|
"78833A783BB2986C254B9370D3C20E5EBA8FA7840CBF63FE17297A0B0119685E", // Atos TrustedRoot Root CA RSA G2 2020
|
||||||
|
"B2FAE53E14CCD7AB9212064701AE279C1D8988FACB775FA8A008914E663988A8", // Atos TrustedRoot Root CA ECC TLS 2021
|
||||||
|
"81A9088EA59FB364C548A6F85559099B6F0405EFBF18E5324EC9F457BA00112F", // Atos TrustedRoot Root CA RSA TLS 2021
|
||||||
|
"E0D3226AEB1163C2E48FF9BE3B50B4C6431BE7BB1EACC5C36B5D5EC509039A08", // TrustAsia Global Root CA G3
|
||||||
|
"BE4B56CB5056C0136A526DF444508DAA36A0B54F42E4AC38F72AF470E479654C", // TrustAsia Global Root CA G4
|
||||||
|
"D92C171F5CF890BA428019292927FE22F3207FD2B54449CB6F675AF4922146E2", // D-Trust SBR Root CA 1 2022
|
||||||
|
"DBA84DD7EF622D485463A90137EA4D574DF8550928F6AFA03B4D8B1141E636CC", // D-Trust SBR Root CA 2 2022
|
||||||
|
"3AE6DF7E0D637A65A8C81612EC6F9A142F85A16834C10280D88E707028518755", // Telekom Security SMIME ECC Root 2021
|
||||||
|
"578AF4DED0853F4E5998DB4AEAF9CBEA8D945F60B620A38D1A3C13B2BC7BA8E1", // Telekom Security TLS ECC Root 2020
|
||||||
|
"78A656344F947E9CC0F734D9053D32F6742086B6B9CD2CAE4FAE1A2E4EFDE048", // Telekom Security SMIME RSA Root 2023
|
||||||
|
"EFC65CADBB59ADB6EFE84DA22311B35624B71B3B1EA0DA8B6655174EC8978646", // Telekom Security TLS RSA Root 2023
|
||||||
|
"BEF256DAF26E9C69BDEC1602359798F3CAF71821A03E018257C53C65617F3D4A", // FIRMAPROFESIONAL CA ROOT-A WEB
|
||||||
|
"3F63BB2814BE174EC8B6439CF08D6D56F0B7C405883A5648A334424D6B3EC558", // TWCA CYBER Root CA
|
||||||
|
"3A0072D49FFC04E996C59AEB75991D3C340F3615D6FD4DCE90AC0B3D88EAD4F4", // TWCA Global Root CA G2
|
||||||
|
"3F034BB5704D44B2D08545A02057DE93EBF3905FCE721ACBC730C06DDAEE904E", // SecureSign Root CA12
|
||||||
|
"4B009C1034494F9AB56BBA3BA1D62731FC4D20D8955ADCEC10A925607261E338", // SecureSign Root CA14
|
||||||
|
"E778F0F095FE843729CD1A0082179E5314A9C291442805E1FB1D8FB6B8886C3A", // SecureSign Root CA15
|
||||||
|
"0552E6F83FDF65E8FA9670E666DF28A4E21340B510CBE52566F97C4FB94B2BD1", // D-TRUST BR Root CA 2 2023
|
||||||
|
"436472C1009A325C54F1A5BBB5468A7BAEECCBE05DE5F099CB70D3FE41E13C16", // TrustAsia SMIME ECC Root CA
|
||||||
|
"C7796BEB62C101BB143D262A7C96A0C6168183223EF50D699632D86E03B8CC9B", // TrustAsia SMIME RSA Root CA
|
||||||
|
"C0076B9EF0531FB1A656D67C4EBE97CD5DBAA41EF44598ACC2489878C92D8711", // TrustAsia TLS ECC Root CA
|
||||||
|
"06C08D7DAFD876971EB1124FE67F847EC0C7A158D3EA53CBE940E2EA9791F4C3", // TrustAsia TLS RSA Root CA
|
||||||
|
"8E8221B2E7D4007836A1672F0DCC299C33BC07D316F132FA1A206D587150F1CE", // D-TRUST EV Root CA 2 2023
|
||||||
|
"9A12C392BFE57891A0C545309D4D9FD567E480CB613D6342278B195C79A7931F", // SwissSign RSA SMIME Root CA 2022 - 1
|
||||||
|
"193144F431E0FDDB740717D4DE926A571133884B4360D30E272913CBE660CE41", // SwissSign RSA TLS Root CA 2022 - 1
|
||||||
|
"D9A32485A8CCA85539CEF12FFFFF711378A17851D73DA2732AB4302D763BD62B", // OISTE Client Root ECC G1
|
||||||
|
"D02A0F994A868C66395F2E7A880DF509BD0C29C96DE16015A0FD501EDA4F96A9", // OISTE Client Root RSA G1
|
||||||
|
"EEC997C0C30F216F7E3B8B307D2BAE42412D753FC8219DAFD1520B2572850F49", // OISTE Server Root ECC G1
|
||||||
|
"9AE36232A5189FFDDB353DFD26520C015395D22777DAC59DB57B98C089A651E6", // OISTE Server Root RSA G1
|
||||||
|
};
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Get certificate in PEM format from a server with CA pinning validation
|
||||||
|
/// </summary>
|
||||||
|
public async Task<(string?, string?)> GetCertPemAsync(string target, string serverName, int timeout = 4)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var (domain, _, port, _) = Utils.ParseUrl(target);
|
||||||
|
|
||||||
|
using var cts = new CancellationTokenSource();
|
||||||
|
cts.CancelAfter(TimeSpan.FromSeconds(timeout));
|
||||||
|
|
||||||
|
using var client = new TcpClient();
|
||||||
|
await client.ConnectAsync(domain, port > 0 ? port : 443, cts.Token);
|
||||||
|
|
||||||
|
using var ssl = new SslStream(client.GetStream(), false, ValidateServerCertificate);
|
||||||
|
|
||||||
|
var sslOptions = new SslClientAuthenticationOptions
|
||||||
|
{
|
||||||
|
TargetHost = serverName,
|
||||||
|
RemoteCertificateValidationCallback = ValidateServerCertificate
|
||||||
|
};
|
||||||
|
|
||||||
|
await ssl.AuthenticateAsClientAsync(sslOptions, cts.Token);
|
||||||
|
|
||||||
|
var remote = ssl.RemoteCertificate;
|
||||||
|
if (remote == null)
|
||||||
|
{
|
||||||
|
return (null, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
var leaf = new X509Certificate2(remote);
|
||||||
|
return (ExportCertToPem(leaf), null);
|
||||||
|
}
|
||||||
|
catch (OperationCanceledException)
|
||||||
|
{
|
||||||
|
Logging.SaveLog(_tag, new TimeoutException($"Connection timeout after {timeout} seconds"));
|
||||||
|
return (null, $"Connection timeout after {timeout} seconds");
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
Logging.SaveLog(_tag, ex);
|
||||||
|
return (null, ex.Message);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Get certificate chain in PEM format from a server with CA pinning validation
|
||||||
|
/// </summary>
|
||||||
|
public async Task<(List<string>, string?)> GetCertChainPemAsync(string target, string serverName, int timeout = 4)
|
||||||
|
{
|
||||||
|
var pemList = new List<string>();
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var (domain, _, port, _) = Utils.ParseUrl(target);
|
||||||
|
|
||||||
|
using var cts = new CancellationTokenSource();
|
||||||
|
cts.CancelAfter(TimeSpan.FromSeconds(timeout));
|
||||||
|
|
||||||
|
using var client = new TcpClient();
|
||||||
|
await client.ConnectAsync(domain, port > 0 ? port : 443, cts.Token);
|
||||||
|
|
||||||
|
using var ssl = new SslStream(client.GetStream(), false, ValidateServerCertificate);
|
||||||
|
|
||||||
|
var sslOptions = new SslClientAuthenticationOptions
|
||||||
|
{
|
||||||
|
TargetHost = serverName,
|
||||||
|
RemoteCertificateValidationCallback = ValidateServerCertificate
|
||||||
|
};
|
||||||
|
|
||||||
|
await ssl.AuthenticateAsClientAsync(sslOptions, cts.Token);
|
||||||
|
|
||||||
|
if (ssl.RemoteCertificate is not X509Certificate2 certChain)
|
||||||
|
{
|
||||||
|
return (pemList, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
var chain = new X509Chain();
|
||||||
|
chain.Build(certChain);
|
||||||
|
|
||||||
|
foreach (var element in chain.ChainElements)
|
||||||
|
{
|
||||||
|
var pem = ExportCertToPem(element.Certificate);
|
||||||
|
pemList.Add(pem);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (pemList, null);
|
||||||
|
}
|
||||||
|
catch (OperationCanceledException)
|
||||||
|
{
|
||||||
|
Logging.SaveLog(_tag, new TimeoutException($"Connection timeout after {timeout} seconds"));
|
||||||
|
return (pemList, $"Connection timeout after {timeout} seconds");
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
Logging.SaveLog(_tag, ex);
|
||||||
|
return (pemList, ex.Message);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Validate server certificate with CA pinning
|
||||||
|
/// </summary>
|
||||||
|
private bool ValidateServerCertificate(
|
||||||
|
object sender,
|
||||||
|
X509Certificate? certificate,
|
||||||
|
X509Chain? chain,
|
||||||
|
SslPolicyErrors sslPolicyErrors)
|
||||||
|
{
|
||||||
|
if (certificate == null)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check certificate name mismatch
|
||||||
|
if (sslPolicyErrors.HasFlag(SslPolicyErrors.RemoteCertificateNameMismatch))
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Build certificate chain
|
||||||
|
var cert2 = certificate as X509Certificate2 ?? new X509Certificate2(certificate);
|
||||||
|
var certChain = chain ?? new X509Chain();
|
||||||
|
|
||||||
|
certChain.ChainPolicy.RevocationMode = X509RevocationMode.Online;
|
||||||
|
certChain.ChainPolicy.RevocationFlag = X509RevocationFlag.ExcludeRoot;
|
||||||
|
certChain.ChainPolicy.VerificationFlags = X509VerificationFlags.NoFlag;
|
||||||
|
certChain.ChainPolicy.VerificationTime = DateTime.Now;
|
||||||
|
|
||||||
|
certChain.Build(cert2);
|
||||||
|
|
||||||
|
// Find root CA
|
||||||
|
if (certChain.ChainElements.Count == 0)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
var rootCert = certChain.ChainElements[certChain.ChainElements.Count - 1].Certificate;
|
||||||
|
var rootThumbprint = rootCert.GetCertHashString(HashAlgorithmName.SHA256);
|
||||||
|
|
||||||
|
return TrustedCaThumbprints.Contains(rootThumbprint);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static string ExportCertToPem(X509Certificate2 cert)
|
||||||
|
{
|
||||||
|
var der = cert.Export(X509ContentType.Cert);
|
||||||
|
var b64 = Convert.ToBase64String(der);
|
||||||
|
return $"-----BEGIN CERTIFICATE-----\n{b64}\n-----END CERTIFICATE-----\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Parse concatenated PEM certificates string into a list of individual certificates
|
||||||
|
/// Normalizes format: removes line breaks from base64 content for better compatibility
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="pemChain">Concatenated PEM certificates string (supports both \r\n and \n line endings)</param>
|
||||||
|
/// <returns>List of individual PEM certificate strings with normalized format</returns>
|
||||||
|
public static List<string> ParsePemChain(string pemChain)
|
||||||
|
{
|
||||||
|
var certs = new List<string>();
|
||||||
|
if (string.IsNullOrWhiteSpace(pemChain))
|
||||||
|
{
|
||||||
|
return certs;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Normalize line endings (CRLF -> LF) at the beginning
|
||||||
|
pemChain = pemChain.Replace("\r\n", "\n").Replace("\r", "\n");
|
||||||
|
|
||||||
|
const string beginMarker = "-----BEGIN CERTIFICATE-----";
|
||||||
|
const string endMarker = "-----END CERTIFICATE-----";
|
||||||
|
|
||||||
|
var index = 0;
|
||||||
|
while (index < pemChain.Length)
|
||||||
|
{
|
||||||
|
var beginIndex = pemChain.IndexOf(beginMarker, index, StringComparison.Ordinal);
|
||||||
|
if (beginIndex == -1)
|
||||||
|
{
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
var endIndex = pemChain.IndexOf(endMarker, beginIndex, StringComparison.Ordinal);
|
||||||
|
if (endIndex == -1)
|
||||||
|
{
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Extract certificate content
|
||||||
|
var base64Start = beginIndex + beginMarker.Length;
|
||||||
|
var base64Content = pemChain.Substring(base64Start, endIndex - base64Start);
|
||||||
|
|
||||||
|
// Remove all whitespace from base64 content
|
||||||
|
base64Content = new string(base64Content.Where(c => !char.IsWhiteSpace(c)).ToArray());
|
||||||
|
|
||||||
|
// Reconstruct with clean format: BEGIN marker + base64 (no line breaks) + END marker
|
||||||
|
var normalizedCert = $"{beginMarker}\n{base64Content}\n{endMarker}\n";
|
||||||
|
certs.Add(normalizedCert);
|
||||||
|
|
||||||
|
// Move to next certificate
|
||||||
|
index = endIndex + endMarker.Length;
|
||||||
|
}
|
||||||
|
|
||||||
|
return certs;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Concatenate a list of PEM certificates into a single string
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="pemList">List of individual PEM certificate strings</param>
|
||||||
|
/// <returns>Concatenated PEM certificates string</returns>
|
||||||
|
public static string ConcatenatePemChain(IEnumerable<string> pemList)
|
||||||
|
{
|
||||||
|
if (pemList == null)
|
||||||
|
{
|
||||||
|
return string.Empty;
|
||||||
|
}
|
||||||
|
|
||||||
|
return string.Concat(pemList);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -34,8 +34,8 @@ public class CoreAdminManager
|
|||||||
StringBuilder sb = new();
|
StringBuilder sb = new();
|
||||||
sb.AppendLine("#!/bin/bash");
|
sb.AppendLine("#!/bin/bash");
|
||||||
var cmdLine = $"{fileName.AppendQuotes()} {string.Format(coreInfo.Arguments, Utils.GetBinConfigPath(configPath).AppendQuotes())}";
|
var cmdLine = $"{fileName.AppendQuotes()} {string.Format(coreInfo.Arguments, Utils.GetBinConfigPath(configPath).AppendQuotes())}";
|
||||||
sb.AppendLine($"sudo -S {cmdLine}");
|
sb.AppendLine($"exec sudo -S -- {cmdLine}");
|
||||||
var shFilePath = await FileManager.CreateLinuxShellFile("run_as_sudo.sh", sb.ToString(), true);
|
var shFilePath = await FileUtils.CreateLinuxShellFile("run_as_sudo.sh", sb.ToString(), true);
|
||||||
|
|
||||||
var procService = new ProcessService(
|
var procService = new ProcessService(
|
||||||
fileName: shFilePath,
|
fileName: shFilePath,
|
||||||
@@ -67,8 +67,8 @@ public class CoreAdminManager
|
|||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
var shellFileName = Utils.IsOSX() ? Global.KillAsSudoOSXShellFileName : Global.KillAsSudoLinuxShellFileName;
|
var shellFileName = Utils.IsMacOS() ? Global.KillAsSudoOSXShellFileName : Global.KillAsSudoLinuxShellFileName;
|
||||||
var shFilePath = await FileManager.CreateLinuxShellFile("kill_as_sudo.sh", EmbedUtils.GetEmbedText(shellFileName), true);
|
var shFilePath = await FileUtils.CreateLinuxShellFile("kill_as_sudo.sh", EmbedUtils.GetEmbedText(shellFileName), true);
|
||||||
if (shFilePath.Contains(' '))
|
if (shFilePath.Contains(' '))
|
||||||
{
|
{
|
||||||
shFilePath = shFilePath.AppendQuotes();
|
shFilePath = shFilePath.AppendQuotes();
|
||||||
|
|||||||
@@ -8,7 +8,7 @@ public class CoreManager
|
|||||||
private static readonly Lazy<CoreManager> _instance = new(() => new());
|
private static readonly Lazy<CoreManager> _instance = new(() => new());
|
||||||
public static CoreManager Instance => _instance.Value;
|
public static CoreManager Instance => _instance.Value;
|
||||||
private Config _config;
|
private Config _config;
|
||||||
private WindowsJob? _processJob;
|
private WindowsJobService? _processJob;
|
||||||
private ProcessService? _processService;
|
private ProcessService? _processService;
|
||||||
private ProcessService? _processPreService;
|
private ProcessService? _processPreService;
|
||||||
private bool _linuxSudo = false;
|
private bool _linuxSudo = false;
|
||||||
@@ -27,7 +27,7 @@ public class CoreManager
|
|||||||
var toPath = Utils.GetBinPath("");
|
var toPath = Utils.GetBinPath("");
|
||||||
if (fromPath != toPath)
|
if (fromPath != toPath)
|
||||||
{
|
{
|
||||||
FileManager.CopyDirectory(fromPath, toPath, true, false);
|
FileUtils.CopyDirectory(fromPath, toPath, true, false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -5,7 +5,6 @@ public class PacManager
|
|||||||
private static readonly Lazy<PacManager> _instance = new(() => new PacManager());
|
private static readonly Lazy<PacManager> _instance = new(() => new PacManager());
|
||||||
public static PacManager Instance => _instance.Value;
|
public static PacManager Instance => _instance.Value;
|
||||||
|
|
||||||
private string _configPath;
|
|
||||||
private int _httpPort;
|
private int _httpPort;
|
||||||
private int _pacPort;
|
private int _pacPort;
|
||||||
private TcpListener? _tcpListener;
|
private TcpListener? _tcpListener;
|
||||||
@@ -13,11 +12,10 @@ public class PacManager
|
|||||||
private bool _isRunning;
|
private bool _isRunning;
|
||||||
private bool _needRestart = true;
|
private bool _needRestart = true;
|
||||||
|
|
||||||
public async Task StartAsync(string configPath, int httpPort, int pacPort)
|
public async Task StartAsync(int httpPort, int pacPort)
|
||||||
{
|
{
|
||||||
_needRestart = configPath != _configPath || httpPort != _httpPort || pacPort != _pacPort || !_isRunning;
|
_needRestart = httpPort != _httpPort || pacPort != _pacPort || !_isRunning;
|
||||||
|
|
||||||
_configPath = configPath;
|
|
||||||
_httpPort = httpPort;
|
_httpPort = httpPort;
|
||||||
_pacPort = pacPort;
|
_pacPort = pacPort;
|
||||||
|
|
||||||
@@ -32,22 +30,22 @@ public class PacManager
|
|||||||
|
|
||||||
private async Task InitText()
|
private async Task InitText()
|
||||||
{
|
{
|
||||||
var path = Path.Combine(_configPath, "pac.txt");
|
var customSystemProxyPacPath = AppManager.Instance.Config.SystemProxyItem?.CustomSystemProxyPacPath;
|
||||||
|
var fileName = (customSystemProxyPacPath.IsNotEmpty() && File.Exists(customSystemProxyPacPath))
|
||||||
|
? customSystemProxyPacPath
|
||||||
|
: Path.Combine(Utils.GetConfigPath(), "pac.txt");
|
||||||
|
|
||||||
// Delete the old pac file
|
// TODO: temporarily notify which script is being used
|
||||||
if (File.Exists(path) && Utils.GetFileHash(path).Equals("b590c07280f058ef05d5394aa2f927fe"))
|
NoticeManager.Instance.SendMessage(fileName);
|
||||||
{
|
|
||||||
File.Delete(path);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!File.Exists(path))
|
if (!File.Exists(fileName))
|
||||||
{
|
{
|
||||||
var pac = EmbedUtils.GetEmbedText(Global.PacFileName);
|
var pac = EmbedUtils.GetEmbedText(Global.PacFileName);
|
||||||
await File.AppendAllTextAsync(path, pac);
|
await File.AppendAllTextAsync(fileName, pac);
|
||||||
}
|
}
|
||||||
|
|
||||||
var pacText =
|
var pacText = await File.ReadAllTextAsync(fileName);
|
||||||
(await File.ReadAllTextAsync(path)).Replace("__PROXY__", $"PROXY 127.0.0.1:{_httpPort};DIRECT;");
|
pacText = pacText.Replace("__PROXY__", $"PROXY 127.0.0.1:{_httpPort};DIRECT;");
|
||||||
|
|
||||||
var sb = new StringBuilder();
|
var sb = new StringBuilder();
|
||||||
sb.AppendLine("HTTP/1.0 200 OK");
|
sb.AppendLine("HTTP/1.0 200 OK");
|
||||||
|
|||||||
@@ -173,13 +173,19 @@ public class ProfileGroupItemManager
|
|||||||
public static bool HasCycle(string? indexId, HashSet<string> visited, HashSet<string> stack)
|
public static bool HasCycle(string? indexId, HashSet<string> visited, HashSet<string> stack)
|
||||||
{
|
{
|
||||||
if (indexId.IsNullOrEmpty())
|
if (indexId.IsNullOrEmpty())
|
||||||
|
{
|
||||||
return false;
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
if (stack.Contains(indexId))
|
if (stack.Contains(indexId))
|
||||||
|
{
|
||||||
return true;
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
if (visited.Contains(indexId))
|
if (visited.Contains(indexId))
|
||||||
|
{
|
||||||
return false;
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
visited.Add(indexId);
|
visited.Add(indexId);
|
||||||
stack.Add(indexId);
|
stack.Add(indexId);
|
||||||
@@ -220,11 +226,14 @@ public class ProfileGroupItemManager
|
|||||||
public static async Task<(List<ProfileItem> Items, ProfileGroupItem? Group)> GetChildProfileItems(string? indexId)
|
public static async Task<(List<ProfileItem> Items, ProfileGroupItem? Group)> GetChildProfileItems(string? indexId)
|
||||||
{
|
{
|
||||||
Instance.TryGet(indexId, out var profileGroupItem);
|
Instance.TryGet(indexId, out var profileGroupItem);
|
||||||
if (profileGroupItem == null || profileGroupItem.ChildItems.IsNullOrEmpty())
|
if (profileGroupItem == null || profileGroupItem.NotHasChild())
|
||||||
{
|
{
|
||||||
return (new List<ProfileItem>(), profileGroupItem);
|
return (new List<ProfileItem>(), profileGroupItem);
|
||||||
}
|
}
|
||||||
var items = await GetChildProfileItems(profileGroupItem);
|
var items = await GetChildProfileItems(profileGroupItem);
|
||||||
|
var subItems = await GetSubChildProfileItems(profileGroupItem);
|
||||||
|
items.AddRange(subItems);
|
||||||
|
|
||||||
return (items, profileGroupItem);
|
return (items, profileGroupItem);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -248,20 +257,47 @@ public class ProfileGroupItemManager
|
|||||||
return childProfiles;
|
return childProfiles;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static async Task<List<ProfileItem>> GetSubChildProfileItems(ProfileGroupItem? group)
|
||||||
|
{
|
||||||
|
if (group == null || group.SubChildItems.IsNullOrEmpty())
|
||||||
|
{
|
||||||
|
return new();
|
||||||
|
}
|
||||||
|
var childProfiles = await AppManager.Instance.ProfileItems(group.SubChildItems);
|
||||||
|
|
||||||
|
return childProfiles.Where(p =>
|
||||||
|
p != null &&
|
||||||
|
p.IsValid() &&
|
||||||
|
!p.ConfigType.IsComplexType() &&
|
||||||
|
(group.Filter.IsNullOrEmpty() || Regex.IsMatch(p.Remarks, group.Filter))
|
||||||
|
)
|
||||||
|
.ToList();
|
||||||
|
}
|
||||||
|
|
||||||
public static async Task<HashSet<string>> GetAllChildDomainAddresses(string indexId)
|
public static async Task<HashSet<string>> GetAllChildDomainAddresses(string indexId)
|
||||||
{
|
{
|
||||||
// include grand children
|
// include grand children
|
||||||
var childAddresses = new HashSet<string>();
|
var childAddresses = new HashSet<string>();
|
||||||
if (!Instance.TryGet(indexId, out var groupItem) || groupItem.ChildItems.IsNullOrEmpty())
|
if (!Instance.TryGet(indexId, out var groupItem) || groupItem == null)
|
||||||
|
{
|
||||||
return childAddresses;
|
return childAddresses;
|
||||||
|
}
|
||||||
|
|
||||||
var childIds = Utils.String2List(groupItem.ChildItems);
|
if (groupItem.SubChildItems.IsNotEmpty())
|
||||||
|
{
|
||||||
|
var subItems = await GetSubChildProfileItems(groupItem);
|
||||||
|
subItems.ForEach(p => childAddresses.Add(p.Address));
|
||||||
|
}
|
||||||
|
|
||||||
|
var childIds = Utils.String2List(groupItem.ChildItems) ?? [];
|
||||||
|
|
||||||
foreach (var childId in childIds)
|
foreach (var childId in childIds)
|
||||||
{
|
{
|
||||||
var childNode = await AppManager.Instance.GetProfileItem(childId);
|
var childNode = await AppManager.Instance.GetProfileItem(childId);
|
||||||
if (childNode == null)
|
if (childNode == null)
|
||||||
|
{
|
||||||
continue;
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
if (!childNode.IsComplex())
|
if (!childNode.IsComplex())
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -56,9 +56,9 @@ public class TaskManager
|
|||||||
{
|
{
|
||||||
//Logging.SaveLog("Execute delete expired files");
|
//Logging.SaveLog("Execute delete expired files");
|
||||||
|
|
||||||
FileManager.DeleteExpiredFiles(Utils.GetBinConfigPath(), DateTime.Now.AddHours(-1));
|
FileUtils.DeleteExpiredFiles(Utils.GetBinConfigPath(), DateTime.Now.AddHours(-1));
|
||||||
FileManager.DeleteExpiredFiles(Utils.GetLogPath(), DateTime.Now.AddMonths(-1));
|
FileUtils.DeleteExpiredFiles(Utils.GetLogPath(), DateTime.Now.AddMonths(-1));
|
||||||
FileManager.DeleteExpiredFiles(Utils.GetTempPath(), DateTime.Now.AddMonths(-1));
|
FileUtils.DeleteExpiredFiles(Utils.GetTempPath(), DateTime.Now.AddMonths(-1));
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
@@ -111,11 +111,10 @@ public class TaskManager
|
|||||||
{
|
{
|
||||||
Logging.SaveLog("Execute update geo files");
|
Logging.SaveLog("Execute update geo files");
|
||||||
|
|
||||||
var updateHandle = new UpdateService();
|
await new UpdateService(_config, async (success, msg) =>
|
||||||
await updateHandle.UpdateGeoFileAll(_config, async (success, msg) =>
|
|
||||||
{
|
{
|
||||||
await _updateFunc?.Invoke(false, msg);
|
await _updateFunc?.Invoke(false, msg);
|
||||||
});
|
}).UpdateGeoFileAll();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -219,6 +219,8 @@ public class SystemProxyItem
|
|||||||
public string SystemProxyExceptions { get; set; }
|
public string SystemProxyExceptions { get; set; }
|
||||||
public bool NotProxyLocalAddress { get; set; } = true;
|
public bool NotProxyLocalAddress { get; set; } = true;
|
||||||
public string SystemProxyAdvancedProtocol { get; set; }
|
public string SystemProxyAdvancedProtocol { get; set; }
|
||||||
|
public string? CustomSystemProxyPacPath { get; set; }
|
||||||
|
public string? CustomSystemProxyScriptPath { get; set; }
|
||||||
}
|
}
|
||||||
|
|
||||||
[Serializable]
|
[Serializable]
|
||||||
|
|||||||
@@ -8,5 +8,14 @@ public class ProfileGroupItem
|
|||||||
|
|
||||||
public string ChildItems { get; set; }
|
public string ChildItems { get; set; }
|
||||||
|
|
||||||
|
public string? SubChildItems { get; set; }
|
||||||
|
|
||||||
|
public string? Filter { get; set; }
|
||||||
|
|
||||||
public EMultipleLoad MultipleLoad { get; set; } = EMultipleLoad.LeastPing;
|
public EMultipleLoad MultipleLoad { get; set; } = EMultipleLoad.LeastPing;
|
||||||
|
|
||||||
|
public bool NotHasChild()
|
||||||
|
{
|
||||||
|
return string.IsNullOrWhiteSpace(ChildItems) && string.IsNullOrWhiteSpace(SubChildItems);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -28,7 +28,7 @@ public class ProfileItem : ReactiveObject
|
|||||||
|
|
||||||
public string GetSummary()
|
public string GetSummary()
|
||||||
{
|
{
|
||||||
var summary = $"[{(ConfigType).ToString()}] ";
|
var summary = $"[{ConfigType.ToString()}] ";
|
||||||
if (IsComplex())
|
if (IsComplex())
|
||||||
{
|
{
|
||||||
summary += $"[{CoreType.ToString()}]{Remarks}";
|
summary += $"[{CoreType.ToString()}]{Remarks}";
|
||||||
@@ -69,30 +69,49 @@ public class ProfileItem : ReactiveObject
|
|||||||
public bool IsValid()
|
public bool IsValid()
|
||||||
{
|
{
|
||||||
if (IsComplex())
|
if (IsComplex())
|
||||||
|
{
|
||||||
return true;
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
if (Address.IsNullOrEmpty() || Port is <= 0 or >= 65536)
|
if (Address.IsNullOrEmpty() || Port is <= 0 or >= 65536)
|
||||||
|
{
|
||||||
return false;
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
switch (ConfigType)
|
switch (ConfigType)
|
||||||
{
|
{
|
||||||
case EConfigType.VMess:
|
case EConfigType.VMess:
|
||||||
if (Id.IsNullOrEmpty() || !Utils.IsGuidByParse(Id))
|
if (Id.IsNullOrEmpty() || !Utils.IsGuidByParse(Id))
|
||||||
|
{
|
||||||
return false;
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case EConfigType.VLESS:
|
case EConfigType.VLESS:
|
||||||
if (Id.IsNullOrEmpty() || (!Utils.IsGuidByParse(Id) && Id.Length > 30))
|
if (Id.IsNullOrEmpty() || (!Utils.IsGuidByParse(Id) && Id.Length > 30))
|
||||||
|
{
|
||||||
return false;
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
if (!Global.Flows.Contains(Flow))
|
if (!Global.Flows.Contains(Flow))
|
||||||
|
{
|
||||||
return false;
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case EConfigType.Shadowsocks:
|
case EConfigType.Shadowsocks:
|
||||||
if (Id.IsNullOrEmpty())
|
if (Id.IsNullOrEmpty())
|
||||||
|
{
|
||||||
return false;
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
if (string.IsNullOrEmpty(Security) || !Global.SsSecuritiesInSingbox.Contains(Security))
|
if (string.IsNullOrEmpty(Security) || !Global.SsSecuritiesInSingbox.Contains(Security))
|
||||||
|
{
|
||||||
return false;
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -141,4 +160,5 @@ public class ProfileItem : ReactiveObject
|
|||||||
public string Mldsa65Verify { get; set; }
|
public string Mldsa65Verify { get; set; }
|
||||||
public string Extra { get; set; }
|
public string Extra { get; set; }
|
||||||
public bool? MuxEnabled { get; set; }
|
public bool? MuxEnabled { get; set; }
|
||||||
|
public string Cert { get; set; }
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
namespace ServiceLib.Common;
|
namespace ServiceLib.Models;
|
||||||
|
|
||||||
public class SemanticVersion
|
public class SemanticVersion
|
||||||
{
|
{
|
||||||
@@ -181,6 +181,7 @@ public class Tls4Sbox
|
|||||||
public bool? fragment { get; set; }
|
public bool? fragment { get; set; }
|
||||||
public string? fragment_fallback_delay { get; set; }
|
public string? fragment_fallback_delay { get; set; }
|
||||||
public bool? record_fragment { get; set; }
|
public bool? record_fragment { get; set; }
|
||||||
|
public List<string>? certificate { get; set; }
|
||||||
}
|
}
|
||||||
|
|
||||||
public class Multiplex4Sbox
|
public class Multiplex4Sbox
|
||||||
@@ -215,6 +216,8 @@ public class Transport4Sbox
|
|||||||
public string? idle_timeout { get; set; }
|
public string? idle_timeout { get; set; }
|
||||||
public string? ping_timeout { get; set; }
|
public string? ping_timeout { get; set; }
|
||||||
public bool? permit_without_stream { get; set; }
|
public bool? permit_without_stream { get; set; }
|
||||||
|
public int? max_early_data { get; set; }
|
||||||
|
public string? early_data_header_name { get; set; }
|
||||||
}
|
}
|
||||||
|
|
||||||
public class Headers4Sbox
|
public class Headers4Sbox
|
||||||
|
|||||||
21
v2rayN/ServiceLib/Models/UpdateResult.cs
Normal file
21
v2rayN/ServiceLib/Models/UpdateResult.cs
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
namespace ServiceLib.Models;
|
||||||
|
|
||||||
|
public class UpdateResult
|
||||||
|
{
|
||||||
|
public bool Success { get; set; }
|
||||||
|
public string? Msg { get; set; }
|
||||||
|
public SemanticVersion? Version { get; set; }
|
||||||
|
public string? Url { get; set; }
|
||||||
|
|
||||||
|
public UpdateResult(bool success, string? msg)
|
||||||
|
{
|
||||||
|
Success = success;
|
||||||
|
Msg = msg;
|
||||||
|
}
|
||||||
|
|
||||||
|
public UpdateResult(bool success, SemanticVersion? version)
|
||||||
|
{
|
||||||
|
Success = success;
|
||||||
|
Version = version;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -215,6 +215,7 @@ public class Dns4Ray
|
|||||||
public class DnsServer4Ray
|
public class DnsServer4Ray
|
||||||
{
|
{
|
||||||
public string? address { get; set; }
|
public string? address { get; set; }
|
||||||
|
public int? port { get; set; }
|
||||||
public List<string>? domains { get; set; }
|
public List<string>? domains { get; set; }
|
||||||
public bool? skipFallback { get; set; }
|
public bool? skipFallback { get; set; }
|
||||||
public List<string>? expectedIPs { get; set; }
|
public List<string>? expectedIPs { get; set; }
|
||||||
@@ -353,6 +354,14 @@ public class TlsSettings4Ray
|
|||||||
public string? shortId { get; set; }
|
public string? shortId { get; set; }
|
||||||
public string? spiderX { get; set; }
|
public string? spiderX { get; set; }
|
||||||
public string? mldsa65Verify { get; set; }
|
public string? mldsa65Verify { get; set; }
|
||||||
|
public List<CertificateSettings4Ray>? certificates { get; set; }
|
||||||
|
public bool? disableSystemRoot { get; set; }
|
||||||
|
}
|
||||||
|
|
||||||
|
public class CertificateSettings4Ray
|
||||||
|
{
|
||||||
|
public List<string>? certificate { get; set; }
|
||||||
|
public string? usage { get; set; }
|
||||||
}
|
}
|
||||||
|
|
||||||
public class TcpSettings4Ray
|
public class TcpSettings4Ray
|
||||||
@@ -402,8 +411,6 @@ public class WsSettings4Ray
|
|||||||
|
|
||||||
public class Headers4Ray
|
public class Headers4Ray
|
||||||
{
|
{
|
||||||
public string Host { get; set; }
|
|
||||||
|
|
||||||
[JsonPropertyName("User-Agent")]
|
[JsonPropertyName("User-Agent")]
|
||||||
public string UserAgent { get; set; }
|
public string UserAgent { get; set; }
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -38,4 +38,6 @@ public class VmessQRCode
|
|||||||
public string alpn { get; set; } = string.Empty;
|
public string alpn { get; set; } = string.Empty;
|
||||||
|
|
||||||
public string fp { get; set; } = string.Empty;
|
public string fp { get; set; } = string.Empty;
|
||||||
|
|
||||||
|
public string insecure { get; set; } = string.Empty;
|
||||||
}
|
}
|
||||||
|
|||||||
241
v2rayN/ServiceLib/Resx/ResUI.Designer.cs
generated
241
v2rayN/ServiceLib/Resx/ResUI.Designer.cs
generated
@@ -87,6 +87,24 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// 查找类似 Certificate not set 的本地化字符串。
|
||||||
|
/// </summary>
|
||||||
|
public static string CertNotSet {
|
||||||
|
get {
|
||||||
|
return ResourceManager.GetString("CertNotSet", resourceCulture);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// 查找类似 Certificate set 的本地化字符串。
|
||||||
|
/// </summary>
|
||||||
|
public static string CertSet {
|
||||||
|
get {
|
||||||
|
return ResourceManager.GetString("CertSet", resourceCulture);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Please check the Configuration settings first. 的本地化字符串。
|
/// 查找类似 Please check the Configuration settings first. 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@@ -511,7 +529,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Next proxy Configuration remarks 的本地化字符串。
|
/// 查找类似 Next proxy remarks 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string LvNextProfile {
|
public static string LvNextProfile {
|
||||||
get {
|
get {
|
||||||
@@ -529,7 +547,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Previous proxy Configuration remarks 的本地化字符串。
|
/// 查找类似 Previous proxy remarks 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string LvPrevProfile {
|
public static string LvPrevProfile {
|
||||||
get {
|
get {
|
||||||
@@ -718,7 +736,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Add [Anytls] Configuration 的本地化字符串。
|
/// 查找类似 Add [Anytls] 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuAddAnytlsServer {
|
public static string menuAddAnytlsServer {
|
||||||
get {
|
get {
|
||||||
@@ -727,7 +745,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Add Child Configuration 的本地化字符串。
|
/// 查找类似 Add Child 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuAddChildServer {
|
public static string menuAddChildServer {
|
||||||
get {
|
get {
|
||||||
@@ -736,7 +754,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Add a custom configuration Configuration 的本地化字符串。
|
/// 查找类似 Add a custom configuration 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuAddCustomServer {
|
public static string menuAddCustomServer {
|
||||||
get {
|
get {
|
||||||
@@ -745,7 +763,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Add [HTTP] Configuration 的本地化字符串。
|
/// 查找类似 Add [HTTP] 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuAddHttpServer {
|
public static string menuAddHttpServer {
|
||||||
get {
|
get {
|
||||||
@@ -754,7 +772,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Add [Hysteria2] Configuration 的本地化字符串。
|
/// 查找类似 Add [Hysteria2] 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuAddHysteria2Server {
|
public static string menuAddHysteria2Server {
|
||||||
get {
|
get {
|
||||||
@@ -763,7 +781,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Add Policy Group Configuration 的本地化字符串。
|
/// 查找类似 Add Policy Group 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuAddPolicyGroupServer {
|
public static string menuAddPolicyGroupServer {
|
||||||
get {
|
get {
|
||||||
@@ -772,7 +790,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Add Proxy Chain Configuration 的本地化字符串。
|
/// 查找类似 Add Proxy Chain 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuAddProxyChainServer {
|
public static string menuAddProxyChainServer {
|
||||||
get {
|
get {
|
||||||
@@ -781,7 +799,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Import Share Links from clipboard (Ctrl+V) 的本地化字符串。
|
/// 查找类似 Import Share Links from clipboard 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuAddServerViaClipboard {
|
public static string menuAddServerViaClipboard {
|
||||||
get {
|
get {
|
||||||
@@ -799,7 +817,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Scan QR code on the screen (Ctrl+S) 的本地化字符串。
|
/// 查找类似 Scan QR code on the screen 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuAddServerViaScan {
|
public static string menuAddServerViaScan {
|
||||||
get {
|
get {
|
||||||
@@ -808,7 +826,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Add [Shadowsocks] Configuration 的本地化字符串。
|
/// 查找类似 Add [Shadowsocks] 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuAddShadowsocksServer {
|
public static string menuAddShadowsocksServer {
|
||||||
get {
|
get {
|
||||||
@@ -817,7 +835,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Add [SOCKS] Configuration 的本地化字符串。
|
/// 查找类似 Add [SOCKS] 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuAddSocksServer {
|
public static string menuAddSocksServer {
|
||||||
get {
|
get {
|
||||||
@@ -826,7 +844,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Add [Trojan] Configuration 的本地化字符串。
|
/// 查找类似 Add [Trojan] 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuAddTrojanServer {
|
public static string menuAddTrojanServer {
|
||||||
get {
|
get {
|
||||||
@@ -835,7 +853,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Add [TUIC] Configuration 的本地化字符串。
|
/// 查找类似 Add [TUIC] 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuAddTuicServer {
|
public static string menuAddTuicServer {
|
||||||
get {
|
get {
|
||||||
@@ -844,7 +862,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Add [VLESS] Configuration 的本地化字符串。
|
/// 查找类似 Add [VLESS] 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuAddVlessServer {
|
public static string menuAddVlessServer {
|
||||||
get {
|
get {
|
||||||
@@ -853,7 +871,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Add [VMess] Configuration 的本地化字符串。
|
/// 查找类似 Add [VMess] 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuAddVmessServer {
|
public static string menuAddVmessServer {
|
||||||
get {
|
get {
|
||||||
@@ -862,7 +880,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Add [WireGuard] Configuration 的本地化字符串。
|
/// 查找类似 Add [WireGuard] 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuAddWireguardServer {
|
public static string menuAddWireguardServer {
|
||||||
get {
|
get {
|
||||||
@@ -934,7 +952,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Clone selected Configuration 的本地化字符串。
|
/// 查找类似 Clone selected 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuCopyServer {
|
public static string menuCopyServer {
|
||||||
get {
|
get {
|
||||||
@@ -952,7 +970,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Edit Configuration (Ctrl+D) 的本地化字符串。
|
/// 查找类似 Edit 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuEditServer {
|
public static string menuEditServer {
|
||||||
get {
|
get {
|
||||||
@@ -979,7 +997,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Export selected Configuration for complete configuration 的本地化字符串。
|
/// 查找类似 Export selected for complete configuration 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuExport2ClientConfig {
|
public static string menuExport2ClientConfig {
|
||||||
get {
|
get {
|
||||||
@@ -988,7 +1006,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Export selected Configuration for complete configuration to clipboard 的本地化字符串。
|
/// 查找类似 Export selected for complete configuration to clipboard 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuExport2ClientConfigClipboard {
|
public static string menuExport2ClientConfigClipboard {
|
||||||
get {
|
get {
|
||||||
@@ -997,7 +1015,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Export Share Link to Clipboard (Ctrl+C) 的本地化字符串。
|
/// 查找类似 Export Share Link to Clipboard 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuExport2ShareUrl {
|
public static string menuExport2ShareUrl {
|
||||||
get {
|
get {
|
||||||
@@ -1015,7 +1033,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Export Configuration 的本地化字符串。
|
/// 查找类似 Export 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuExportConfig {
|
public static string menuExportConfig {
|
||||||
get {
|
get {
|
||||||
@@ -1023,6 +1041,15 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// 查找类似 Test real delay 的本地化字符串。
|
||||||
|
/// </summary>
|
||||||
|
public static string menuFastRealPing {
|
||||||
|
get {
|
||||||
|
return ResourceManager.GetString("menuFastRealPing", resourceCulture);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Full Config Template Setting 的本地化字符串。
|
/// 查找类似 Full Config Template Setting 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@@ -1042,7 +1069,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Multi-Configuration Fallback by sing-box 的本地化字符串。
|
/// 查找类似 Fallback by sing-box 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuGenGroupMultipleServerSingBoxFallback {
|
public static string menuGenGroupMultipleServerSingBoxFallback {
|
||||||
get {
|
get {
|
||||||
@@ -1051,7 +1078,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Multi-Configuration LeastPing by sing-box 的本地化字符串。
|
/// 查找类似 LeastPing by sing-box 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuGenGroupMultipleServerSingBoxLeastPing {
|
public static string menuGenGroupMultipleServerSingBoxLeastPing {
|
||||||
get {
|
get {
|
||||||
@@ -1060,7 +1087,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Multi-Configuration Fallback by Xray 的本地化字符串。
|
/// 查找类似 Fallback by Xray 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuGenGroupMultipleServerXrayFallback {
|
public static string menuGenGroupMultipleServerXrayFallback {
|
||||||
get {
|
get {
|
||||||
@@ -1069,7 +1096,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Multi-Configuration LeastLoad by Xray 的本地化字符串。
|
/// 查找类似 LeastLoad by Xray 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuGenGroupMultipleServerXrayLeastLoad {
|
public static string menuGenGroupMultipleServerXrayLeastLoad {
|
||||||
get {
|
get {
|
||||||
@@ -1078,7 +1105,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Multi-Configuration LeastPing by Xray 的本地化字符串。
|
/// 查找类似 LeastPing by Xray 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuGenGroupMultipleServerXrayLeastPing {
|
public static string menuGenGroupMultipleServerXrayLeastPing {
|
||||||
get {
|
get {
|
||||||
@@ -1087,7 +1114,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Multi-Configuration Random by Xray 的本地化字符串。
|
/// 查找类似 Random by Xray 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuGenGroupMultipleServerXrayRandom {
|
public static string menuGenGroupMultipleServerXrayRandom {
|
||||||
get {
|
get {
|
||||||
@@ -1096,7 +1123,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Multi-Configuration RoundRobin by Xray 的本地化字符串。
|
/// 查找类似 RoundRobin by Xray 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuGenGroupMultipleServerXrayRoundRobin {
|
public static string menuGenGroupMultipleServerXrayRoundRobin {
|
||||||
get {
|
get {
|
||||||
@@ -1222,7 +1249,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Move to bottom (B) 的本地化字符串。
|
/// 查找类似 Move to bottom 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuMoveBottom {
|
public static string menuMoveBottom {
|
||||||
get {
|
get {
|
||||||
@@ -1231,7 +1258,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Down (D) 的本地化字符串。
|
/// 查找类似 Down 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuMoveDown {
|
public static string menuMoveDown {
|
||||||
get {
|
get {
|
||||||
@@ -1258,7 +1285,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Move to top (T) 的本地化字符串。
|
/// 查找类似 Move to top 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuMoveTop {
|
public static string menuMoveTop {
|
||||||
get {
|
get {
|
||||||
@@ -1267,7 +1294,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Up (U) 的本地化字符串。
|
/// 查找类似 Up 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuMoveUp {
|
public static string menuMoveUp {
|
||||||
get {
|
get {
|
||||||
@@ -1285,7 +1312,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Copy (Ctrl+C) 的本地化字符串。
|
/// 查找类似 Copy 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuMsgViewCopy {
|
public static string menuMsgViewCopy {
|
||||||
get {
|
get {
|
||||||
@@ -1303,7 +1330,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Select all (Ctrl+A) 的本地化字符串。
|
/// 查找类似 Select all 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuMsgViewSelectAll {
|
public static string menuMsgViewSelectAll {
|
||||||
get {
|
get {
|
||||||
@@ -1375,7 +1402,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Select active node (Enter) 的本地化字符串。
|
/// 查找类似 Select active node 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuProxiesSelectActivity {
|
public static string menuProxiesSelectActivity {
|
||||||
get {
|
get {
|
||||||
@@ -1384,7 +1411,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Test Configurations real delay (Ctrl+R) 的本地化字符串。
|
/// 查找类似 Test real delay 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuRealPingServer {
|
public static string menuRealPingServer {
|
||||||
get {
|
get {
|
||||||
@@ -1474,7 +1501,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Remove Child Configuration 的本地化字符串。
|
/// 查找类似 Remove Child 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuRemoveChildServer {
|
public static string menuRemoveChildServer {
|
||||||
get {
|
get {
|
||||||
@@ -1483,7 +1510,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Remove duplicate Configurations 的本地化字符串。
|
/// 查找类似 Remove duplicate 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuRemoveDuplicateServer {
|
public static string menuRemoveDuplicateServer {
|
||||||
get {
|
get {
|
||||||
@@ -1501,7 +1528,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Remove selected Configurations (Delete) 的本地化字符串。
|
/// 查找类似 Remove selected 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuRemoveServer {
|
public static string menuRemoveServer {
|
||||||
get {
|
get {
|
||||||
@@ -1537,7 +1564,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Remove selected (Delete) 的本地化字符串。
|
/// 查找类似 Remove selected 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuRoutingAdvancedRemove {
|
public static string menuRoutingAdvancedRemove {
|
||||||
get {
|
get {
|
||||||
@@ -1546,7 +1573,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Set as active rule (Enter) 的本地化字符串。
|
/// 查找类似 Set as active rule 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuRoutingAdvancedSetDefault {
|
public static string menuRoutingAdvancedSetDefault {
|
||||||
get {
|
get {
|
||||||
@@ -1618,7 +1645,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Remove Rule (Delete) 的本地化字符串。
|
/// 查找类似 Remove Rule 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuRuleRemove {
|
public static string menuRuleRemove {
|
||||||
get {
|
get {
|
||||||
@@ -1627,7 +1654,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Select all (Ctrl+A) 的本地化字符串。
|
/// 查找类似 Select all 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuSelectAll {
|
public static string menuSelectAll {
|
||||||
get {
|
get {
|
||||||
@@ -1636,7 +1663,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Server List 的本地化字符串。
|
/// 查找类似 Configuration item 1, Auto add from subscription group 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuServerList {
|
public static string menuServerList {
|
||||||
get {
|
get {
|
||||||
@@ -1645,7 +1672,16 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Configurations 的本地化字符串。
|
/// 查找类似 Configuration Item 2, Select and add from self-built 的本地化字符串。
|
||||||
|
/// </summary>
|
||||||
|
public static string menuServerList2 {
|
||||||
|
get {
|
||||||
|
return ResourceManager.GetString("menuServerList2", resourceCulture);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// 查找类似 Configuration 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuServers {
|
public static string menuServers {
|
||||||
get {
|
get {
|
||||||
@@ -1654,7 +1690,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Set as active Configuration (Enter) 的本地化字符串。
|
/// 查找类似 Set as active 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuSetDefaultServer {
|
public static string menuSetDefaultServer {
|
||||||
get {
|
get {
|
||||||
@@ -1672,7 +1708,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Share Configuration (Ctrl+F) 的本地化字符串。
|
/// 查找类似 Share 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuShareServer {
|
public static string menuShareServer {
|
||||||
get {
|
get {
|
||||||
@@ -1699,7 +1735,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Test Configurations download speed (Ctrl+T) 的本地化字符串。
|
/// 查找类似 Test download speed 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuSpeedServer {
|
public static string menuSpeedServer {
|
||||||
get {
|
get {
|
||||||
@@ -1843,7 +1879,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Test Configurations with tcping (Ctrl+O) 的本地化字符串。
|
/// 查找类似 Test tcping 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string menuTcpingServer {
|
public static string menuTcpingServer {
|
||||||
get {
|
get {
|
||||||
@@ -1951,7 +1987,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Configuration filter, press Enter to execute 的本地化字符串。
|
/// 查找类似 Filter, press Enter to execute 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string MsgServerTitle {
|
public static string MsgServerTitle {
|
||||||
get {
|
get {
|
||||||
@@ -2248,7 +2284,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Are you sure you want to remove the Configuration? 的本地化字符串。
|
/// 查找类似 Are you sure you want to remove? 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string RemoveServer {
|
public static string RemoveServer {
|
||||||
get {
|
get {
|
||||||
@@ -2292,6 +2328,15 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// 查找类似 Please set a valid domain 的本地化字符串。
|
||||||
|
/// </summary>
|
||||||
|
public static string ServerNameMustBeValidDomain {
|
||||||
|
get {
|
||||||
|
return ResourceManager.GetString("ServerNameMustBeValidDomain", resourceCulture);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 {0} : {1}/s↑ | {2}/s↓ 的本地化字符串。
|
/// 查找类似 {0} : {1}/s↑ | {2}/s↓ 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@@ -2319,6 +2364,15 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// 查找类似 Press ESC to terminate the test 的本地化字符串。
|
||||||
|
/// </summary>
|
||||||
|
public static string SpeedtestingPressEscToExit {
|
||||||
|
get {
|
||||||
|
return ResourceManager.GetString("SpeedtestingPressEscToExit", resourceCulture);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Skip test 的本地化字符串。
|
/// 查找类似 Skip test 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@@ -2347,7 +2401,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Waiting for testing (press ESC to terminate)... 的本地化字符串。
|
/// 查找类似 Waiting... 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string SpeedtestingWait {
|
public static string SpeedtestingWait {
|
||||||
get {
|
get {
|
||||||
@@ -2553,6 +2607,27 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// 查找类似 Certificate Pinning 的本地化字符串。
|
||||||
|
/// </summary>
|
||||||
|
public static string TbCertPinning {
|
||||||
|
get {
|
||||||
|
return ResourceManager.GetString("TbCertPinning", resourceCulture);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// 查找类似 Server Certificate (PEM format, optional)
|
||||||
|
///When specified, the certificate will be pinned, and "Allow Insecure" will be disabled.
|
||||||
|
///
|
||||||
|
///The "Get Certificate" action may fail if a self-signed certificate is used or if the system contains an untrusted or malicious CA. 的本地化字符串。
|
||||||
|
/// </summary>
|
||||||
|
public static string TbCertPinningTips {
|
||||||
|
get {
|
||||||
|
return ResourceManager.GetString("TbCertPinningTips", resourceCulture);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Clear system proxy 的本地化字符串。
|
/// 查找类似 Clear system proxy 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@@ -2760,6 +2835,24 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// 查找类似 Fetch Certificate 的本地化字符串。
|
||||||
|
/// </summary>
|
||||||
|
public static string TbFetchCert {
|
||||||
|
get {
|
||||||
|
return ResourceManager.GetString("TbFetchCert", resourceCulture);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// 查找类似 Fetch Certificate Chain 的本地化字符串。
|
||||||
|
/// </summary>
|
||||||
|
public static string TbFetchCertChain {
|
||||||
|
get {
|
||||||
|
return ResourceManager.GetString("TbFetchCertChain", resourceCulture);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Fingerprint 的本地化字符串。
|
/// 查找类似 Fingerprint 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@@ -2949,6 +3042,15 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// 查找类似 Auto add filtered configuration from subscription groups 的本地化字符串。
|
||||||
|
/// </summary>
|
||||||
|
public static string TbPolicyGroupSubChildTip {
|
||||||
|
get {
|
||||||
|
return ResourceManager.GetString("TbPolicyGroupSubChildTip", resourceCulture);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Policy Group Type 的本地化字符串。
|
/// 查找类似 Policy Group Type 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@@ -3426,6 +3528,24 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// 查找类似 Custom PAC file path 的本地化字符串。
|
||||||
|
/// </summary>
|
||||||
|
public static string TbSettingsCustomSystemProxyPacPath {
|
||||||
|
get {
|
||||||
|
return ResourceManager.GetString("TbSettingsCustomSystemProxyPacPath", resourceCulture);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// 查找类似 Custom system proxy script file path 的本地化字符串。
|
||||||
|
/// </summary>
|
||||||
|
public static string TbSettingsCustomSystemProxyScriptPath {
|
||||||
|
get {
|
||||||
|
return ResourceManager.GetString("TbSettingsCustomSystemProxyScriptPath", resourceCulture);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Allow Insecure 的本地化字符串。
|
/// 查找类似 Allow Insecure 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@@ -3750,6 +3870,15 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// 查找类似 macOS displays this in the Dock (requires restart) 的本地化字符串。
|
||||||
|
/// </summary>
|
||||||
|
public static string TbSettingsMacOSShowInDock {
|
||||||
|
get {
|
||||||
|
return ResourceManager.GetString("TbSettingsMacOSShowInDock", resourceCulture);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Main layout orientation (requires restart) 的本地化字符串。
|
/// 查找类似 Main layout orientation (requires restart) 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@@ -3985,7 +4114,7 @@ namespace ServiceLib.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Tray right-click menu Configurations display limit 的本地化字符串。
|
/// 查找类似 Tray right-click menu display limit 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string TbSettingsTrayMenuServersLimit {
|
public static string TbSettingsTrayMenuServersLimit {
|
||||||
get {
|
get {
|
||||||
|
|||||||
@@ -472,10 +472,10 @@
|
|||||||
<value>زبان</value>
|
<value>زبان</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuAddServerViaClipboard" xml:space="preserve">
|
<data name="menuAddServerViaClipboard" xml:space="preserve">
|
||||||
<value>وارد کردن URL انبوه از کلیپ بورد (Ctrl+V)</value>
|
<value>وارد کردن URL انبوه از کلیپ بورد</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuAddServerViaScan" xml:space="preserve">
|
<data name="menuAddServerViaScan" xml:space="preserve">
|
||||||
<value>اسکن کد QR روی صفحه (Ctrl+S)</value>
|
<value>اسکن کد QR روی صفحه</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuCopyServer" xml:space="preserve">
|
<data name="menuCopyServer" xml:space="preserve">
|
||||||
<value>سرور انتخاب شده را شبیه سازی کنید</value>
|
<value>سرور انتخاب شده را شبیه سازی کنید</value>
|
||||||
@@ -484,31 +484,31 @@
|
|||||||
<value>سرورهای تکراری را حذف کنید</value>
|
<value>سرورهای تکراری را حذف کنید</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuRemoveServer" xml:space="preserve">
|
<data name="menuRemoveServer" xml:space="preserve">
|
||||||
<value>حذف سرورهای انتخابی (Delete)</value>
|
<value>حذف سرورهای انتخابی</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuSetDefaultServer" xml:space="preserve">
|
<data name="menuSetDefaultServer" xml:space="preserve">
|
||||||
<value>به عنوان سرور فعال تنظیم کنید (Enter)</value>
|
<value>به عنوان سرور فعال تنظیم کنید</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuClearServerStatistics" xml:space="preserve">
|
<data name="menuClearServerStatistics" xml:space="preserve">
|
||||||
<value>تمام آمار خدمات را پاک کنید</value>
|
<value>تمام آمار خدمات را پاک کنید</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuRealPingServer" xml:space="preserve">
|
<data name="menuRealPingServer" xml:space="preserve">
|
||||||
<value>آزمایش سرورها با تاخیر واقعی (Ctrl+R)</value>
|
<value>آزمایش سرورها با تاخیر واقعی</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuSortServerResult" xml:space="preserve">
|
<data name="menuSortServerResult" xml:space="preserve">
|
||||||
<value>مرتب سازی بر اساس نتیجه تست</value>
|
<value>مرتب سازی بر اساس نتیجه تست</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuSpeedServer" xml:space="preserve">
|
<data name="menuSpeedServer" xml:space="preserve">
|
||||||
<value>تست سرعت دانلود سرورها (Ctrl+T)</value>
|
<value>تست سرعت دانلود سرورها</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuTcpingServer" xml:space="preserve">
|
<data name="menuTcpingServer" xml:space="preserve">
|
||||||
<value>تست سرورها با tcping (Ctrl+O)</value>
|
<value>تست سرورها با tcping</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuExport2ClientConfig" xml:space="preserve">
|
<data name="menuExport2ClientConfig" xml:space="preserve">
|
||||||
<value>سرور انتخابی را برای پیکربندی کلاینت صادر کنید</value>
|
<value>سرور انتخابی را برای پیکربندی کلاینت صادر کنید</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuExport2ShareUrl" xml:space="preserve">
|
<data name="menuExport2ShareUrl" xml:space="preserve">
|
||||||
<value>URL های اشتراک گذاری را به کلیپ بورد صادر کنید (Ctrl+C)</value>
|
<value>URL های اشتراک گذاری را به کلیپ بورد صادر کنید</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuAddCustomServer" xml:space="preserve">
|
<data name="menuAddCustomServer" xml:space="preserve">
|
||||||
<value>یک سرور پیکربندی سفارشی اضافه شود</value>
|
<value>یک سرور پیکربندی سفارشی اضافه شود</value>
|
||||||
@@ -529,19 +529,19 @@
|
|||||||
<value>سرور [VMess] را اضافه کنید</value>
|
<value>سرور [VMess] را اضافه کنید</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuSelectAll" xml:space="preserve">
|
<data name="menuSelectAll" xml:space="preserve">
|
||||||
<value>انتخاب همه (Ctrl+A)</value>
|
<value>انتخاب همه</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMsgViewClear" xml:space="preserve">
|
<data name="menuMsgViewClear" xml:space="preserve">
|
||||||
<value>همه را پاک کن</value>
|
<value>همه را پاک کن</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMsgViewCopy" xml:space="preserve">
|
<data name="menuMsgViewCopy" xml:space="preserve">
|
||||||
<value>کپی (Ctrl+C)</value>
|
<value>کپی</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMsgViewCopyAll" xml:space="preserve">
|
<data name="menuMsgViewCopyAll" xml:space="preserve">
|
||||||
<value>کپی همه</value>
|
<value>کپی همه</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMsgViewSelectAll" xml:space="preserve">
|
<data name="menuMsgViewSelectAll" xml:space="preserve">
|
||||||
<value>انتخاب همه (Ctrl+A)</value>
|
<value>انتخاب همه</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuSubAdd" xml:space="preserve">
|
<data name="menuSubAdd" xml:space="preserve">
|
||||||
<value>اضافه کردن</value>
|
<value>اضافه کردن</value>
|
||||||
@@ -796,13 +796,13 @@
|
|||||||
<value>به پایین حرکت شود(B)</value>
|
<value>به پایین حرکت شود(B)</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMoveDown" xml:space="preserve">
|
<data name="menuMoveDown" xml:space="preserve">
|
||||||
<value>پایین (D)</value>
|
<value>پایین</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMoveTop" xml:space="preserve">
|
<data name="menuMoveTop" xml:space="preserve">
|
||||||
<value>حرکت به بالا (T)</value>
|
<value>حرکت به بالا</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMoveUp" xml:space="preserve">
|
<data name="menuMoveUp" xml:space="preserve">
|
||||||
<value>بالا (U)</value>
|
<value>بالا</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="MsgFilterTitle" xml:space="preserve">
|
<data name="MsgFilterTitle" xml:space="preserve">
|
||||||
<value>فیلتر، از عبارات منظم پشتیبانی می کند</value>
|
<value>فیلتر، از عبارات منظم پشتیبانی می کند</value>
|
||||||
@@ -922,7 +922,7 @@
|
|||||||
<value>رد شدن از آزمون</value>
|
<value>رد شدن از آزمون</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuEditServer" xml:space="preserve">
|
<data name="menuEditServer" xml:space="preserve">
|
||||||
<value>ویرایش سرور (Ctrl+D)</value>
|
<value>ویرایش سرور</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbSettingsDoubleClick2Activate" xml:space="preserve">
|
<data name="TbSettingsDoubleClick2Activate" xml:space="preserve">
|
||||||
<value>دوبار کلیک کردن سرور باعث فعال شدن آن می شود</value>
|
<value>دوبار کلیک کردن سرور باعث فعال شدن آن می شود</value>
|
||||||
@@ -976,7 +976,10 @@
|
|||||||
<value>فعال سازی شتاب دهنده سخت افزاری (نیاز به راهاندازی مجدد)</value>
|
<value>فعال سازی شتاب دهنده سخت افزاری (نیاز به راهاندازی مجدد)</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="SpeedtestingWait" xml:space="preserve">
|
<data name="SpeedtestingWait" xml:space="preserve">
|
||||||
<value>در انتظار آزمایش (برای پایان دادن به ESC فشار دهید)...</value>
|
<value>در انتظار آزمایش...</value>
|
||||||
|
</data>
|
||||||
|
<data name="SpeedtestingPressEscToExit" xml:space="preserve">
|
||||||
|
<value>برای پایان دادن به ESC فشار دهید</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TipDisplayLog" xml:space="preserve">
|
<data name="TipDisplayLog" xml:space="preserve">
|
||||||
<value>لطفاً در صورت قطع غیرعادی آن را خاموش کنید</value>
|
<value>لطفاً در صورت قطع غیرعادی آن را خاموش کنید</value>
|
||||||
@@ -1204,7 +1207,7 @@
|
|||||||
<value>تازه سازی پروکسی ها</value>
|
<value>تازه سازی پروکسی ها</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuProxiesSelectActivity" xml:space="preserve">
|
<data name="menuProxiesSelectActivity" xml:space="preserve">
|
||||||
<value>انتخاب گره فعال (Enter)</value>
|
<value>انتخاب گره فعال</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbSettingsDomainStrategy4Out" xml:space="preserve">
|
<data name="TbSettingsDomainStrategy4Out" xml:space="preserve">
|
||||||
<value>استراتژی دامنه پیش فرض برای خروجی</value>
|
<value>استراتژی دامنه پیش فرض برای خروجی</value>
|
||||||
@@ -1537,7 +1540,7 @@
|
|||||||
<value>Remove Child Configuration</value>
|
<value>Remove Child Configuration</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuServerList" xml:space="preserve">
|
<data name="menuServerList" xml:space="preserve">
|
||||||
<value>Server List</value>
|
<value>Configuration item 1, Auto add from subscription group</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbFallback" xml:space="preserve">
|
<data name="TbFallback" xml:space="preserve">
|
||||||
<value>Fallback</value>
|
<value>Fallback</value>
|
||||||
@@ -1596,4 +1599,46 @@
|
|||||||
<data name="TbBootstrapDNSTips" xml:space="preserve">
|
<data name="TbBootstrapDNSTips" xml:space="preserve">
|
||||||
<value>Resolve DNS server domains, requires IP</value>
|
<value>Resolve DNS server domains, requires IP</value>
|
||||||
</data>
|
</data>
|
||||||
|
<data name="menuFastRealPing" xml:space="preserve">
|
||||||
|
<value>Test real delay</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbPolicyGroupSubChildTip" xml:space="preserve">
|
||||||
|
<value>Auto add filtered configuration from subscription groups</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbCertPinning" xml:space="preserve">
|
||||||
|
<value>Certificate Pinning</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbCertPinningTips" xml:space="preserve">
|
||||||
|
<value>Server Certificate (PEM format, optional)
|
||||||
|
When specified, the certificate will be pinned, and "Allow Insecure" will be disabled.
|
||||||
|
|
||||||
|
The "Get Certificate" action may fail if a self-signed certificate is used or if the system contains an untrusted or malicious CA.</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbFetchCert" xml:space="preserve">
|
||||||
|
<value>Fetch Certificate</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbFetchCertChain" xml:space="preserve">
|
||||||
|
<value>Fetch Certificate Chain</value>
|
||||||
|
</data>
|
||||||
|
<data name="ServerNameMustBeValidDomain" xml:space="preserve">
|
||||||
|
<value>Please set a valid domain</value>
|
||||||
|
</data>
|
||||||
|
<data name="CertNotSet" xml:space="preserve">
|
||||||
|
<value>Certificate not set</value>
|
||||||
|
</data>
|
||||||
|
<data name="CertSet" xml:space="preserve">
|
||||||
|
<value>Certificate set</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbSettingsCustomSystemProxyPacPath" xml:space="preserve">
|
||||||
|
<value>Custom PAC file path</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbSettingsCustomSystemProxyScriptPath" xml:space="preserve">
|
||||||
|
<value>Custom system proxy script file path</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbSettingsMacOSShowInDock" xml:space="preserve">
|
||||||
|
<value>macOS displays this in the Dock (requires restart)</value>
|
||||||
|
</data>
|
||||||
|
<data name="menuServerList2" xml:space="preserve">
|
||||||
|
<value>Configuration Item 2, Select and add from self-built</value>
|
||||||
|
</data>
|
||||||
</root>
|
</root>
|
||||||
1641
v2rayN/ServiceLib/Resx/ResUI.fr.resx
Normal file
1641
v2rayN/ServiceLib/Resx/ResUI.fr.resx
Normal file
File diff suppressed because it is too large
Load Diff
@@ -472,10 +472,10 @@
|
|||||||
<value>Nyelv (Újraindítás)</value>
|
<value>Nyelv (Újraindítás)</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuAddServerViaClipboard" xml:space="preserve">
|
<data name="menuAddServerViaClipboard" xml:space="preserve">
|
||||||
<value>Megosztási linkek importálása vágólapról (Ctrl+V)</value>
|
<value>Megosztási linkek importálása vágólapról</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuAddServerViaScan" xml:space="preserve">
|
<data name="menuAddServerViaScan" xml:space="preserve">
|
||||||
<value>QR kód beolvasása a képernyőről (Ctrl+S)</value>
|
<value>QR kód beolvasása a képernyőről</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuCopyServer" xml:space="preserve">
|
<data name="menuCopyServer" xml:space="preserve">
|
||||||
<value>Kijelölt konfiguráció klónozása</value>
|
<value>Kijelölt konfiguráció klónozása</value>
|
||||||
@@ -484,31 +484,31 @@
|
|||||||
<value>Ismétlődő konfigurációk eltávolítása</value>
|
<value>Ismétlődő konfigurációk eltávolítása</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuRemoveServer" xml:space="preserve">
|
<data name="menuRemoveServer" xml:space="preserve">
|
||||||
<value>Kijelölt konfigurációk eltávolítása (Delete)</value>
|
<value>Kijelölt konfigurációk eltávolítása</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuSetDefaultServer" xml:space="preserve">
|
<data name="menuSetDefaultServer" xml:space="preserve">
|
||||||
<value>Beállítás aktív konfigurációként (Enter)</value>
|
<value>Beállítás aktív konfigurációként</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuClearServerStatistics" xml:space="preserve">
|
<data name="menuClearServerStatistics" xml:space="preserve">
|
||||||
<value>Összes szolgáltatás statisztika törlése</value>
|
<value>Összes szolgáltatás statisztika törlése</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuRealPingServer" xml:space="preserve">
|
<data name="menuRealPingServer" xml:space="preserve">
|
||||||
<value>Konfigurációk valós késleltetésének tesztelése (Ctrl+R)</value>
|
<value>Konfigurációk valós késleltetésének tesztelése</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuSortServerResult" xml:space="preserve">
|
<data name="menuSortServerResult" xml:space="preserve">
|
||||||
<value>Rendezés teszteredmény szerint</value>
|
<value>Rendezés teszteredmény szerint</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuSpeedServer" xml:space="preserve">
|
<data name="menuSpeedServer" xml:space="preserve">
|
||||||
<value>Konfigurációk letöltési sebességének tesztelése (Ctrl+T)</value>
|
<value>Konfigurációk letöltési sebességének tesztelése</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuTcpingServer" xml:space="preserve">
|
<data name="menuTcpingServer" xml:space="preserve">
|
||||||
<value>Konfigurációk tesztelése tcpinggel (Ctrl+O)</value>
|
<value>Konfigurációk tesztelése tcpinggel</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuExport2ClientConfig" xml:space="preserve">
|
<data name="menuExport2ClientConfig" xml:space="preserve">
|
||||||
<value>Kijelölt konfiguráció exportálása teljes konfigurációként</value>
|
<value>Kijelölt konfiguráció exportálása teljes konfigurációként</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuExport2ShareUrl" xml:space="preserve">
|
<data name="menuExport2ShareUrl" xml:space="preserve">
|
||||||
<value>Megosztási link exportálása vágólapra (Ctrl+C)</value>
|
<value>Megosztási link exportálása vágólapra</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuAddCustomServer" xml:space="preserve">
|
<data name="menuAddCustomServer" xml:space="preserve">
|
||||||
<value>Egyéni konfiguráció hozzáadása</value>
|
<value>Egyéni konfiguráció hozzáadása</value>
|
||||||
@@ -529,19 +529,19 @@
|
|||||||
<value>[VMess] konfiguráció hozzáadása</value>
|
<value>[VMess] konfiguráció hozzáadása</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuSelectAll" xml:space="preserve">
|
<data name="menuSelectAll" xml:space="preserve">
|
||||||
<value>Összes kijelölése (Ctrl+A)</value>
|
<value>Összes kijelölése</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMsgViewClear" xml:space="preserve">
|
<data name="menuMsgViewClear" xml:space="preserve">
|
||||||
<value>Összes törlése</value>
|
<value>Összes törlése</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMsgViewCopy" xml:space="preserve">
|
<data name="menuMsgViewCopy" xml:space="preserve">
|
||||||
<value>Másolás (Ctrl+C)</value>
|
<value>Másolás</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMsgViewCopyAll" xml:space="preserve">
|
<data name="menuMsgViewCopyAll" xml:space="preserve">
|
||||||
<value>Összes másolása</value>
|
<value>Összes másolása</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMsgViewSelectAll" xml:space="preserve">
|
<data name="menuMsgViewSelectAll" xml:space="preserve">
|
||||||
<value>Összes kijelölése (Ctrl+A)</value>
|
<value>Összes kijelölése</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuSubAdd" xml:space="preserve">
|
<data name="menuSubAdd" xml:space="preserve">
|
||||||
<value>Hozzáadás</value>
|
<value>Hozzáadás</value>
|
||||||
@@ -781,7 +781,7 @@
|
|||||||
<value>PAC mód</value>
|
<value>PAC mód</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuShareServer" xml:space="preserve">
|
<data name="menuShareServer" xml:space="preserve">
|
||||||
<value>Konfiguráció megosztása (Ctrl+F)</value>
|
<value>Konfiguráció megosztása</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuRouting" xml:space="preserve">
|
<data name="menuRouting" xml:space="preserve">
|
||||||
<value>Útválasztás</value>
|
<value>Útválasztás</value>
|
||||||
@@ -793,16 +793,16 @@
|
|||||||
<value>Futtatás rendszergazdaként</value>
|
<value>Futtatás rendszergazdaként</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMoveBottom" xml:space="preserve">
|
<data name="menuMoveBottom" xml:space="preserve">
|
||||||
<value>Mozgatás alulra (B)</value>
|
<value>Mozgatás alulra</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMoveDown" xml:space="preserve">
|
<data name="menuMoveDown" xml:space="preserve">
|
||||||
<value>Le (D)</value>
|
<value>Le</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMoveTop" xml:space="preserve">
|
<data name="menuMoveTop" xml:space="preserve">
|
||||||
<value>Mozgatás felülre (T)</value>
|
<value>Mozgatás felülre</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMoveUp" xml:space="preserve">
|
<data name="menuMoveUp" xml:space="preserve">
|
||||||
<value>Fel (U)</value>
|
<value>Fel</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="MsgFilterTitle" xml:space="preserve">
|
<data name="MsgFilterTitle" xml:space="preserve">
|
||||||
<value>Szűrő, támogatja a reguláris kifejezéseket</value>
|
<value>Szűrő, támogatja a reguláris kifejezéseket</value>
|
||||||
@@ -817,10 +817,10 @@
|
|||||||
<value>Szabályok importálása</value>
|
<value>Szabályok importálása</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuRoutingAdvancedRemove" xml:space="preserve">
|
<data name="menuRoutingAdvancedRemove" xml:space="preserve">
|
||||||
<value>Kijelölt eltávolítása (Delete)</value>
|
<value>Kijelölt eltávolítása</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuRoutingAdvancedSetDefault" xml:space="preserve">
|
<data name="menuRoutingAdvancedSetDefault" xml:space="preserve">
|
||||||
<value>Beállítás aktív szabályként (Enter)</value>
|
<value>Beállítás aktív szabályként</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbdomainStrategy" xml:space="preserve">
|
<data name="TbdomainStrategy" xml:space="preserve">
|
||||||
<value>Tartomány stratégia</value>
|
<value>Tartomány stratégia</value>
|
||||||
@@ -853,7 +853,7 @@
|
|||||||
<value>Szabálylista</value>
|
<value>Szabálylista</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuRuleRemove" xml:space="preserve">
|
<data name="menuRuleRemove" xml:space="preserve">
|
||||||
<value>Szabály eltávolítása (Delete)</value>
|
<value>Szabály eltávolítása</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuRoutingRuleDetailsSetting" xml:space="preserve">
|
<data name="menuRoutingRuleDetailsSetting" xml:space="preserve">
|
||||||
<value>Útválasztási szabály részleteinek beállítása</value>
|
<value>Útválasztási szabály részleteinek beállítása</value>
|
||||||
@@ -922,7 +922,7 @@
|
|||||||
<value>Teszt kihagyása</value>
|
<value>Teszt kihagyása</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuEditServer" xml:space="preserve">
|
<data name="menuEditServer" xml:space="preserve">
|
||||||
<value>Konfiguráció szerkesztése (Ctrl+D)</value>
|
<value>Konfiguráció szerkesztése</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbSettingsDoubleClick2Activate" xml:space="preserve">
|
<data name="TbSettingsDoubleClick2Activate" xml:space="preserve">
|
||||||
<value>Dupla kattintás a konfigurációra aktiválja</value>
|
<value>Dupla kattintás a konfigurációra aktiválja</value>
|
||||||
@@ -976,7 +976,10 @@
|
|||||||
<value>Hardveres gyorsítás engedélyezése (újraindítást igényel)</value>
|
<value>Hardveres gyorsítás engedélyezése (újraindítást igényel)</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="SpeedtestingWait" xml:space="preserve">
|
<data name="SpeedtestingWait" xml:space="preserve">
|
||||||
<value>Tesztelésre vár (ESC megnyomásával megszakítható)...</value>
|
<value>Tesztelésre vár...</value>
|
||||||
|
</data>
|
||||||
|
<data name="SpeedtestingPressEscToExit" xml:space="preserve">
|
||||||
|
<value>ESC megnyomásával megszakítható</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TipDisplayLog" xml:space="preserve">
|
<data name="TipDisplayLog" xml:space="preserve">
|
||||||
<value>Kérjük, kapcsolja ki rendellenes megszakadás esetén</value>
|
<value>Kérjük, kapcsolja ki rendellenes megszakadás esetén</value>
|
||||||
@@ -1204,7 +1207,7 @@
|
|||||||
<value>Proxyk frissítése</value>
|
<value>Proxyk frissítése</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuProxiesSelectActivity" xml:space="preserve">
|
<data name="menuProxiesSelectActivity" xml:space="preserve">
|
||||||
<value>Aktív csomópont kiválasztása (Enter)</value>
|
<value>Aktív csomópont kiválasztása</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbSettingsDomainStrategy4Out" xml:space="preserve">
|
<data name="TbSettingsDomainStrategy4Out" xml:space="preserve">
|
||||||
<value>Alapértelmezett tartomány stratégia kimenő forgalomhoz</value>
|
<value>Alapértelmezett tartomány stratégia kimenő forgalomhoz</value>
|
||||||
@@ -1537,7 +1540,7 @@
|
|||||||
<value>Remove Child Configuration</value>
|
<value>Remove Child Configuration</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuServerList" xml:space="preserve">
|
<data name="menuServerList" xml:space="preserve">
|
||||||
<value>Server List</value>
|
<value>Configuration item 1, Auto add from subscription group</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbFallback" xml:space="preserve">
|
<data name="TbFallback" xml:space="preserve">
|
||||||
<value>Fallback</value>
|
<value>Fallback</value>
|
||||||
@@ -1596,4 +1599,46 @@
|
|||||||
<data name="TbBootstrapDNSTips" xml:space="preserve">
|
<data name="TbBootstrapDNSTips" xml:space="preserve">
|
||||||
<value>Resolve DNS server domains, requires IP</value>
|
<value>Resolve DNS server domains, requires IP</value>
|
||||||
</data>
|
</data>
|
||||||
|
<data name="menuFastRealPing" xml:space="preserve">
|
||||||
|
<value>Test real delay</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbPolicyGroupSubChildTip" xml:space="preserve">
|
||||||
|
<value>Auto add filtered configuration from subscription groups</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbCertPinning" xml:space="preserve">
|
||||||
|
<value>Certificate Pinning</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbCertPinningTips" xml:space="preserve">
|
||||||
|
<value>Server Certificate (PEM format, optional)
|
||||||
|
When specified, the certificate will be pinned, and "Allow Insecure" will be disabled.
|
||||||
|
|
||||||
|
The "Get Certificate" action may fail if a self-signed certificate is used or if the system contains an untrusted or malicious CA.</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbFetchCert" xml:space="preserve">
|
||||||
|
<value>Fetch Certificate</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbFetchCertChain" xml:space="preserve">
|
||||||
|
<value>Fetch Certificate Chain</value>
|
||||||
|
</data>
|
||||||
|
<data name="ServerNameMustBeValidDomain" xml:space="preserve">
|
||||||
|
<value>Please set a valid domain</value>
|
||||||
|
</data>
|
||||||
|
<data name="CertNotSet" xml:space="preserve">
|
||||||
|
<value>Certificate not set</value>
|
||||||
|
</data>
|
||||||
|
<data name="CertSet" xml:space="preserve">
|
||||||
|
<value>Certificate set</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbSettingsCustomSystemProxyPacPath" xml:space="preserve">
|
||||||
|
<value>Custom PAC file path</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbSettingsCustomSystemProxyScriptPath" xml:space="preserve">
|
||||||
|
<value>Custom system proxy script file path</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbSettingsMacOSShowInDock" xml:space="preserve">
|
||||||
|
<value>macOS displays this in the Dock (requires restart)</value>
|
||||||
|
</data>
|
||||||
|
<data name="menuServerList2" xml:space="preserve">
|
||||||
|
<value>Configuration Item 2, Select and add from self-built</value>
|
||||||
|
</data>
|
||||||
</root>
|
</root>
|
||||||
@@ -271,7 +271,7 @@
|
|||||||
<value>Configurations deduplication completed. Old: {0}, New: {1}.</value>
|
<value>Configurations deduplication completed. Old: {0}, New: {1}.</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="RemoveServer" xml:space="preserve">
|
<data name="RemoveServer" xml:space="preserve">
|
||||||
<value>Are you sure you want to remove the Configuration?</value>
|
<value>Are you sure you want to remove?</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="SaveClientConfigurationIn" xml:space="preserve">
|
<data name="SaveClientConfigurationIn" xml:space="preserve">
|
||||||
<value>The client configuration file is saved at: {0}</value>
|
<value>The client configuration file is saved at: {0}</value>
|
||||||
@@ -397,7 +397,7 @@
|
|||||||
<value>Local</value>
|
<value>Local</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="MsgServerTitle" xml:space="preserve">
|
<data name="MsgServerTitle" xml:space="preserve">
|
||||||
<value>Configuration filter, press Enter to execute</value>
|
<value>Filter, press Enter to execute</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuCheckUpdate" xml:space="preserve">
|
<data name="menuCheckUpdate" xml:space="preserve">
|
||||||
<value>Check Update</value>
|
<value>Check Update</value>
|
||||||
@@ -427,7 +427,7 @@
|
|||||||
<value>Routing Setting</value>
|
<value>Routing Setting</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuServers" xml:space="preserve">
|
<data name="menuServers" xml:space="preserve">
|
||||||
<value>Configurations</value>
|
<value>Configuration</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuSetting" xml:space="preserve">
|
<data name="menuSetting" xml:space="preserve">
|
||||||
<value>Settings</value>
|
<value>Settings</value>
|
||||||
@@ -472,76 +472,76 @@
|
|||||||
<value>Language (Restart)</value>
|
<value>Language (Restart)</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuAddServerViaClipboard" xml:space="preserve">
|
<data name="menuAddServerViaClipboard" xml:space="preserve">
|
||||||
<value>Import Share Links from clipboard (Ctrl+V)</value>
|
<value>Import Share Links from clipboard</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuAddServerViaScan" xml:space="preserve">
|
<data name="menuAddServerViaScan" xml:space="preserve">
|
||||||
<value>Scan QR code on the screen (Ctrl+S)</value>
|
<value>Scan QR code on the screen</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuCopyServer" xml:space="preserve">
|
<data name="menuCopyServer" xml:space="preserve">
|
||||||
<value>Clone selected Configuration</value>
|
<value>Clone selected</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuRemoveDuplicateServer" xml:space="preserve">
|
<data name="menuRemoveDuplicateServer" xml:space="preserve">
|
||||||
<value>Remove duplicate Configurations</value>
|
<value>Remove duplicate</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuRemoveServer" xml:space="preserve">
|
<data name="menuRemoveServer" xml:space="preserve">
|
||||||
<value>Remove selected Configurations (Delete)</value>
|
<value>Remove selected</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuSetDefaultServer" xml:space="preserve">
|
<data name="menuSetDefaultServer" xml:space="preserve">
|
||||||
<value>Set as active Configuration (Enter)</value>
|
<value>Set as active</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuClearServerStatistics" xml:space="preserve">
|
<data name="menuClearServerStatistics" xml:space="preserve">
|
||||||
<value>Clear all service statistics</value>
|
<value>Clear all service statistics</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuRealPingServer" xml:space="preserve">
|
<data name="menuRealPingServer" xml:space="preserve">
|
||||||
<value>Test Configurations real delay (Ctrl+R)</value>
|
<value>Test real delay</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuSortServerResult" xml:space="preserve">
|
<data name="menuSortServerResult" xml:space="preserve">
|
||||||
<value>Sort by test result</value>
|
<value>Sort by test result</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuSpeedServer" xml:space="preserve">
|
<data name="menuSpeedServer" xml:space="preserve">
|
||||||
<value>Test Configurations download speed (Ctrl+T)</value>
|
<value>Test download speed</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuTcpingServer" xml:space="preserve">
|
<data name="menuTcpingServer" xml:space="preserve">
|
||||||
<value>Test Configurations with tcping (Ctrl+O)</value>
|
<value>Test tcping</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuExport2ClientConfig" xml:space="preserve">
|
<data name="menuExport2ClientConfig" xml:space="preserve">
|
||||||
<value>Export selected Configuration for complete configuration</value>
|
<value>Export selected for complete configuration</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuExport2ShareUrl" xml:space="preserve">
|
<data name="menuExport2ShareUrl" xml:space="preserve">
|
||||||
<value>Export Share Link to Clipboard (Ctrl+C)</value>
|
<value>Export Share Link to Clipboard</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuAddCustomServer" xml:space="preserve">
|
<data name="menuAddCustomServer" xml:space="preserve">
|
||||||
<value>Add a custom configuration Configuration</value>
|
<value>Add a custom configuration</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuAddShadowsocksServer" xml:space="preserve">
|
<data name="menuAddShadowsocksServer" xml:space="preserve">
|
||||||
<value>Add [Shadowsocks] Configuration</value>
|
<value>Add [Shadowsocks] </value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuAddSocksServer" xml:space="preserve">
|
<data name="menuAddSocksServer" xml:space="preserve">
|
||||||
<value>Add [SOCKS] Configuration</value>
|
<value>Add [SOCKS] </value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuAddTrojanServer" xml:space="preserve">
|
<data name="menuAddTrojanServer" xml:space="preserve">
|
||||||
<value>Add [Trojan] Configuration</value>
|
<value>Add [Trojan] </value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuAddVlessServer" xml:space="preserve">
|
<data name="menuAddVlessServer" xml:space="preserve">
|
||||||
<value>Add [VLESS] Configuration</value>
|
<value>Add [VLESS] </value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuAddVmessServer" xml:space="preserve">
|
<data name="menuAddVmessServer" xml:space="preserve">
|
||||||
<value>Add [VMess] Configuration</value>
|
<value>Add [VMess] </value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuSelectAll" xml:space="preserve">
|
<data name="menuSelectAll" xml:space="preserve">
|
||||||
<value>Select all (Ctrl+A)</value>
|
<value>Select all</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMsgViewClear" xml:space="preserve">
|
<data name="menuMsgViewClear" xml:space="preserve">
|
||||||
<value>Clear all</value>
|
<value>Clear all</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMsgViewCopy" xml:space="preserve">
|
<data name="menuMsgViewCopy" xml:space="preserve">
|
||||||
<value>Copy (Ctrl+C)</value>
|
<value>Copy</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMsgViewCopyAll" xml:space="preserve">
|
<data name="menuMsgViewCopyAll" xml:space="preserve">
|
||||||
<value>Copy all</value>
|
<value>Copy all</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMsgViewSelectAll" xml:space="preserve">
|
<data name="menuMsgViewSelectAll" xml:space="preserve">
|
||||||
<value>Select all (Ctrl+A)</value>
|
<value>Select all</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuSubAdd" xml:space="preserve">
|
<data name="menuSubAdd" xml:space="preserve">
|
||||||
<value>Add</value>
|
<value>Add</value>
|
||||||
@@ -748,7 +748,7 @@
|
|||||||
<value>System proxy settings</value>
|
<value>System proxy settings</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbSettingsTrayMenuServersLimit" xml:space="preserve">
|
<data name="TbSettingsTrayMenuServersLimit" xml:space="preserve">
|
||||||
<value>Tray right-click menu Configurations display limit</value>
|
<value>Tray right-click menu display limit</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbSettingsUdpEnabled" xml:space="preserve">
|
<data name="TbSettingsUdpEnabled" xml:space="preserve">
|
||||||
<value>Enable UDP</value>
|
<value>Enable UDP</value>
|
||||||
@@ -781,7 +781,7 @@
|
|||||||
<value>PAC mode</value>
|
<value>PAC mode</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuShareServer" xml:space="preserve">
|
<data name="menuShareServer" xml:space="preserve">
|
||||||
<value>Share Configuration (Ctrl+F)</value>
|
<value>Share</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuRouting" xml:space="preserve">
|
<data name="menuRouting" xml:space="preserve">
|
||||||
<value>Routing</value>
|
<value>Routing</value>
|
||||||
@@ -793,16 +793,16 @@
|
|||||||
<value>Run as Admin</value>
|
<value>Run as Admin</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMoveBottom" xml:space="preserve">
|
<data name="menuMoveBottom" xml:space="preserve">
|
||||||
<value>Move to bottom (B)</value>
|
<value>Move to bottom</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMoveDown" xml:space="preserve">
|
<data name="menuMoveDown" xml:space="preserve">
|
||||||
<value>Down (D)</value>
|
<value>Down</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMoveTop" xml:space="preserve">
|
<data name="menuMoveTop" xml:space="preserve">
|
||||||
<value>Move to top (T)</value>
|
<value>Move to top</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMoveUp" xml:space="preserve">
|
<data name="menuMoveUp" xml:space="preserve">
|
||||||
<value>Up (U)</value>
|
<value>Up</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="MsgFilterTitle" xml:space="preserve">
|
<data name="MsgFilterTitle" xml:space="preserve">
|
||||||
<value>Filter, supports regular expressions</value>
|
<value>Filter, supports regular expressions</value>
|
||||||
@@ -817,10 +817,10 @@
|
|||||||
<value>Import Rules</value>
|
<value>Import Rules</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuRoutingAdvancedRemove" xml:space="preserve">
|
<data name="menuRoutingAdvancedRemove" xml:space="preserve">
|
||||||
<value>Remove selected (Delete)</value>
|
<value>Remove selected</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuRoutingAdvancedSetDefault" xml:space="preserve">
|
<data name="menuRoutingAdvancedSetDefault" xml:space="preserve">
|
||||||
<value>Set as active rule (Enter)</value>
|
<value>Set as active rule</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbdomainStrategy" xml:space="preserve">
|
<data name="TbdomainStrategy" xml:space="preserve">
|
||||||
<value>Domain strategy</value>
|
<value>Domain strategy</value>
|
||||||
@@ -853,7 +853,7 @@
|
|||||||
<value>Rule List</value>
|
<value>Rule List</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuRuleRemove" xml:space="preserve">
|
<data name="menuRuleRemove" xml:space="preserve">
|
||||||
<value>Remove Rule (Delete)</value>
|
<value>Remove Rule</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuRoutingRuleDetailsSetting" xml:space="preserve">
|
<data name="menuRoutingRuleDetailsSetting" xml:space="preserve">
|
||||||
<value>Routing Rule Details Setting</value>
|
<value>Routing Rule Details Setting</value>
|
||||||
@@ -922,7 +922,7 @@
|
|||||||
<value>Skip test</value>
|
<value>Skip test</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuEditServer" xml:space="preserve">
|
<data name="menuEditServer" xml:space="preserve">
|
||||||
<value>Edit Configuration (Ctrl+D)</value>
|
<value>Edit </value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbSettingsDoubleClick2Activate" xml:space="preserve">
|
<data name="TbSettingsDoubleClick2Activate" xml:space="preserve">
|
||||||
<value>Double-clicking Configuration makes it active</value>
|
<value>Double-clicking Configuration makes it active</value>
|
||||||
@@ -976,7 +976,10 @@
|
|||||||
<value>Enable hardware acceleration (requires restart)</value>
|
<value>Enable hardware acceleration (requires restart)</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="SpeedtestingWait" xml:space="preserve">
|
<data name="SpeedtestingWait" xml:space="preserve">
|
||||||
<value>Waiting for testing (press ESC to terminate)...</value>
|
<value>Waiting...</value>
|
||||||
|
</data>
|
||||||
|
<data name="SpeedtestingPressEscToExit" xml:space="preserve">
|
||||||
|
<value>Press ESC to terminate the test</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TipDisplayLog" xml:space="preserve">
|
<data name="TipDisplayLog" xml:space="preserve">
|
||||||
<value>Please turn off when there is an abnormal disconnection</value>
|
<value>Please turn off when there is an abnormal disconnection</value>
|
||||||
@@ -1033,7 +1036,7 @@
|
|||||||
<value>Domain</value>
|
<value>Domain</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuAddHysteria2Server" xml:space="preserve">
|
<data name="menuAddHysteria2Server" xml:space="preserve">
|
||||||
<value>Add [Hysteria2] Configuration</value>
|
<value>Add [Hysteria2] </value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbSettingsHysteriaBandwidth" xml:space="preserve">
|
<data name="TbSettingsHysteriaBandwidth" xml:space="preserve">
|
||||||
<value>Hysteria Max bandwidth (Up/Down)</value>
|
<value>Hysteria Max bandwidth (Up/Down)</value>
|
||||||
@@ -1042,16 +1045,16 @@
|
|||||||
<value>Use System Hosts</value>
|
<value>Use System Hosts</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuAddTuicServer" xml:space="preserve">
|
<data name="menuAddTuicServer" xml:space="preserve">
|
||||||
<value>Add [TUIC] Configuration</value>
|
<value>Add [TUIC] </value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbHeaderType8" xml:space="preserve">
|
<data name="TbHeaderType8" xml:space="preserve">
|
||||||
<value>Congestion control</value>
|
<value>Congestion control</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="LvPrevProfile" xml:space="preserve">
|
<data name="LvPrevProfile" xml:space="preserve">
|
||||||
<value>Previous proxy Configuration remarks</value>
|
<value>Previous proxy remarks</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="LvNextProfile" xml:space="preserve">
|
<data name="LvNextProfile" xml:space="preserve">
|
||||||
<value>Next proxy Configuration remarks</value>
|
<value>Next proxy remarks</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="LvPrevProfileTip" xml:space="preserve">
|
<data name="LvPrevProfileTip" xml:space="preserve">
|
||||||
<value>Please make sure the Configuration remarks exist and are unique</value>
|
<value>Please make sure the Configuration remarks exist and are unique</value>
|
||||||
@@ -1075,7 +1078,7 @@
|
|||||||
<value>Enable IPv6 Address</value>
|
<value>Enable IPv6 Address</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuAddWireguardServer" xml:space="preserve">
|
<data name="menuAddWireguardServer" xml:space="preserve">
|
||||||
<value>Add [WireGuard] Configuration</value>
|
<value>Add [WireGuard] </value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbPrivateKey" xml:space="preserve">
|
<data name="TbPrivateKey" xml:space="preserve">
|
||||||
<value>Private Key</value>
|
<value>Private Key</value>
|
||||||
@@ -1108,7 +1111,7 @@
|
|||||||
<value>*grpc Authority</value>
|
<value>*grpc Authority</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuAddHttpServer" xml:space="preserve">
|
<data name="menuAddHttpServer" xml:space="preserve">
|
||||||
<value>Add [HTTP] Configuration</value>
|
<value>Add [HTTP]</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbSettingsEnableFragmentTips" xml:space="preserve">
|
<data name="TbSettingsEnableFragmentTips" xml:space="preserve">
|
||||||
<value>which conflicts with the group previous proxy</value>
|
<value>which conflicts with the group previous proxy</value>
|
||||||
@@ -1204,7 +1207,7 @@
|
|||||||
<value>Refresh Proxies</value>
|
<value>Refresh Proxies</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuProxiesSelectActivity" xml:space="preserve">
|
<data name="menuProxiesSelectActivity" xml:space="preserve">
|
||||||
<value>Select active node (Enter)</value>
|
<value>Select active node</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbSettingsDomainStrategy4Out" xml:space="preserve">
|
<data name="TbSettingsDomainStrategy4Out" xml:space="preserve">
|
||||||
<value>Default domain strategy for outbound</value>
|
<value>Default domain strategy for outbound</value>
|
||||||
@@ -1222,7 +1225,7 @@
|
|||||||
<value>Export Base64-encoded Share Links to Clipboard</value>
|
<value>Export Base64-encoded Share Links to Clipboard</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuExport2ClientConfigClipboard" xml:space="preserve">
|
<data name="menuExport2ClientConfigClipboard" xml:space="preserve">
|
||||||
<value>Export selected Configuration for complete configuration to clipboard</value>
|
<value>Export selected for complete configuration to clipboard</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuShowOrHideMainWindow" xml:space="preserve">
|
<data name="menuShowOrHideMainWindow" xml:space="preserve">
|
||||||
<value>Show or hide the main window</value>
|
<value>Show or hide the main window</value>
|
||||||
@@ -1378,22 +1381,22 @@
|
|||||||
<value>Generate Policy Group from Multiple Profiles</value>
|
<value>Generate Policy Group from Multiple Profiles</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuGenGroupMultipleServerXrayRandom" xml:space="preserve">
|
<data name="menuGenGroupMultipleServerXrayRandom" xml:space="preserve">
|
||||||
<value>Multi-Configuration Random by Xray</value>
|
<value>Random by Xray</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuGenGroupMultipleServerXrayRoundRobin" xml:space="preserve">
|
<data name="menuGenGroupMultipleServerXrayRoundRobin" xml:space="preserve">
|
||||||
<value>Multi-Configuration RoundRobin by Xray</value>
|
<value>RoundRobin by Xray</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuGenGroupMultipleServerXrayLeastPing" xml:space="preserve">
|
<data name="menuGenGroupMultipleServerXrayLeastPing" xml:space="preserve">
|
||||||
<value>Multi-Configuration LeastPing by Xray</value>
|
<value>LeastPing by Xray</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuGenGroupMultipleServerXrayLeastLoad" xml:space="preserve">
|
<data name="menuGenGroupMultipleServerXrayLeastLoad" xml:space="preserve">
|
||||||
<value>Multi-Configuration LeastLoad by Xray</value>
|
<value>LeastLoad by Xray</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuGenGroupMultipleServerSingBoxLeastPing" xml:space="preserve">
|
<data name="menuGenGroupMultipleServerSingBoxLeastPing" xml:space="preserve">
|
||||||
<value>Multi-Configuration LeastPing by sing-box</value>
|
<value>LeastPing by sing-box</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuExportConfig" xml:space="preserve">
|
<data name="menuExportConfig" xml:space="preserve">
|
||||||
<value>Export Configuration</value>
|
<value>Export</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbSettingsIPAPIUrl" xml:space="preserve">
|
<data name="TbSettingsIPAPIUrl" xml:space="preserve">
|
||||||
<value>Current connection info test URL</value>
|
<value>Current connection info test URL</value>
|
||||||
@@ -1408,7 +1411,7 @@
|
|||||||
<value>Mldsa65Verify</value>
|
<value>Mldsa65Verify</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuAddAnytlsServer" xml:space="preserve">
|
<data name="menuAddAnytlsServer" xml:space="preserve">
|
||||||
<value>Add [Anytls] Configuration</value>
|
<value>Add [Anytls]</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbRemoteDNS" xml:space="preserve">
|
<data name="TbRemoteDNS" xml:space="preserve">
|
||||||
<value>Remote DNS</value>
|
<value>Remote DNS</value>
|
||||||
@@ -1525,28 +1528,28 @@
|
|||||||
<value>Policy Group Type</value>
|
<value>Policy Group Type</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuAddPolicyGroupServer" xml:space="preserve">
|
<data name="menuAddPolicyGroupServer" xml:space="preserve">
|
||||||
<value>Add Policy Group Configuration</value>
|
<value>Add Policy Group </value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuAddProxyChainServer" xml:space="preserve">
|
<data name="menuAddProxyChainServer" xml:space="preserve">
|
||||||
<value>Add Proxy Chain Configuration</value>
|
<value>Add Proxy Chain</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuAddChildServer" xml:space="preserve">
|
<data name="menuAddChildServer" xml:space="preserve">
|
||||||
<value>Add Child Configuration</value>
|
<value>Add Child </value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuRemoveChildServer" xml:space="preserve">
|
<data name="menuRemoveChildServer" xml:space="preserve">
|
||||||
<value>Remove Child Configuration</value>
|
<value>Remove Child </value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuServerList" xml:space="preserve">
|
<data name="menuServerList" xml:space="preserve">
|
||||||
<value>Server List</value>
|
<value>Configuration item 1, Auto add from subscription group</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbFallback" xml:space="preserve">
|
<data name="TbFallback" xml:space="preserve">
|
||||||
<value>Fallback</value>
|
<value>Fallback</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuGenGroupMultipleServerSingBoxFallback" xml:space="preserve">
|
<data name="menuGenGroupMultipleServerSingBoxFallback" xml:space="preserve">
|
||||||
<value>Multi-Configuration Fallback by sing-box</value>
|
<value>Fallback by sing-box</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuGenGroupMultipleServerXrayFallback" xml:space="preserve">
|
<data name="menuGenGroupMultipleServerXrayFallback" xml:space="preserve">
|
||||||
<value>Multi-Configuration Fallback by Xray</value>
|
<value>Fallback by Xray</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="CoreNotSupportNetwork" xml:space="preserve">
|
<data name="CoreNotSupportNetwork" xml:space="preserve">
|
||||||
<value>Core '{0}' does not support network type '{1}'.</value>
|
<value>Core '{0}' does not support network type '{1}'.</value>
|
||||||
@@ -1596,4 +1599,46 @@
|
|||||||
<data name="TbBootstrapDNSTips" xml:space="preserve">
|
<data name="TbBootstrapDNSTips" xml:space="preserve">
|
||||||
<value>Resolve DNS server domains, requires IP</value>
|
<value>Resolve DNS server domains, requires IP</value>
|
||||||
</data>
|
</data>
|
||||||
|
<data name="menuFastRealPing" xml:space="preserve">
|
||||||
|
<value>Test real delay</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbPolicyGroupSubChildTip" xml:space="preserve">
|
||||||
|
<value>Auto add filtered configuration from subscription groups</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbCertPinning" xml:space="preserve">
|
||||||
|
<value>Certificate Pinning</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbCertPinningTips" xml:space="preserve">
|
||||||
|
<value>Server Certificate (PEM format, optional)
|
||||||
|
When specified, the certificate will be pinned, and "Allow Insecure" will be disabled.
|
||||||
|
|
||||||
|
The "Get Certificate" action may fail if a self-signed certificate is used or if the system contains an untrusted or malicious CA.</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbFetchCert" xml:space="preserve">
|
||||||
|
<value>Fetch Certificate</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbFetchCertChain" xml:space="preserve">
|
||||||
|
<value>Fetch Certificate Chain</value>
|
||||||
|
</data>
|
||||||
|
<data name="ServerNameMustBeValidDomain" xml:space="preserve">
|
||||||
|
<value>Please set a valid domain</value>
|
||||||
|
</data>
|
||||||
|
<data name="CertNotSet" xml:space="preserve">
|
||||||
|
<value>Certificate not set</value>
|
||||||
|
</data>
|
||||||
|
<data name="CertSet" xml:space="preserve">
|
||||||
|
<value>Certificate set</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbSettingsCustomSystemProxyPacPath" xml:space="preserve">
|
||||||
|
<value>Custom PAC file path</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbSettingsCustomSystemProxyScriptPath" xml:space="preserve">
|
||||||
|
<value>Custom system proxy script file path</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbSettingsMacOSShowInDock" xml:space="preserve">
|
||||||
|
<value>macOS displays this in the Dock (requires restart)</value>
|
||||||
|
</data>
|
||||||
|
<data name="menuServerList2" xml:space="preserve">
|
||||||
|
<value>Configuration Item 2, Select and add from self-built</value>
|
||||||
|
</data>
|
||||||
</root>
|
</root>
|
||||||
@@ -472,10 +472,10 @@
|
|||||||
<value>Язык (требуется перезапуск)</value>
|
<value>Язык (требуется перезапуск)</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuAddServerViaClipboard" xml:space="preserve">
|
<data name="menuAddServerViaClipboard" xml:space="preserve">
|
||||||
<value>Импорт массива URL из буфера обмена (Ctrl+V)</value>
|
<value>Импорт массива URL из буфера обмена</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuAddServerViaScan" xml:space="preserve">
|
<data name="menuAddServerViaScan" xml:space="preserve">
|
||||||
<value>Сканировать QR-код с экрана (Ctrl+S)</value>
|
<value>Сканировать QR-код с экрана</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuCopyServer" xml:space="preserve">
|
<data name="menuCopyServer" xml:space="preserve">
|
||||||
<value>Клонировать выбранный сервер</value>
|
<value>Клонировать выбранный сервер</value>
|
||||||
@@ -484,31 +484,31 @@
|
|||||||
<value>Удалить дубликаты серверов</value>
|
<value>Удалить дубликаты серверов</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuRemoveServer" xml:space="preserve">
|
<data name="menuRemoveServer" xml:space="preserve">
|
||||||
<value>Удалить выбранные серверы (Delete)</value>
|
<value>Удалить выбранные серверы</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuSetDefaultServer" xml:space="preserve">
|
<data name="menuSetDefaultServer" xml:space="preserve">
|
||||||
<value>Установить как активный сервер (Enter)</value>
|
<value>Установить как активный сервер</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuClearServerStatistics" xml:space="preserve">
|
<data name="menuClearServerStatistics" xml:space="preserve">
|
||||||
<value>Очистить всю статистику</value>
|
<value>Очистить всю статистику</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuRealPingServer" xml:space="preserve">
|
<data name="menuRealPingServer" xml:space="preserve">
|
||||||
<value>Тест на реальную задержку сервера (Ctrl+R)</value>
|
<value>Тест на реальную задержку сервера</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuSortServerResult" xml:space="preserve">
|
<data name="menuSortServerResult" xml:space="preserve">
|
||||||
<value>Сортировать по результату теста</value>
|
<value>Сортировать по результату теста</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuSpeedServer" xml:space="preserve">
|
<data name="menuSpeedServer" xml:space="preserve">
|
||||||
<value>Тест на скорость загрузки сервера (Ctrl+T)</value>
|
<value>Тест на скорость загрузки сервера</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuTcpingServer" xml:space="preserve">
|
<data name="menuTcpingServer" xml:space="preserve">
|
||||||
<value>Тест задержки с tcping (Ctrl+O)</value>
|
<value>Тест задержки с tcping</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuExport2ClientConfig" xml:space="preserve">
|
<data name="menuExport2ClientConfig" xml:space="preserve">
|
||||||
<value>Экспортировать выбранный сервер для клиента</value>
|
<value>Экспортировать выбранный сервер для клиента</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuExport2ShareUrl" xml:space="preserve">
|
<data name="menuExport2ShareUrl" xml:space="preserve">
|
||||||
<value>Экспорт URL-адресов общего доступа в буфер обмена (Ctrl+C)</value>
|
<value>Экспорт URL-адресов общего доступа в буфер обмена</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuAddCustomServer" xml:space="preserve">
|
<data name="menuAddCustomServer" xml:space="preserve">
|
||||||
<value>Добавить сервер пользовательской конфигурации</value>
|
<value>Добавить сервер пользовательской конфигурации</value>
|
||||||
@@ -529,19 +529,19 @@
|
|||||||
<value>Добавить сервер [VMess]</value>
|
<value>Добавить сервер [VMess]</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuSelectAll" xml:space="preserve">
|
<data name="menuSelectAll" xml:space="preserve">
|
||||||
<value>Выбрать все (Ctrl+A)</value>
|
<value>Выбрать все</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMsgViewClear" xml:space="preserve">
|
<data name="menuMsgViewClear" xml:space="preserve">
|
||||||
<value>Очистить все</value>
|
<value>Очистить все</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMsgViewCopy" xml:space="preserve">
|
<data name="menuMsgViewCopy" xml:space="preserve">
|
||||||
<value>Скопировать (Ctrl+C)</value>
|
<value>Скопировать</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMsgViewCopyAll" xml:space="preserve">
|
<data name="menuMsgViewCopyAll" xml:space="preserve">
|
||||||
<value>Скопировать все</value>
|
<value>Скопировать все</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMsgViewSelectAll" xml:space="preserve">
|
<data name="menuMsgViewSelectAll" xml:space="preserve">
|
||||||
<value>Выбрать все (Ctrl+A)</value>
|
<value>Выбрать все</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuSubAdd" xml:space="preserve">
|
<data name="menuSubAdd" xml:space="preserve">
|
||||||
<value>Добавить</value>
|
<value>Добавить</value>
|
||||||
@@ -781,7 +781,7 @@
|
|||||||
<value>Режим PAC</value>
|
<value>Режим PAC</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuShareServer" xml:space="preserve">
|
<data name="menuShareServer" xml:space="preserve">
|
||||||
<value>Поделиться сервером (Ctrl+F)</value>
|
<value>Поделиться сервером</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuRouting" xml:space="preserve">
|
<data name="menuRouting" xml:space="preserve">
|
||||||
<value>Маршрутизация</value>
|
<value>Маршрутизация</value>
|
||||||
@@ -793,16 +793,16 @@
|
|||||||
<value>Администратор</value>
|
<value>Администратор</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMoveBottom" xml:space="preserve">
|
<data name="menuMoveBottom" xml:space="preserve">
|
||||||
<value>Спуститься вниз (B)</value>
|
<value>Спуститься вниз</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMoveDown" xml:space="preserve">
|
<data name="menuMoveDown" xml:space="preserve">
|
||||||
<value>Вниз (D)</value>
|
<value>Вниз</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMoveTop" xml:space="preserve">
|
<data name="menuMoveTop" xml:space="preserve">
|
||||||
<value>Подняться наверх (T)</value>
|
<value>Подняться наверх</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMoveUp" xml:space="preserve">
|
<data name="menuMoveUp" xml:space="preserve">
|
||||||
<value>Вверх (U)</value>
|
<value>Вверх</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="MsgFilterTitle" xml:space="preserve">
|
<data name="MsgFilterTitle" xml:space="preserve">
|
||||||
<value>Фильтр, поддерживает regex</value>
|
<value>Фильтр, поддерживает regex</value>
|
||||||
@@ -853,7 +853,7 @@
|
|||||||
<value>Список правил</value>
|
<value>Список правил</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuRuleRemove" xml:space="preserve">
|
<data name="menuRuleRemove" xml:space="preserve">
|
||||||
<value>Удалить правила (Delete)</value>
|
<value>Удалить правила</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuRoutingRuleDetailsSetting" xml:space="preserve">
|
<data name="menuRoutingRuleDetailsSetting" xml:space="preserve">
|
||||||
<value>Детальные настройки правил маршрутизации</value>
|
<value>Детальные настройки правил маршрутизации</value>
|
||||||
@@ -922,7 +922,7 @@
|
|||||||
<value>Пропустить тест</value>
|
<value>Пропустить тест</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuEditServer" xml:space="preserve">
|
<data name="menuEditServer" xml:space="preserve">
|
||||||
<value>Редактировать сервер (Ctrl+D)</value>
|
<value>Редактировать сервер</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbSettingsDoubleClick2Activate" xml:space="preserve">
|
<data name="TbSettingsDoubleClick2Activate" xml:space="preserve">
|
||||||
<value>Двойной клик чтобы сделать сервер активным</value>
|
<value>Двойной клик чтобы сделать сервер активным</value>
|
||||||
@@ -976,7 +976,10 @@
|
|||||||
<value>Включить аппаратное ускорение (требуется перезагрузка)</value>
|
<value>Включить аппаратное ускорение (требуется перезагрузка)</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="SpeedtestingWait" xml:space="preserve">
|
<data name="SpeedtestingWait" xml:space="preserve">
|
||||||
<value>Ожидание тестирования (нажмите ESC для отмены)…</value>
|
<value>Ожидание тестирования…</value>
|
||||||
|
</data>
|
||||||
|
<data name="SpeedtestingPressEscToExit" xml:space="preserve">
|
||||||
|
<value>нажмите ESC для отмены</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TipDisplayLog" xml:space="preserve">
|
<data name="TipDisplayLog" xml:space="preserve">
|
||||||
<value>Отключите при аномальном разрыве соединения</value>
|
<value>Отключите при аномальном разрыве соединения</value>
|
||||||
@@ -1204,7 +1207,7 @@
|
|||||||
<value>Обновить прокси</value>
|
<value>Обновить прокси</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuProxiesSelectActivity" xml:space="preserve">
|
<data name="menuProxiesSelectActivity" xml:space="preserve">
|
||||||
<value>Сделать узел активным (Enter)</value>
|
<value>Сделать узел активным</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbSettingsDomainStrategy4Out" xml:space="preserve">
|
<data name="TbSettingsDomainStrategy4Out" xml:space="preserve">
|
||||||
<value>Стратегия домена по умолчанию для исходящих</value>
|
<value>Стратегия домена по умолчанию для исходящих</value>
|
||||||
@@ -1537,7 +1540,7 @@
|
|||||||
<value>Remove Child Configuration</value>
|
<value>Remove Child Configuration</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuServerList" xml:space="preserve">
|
<data name="menuServerList" xml:space="preserve">
|
||||||
<value>Server List</value>
|
<value>Configuration item 1, Auto add from subscription group</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbFallback" xml:space="preserve">
|
<data name="TbFallback" xml:space="preserve">
|
||||||
<value>Fallback</value>
|
<value>Fallback</value>
|
||||||
@@ -1596,4 +1599,46 @@
|
|||||||
<data name="TbBootstrapDNSTips" xml:space="preserve">
|
<data name="TbBootstrapDNSTips" xml:space="preserve">
|
||||||
<value>Resolve DNS server domains, requires IP</value>
|
<value>Resolve DNS server domains, requires IP</value>
|
||||||
</data>
|
</data>
|
||||||
|
<data name="menuFastRealPing" xml:space="preserve">
|
||||||
|
<value>Test real delay</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbPolicyGroupSubChildTip" xml:space="preserve">
|
||||||
|
<value>Auto add filtered configuration from subscription groups</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbCertPinning" xml:space="preserve">
|
||||||
|
<value>Certificate Pinning</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbCertPinningTips" xml:space="preserve">
|
||||||
|
<value>Server Certificate (PEM format, optional)
|
||||||
|
When specified, the certificate will be pinned, and "Allow Insecure" will be disabled.
|
||||||
|
|
||||||
|
The "Get Certificate" action may fail if a self-signed certificate is used or if the system contains an untrusted or malicious CA.</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbFetchCert" xml:space="preserve">
|
||||||
|
<value>Fetch Certificate</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbFetchCertChain" xml:space="preserve">
|
||||||
|
<value>Fetch Certificate Chain</value>
|
||||||
|
</data>
|
||||||
|
<data name="ServerNameMustBeValidDomain" xml:space="preserve">
|
||||||
|
<value>Please set a valid domain</value>
|
||||||
|
</data>
|
||||||
|
<data name="CertNotSet" xml:space="preserve">
|
||||||
|
<value>Certificate not set</value>
|
||||||
|
</data>
|
||||||
|
<data name="CertSet" xml:space="preserve">
|
||||||
|
<value>Certificate set</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbSettingsCustomSystemProxyPacPath" xml:space="preserve">
|
||||||
|
<value>Custom PAC file path</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbSettingsCustomSystemProxyScriptPath" xml:space="preserve">
|
||||||
|
<value>Custom system proxy script file path</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbSettingsMacOSShowInDock" xml:space="preserve">
|
||||||
|
<value>macOS displays this in the Dock (requires restart)</value>
|
||||||
|
</data>
|
||||||
|
<data name="menuServerList2" xml:space="preserve">
|
||||||
|
<value>Configuration Item 2, Select and add from self-built</value>
|
||||||
|
</data>
|
||||||
</root>
|
</root>
|
||||||
@@ -427,7 +427,7 @@
|
|||||||
<value>路由设置</value>
|
<value>路由设置</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuServers" xml:space="preserve">
|
<data name="menuServers" xml:space="preserve">
|
||||||
<value>配置文件</value>
|
<value>配置项</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuSetting" xml:space="preserve">
|
<data name="menuSetting" xml:space="preserve">
|
||||||
<value>设置</value>
|
<value>设置</value>
|
||||||
@@ -472,10 +472,10 @@
|
|||||||
<value>语言 (需重启)</value>
|
<value>语言 (需重启)</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuAddServerViaClipboard" xml:space="preserve">
|
<data name="menuAddServerViaClipboard" xml:space="preserve">
|
||||||
<value>从剪贴板导入分享链接 (Ctrl+V)</value>
|
<value>从剪贴板导入分享链接</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuAddServerViaScan" xml:space="preserve">
|
<data name="menuAddServerViaScan" xml:space="preserve">
|
||||||
<value>扫描屏幕上的二维码 (Ctrl+S)</value>
|
<value>扫描屏幕上的二维码</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuCopyServer" xml:space="preserve">
|
<data name="menuCopyServer" xml:space="preserve">
|
||||||
<value>克隆所选</value>
|
<value>克隆所选</value>
|
||||||
@@ -484,31 +484,31 @@
|
|||||||
<value>移除重复</value>
|
<value>移除重复</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuRemoveServer" xml:space="preserve">
|
<data name="menuRemoveServer" xml:space="preserve">
|
||||||
<value>移除所选 (多选) (Delete)</value>
|
<value>移除所选 (多选)</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuSetDefaultServer" xml:space="preserve">
|
<data name="menuSetDefaultServer" xml:space="preserve">
|
||||||
<value>设为活动 (Enter)</value>
|
<value>设为活动</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuClearServerStatistics" xml:space="preserve">
|
<data name="menuClearServerStatistics" xml:space="preserve">
|
||||||
<value>清除所有服务统计数据</value>
|
<value>清除所有服务统计数据</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuRealPingServer" xml:space="preserve">
|
<data name="menuRealPingServer" xml:space="preserve">
|
||||||
<value>测试真连接延迟 (多选) (Ctrl+R)</value>
|
<value>测试真连接延迟 (多选)</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuSortServerResult" xml:space="preserve">
|
<data name="menuSortServerResult" xml:space="preserve">
|
||||||
<value>按测试结果排序</value>
|
<value>按测试结果排序</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuSpeedServer" xml:space="preserve">
|
<data name="menuSpeedServer" xml:space="preserve">
|
||||||
<value>测试速度 (多选) (Ctrl+T)</value>
|
<value>测试速度 (多选)</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuTcpingServer" xml:space="preserve">
|
<data name="menuTcpingServer" xml:space="preserve">
|
||||||
<value>测试延迟 Tcping (多选) (Ctrl+O)</value>
|
<value>测试延迟 Tcping (多选)</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuExport2ClientConfig" xml:space="preserve">
|
<data name="menuExport2ClientConfig" xml:space="preserve">
|
||||||
<value>导出所选完整配置</value>
|
<value>导出所选完整配置</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuExport2ShareUrl" xml:space="preserve">
|
<data name="menuExport2ShareUrl" xml:space="preserve">
|
||||||
<value>导出分享链接至剪贴板 (多选) (Ctrl+C)</value>
|
<value>导出分享链接至剪贴板 (多选)</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuAddCustomServer" xml:space="preserve">
|
<data name="menuAddCustomServer" xml:space="preserve">
|
||||||
<value>添加自定义配置</value>
|
<value>添加自定义配置</value>
|
||||||
@@ -529,19 +529,19 @@
|
|||||||
<value>添加 [VMess] </value>
|
<value>添加 [VMess] </value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuSelectAll" xml:space="preserve">
|
<data name="menuSelectAll" xml:space="preserve">
|
||||||
<value>全选 (Ctrl+A)</value>
|
<value>全选</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMsgViewClear" xml:space="preserve">
|
<data name="menuMsgViewClear" xml:space="preserve">
|
||||||
<value>清除所有</value>
|
<value>清除所有</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMsgViewCopy" xml:space="preserve">
|
<data name="menuMsgViewCopy" xml:space="preserve">
|
||||||
<value>复制 (Ctrl+C)</value>
|
<value>复制</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMsgViewCopyAll" xml:space="preserve">
|
<data name="menuMsgViewCopyAll" xml:space="preserve">
|
||||||
<value>复制所有</value>
|
<value>复制所有</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMsgViewSelectAll" xml:space="preserve">
|
<data name="menuMsgViewSelectAll" xml:space="preserve">
|
||||||
<value>全选 (Ctrl+A)</value>
|
<value>全选</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuSubAdd" xml:space="preserve">
|
<data name="menuSubAdd" xml:space="preserve">
|
||||||
<value>添加</value>
|
<value>添加</value>
|
||||||
@@ -781,7 +781,7 @@
|
|||||||
<value>Pac 模式</value>
|
<value>Pac 模式</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuShareServer" xml:space="preserve">
|
<data name="menuShareServer" xml:space="preserve">
|
||||||
<value>分享 (Ctrl+F)</value>
|
<value>分享</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuRouting" xml:space="preserve">
|
<data name="menuRouting" xml:space="preserve">
|
||||||
<value>路由</value>
|
<value>路由</value>
|
||||||
@@ -793,16 +793,16 @@
|
|||||||
<value>以管理员身份运行</value>
|
<value>以管理员身份运行</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMoveBottom" xml:space="preserve">
|
<data name="menuMoveBottom" xml:space="preserve">
|
||||||
<value>下移至底 (B)</value>
|
<value>下移至底</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMoveDown" xml:space="preserve">
|
<data name="menuMoveDown" xml:space="preserve">
|
||||||
<value>下移 (D)</value>
|
<value>下移</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMoveTop" xml:space="preserve">
|
<data name="menuMoveTop" xml:space="preserve">
|
||||||
<value>上移至顶 (T)</value>
|
<value>上移至顶</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMoveUp" xml:space="preserve">
|
<data name="menuMoveUp" xml:space="preserve">
|
||||||
<value>上移 (U)</value>
|
<value>上移</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="MsgFilterTitle" xml:space="preserve">
|
<data name="MsgFilterTitle" xml:space="preserve">
|
||||||
<value>过滤器 (支持正则)</value>
|
<value>过滤器 (支持正则)</value>
|
||||||
@@ -817,10 +817,10 @@
|
|||||||
<value>一键导入规则集</value>
|
<value>一键导入规则集</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuRoutingAdvancedRemove" xml:space="preserve">
|
<data name="menuRoutingAdvancedRemove" xml:space="preserve">
|
||||||
<value>移除所选规则 (Delete)</value>
|
<value>移除所选规则</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuRoutingAdvancedSetDefault" xml:space="preserve">
|
<data name="menuRoutingAdvancedSetDefault" xml:space="preserve">
|
||||||
<value>设为活动规则 (Enter)</value>
|
<value>设为活动规则</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbdomainStrategy" xml:space="preserve">
|
<data name="TbdomainStrategy" xml:space="preserve">
|
||||||
<value>域名解析策略</value>
|
<value>域名解析策略</value>
|
||||||
@@ -853,7 +853,7 @@
|
|||||||
<value>规则列表</value>
|
<value>规则列表</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuRuleRemove" xml:space="preserve">
|
<data name="menuRuleRemove" xml:space="preserve">
|
||||||
<value>移除所选规则 (Delete)</value>
|
<value>移除所选规则</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuRoutingRuleDetailsSetting" xml:space="preserve">
|
<data name="menuRoutingRuleDetailsSetting" xml:space="preserve">
|
||||||
<value>路由规则详情设置</value>
|
<value>路由规则详情设置</value>
|
||||||
@@ -922,7 +922,7 @@
|
|||||||
<value>跳过测试</value>
|
<value>跳过测试</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuEditServer" xml:space="preserve">
|
<data name="menuEditServer" xml:space="preserve">
|
||||||
<value>编辑 (Ctrl+D)</value>
|
<value>编辑</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbSettingsDoubleClick2Activate" xml:space="preserve">
|
<data name="TbSettingsDoubleClick2Activate" xml:space="preserve">
|
||||||
<value>主界面双击设为活动</value>
|
<value>主界面双击设为活动</value>
|
||||||
@@ -976,7 +976,10 @@
|
|||||||
<value>启用硬件加速 (需重启)</value>
|
<value>启用硬件加速 (需重启)</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="SpeedtestingWait" xml:space="preserve">
|
<data name="SpeedtestingWait" xml:space="preserve">
|
||||||
<value>等待测试中 (按 ESC 终止)...</value>
|
<value>等待测试...</value>
|
||||||
|
</data>
|
||||||
|
<data name="SpeedtestingPressEscToExit" xml:space="preserve">
|
||||||
|
<value>按 ESC 可终止测试</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TipDisplayLog" xml:space="preserve">
|
<data name="TipDisplayLog" xml:space="preserve">
|
||||||
<value>当有异常断流时请关闭</value>
|
<value>当有异常断流时请关闭</value>
|
||||||
@@ -1201,7 +1204,7 @@
|
|||||||
<value>刷新</value>
|
<value>刷新</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuProxiesSelectActivity" xml:space="preserve">
|
<data name="menuProxiesSelectActivity" xml:space="preserve">
|
||||||
<value>设为活动 (Enter)</value>
|
<value>设为活动</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbSettingsDomainStrategy4Out" xml:space="preserve">
|
<data name="TbSettingsDomainStrategy4Out" xml:space="preserve">
|
||||||
<value>Outbound 默认解析策略</value>
|
<value>Outbound 默认解析策略</value>
|
||||||
@@ -1528,13 +1531,13 @@
|
|||||||
<value>添加链式代理</value>
|
<value>添加链式代理</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuAddChildServer" xml:space="preserve">
|
<data name="menuAddChildServer" xml:space="preserve">
|
||||||
<value>添加子项</value>
|
<value>添加子配置</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuRemoveChildServer" xml:space="preserve">
|
<data name="menuRemoveChildServer" xml:space="preserve">
|
||||||
<value>删除子项</value>
|
<value>删除子配置</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuServerList" xml:space="preserve">
|
<data name="menuServerList" xml:space="preserve">
|
||||||
<value>子项列表</value>
|
<value>子配置项一,从订阅分组中自动添加</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbFallback" xml:space="preserve">
|
<data name="TbFallback" xml:space="preserve">
|
||||||
<value>故障转移</value>
|
<value>故障转移</value>
|
||||||
@@ -1593,4 +1596,46 @@
|
|||||||
<data name="TbBootstrapDNSTips" xml:space="preserve">
|
<data name="TbBootstrapDNSTips" xml:space="preserve">
|
||||||
<value>解析 DNS 服务器域名,需指定为 IP</value>
|
<value>解析 DNS 服务器域名,需指定为 IP</value>
|
||||||
</data>
|
</data>
|
||||||
|
<data name="menuFastRealPing" xml:space="preserve">
|
||||||
|
<value>一键测试真连接延迟</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbPolicyGroupSubChildTip" xml:space="preserve">
|
||||||
|
<value>自动从订阅分组添加过滤后的配置</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbCertPinning" xml:space="preserve">
|
||||||
|
<value>固定证书</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbCertPinningTips" xml:space="preserve">
|
||||||
|
<value>服务器证书(PEM 格式,可选)
|
||||||
|
当指定此证书后,将固定该证书,并禁用“跳过证书验证”选项。
|
||||||
|
|
||||||
|
“获取证书”操作可能失败,原因可能是使用了自签证书,或系统中存在不受信任或恶意的 CA。</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbFetchCert" xml:space="preserve">
|
||||||
|
<value>获取证书</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbFetchCertChain" xml:space="preserve">
|
||||||
|
<value>获取证书链</value>
|
||||||
|
</data>
|
||||||
|
<data name="ServerNameMustBeValidDomain" xml:space="preserve">
|
||||||
|
<value>请设置有效的域名</value>
|
||||||
|
</data>
|
||||||
|
<data name="CertNotSet" xml:space="preserve">
|
||||||
|
<value>证书未设置</value>
|
||||||
|
</data>
|
||||||
|
<data name="CertSet" xml:space="preserve">
|
||||||
|
<value>证书已设置</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbSettingsCustomSystemProxyPacPath" xml:space="preserve">
|
||||||
|
<value>自定义 PAC 文件路径</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbSettingsCustomSystemProxyScriptPath" xml:space="preserve">
|
||||||
|
<value>自定义系统代理脚本文件路径</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbSettingsMacOSShowInDock" xml:space="preserve">
|
||||||
|
<value>macOS 在 Dock 栏中显示 (需重启)</value>
|
||||||
|
</data>
|
||||||
|
<data name="menuServerList2" xml:space="preserve">
|
||||||
|
<value>子配置项二,从自建中选择添加</value>
|
||||||
|
</data>
|
||||||
</root>
|
</root>
|
||||||
@@ -121,7 +121,7 @@
|
|||||||
<value>匯出分享連結至剪貼簿成功</value>
|
<value>匯出分享連結至剪貼簿成功</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="CheckServerSettings" xml:space="preserve">
|
<data name="CheckServerSettings" xml:space="preserve">
|
||||||
<value>請先檢查設定檔設定</value>
|
<value>請先檢查設定</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="ConfigurationFormatIncorrect" xml:space="preserve">
|
<data name="ConfigurationFormatIncorrect" xml:space="preserve">
|
||||||
<value>設定格式不正確</value>
|
<value>設定格式不正確</value>
|
||||||
@@ -133,7 +133,7 @@
|
|||||||
<value>下載開始...</value>
|
<value>下載開始...</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="FailedConversionConfiguration" xml:space="preserve">
|
<data name="FailedConversionConfiguration" xml:space="preserve">
|
||||||
<value>轉換設定檔失敗</value>
|
<value>轉換設定失敗</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="FailedGenDefaultConfiguration" xml:space="preserve">
|
<data name="FailedGenDefaultConfiguration" xml:space="preserve">
|
||||||
<value>生成預設設定檔失敗</value>
|
<value>生成預設設定檔失敗</value>
|
||||||
@@ -142,10 +142,10 @@
|
|||||||
<value>獲取預設設定失敗</value>
|
<value>獲取預設設定失敗</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="FailedImportedCustomServer" xml:space="preserve">
|
<data name="FailedImportedCustomServer" xml:space="preserve">
|
||||||
<value>匯入自訂設定設定檔失敗</value>
|
<value>匯入自訂設定失敗</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="FailedReadConfiguration" xml:space="preserve">
|
<data name="FailedReadConfiguration" xml:space="preserve">
|
||||||
<value>讀取設定檔失敗</value>
|
<value>讀取設定失敗</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="FillCorrectServerPort" xml:space="preserve">
|
<data name="FillCorrectServerPort" xml:space="preserve">
|
||||||
<value>請填寫正確格式的埠</value>
|
<value>請填寫正確格式的埠</value>
|
||||||
@@ -265,13 +265,13 @@
|
|||||||
<value>請選擇協定</value>
|
<value>請選擇協定</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="PleaseSelectServer" xml:space="preserve">
|
<data name="PleaseSelectServer" xml:space="preserve">
|
||||||
<value>請先選擇設定檔</value>
|
<value>請先選擇設定</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="RemoveDuplicateServerResult" xml:space="preserve">
|
<data name="RemoveDuplicateServerResult" xml:space="preserve">
|
||||||
<value>設定檔去重完成。原數量: {0},現數量: {1}。</value>
|
<value>去重完成。原數量: {0},現數量: {1}。</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="RemoveServer" xml:space="preserve">
|
<data name="RemoveServer" xml:space="preserve">
|
||||||
<value>是否確定移除設定檔?</value>
|
<value>是否確定移除?</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="SaveClientConfigurationIn" xml:space="preserve">
|
<data name="SaveClientConfigurationIn" xml:space="preserve">
|
||||||
<value>用戶端設定檔儲存在:{0}</value>
|
<value>用戶端設定檔儲存在:{0}</value>
|
||||||
@@ -283,10 +283,10 @@
|
|||||||
<value>設定成功。{0}</value>
|
<value>設定成功。{0}</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="SuccessfullyImportedCustomServer" xml:space="preserve">
|
<data name="SuccessfullyImportedCustomServer" xml:space="preserve">
|
||||||
<value>成功匯入自訂設定設定檔</value>
|
<value>成功匯入自訂節點</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="SuccessfullyImportedServerViaClipboard" xml:space="preserve">
|
<data name="SuccessfullyImportedServerViaClipboard" xml:space="preserve">
|
||||||
<value>成功從剪貼簿匯入 {0} 個設定檔</value>
|
<value>成功從剪貼簿匯入 {0} 個節點</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="SuccessfullyImportedServerViaScan" xml:space="preserve">
|
<data name="SuccessfullyImportedServerViaScan" xml:space="preserve">
|
||||||
<value>掃描匯入分享連結成功</value>
|
<value>掃描匯入分享連結成功</value>
|
||||||
@@ -385,7 +385,7 @@
|
|||||||
<value>所有</value>
|
<value>所有</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="FillServerAddressCustom" xml:space="preserve">
|
<data name="FillServerAddressCustom" xml:space="preserve">
|
||||||
<value>請瀏覽匯入設定檔設定</value>
|
<value>請瀏覽匯入設定</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="Speedtesting" xml:space="preserve">
|
<data name="Speedtesting" xml:space="preserve">
|
||||||
<value>測試中...</value>
|
<value>測試中...</value>
|
||||||
@@ -397,7 +397,7 @@
|
|||||||
<value>本機</value>
|
<value>本機</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="MsgServerTitle" xml:space="preserve">
|
<data name="MsgServerTitle" xml:space="preserve">
|
||||||
<value>設定檔過濾,按 Enter 執行</value>
|
<value>過濾器,按 Enter 執行</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuCheckUpdate" xml:space="preserve">
|
<data name="menuCheckUpdate" xml:space="preserve">
|
||||||
<value>檢查更新</value>
|
<value>檢查更新</value>
|
||||||
@@ -472,76 +472,76 @@
|
|||||||
<value>語言 (需重啟)</value>
|
<value>語言 (需重啟)</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuAddServerViaClipboard" xml:space="preserve">
|
<data name="menuAddServerViaClipboard" xml:space="preserve">
|
||||||
<value>從剪貼簿導入分享連結 (Ctrl+V)</value>
|
<value>從剪貼簿導入分享連結</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuAddServerViaScan" xml:space="preserve">
|
<data name="menuAddServerViaScan" xml:space="preserve">
|
||||||
<value>掃描螢幕上的二維碼 (Ctrl+S)</value>
|
<value>掃描螢幕上的二維碼</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuCopyServer" xml:space="preserve">
|
<data name="menuCopyServer" xml:space="preserve">
|
||||||
<value>複製所選設定檔</value>
|
<value>複製所選</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuRemoveDuplicateServer" xml:space="preserve">
|
<data name="menuRemoveDuplicateServer" xml:space="preserve">
|
||||||
<value>移除重複的設定檔</value>
|
<value>移除重複</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuRemoveServer" xml:space="preserve">
|
<data name="menuRemoveServer" xml:space="preserve">
|
||||||
<value>移除所選設定檔 (多選) (Delete)</value>
|
<value>移除所選 (多選)</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuSetDefaultServer" xml:space="preserve">
|
<data name="menuSetDefaultServer" xml:space="preserve">
|
||||||
<value>設為活動設定檔 (Enter)</value>
|
<value>設為活動</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuClearServerStatistics" xml:space="preserve">
|
<data name="menuClearServerStatistics" xml:space="preserve">
|
||||||
<value>清除所有服務統計資料</value>
|
<value>清除所有服務統計資料</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuRealPingServer" xml:space="preserve">
|
<data name="menuRealPingServer" xml:space="preserve">
|
||||||
<value>測試設定檔真連線延遲 (多選) (Ctrl+R)</value>
|
<value>測試真連線延遲 (多選)</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuSortServerResult" xml:space="preserve">
|
<data name="menuSortServerResult" xml:space="preserve">
|
||||||
<value>按測試結果排序</value>
|
<value>按測試結果排序</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuSpeedServer" xml:space="preserve">
|
<data name="menuSpeedServer" xml:space="preserve">
|
||||||
<value>測試設定檔速度 (多選) (Ctrl+T)</value>
|
<value>測試速度 (多選)</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuTcpingServer" xml:space="preserve">
|
<data name="menuTcpingServer" xml:space="preserve">
|
||||||
<value>測試設定檔延遲 Tcping (多選) (Ctrl+O)</value>
|
<value>測試延遲 Tcping (多選)</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuExport2ClientConfig" xml:space="preserve">
|
<data name="menuExport2ClientConfig" xml:space="preserve">
|
||||||
<value>匯出所選設定檔完整設定</value>
|
<value>匯出所選完整設定</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuExport2ShareUrl" xml:space="preserve">
|
<data name="menuExport2ShareUrl" xml:space="preserve">
|
||||||
<value>匯出分享連結至剪貼簿 (多選) (Ctrl+C)</value>
|
<value>匯出分享連結至剪貼簿 (多選)</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuAddCustomServer" xml:space="preserve">
|
<data name="menuAddCustomServer" xml:space="preserve">
|
||||||
<value>新增自訂設定設定檔</value>
|
<value>新增自訂節點</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuAddShadowsocksServer" xml:space="preserve">
|
<data name="menuAddShadowsocksServer" xml:space="preserve">
|
||||||
<value>新增 [Shadowsocks] 設定檔</value>
|
<value>新增 [Shadowsocks] 節點</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuAddSocksServer" xml:space="preserve">
|
<data name="menuAddSocksServer" xml:space="preserve">
|
||||||
<value>新增 [SOCKS] 設定檔</value>
|
<value>新增 [SOCKS] 節點</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuAddTrojanServer" xml:space="preserve">
|
<data name="menuAddTrojanServer" xml:space="preserve">
|
||||||
<value>新增 [Trojan] 設定檔</value>
|
<value>新增 [Trojan] 節點</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuAddVlessServer" xml:space="preserve">
|
<data name="menuAddVlessServer" xml:space="preserve">
|
||||||
<value>新增 [VLESS] 設定檔</value>
|
<value>新增 [VLESS] 節點</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuAddVmessServer" xml:space="preserve">
|
<data name="menuAddVmessServer" xml:space="preserve">
|
||||||
<value>新增 [VMess] 設定檔</value>
|
<value>新增 [VMess] 節點</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuSelectAll" xml:space="preserve">
|
<data name="menuSelectAll" xml:space="preserve">
|
||||||
<value>全選 (Ctrl+A)</value>
|
<value>全選</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMsgViewClear" xml:space="preserve">
|
<data name="menuMsgViewClear" xml:space="preserve">
|
||||||
<value>清除所有</value>
|
<value>清除所有</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMsgViewCopy" xml:space="preserve">
|
<data name="menuMsgViewCopy" xml:space="preserve">
|
||||||
<value>複製 (Ctrl+C)</value>
|
<value>複製</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMsgViewCopyAll" xml:space="preserve">
|
<data name="menuMsgViewCopyAll" xml:space="preserve">
|
||||||
<value>複製所有</value>
|
<value>複製所有</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMsgViewSelectAll" xml:space="preserve">
|
<data name="menuMsgViewSelectAll" xml:space="preserve">
|
||||||
<value>全選 (Ctrl+A)</value>
|
<value>全選</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuSubAdd" xml:space="preserve">
|
<data name="menuSubAdd" xml:space="preserve">
|
||||||
<value>新增</value>
|
<value>新增</value>
|
||||||
@@ -676,7 +676,7 @@
|
|||||||
<value>Core: 基礎設定</value>
|
<value>Core: 基礎設定</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbCustomDnsRay" xml:space="preserve">
|
<data name="TbCustomDnsRay" xml:space="preserve">
|
||||||
<value>V2ray Custom DNS</value>
|
<value>v2ray 自訂 DNS</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbSettingsCoreKcp" xml:space="preserve">
|
<data name="TbSettingsCoreKcp" xml:space="preserve">
|
||||||
<value>Core: KCP 設定</value>
|
<value>Core: KCP 設定</value>
|
||||||
@@ -691,7 +691,7 @@
|
|||||||
<value>Outbound Freedom domainStrategy</value>
|
<value>Outbound Freedom domainStrategy</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbSettingsEnableAutoAdjustMainLvColWidth" xml:space="preserve">
|
<data name="TbSettingsEnableAutoAdjustMainLvColWidth" xml:space="preserve">
|
||||||
<value>在更新訂閱後自動調整設定檔列寬</value>
|
<value>在更新訂閱後自動調整列寬</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbSettingsEnableCheckPreReleaseUpdate" xml:space="preserve">
|
<data name="TbSettingsEnableCheckPreReleaseUpdate" xml:space="preserve">
|
||||||
<value>檢查 Pre-Release 更新 (請謹慎啟用)</value>
|
<value>檢查 Pre-Release 更新 (請謹慎啟用)</value>
|
||||||
@@ -700,7 +700,7 @@
|
|||||||
<value>例外</value>
|
<value>例外</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbSettingsExceptionTip" xml:space="preserve">
|
<data name="TbSettingsExceptionTip" xml:space="preserve">
|
||||||
<value>例外:對於下列字元開頭的位址,不使用代理設定檔。使用分號 (;) 分隔。</value>
|
<value>例外:對於下列字元開頭的位址,不使用代理。使用分號 (;) 分隔。</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbSettingsDisplayRealTimeSpeed" xml:space="preserve">
|
<data name="TbSettingsDisplayRealTimeSpeed" xml:space="preserve">
|
||||||
<value>顯示即時速度(需重啟)</value>
|
<value>顯示即時速度(需重啟)</value>
|
||||||
@@ -748,7 +748,7 @@
|
|||||||
<value>系統代理設定</value>
|
<value>系統代理設定</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbSettingsTrayMenuServersLimit" xml:space="preserve">
|
<data name="TbSettingsTrayMenuServersLimit" xml:space="preserve">
|
||||||
<value>工具列右鍵選單設定檔展示數量限制</value>
|
<value>工具列右鍵選單設定展示數量限制</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbSettingsUdpEnabled" xml:space="preserve">
|
<data name="TbSettingsUdpEnabled" xml:space="preserve">
|
||||||
<value>開啟 UDP</value>
|
<value>開啟 UDP</value>
|
||||||
@@ -781,7 +781,7 @@
|
|||||||
<value>PAC 模式</value>
|
<value>PAC 模式</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuShareServer" xml:space="preserve">
|
<data name="menuShareServer" xml:space="preserve">
|
||||||
<value>分享設定檔 (Ctrl+F)</value>
|
<value>分享</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuRouting" xml:space="preserve">
|
<data name="menuRouting" xml:space="preserve">
|
||||||
<value>路由</value>
|
<value>路由</value>
|
||||||
@@ -793,16 +793,16 @@
|
|||||||
<value>以管理員身份執行</value>
|
<value>以管理員身份執行</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMoveBottom" xml:space="preserve">
|
<data name="menuMoveBottom" xml:space="preserve">
|
||||||
<value>下移至底部 (B)</value>
|
<value>下移至底部</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMoveDown" xml:space="preserve">
|
<data name="menuMoveDown" xml:space="preserve">
|
||||||
<value>下移 (D)</value>
|
<value>下移</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMoveTop" xml:space="preserve">
|
<data name="menuMoveTop" xml:space="preserve">
|
||||||
<value>上移至頂部 (T)</value>
|
<value>上移至頂部</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMoveUp" xml:space="preserve">
|
<data name="menuMoveUp" xml:space="preserve">
|
||||||
<value>上移 (U)</value>
|
<value>上移</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="MsgFilterTitle" xml:space="preserve">
|
<data name="MsgFilterTitle" xml:space="preserve">
|
||||||
<value>過濾 (允許正則)</value>
|
<value>過濾 (允許正則)</value>
|
||||||
@@ -817,10 +817,10 @@
|
|||||||
<value>一鍵匯入規則集</value>
|
<value>一鍵匯入規則集</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuRoutingAdvancedRemove" xml:space="preserve">
|
<data name="menuRoutingAdvancedRemove" xml:space="preserve">
|
||||||
<value>移除所選規則 (Delete)</value>
|
<value>移除所選規則</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuRoutingAdvancedSetDefault" xml:space="preserve">
|
<data name="menuRoutingAdvancedSetDefault" xml:space="preserve">
|
||||||
<value>設為活動規則 (Enter)</value>
|
<value>設為活動規則</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbdomainStrategy" xml:space="preserve">
|
<data name="TbdomainStrategy" xml:space="preserve">
|
||||||
<value>域名解析策略</value>
|
<value>域名解析策略</value>
|
||||||
@@ -853,7 +853,7 @@
|
|||||||
<value>規則列表</value>
|
<value>規則列表</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuRuleRemove" xml:space="preserve">
|
<data name="menuRuleRemove" xml:space="preserve">
|
||||||
<value>移除所選規則 (Delete)</value>
|
<value>移除所選規則</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuRoutingRuleDetailsSetting" xml:space="preserve">
|
<data name="menuRoutingRuleDetailsSetting" xml:space="preserve">
|
||||||
<value>路由規則詳情設定</value>
|
<value>路由規則詳情設定</value>
|
||||||
@@ -883,7 +883,7 @@
|
|||||||
<value>請勿將代理伺服器用於本機(Intranet)位址</value>
|
<value>請勿將代理伺服器用於本機(Intranet)位址</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuMixedTestServer" xml:space="preserve">
|
<data name="menuMixedTestServer" xml:space="preserve">
|
||||||
<value>一鍵多執行緒測試延遲和速度 (Ctrl+E)</value>
|
<value>一鍵延遲與速度測試 (Ctrl+E)</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="LvTestDelay" xml:space="preserve">
|
<data name="LvTestDelay" xml:space="preserve">
|
||||||
<value>延遲 (ms)</value>
|
<value>延遲 (ms)</value>
|
||||||
@@ -913,7 +913,7 @@
|
|||||||
<value>移至訂閱分組</value>
|
<value>移至訂閱分組</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbSettingsEnableDragDropSort" xml:space="preserve">
|
<data name="TbSettingsEnableDragDropSort" xml:space="preserve">
|
||||||
<value>啟動設定檔拖放排序 (需重啟)</value>
|
<value>啟用拖放排序 (需重啟)</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbAutoRefresh" xml:space="preserve">
|
<data name="TbAutoRefresh" xml:space="preserve">
|
||||||
<value>自動重新整理</value>
|
<value>自動重新整理</value>
|
||||||
@@ -922,10 +922,10 @@
|
|||||||
<value>跳過測試</value>
|
<value>跳過測試</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuEditServer" xml:space="preserve">
|
<data name="menuEditServer" xml:space="preserve">
|
||||||
<value>編輯設定檔 (Ctrl+D)</value>
|
<value>編輯</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbSettingsDoubleClick2Activate" xml:space="preserve">
|
<data name="TbSettingsDoubleClick2Activate" xml:space="preserve">
|
||||||
<value>主介面輕按兩下設為活動設定檔</value>
|
<value>主介面輕按兩下設為活動</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="SpeedtestingCompleted" xml:space="preserve">
|
<data name="SpeedtestingCompleted" xml:space="preserve">
|
||||||
<value>測試完成</value>
|
<value>測試完成</value>
|
||||||
@@ -943,7 +943,7 @@
|
|||||||
<value>目前字型 (需重啟)</value>
|
<value>目前字型 (需重啟)</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbSettingsCurrentFontFamilyTip" xml:space="preserve">
|
<data name="TbSettingsCurrentFontFamilyTip" xml:space="preserve">
|
||||||
<value>複製字型 TTF/TTC 檔案到目錄 guiFonts,重啟設定</value>
|
<value>複製字型 TTF/TTC 檔案到目錄 guiFonts,重新啟動後生效</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbSettingsSocksPortTip" xml:space="preserve">
|
<data name="TbSettingsSocksPortTip" xml:space="preserve">
|
||||||
<value>Pac 連接埠 = +3;Xray API 連接埠 = +4;mihomo API 連接埠 = +5;</value>
|
<value>Pac 連接埠 = +3;Xray API 連接埠 = +4;mihomo API 連接埠 = +5;</value>
|
||||||
@@ -976,7 +976,10 @@
|
|||||||
<value>啟用硬體加速 (需重啟)</value>
|
<value>啟用硬體加速 (需重啟)</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="SpeedtestingWait" xml:space="preserve">
|
<data name="SpeedtestingWait" xml:space="preserve">
|
||||||
<value>等待測試中(按 ESC 終止)...</value>
|
<value>等待測試中...</value>
|
||||||
|
</data>
|
||||||
|
<data name="SpeedtestingPressEscToExit" xml:space="preserve">
|
||||||
|
<value>按 ECS 以終止測試</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TipDisplayLog" xml:space="preserve">
|
<data name="TipDisplayLog" xml:space="preserve">
|
||||||
<value>當有異常斷流時請關閉</value>
|
<value>當有異常斷流時請關閉</value>
|
||||||
@@ -1003,10 +1006,10 @@
|
|||||||
<value>不需要轉換時請留空</value>
|
<value>不需要轉換時請留空</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuDNSSetting" xml:space="preserve">
|
<data name="menuDNSSetting" xml:space="preserve">
|
||||||
<value>DNS 設定</value>
|
<value>DNS設定</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbCustomDnsSingbox" xml:space="preserve">
|
<data name="TbCustomDnsSingbox" xml:space="preserve">
|
||||||
<value>sing-box Custom DNS</value>
|
<value>sing-box 自訂 DNS</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbDnsSingboxObjectDoc" xml:space="preserve">
|
<data name="TbDnsSingboxObjectDoc" xml:space="preserve">
|
||||||
<value>請填寫 DNS JSON 結構,點擊查看檔案</value>
|
<value>請填寫 DNS JSON 結構,點擊查看檔案</value>
|
||||||
@@ -1030,7 +1033,7 @@
|
|||||||
<value>Domain</value>
|
<value>Domain</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuAddHysteria2Server" xml:space="preserve">
|
<data name="menuAddHysteria2Server" xml:space="preserve">
|
||||||
<value>添加 [Hysteria2] 設定檔</value>
|
<value>新增 [Hysteria2] 節點</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbSettingsHysteriaBandwidth" xml:space="preserve">
|
<data name="TbSettingsHysteriaBandwidth" xml:space="preserve">
|
||||||
<value>Hysteria 最大頻寬 (Up/Dw)</value>
|
<value>Hysteria 最大頻寬 (Up/Dw)</value>
|
||||||
@@ -1039,19 +1042,19 @@
|
|||||||
<value>使用系統 hosts</value>
|
<value>使用系統 hosts</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuAddTuicServer" xml:space="preserve">
|
<data name="menuAddTuicServer" xml:space="preserve">
|
||||||
<value>新增 [TUIC] 設定檔</value>
|
<value>新增 [TUIC] 節點</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbHeaderType8" xml:space="preserve">
|
<data name="TbHeaderType8" xml:space="preserve">
|
||||||
<value>擁塞控制算法</value>
|
<value>擁塞控制算法</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="LvPrevProfile" xml:space="preserve">
|
<data name="LvPrevProfile" xml:space="preserve">
|
||||||
<value>前置代理設定檔別名</value>
|
<value>前置代理節點別名</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="LvNextProfile" xml:space="preserve">
|
<data name="LvNextProfile" xml:space="preserve">
|
||||||
<value>落地代理設定檔別名</value>
|
<value>落地代理節點別名</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="LvPrevProfileTip" xml:space="preserve">
|
<data name="LvPrevProfileTip" xml:space="preserve">
|
||||||
<value>請確保設定檔別名存在並且唯一</value>
|
<value>請確保節點別名存在並且唯一</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbSettingsTunAutoRoute" xml:space="preserve">
|
<data name="TbSettingsTunAutoRoute" xml:space="preserve">
|
||||||
<value>自動路由</value>
|
<value>自動路由</value>
|
||||||
@@ -1072,7 +1075,7 @@
|
|||||||
<value>啟用 IPv6</value>
|
<value>啟用 IPv6</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuAddWireguardServer" xml:space="preserve">
|
<data name="menuAddWireguardServer" xml:space="preserve">
|
||||||
<value>添加 [WireGuard] 設定檔</value>
|
<value>新增 [WireGuard] 節點</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbPrivateKey" xml:space="preserve">
|
<data name="TbPrivateKey" xml:space="preserve">
|
||||||
<value>PrivateKey</value>
|
<value>PrivateKey</value>
|
||||||
@@ -1105,7 +1108,7 @@
|
|||||||
<value>*grpc Authority</value>
|
<value>*grpc Authority</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuAddHttpServer" xml:space="preserve">
|
<data name="menuAddHttpServer" xml:space="preserve">
|
||||||
<value>新增 [HTTP] 設定檔</value>
|
<value>新增 [HTTP] 節點</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbSettingsEnableFragmentTips" xml:space="preserve">
|
<data name="TbSettingsEnableFragmentTips" xml:space="preserve">
|
||||||
<value>和分組前置代理衝突</value>
|
<value>和分組前置代理衝突</value>
|
||||||
@@ -1201,7 +1204,7 @@
|
|||||||
<value>重新整理</value>
|
<value>重新整理</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuProxiesSelectActivity" xml:space="preserve">
|
<data name="menuProxiesSelectActivity" xml:space="preserve">
|
||||||
<value>設為活動節點 (Enter)</value>
|
<value>設為活動節點</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbSettingsDomainStrategy4Out" xml:space="preserve">
|
<data name="TbSettingsDomainStrategy4Out" xml:space="preserve">
|
||||||
<value>Outbound 預設解析策略</value>
|
<value>Outbound 預設解析策略</value>
|
||||||
@@ -1219,13 +1222,13 @@
|
|||||||
<value>匯出分享連結至剪貼簿 (多選) Base64 編碼</value>
|
<value>匯出分享連結至剪貼簿 (多選) Base64 編碼</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuExport2ClientConfigClipboard" xml:space="preserve">
|
<data name="menuExport2ClientConfigClipboard" xml:space="preserve">
|
||||||
<value>匯出所選設定檔完整設定至剪貼簿</value>
|
<value>匯出所選完整設定至剪貼簿</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuShowOrHideMainWindow" xml:space="preserve">
|
<data name="menuShowOrHideMainWindow" xml:space="preserve">
|
||||||
<value>顯示或隱藏主介面</value>
|
<value>顯示或隱藏主介面</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbPreSocksPort4Sub" xml:space="preserve">
|
<data name="TbPreSocksPort4Sub" xml:space="preserve">
|
||||||
<value>自訂設定的 Socks 連接埠</value>
|
<value>自訂 Socks 連接埠</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuBackupAndRestore" xml:space="preserve">
|
<data name="menuBackupAndRestore" xml:space="preserve">
|
||||||
<value>備份和還原</value>
|
<value>備份和還原</value>
|
||||||
@@ -1309,7 +1312,7 @@
|
|||||||
<value>請不要使用不安全的 HTTP 協定訂閱位址</value>
|
<value>請不要使用不安全的 HTTP 協定訂閱位址</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbSettingsCurrentFontFamilyLinuxTip" xml:space="preserve">
|
<data name="TbSettingsCurrentFontFamilyLinuxTip" xml:space="preserve">
|
||||||
<value>安裝字體到系統中,選擇或填入字體名稱,重新啟動設定</value>
|
<value>安裝字體到系統中,選擇或填入字體名稱,重新啟動後生效</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuExitTips" xml:space="preserve">
|
<data name="menuExitTips" xml:space="preserve">
|
||||||
<value>是否確定退出?</value>
|
<value>是否確定退出?</value>
|
||||||
@@ -1336,7 +1339,7 @@
|
|||||||
<value>多執行緒測試時的並發數量</value>
|
<value>多執行緒測試時的並發數量</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbSettingsExceptionTip2" xml:space="preserve">
|
<data name="TbSettingsExceptionTip2" xml:space="preserve">
|
||||||
<value>例外:對於下列位址不使用代理設定檔,使用逗號 (,) 分隔。</value>
|
<value>例外:對於下列位址不使用代理,使用逗號 (,) 分隔。</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbSettingsDestOverride" xml:space="preserve">
|
<data name="TbSettingsDestOverride" xml:space="preserve">
|
||||||
<value>流量探測類型</value>
|
<value>流量探測類型</value>
|
||||||
@@ -1372,31 +1375,31 @@
|
|||||||
<value>會覆蓋埠,多組時用逗號 (,) 隔開</value>
|
<value>會覆蓋埠,多組時用逗號 (,) 隔開</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuGenGroupMultipleServer" xml:space="preserve">
|
<data name="menuGenGroupMultipleServer" xml:space="preserve">
|
||||||
<value>Generate Policy Group from Multiple Profiles</value>
|
<value>多選生成策略組</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuGenGroupMultipleServerXrayRandom" xml:space="preserve">
|
<data name="menuGenGroupMultipleServerXrayRandom" xml:space="preserve">
|
||||||
<value>多設定檔隨機 Xray</value>
|
<value>多選隨機 Xray</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuGenGroupMultipleServerXrayRoundRobin" xml:space="preserve">
|
<data name="menuGenGroupMultipleServerXrayRoundRobin" xml:space="preserve">
|
||||||
<value>多設定檔負載平衡 Xray</value>
|
<value>多選負載平衡 Xray</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuGenGroupMultipleServerXrayLeastPing" xml:space="preserve">
|
<data name="menuGenGroupMultipleServerXrayLeastPing" xml:space="preserve">
|
||||||
<value>多設定檔最低延遲 Xray</value>
|
<value>多選最低延遲 Xray</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuGenGroupMultipleServerXrayLeastLoad" xml:space="preserve">
|
<data name="menuGenGroupMultipleServerXrayLeastLoad" xml:space="preserve">
|
||||||
<value>多設定檔最穩定 Xray</value>
|
<value>多選最穩定 Xray</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuGenGroupMultipleServerSingBoxLeastPing" xml:space="preserve">
|
<data name="menuGenGroupMultipleServerSingBoxLeastPing" xml:space="preserve">
|
||||||
<value>多設定檔最低延遲 sing-box</value>
|
<value>多選最低延遲 sing-box</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuExportConfig" xml:space="preserve">
|
<data name="menuExportConfig" xml:space="preserve">
|
||||||
<value>匯出設定檔</value>
|
<value>匯出</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbSettingsIPAPIUrl" xml:space="preserve">
|
<data name="TbSettingsIPAPIUrl" xml:space="preserve">
|
||||||
<value>目前連接資訊測試地址</value>
|
<value>目前連接資訊測試地址</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbRuleOutboundTagTip" xml:space="preserve">
|
<data name="TbRuleOutboundTagTip" xml:space="preserve">
|
||||||
<value>可以填寫設定檔別名,請確保存在並唯一</value>
|
<value>可以填寫節點別名,請確保存在並唯一</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="SudoIncorrectPasswordTip" xml:space="preserve">
|
<data name="SudoIncorrectPasswordTip" xml:space="preserve">
|
||||||
<value>密碼錯誤,請重試。</value>
|
<value>密碼錯誤,請重試。</value>
|
||||||
@@ -1405,192 +1408,234 @@
|
|||||||
<value>Mldsa65Verify</value>
|
<value>Mldsa65Verify</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuAddAnytlsServer" xml:space="preserve">
|
<data name="menuAddAnytlsServer" xml:space="preserve">
|
||||||
<value>新增 [Anytls] 設定檔</value>
|
<value>新增 [Anytls] 節點</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbRemoteDNS" xml:space="preserve">
|
<data name="TbRemoteDNS" xml:space="preserve">
|
||||||
<value>Remote DNS</value>
|
<value>遠程 DNS</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbDomesticDNS" xml:space="preserve">
|
<data name="TbDomesticDNS" xml:space="preserve">
|
||||||
<value>Domestic DNS</value>
|
<value>直連 DNS</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbRemoteDNSTips" xml:space="preserve">
|
<data name="TbRemoteDNSTips" xml:space="preserve">
|
||||||
<value>Via proxy — please ensure remote availability</value>
|
<value>通过代理,请确保远程可用</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbXrayFreedomStrategy" xml:space="preserve">
|
<data name="TbXrayFreedomStrategy" xml:space="preserve">
|
||||||
<value>xray Freedom Resolution Strategy</value>
|
<value>xray freedom 解析策略</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbSBDirectResolveStrategy" xml:space="preserve">
|
<data name="TbSBDirectResolveStrategy" xml:space="preserve">
|
||||||
<value>sing-box Direct Resolution Strategy</value>
|
<value>sing-box 直連解析策略</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbSBRemoteResolveStrategy" xml:space="preserve">
|
<data name="TbSBRemoteResolveStrategy" xml:space="preserve">
|
||||||
<value>sing-box Remote Resolution Strategy</value>
|
<value>sing-box 遠程解析策略</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbAddCommonDNSHosts" xml:space="preserve">
|
<data name="TbAddCommonDNSHosts" xml:space="preserve">
|
||||||
<value>Add Common DNS Hosts</value>
|
<value>新增常用 DNS Hosts</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbFakeIP" xml:space="preserve">
|
<data name="TbFakeIP" xml:space="preserve">
|
||||||
<value>FakeIP</value>
|
<value>FakeIP</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbBlockSVCBHTTPSQueries" xml:space="preserve">
|
<data name="TbBlockSVCBHTTPSQueries" xml:space="preserve">
|
||||||
<value>Block SVCB and HTTPS Queries</value>
|
<value>阻止 SVCB 和 HTTPS 查詢</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbDNSHostsConfig" xml:space="preserve">
|
<data name="TbDNSHostsConfig" xml:space="preserve">
|
||||||
<value>DNS Hosts: ("domain1 ip1 ip2" per line)</value>
|
<value>DNS Hosts:(“網域名稱1 ip1 ip2” 一行一個)</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="ThBasicDNSSettings" xml:space="preserve">
|
<data name="ThBasicDNSSettings" xml:space="preserve">
|
||||||
<value>Basic DNS Settings</value>
|
<value>DNS 基礎設定</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="ThAdvancedDNSSettings" xml:space="preserve">
|
<data name="ThAdvancedDNSSettings" xml:space="preserve">
|
||||||
<value>Advanced DNS Settings</value>
|
<value>DNS 進階設定</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbValidateDirectExpectedIPs" xml:space="preserve">
|
<data name="TbValidateDirectExpectedIPs" xml:space="preserve">
|
||||||
<value>Validate Regional Domain IPs</value>
|
<value>校驗相應地區域名 IP</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbValidateDirectExpectedIPsDesc" xml:space="preserve">
|
<data name="TbValidateDirectExpectedIPsDesc" xml:space="preserve">
|
||||||
<value>When configured, validates IPs returned for regional domains (e.g., geosite:cn), returning only expected IPs</value>
|
<value>配置後,會對相應地區域名(如 geosite:cn)的返回 IP 進行校驗,僅返回期望 IP</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbCustomDNSEnable" xml:space="preserve">
|
<data name="TbCustomDNSEnable" xml:space="preserve">
|
||||||
<value>Enable Custom DNS</value>
|
<value>啟用自訂 DNS</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbCustomDNSEnabledPageInvalid" xml:space="preserve">
|
<data name="TbCustomDNSEnabledPageInvalid" xml:space="preserve">
|
||||||
<value>Custom DNS Enabled, This Page's Settings Invalid</value>
|
<value>自訂 DNS 已啟用,此頁面配置將無效</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbBlockSVCBHTTPSQueriesTips" xml:space="preserve">
|
<data name="TbBlockSVCBHTTPSQueriesTips" xml:space="preserve">
|
||||||
<value>Block ECH and HTTP/3 availability checks when enabled</value>
|
<value>開啟後將阻止 ECH 和 HTTP/3 可用性查詢</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="FillCorrectConfigTemplateText" xml:space="preserve">
|
<data name="FillCorrectConfigTemplateText" xml:space="preserve">
|
||||||
<value>Please fill in the correct config template</value>
|
<value>請填寫正確的配置範本</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuFullConfigTemplate" xml:space="preserve">
|
<data name="menuFullConfigTemplate" xml:space="preserve">
|
||||||
<value>Full Config Template Setting</value>
|
<value>完整配置範本設定</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbFullConfigTemplateEnable" xml:space="preserve">
|
<data name="TbFullConfigTemplateEnable" xml:space="preserve">
|
||||||
<value>Enable Full Config Template</value>
|
<value>啟用完整配置範本</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbRayFullConfigTemplate" xml:space="preserve">
|
<data name="TbRayFullConfigTemplate" xml:space="preserve">
|
||||||
<value>v2ray Full Config Template</value>
|
<value>v2ray 完整配置範本</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbRayFullConfigTemplateDesc" xml:space="preserve">
|
<data name="TbRayFullConfigTemplateDesc" xml:space="preserve">
|
||||||
<value>Add Outbound Config Only, routing.balancers and routing.rules.outboundTag, Click to view the document</value>
|
<value>僅添加出站配置,routing.balancers 和 routing.rules.outboundTag,點擊查看文檔</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbAddProxyProtocolOutboundOnly" xml:space="preserve">
|
<data name="TbAddProxyProtocolOutboundOnly" xml:space="preserve">
|
||||||
<value>Do Not Add Non-Proxy Protocol Outbound</value>
|
<value>不添加非代理協定出站</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbSetUpstreamProxyDetour" xml:space="preserve">
|
<data name="TbSetUpstreamProxyDetour" xml:space="preserve">
|
||||||
<value>Set Upstream Proxy Tag</value>
|
<value>設定上游代理 tag</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbSBFullConfigTemplate" xml:space="preserve">
|
<data name="TbSBFullConfigTemplate" xml:space="preserve">
|
||||||
<value>sing-box Full Config Template</value>
|
<value>sing-box 完整配置範本</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbSBFullConfigTemplateDesc" xml:space="preserve">
|
<data name="TbSBFullConfigTemplateDesc" xml:space="preserve">
|
||||||
<value>Add Outbound and Endpoint Config Only, Click to view the document</value>
|
<value>僅添加出站和端點配置,點擊查看文檔</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbFullConfigTemplateDesc" xml:space="preserve">
|
<data name="TbFullConfigTemplateDesc" xml:space="preserve">
|
||||||
<value>This feature is intended for advanced users and those with special requirements. Once enabled, it will ignore the Core's basic settings, DNS settings, and routing settings. You must ensure that the system proxy port, traffic statistics, and other related configurations are set correctly — everything will be configured by you.</value>
|
<value>此功能供高級用戶和有特殊需求的用戶使用。 啟用此功能後,將忽略 Core 的基礎設定,DNS 設定 ,路由設定。你需要保證系統代理的埠和流量統計等功能的配置正確,一切都由你來設定。</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="MsgStartParsingSubscription" xml:space="preserve">
|
<data name="MsgStartParsingSubscription" xml:space="preserve">
|
||||||
<value>開始解析和處理訂閱內容</value>
|
<value>開始解析和處理訂閱內容</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbSelectProfile" xml:space="preserve">
|
<data name="TbSelectProfile" xml:space="preserve">
|
||||||
<value>Select Profile</value>
|
<value>選擇節點</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbFakeIPTips" xml:space="preserve">
|
<data name="TbFakeIPTips" xml:space="preserve">
|
||||||
<value>Applies globally by default, with built-in FakeIP filtering (sing-box only).</value>
|
<value>默認全域生效,內置 FakeIP 過濾,僅在 sing-box 中生效</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="PleaseAddAtLeastOneServer" xml:space="preserve">
|
<data name="PleaseAddAtLeastOneServer" xml:space="preserve">
|
||||||
<value>Please Add At Least One Configuration</value>
|
<value>請至少添加一個節點</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbConfigTypePolicyGroup" xml:space="preserve">
|
<data name="TbConfigTypePolicyGroup" xml:space="preserve">
|
||||||
<value>Policy Group</value>
|
<value>策略組</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbConfigTypeProxyChain" xml:space="preserve">
|
<data name="TbConfigTypeProxyChain" xml:space="preserve">
|
||||||
<value>Proxy Chain</value>
|
<value>鏈式代理</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbLeastPing" xml:space="preserve">
|
<data name="TbLeastPing" xml:space="preserve">
|
||||||
<value>Lowest Latency</value>
|
<value>最低延遲</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbRandom" xml:space="preserve">
|
<data name="TbRandom" xml:space="preserve">
|
||||||
<value>Random</value>
|
<value>隨機</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbRoundRobin" xml:space="preserve">
|
<data name="TbRoundRobin" xml:space="preserve">
|
||||||
<value>Round Robin</value>
|
<value>負載均衡</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbLeastLoad" xml:space="preserve">
|
<data name="TbLeastLoad" xml:space="preserve">
|
||||||
<value>Most Stable</value>
|
<value>最穩定</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbPolicyGroupType" xml:space="preserve">
|
<data name="TbPolicyGroupType" xml:space="preserve">
|
||||||
<value>Policy Group Type</value>
|
<value>策略組類型</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuAddPolicyGroupServer" xml:space="preserve">
|
<data name="menuAddPolicyGroupServer" xml:space="preserve">
|
||||||
<value>Add Policy Group Configuration</value>
|
<value>添加策略組</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuAddProxyChainServer" xml:space="preserve">
|
<data name="menuAddProxyChainServer" xml:space="preserve">
|
||||||
<value>Add Proxy Chain Configuration</value>
|
<value>添加鏈式代理</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuAddChildServer" xml:space="preserve">
|
<data name="menuAddChildServer" xml:space="preserve">
|
||||||
<value>Add Child Configuration</value>
|
<value>添加子配置</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuRemoveChildServer" xml:space="preserve">
|
<data name="menuRemoveChildServer" xml:space="preserve">
|
||||||
<value>Remove Child Configuration</value>
|
<value>刪除子配置</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuServerList" xml:space="preserve">
|
<data name="menuServerList" xml:space="preserve">
|
||||||
<value>Server List</value>
|
<value>子配置項目一,從訂閱分組中自動新增</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbFallback" xml:space="preserve">
|
<data name="TbFallback" xml:space="preserve">
|
||||||
<value>Fallback</value>
|
<value>容錯移轉</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuGenGroupMultipleServerSingBoxFallback" xml:space="preserve">
|
<data name="menuGenGroupMultipleServerSingBoxFallback" xml:space="preserve">
|
||||||
<value>Multi-Configuration Fallback by sing-box</value>
|
<value>多選容錯移轉 sing-box</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="menuGenGroupMultipleServerXrayFallback" xml:space="preserve">
|
<data name="menuGenGroupMultipleServerXrayFallback" xml:space="preserve">
|
||||||
<value>Multi-Configuration Fallback by Xray</value>
|
<value>多選容錯移轉 Xray</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="CoreNotSupportNetwork" xml:space="preserve">
|
<data name="CoreNotSupportNetwork" xml:space="preserve">
|
||||||
<value>Core '{0}' does not support network type '{1}'.</value>
|
<value>核心 '{0}' 不支援網路類型 '{1}'.</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="CoreNotSupportProtocolTransport" xml:space="preserve">
|
<data name="CoreNotSupportProtocolTransport" xml:space="preserve">
|
||||||
<value>Core '{0}' does not support protocol '{1}' when using transport '{2}'.</value>
|
<value>核心 '{0}' 在使用傳輸方式 '{2}' 時不支援協定 '{1}'.</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="CoreNotSupportProtocol" xml:space="preserve">
|
<data name="CoreNotSupportProtocol" xml:space="preserve">
|
||||||
<value>Core '{0}' does not support protocol '{1}'.</value>
|
<value>核心 '{0}' 不支援協定 '{1}'.</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="ProxyChainedPrefix" xml:space="preserve">
|
<data name="ProxyChainedPrefix" xml:space="preserve">
|
||||||
<value>Proxy chained: </value>
|
<value>代理鏈: </value>
|
||||||
</data>
|
</data>
|
||||||
<data name="RoutingRuleOutboundPrefix" xml:space="preserve">
|
<data name="RoutingRuleOutboundPrefix" xml:space="preserve">
|
||||||
<value>Routing rule outbound: </value>
|
<value>路由規則出站: </value>
|
||||||
</data>
|
</data>
|
||||||
<data name="PolicyGroupPrefix" xml:space="preserve">
|
<data name="PolicyGroupPrefix" xml:space="preserve">
|
||||||
<value>Policy group: </value>
|
<value>策略組: </value>
|
||||||
</data>
|
</data>
|
||||||
<data name="NodeTagNotExist" xml:space="preserve">
|
<data name="NodeTagNotExist" xml:space="preserve">
|
||||||
<value>Node alias '{0}' does not exist.</value>
|
<value>別名 '{0}' 不存在。</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="GroupEmpty" xml:space="preserve">
|
<data name="GroupEmpty" xml:space="preserve">
|
||||||
<value>Group '{0}' is empty. Please add at least one node.</value>
|
<value>組“{0}”為空.請至少添加一個配置。</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="InvalidProperty" xml:space="preserve">
|
<data name="InvalidProperty" xml:space="preserve">
|
||||||
<value>The {0} property is invalid, please check.</value>
|
<value>{0}屬性無效,請檢查</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="GroupSelfReference" xml:space="preserve">
|
<data name="GroupSelfReference" xml:space="preserve">
|
||||||
<value>{0} 分組不能引用自身或循環引用</value>
|
<value>{0} 分組不能引用自身或循環引用</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="NotSupportProtocol" xml:space="preserve">
|
<data name="NotSupportProtocol" xml:space="preserve">
|
||||||
<value>Not support protocol '{0}'.</value>
|
<value>不支援協定 '{0}'.</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbSettingsHide2TrayWhenCloseTip" xml:space="preserve">
|
<data name="TbSettingsHide2TrayWhenCloseTip" xml:space="preserve">
|
||||||
<value>如果系統沒有托盤功能,請不要開啟</value>
|
<value>如果系統沒有託盤功能,請不要開啟</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbRuleType" xml:space="preserve">
|
<data name="TbRuleType" xml:space="preserve">
|
||||||
<value>规则类型</value>
|
<value>規則類型</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbRuleTypeTips" xml:space="preserve">
|
<data name="TbRuleTypeTips" xml:space="preserve">
|
||||||
<value>可对 Routing 和 DNS 单独设定规则,ALL 则都生效</value>
|
<value>可對 Routing 和 DNS 單獨設定規則,ALL 則都生效</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbBootstrapDNS" xml:space="preserve">
|
<data name="TbBootstrapDNS" xml:space="preserve">
|
||||||
<value>Bootstrap DNS</value>
|
<value>Bootstrap DNS</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbBootstrapDNSTips" xml:space="preserve">
|
<data name="TbBootstrapDNSTips" xml:space="preserve">
|
||||||
<value>Resolve DNS server domains, requires IP</value>
|
<value>解析 DNS 伺服器網域名稱,需指定為 IP</value>
|
||||||
|
</data>
|
||||||
|
<data name="menuFastRealPing" xml:space="preserve">
|
||||||
|
<value>一鍵測試真連線延遲</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbPolicyGroupSubChildTip" xml:space="preserve">
|
||||||
|
<value>自動從訂閱分組新增過濾後的配置</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbCertPinning" xml:space="preserve">
|
||||||
|
<value>憑證綁定</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbCertPinningTips" xml:space="preserve">
|
||||||
|
<value>伺服器憑證(PEM 格式,可選)
|
||||||
|
若已指定,憑證將會被綁定,並且「跳過憑證驗證」將被停用。
|
||||||
|
|
||||||
|
若使用自簽憑證,或系統中存在不受信任或惡意的 CA,「取得憑證」動作可能會失敗。</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbFetchCert" xml:space="preserve">
|
||||||
|
<value>獲取憑證</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbFetchCertChain" xml:space="preserve">
|
||||||
|
<value>獲取憑證鏈</value>
|
||||||
|
</data>
|
||||||
|
<data name="ServerNameMustBeValidDomain" xml:space="preserve">
|
||||||
|
<value>請設定有效的網域名稱</value>
|
||||||
|
</data>
|
||||||
|
<data name="CertNotSet" xml:space="preserve">
|
||||||
|
<value>尚未設定憑證</value>
|
||||||
|
</data>
|
||||||
|
<data name="CertSet" xml:space="preserve">
|
||||||
|
<value>已設定憑證</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbSettingsCustomSystemProxyPacPath" xml:space="preserve">
|
||||||
|
<value>自訂 PAC 檔案路徑</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbSettingsCustomSystemProxyScriptPath" xml:space="preserve">
|
||||||
|
<value>自訂系統代理程式腳本檔案路徑</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbSettingsMacOSShowInDock" xml:space="preserve">
|
||||||
|
<value>macOS 在 Dock 欄顯示 (需重啟)</value>
|
||||||
|
</data>
|
||||||
|
<data name="menuServerList2" xml:space="preserve">
|
||||||
|
<value>子配置項二,從自建中選擇新增</value>
|
||||||
</data>
|
</data>
|
||||||
</root>
|
</root>
|
||||||
@@ -13,20 +13,19 @@
|
|||||||
"api.ip.sb"
|
"api.ip.sb"
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
{
|
|
||||||
"remarks": "Google cn",
|
|
||||||
"outboundTag": "proxy",
|
|
||||||
"domain": [
|
|
||||||
"domain:googleapis.cn",
|
|
||||||
"domain:gstatic.com"
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
"remarks": "阻断udp443",
|
"remarks": "阻断udp443",
|
||||||
"outboundTag": "block",
|
"outboundTag": "block",
|
||||||
"port": "443",
|
"port": "443",
|
||||||
"network": "udp"
|
"network": "udp"
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
"remarks": "代理Google",
|
||||||
|
"outboundTag": "proxy",
|
||||||
|
"domain": [
|
||||||
|
"geosite:google"
|
||||||
|
]
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"remarks": "绕过局域网IP",
|
"remarks": "绕过局域网IP",
|
||||||
"outboundTag": "direct",
|
"outboundTag": "direct",
|
||||||
|
|||||||
@@ -1,18 +1,17 @@
|
|||||||
[
|
[
|
||||||
{
|
|
||||||
"remarks": "Google cn",
|
|
||||||
"outboundTag": "proxy",
|
|
||||||
"domain": [
|
|
||||||
"domain:googleapis.cn",
|
|
||||||
"domain:gstatic.com"
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
"remarks": "阻断udp443",
|
"remarks": "阻断udp443",
|
||||||
"outboundTag": "block",
|
"outboundTag": "block",
|
||||||
"port": "443",
|
"port": "443",
|
||||||
"network": "udp"
|
"network": "udp"
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
"remarks": "代理Google",
|
||||||
|
"outboundTag": "proxy",
|
||||||
|
"domain": [
|
||||||
|
"geosite:google"
|
||||||
|
]
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"remarks": "绕过局域网IP",
|
"remarks": "绕过局域网IP",
|
||||||
"outboundTag": "direct",
|
"outboundTag": "direct",
|
||||||
|
|||||||
@@ -14,9 +14,8 @@
|
|||||||
],
|
],
|
||||||
"rules": [
|
"rules": [
|
||||||
{
|
{
|
||||||
"domain_suffix": [
|
"rule_set": [
|
||||||
"googleapis.cn",
|
"geosite-google"
|
||||||
"gstatic.com"
|
|
||||||
],
|
],
|
||||||
"server": "remote",
|
"server": "remote",
|
||||||
"strategy": "prefer_ipv4"
|
"strategy": "prefer_ipv4"
|
||||||
|
|||||||
@@ -8,8 +8,7 @@
|
|||||||
"address": "1.1.1.1",
|
"address": "1.1.1.1",
|
||||||
"skipFallback": true,
|
"skipFallback": true,
|
||||||
"domains": [
|
"domains": [
|
||||||
"domain:googleapis.cn",
|
"geosite:google"
|
||||||
"domain:gstatic.com"
|
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -28,15 +28,15 @@ fi
|
|||||||
kill_children() {
|
kill_children() {
|
||||||
local parent=$1
|
local parent=$1
|
||||||
local children=$(ps -o pid --no-headers --ppid "$parent")
|
local children=$(ps -o pid --no-headers --ppid "$parent")
|
||||||
|
|
||||||
# Output information about processes being terminated
|
# Output information about processes being terminated
|
||||||
echo "Processing children of PID: $parent..."
|
echo "Processing children of PID: $parent..."
|
||||||
|
|
||||||
# Process each child
|
# Process each child
|
||||||
for child in $children; do
|
for child in $children; do
|
||||||
# Recursively find and kill child's children first
|
# Recursively find and kill child's children first
|
||||||
kill_children "$child"
|
kill_children "$child"
|
||||||
|
|
||||||
# Force kill the child process
|
# Force kill the child process
|
||||||
echo "Terminating child process: $child"
|
echo "Terminating child process: $child"
|
||||||
kill -9 "$child" 2>/dev/null || true
|
kill -9 "$child" 2>/dev/null || true
|
||||||
@@ -47,6 +47,18 @@ echo "============================================"
|
|||||||
echo "Starting termination of process $PID and all its children"
|
echo "Starting termination of process $PID and all its children"
|
||||||
echo "============================================"
|
echo "============================================"
|
||||||
|
|
||||||
|
# Try graceful termination first
|
||||||
|
echo "Attempting graceful termination (SIGTERM) of PID: $PID"
|
||||||
|
kill -15 "$PID" 2>/dev/null || true
|
||||||
|
sleep 1
|
||||||
|
# If still running, fall back to kill_children
|
||||||
|
if ps -p $PID > /dev/null; then
|
||||||
|
echo "Process $PID did not exit after SIGTERM; proceeding with forced termination of its children and itself"
|
||||||
|
else
|
||||||
|
echo "Process $PID exited cleanly after SIGTERM"
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
# Find and kill all child processes
|
# Find and kill all child processes
|
||||||
kill_children "$PID"
|
kill_children "$PID"
|
||||||
|
|
||||||
|
|||||||
@@ -42,6 +42,20 @@ echo "============================================"
|
|||||||
echo "Starting termination of process $PID and all its descendants"
|
echo "Starting termination of process $PID and all its descendants"
|
||||||
echo "============================================"
|
echo "============================================"
|
||||||
|
|
||||||
|
# Try graceful termination first
|
||||||
|
echo "Attempting graceful termination (SIGTERM) of PID: $PID"
|
||||||
|
kill -15 "$PID" 2>/dev/null || true
|
||||||
|
sleep 1
|
||||||
|
|
||||||
|
# If still running, fall back to kill_descendants
|
||||||
|
# Use the macOS-native 'kill -0' check
|
||||||
|
if kill -0 $PID 2>/dev/null; then
|
||||||
|
echo "Process $PID did not exit after SIGTERM; proceeding with forced termination of its descendants and itself"
|
||||||
|
else
|
||||||
|
echo "Process $PID exited cleanly after SIGTERM"
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
# Find and kill all descendant processes
|
# Find and kill all descendant processes
|
||||||
kill_descendants "$PID"
|
kill_descendants "$PID"
|
||||||
|
|
||||||
|
|||||||
@@ -14,9 +14,8 @@
|
|||||||
],
|
],
|
||||||
"rules": [
|
"rules": [
|
||||||
{
|
{
|
||||||
"domain_suffix": [
|
"rule_set": [
|
||||||
"googleapis.cn",
|
"geosite-google"
|
||||||
"gstatic.com"
|
|
||||||
],
|
],
|
||||||
"server": "remote",
|
"server": "remote",
|
||||||
"strategy": "prefer_ipv4"
|
"strategy": "prefer_ipv4"
|
||||||
|
|||||||
@@ -57,6 +57,9 @@
|
|||||||
<SubType>Designer</SubType>
|
<SubType>Designer</SubType>
|
||||||
<Generator>PublicResXFileCodeGenerator</Generator>
|
<Generator>PublicResXFileCodeGenerator</Generator>
|
||||||
</EmbeddedResource>
|
</EmbeddedResource>
|
||||||
|
<EmbeddedResource Update="Resx\ResUI.fr.resx">
|
||||||
|
<Generator>PublicResXFileCodeGenerator</Generator>
|
||||||
|
</EmbeddedResource>
|
||||||
<EmbeddedResource Update="Resx\ResUI.hu.resx">
|
<EmbeddedResource Update="Resx\ResUI.hu.resx">
|
||||||
<Generator>PublicResXFileCodeGenerator</Generator>
|
<Generator>PublicResXFileCodeGenerator</Generator>
|
||||||
</EmbeddedResource>
|
</EmbeddedResource>
|
||||||
@@ -79,4 +82,4 @@
|
|||||||
</EmbeddedResource>
|
</EmbeddedResource>
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
</Project>
|
</Project>
|
||||||
|
|||||||
@@ -202,7 +202,9 @@ public partial class CoreConfigSingboxService
|
|||||||
|
|
||||||
var routing = await ConfigHandler.GetDefaultRouting(_config);
|
var routing = await ConfigHandler.GetDefaultRouting(_config);
|
||||||
if (routing == null)
|
if (routing == null)
|
||||||
|
{
|
||||||
return 0;
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
var rules = JsonUtils.Deserialize<List<RulesItem>>(routing.RuleSet) ?? [];
|
var rules = JsonUtils.Deserialize<List<RulesItem>>(routing.RuleSet) ?? [];
|
||||||
var expectedIPCidr = new List<string>();
|
var expectedIPCidr = new List<string>();
|
||||||
@@ -470,14 +472,16 @@ public partial class CoreConfigSingboxService
|
|||||||
{
|
{
|
||||||
// udp dns
|
// udp dns
|
||||||
server.type = "udp";
|
server.type = "udp";
|
||||||
server.server = addressFirst;
|
}
|
||||||
return server;
|
else
|
||||||
|
{
|
||||||
|
// server.type = scheme.ToLower();
|
||||||
|
|
||||||
|
// remove "+local" suffix
|
||||||
|
// TODO: "+local" suffix decide server.detour = "direct" ?
|
||||||
|
server.type = scheme.Replace("+local", "", StringComparison.OrdinalIgnoreCase).ToLower();
|
||||||
}
|
}
|
||||||
|
|
||||||
//server.type = scheme.ToLower();
|
|
||||||
// remove "+local" suffix
|
|
||||||
// TODO: "+local" suffix decide server.detour = "direct" ?
|
|
||||||
server.type = scheme.Replace("+local", "", StringComparison.OrdinalIgnoreCase).ToLower();
|
|
||||||
server.server = domain;
|
server.server = domain;
|
||||||
if (port != 0)
|
if (port != 0)
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -61,7 +61,7 @@ public partial class CoreConfigSingboxService
|
|||||||
}
|
}
|
||||||
|
|
||||||
var tunInbound = JsonUtils.Deserialize<Inbound4Sbox>(EmbedUtils.GetEmbedText(Global.TunSingboxInboundFileName)) ?? new Inbound4Sbox { };
|
var tunInbound = JsonUtils.Deserialize<Inbound4Sbox>(EmbedUtils.GetEmbedText(Global.TunSingboxInboundFileName)) ?? new Inbound4Sbox { };
|
||||||
tunInbound.interface_name = Utils.IsOSX() ? $"utun{new Random().Next(99)}" : "singbox_tun";
|
tunInbound.interface_name = Utils.IsMacOS() ? $"utun{new Random().Next(99)}" : "singbox_tun";
|
||||||
tunInbound.mtu = _config.TunModeItem.Mtu;
|
tunInbound.mtu = _config.TunModeItem.Mtu;
|
||||||
tunInbound.auto_route = _config.TunModeItem.AutoRoute;
|
tunInbound.auto_route = _config.TunModeItem.AutoRoute;
|
||||||
tunInbound.strict_route = _config.TunModeItem.StrictRoute;
|
tunInbound.strict_route = _config.TunModeItem.StrictRoute;
|
||||||
|
|||||||
@@ -26,6 +26,7 @@ public partial class CoreConfigSingboxService
|
|||||||
}
|
}
|
||||||
|
|
||||||
await GenOutboundMux(node, outbound);
|
await GenOutboundMux(node, outbound);
|
||||||
|
await GenOutboundTransport(node, outbound);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case EConfigType.Shadowsocks:
|
case EConfigType.Shadowsocks:
|
||||||
@@ -33,6 +34,50 @@ public partial class CoreConfigSingboxService
|
|||||||
outbound.method = AppManager.Instance.GetShadowsocksSecurities(node).Contains(node.Security) ? node.Security : Global.None;
|
outbound.method = AppManager.Instance.GetShadowsocksSecurities(node).Contains(node.Security) ? node.Security : Global.None;
|
||||||
outbound.password = node.Id;
|
outbound.password = node.Id;
|
||||||
|
|
||||||
|
if (node.Network == nameof(ETransport.tcp) && node.HeaderType == Global.TcpHeaderHttp)
|
||||||
|
{
|
||||||
|
outbound.plugin = "obfs-local";
|
||||||
|
outbound.plugin_opts = $"obfs=http;obfs-host={node.RequestHost};";
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
var pluginArgs = string.Empty;
|
||||||
|
if (node.Network == nameof(ETransport.ws))
|
||||||
|
{
|
||||||
|
pluginArgs += "mode=websocket;";
|
||||||
|
pluginArgs += $"host={node.RequestHost};";
|
||||||
|
pluginArgs += $"path={node.Path};";
|
||||||
|
}
|
||||||
|
else if (node.Network == nameof(ETransport.quic))
|
||||||
|
{
|
||||||
|
pluginArgs += "mode=quic;";
|
||||||
|
}
|
||||||
|
if (node.StreamSecurity == Global.StreamSecurity)
|
||||||
|
{
|
||||||
|
pluginArgs += "tls;";
|
||||||
|
var certs = CertPemManager.ParsePemChain(node.Cert);
|
||||||
|
if (certs.Count > 0)
|
||||||
|
{
|
||||||
|
var cert = certs.First();
|
||||||
|
const string beginMarker = "-----BEGIN CERTIFICATE-----\n";
|
||||||
|
const string endMarker = "\n-----END CERTIFICATE-----";
|
||||||
|
|
||||||
|
var base64Content = cert.Replace(beginMarker, "").Replace(endMarker, "").Trim();
|
||||||
|
|
||||||
|
// https://github.com/shadowsocks/v2ray-plugin/blob/e9af1cdd2549d528deb20a4ab8d61c5fbe51f306/args.go#L172
|
||||||
|
// Equal signs and commas [and backslashes] must be escaped with a backslash.
|
||||||
|
base64Content = base64Content.Replace("=", "\\=");
|
||||||
|
|
||||||
|
pluginArgs += $"certRaw={base64Content};";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (pluginArgs.Length > 0)
|
||||||
|
{
|
||||||
|
outbound.plugin = "v2ray-plugin";
|
||||||
|
outbound.plugin_opts = pluginArgs;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
await GenOutboundMux(node, outbound);
|
await GenOutboundMux(node, outbound);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@@ -71,6 +116,8 @@ public partial class CoreConfigSingboxService
|
|||||||
{
|
{
|
||||||
outbound.flow = node.Flow;
|
outbound.flow = node.Flow;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
await GenOutboundTransport(node, outbound);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case EConfigType.Trojan:
|
case EConfigType.Trojan:
|
||||||
@@ -78,6 +125,7 @@ public partial class CoreConfigSingboxService
|
|||||||
outbound.password = node.Id;
|
outbound.password = node.Id;
|
||||||
|
|
||||||
await GenOutboundMux(node, outbound);
|
await GenOutboundMux(node, outbound);
|
||||||
|
await GenOutboundTransport(node, outbound);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case EConfigType.Hysteria2:
|
case EConfigType.Hysteria2:
|
||||||
@@ -127,8 +175,6 @@ public partial class CoreConfigSingboxService
|
|||||||
}
|
}
|
||||||
|
|
||||||
await GenOutboundTls(node, outbound);
|
await GenOutboundTls(node, outbound);
|
||||||
|
|
||||||
await GenOutboundTransport(node, outbound);
|
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
@@ -204,6 +250,192 @@ public partial class CoreConfigSingboxService
|
|||||||
return await Task.FromResult<BaseServer4Sbox?>(null);
|
return await Task.FromResult<BaseServer4Sbox?>(null);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private async Task<int> GenOutboundMux(ProfileItem node, Outbound4Sbox outbound)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var muxEnabled = node.MuxEnabled ?? _config.CoreBasicItem.MuxEnabled;
|
||||||
|
if (muxEnabled && _config.Mux4SboxItem.Protocol.IsNotEmpty())
|
||||||
|
{
|
||||||
|
var mux = new Multiplex4Sbox()
|
||||||
|
{
|
||||||
|
enabled = true,
|
||||||
|
protocol = _config.Mux4SboxItem.Protocol,
|
||||||
|
max_connections = _config.Mux4SboxItem.MaxConnections,
|
||||||
|
padding = _config.Mux4SboxItem.Padding,
|
||||||
|
};
|
||||||
|
outbound.multiplex = mux;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
Logging.SaveLog(_tag, ex);
|
||||||
|
}
|
||||||
|
return await Task.FromResult(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task<int> GenOutboundTls(ProfileItem node, Outbound4Sbox outbound)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
if (node.StreamSecurity is not (Global.StreamSecurityReality or Global.StreamSecurity))
|
||||||
|
{
|
||||||
|
return await Task.FromResult(0);
|
||||||
|
}
|
||||||
|
if (node.ConfigType is EConfigType.Shadowsocks or EConfigType.SOCKS or EConfigType.WireGuard)
|
||||||
|
{
|
||||||
|
return await Task.FromResult(0);
|
||||||
|
}
|
||||||
|
var server_name = string.Empty;
|
||||||
|
if (node.Sni.IsNotEmpty())
|
||||||
|
{
|
||||||
|
server_name = node.Sni;
|
||||||
|
}
|
||||||
|
else if (node.RequestHost.IsNotEmpty())
|
||||||
|
{
|
||||||
|
server_name = Utils.String2List(node.RequestHost)?.First();
|
||||||
|
}
|
||||||
|
var tls = new Tls4Sbox()
|
||||||
|
{
|
||||||
|
enabled = true,
|
||||||
|
record_fragment = _config.CoreBasicItem.EnableFragment ? true : null,
|
||||||
|
server_name = server_name,
|
||||||
|
insecure = Utils.ToBool(node.AllowInsecure.IsNullOrEmpty() ? _config.CoreBasicItem.DefAllowInsecure.ToString().ToLower() : node.AllowInsecure),
|
||||||
|
alpn = node.GetAlpn(),
|
||||||
|
};
|
||||||
|
if (node.Fingerprint.IsNotEmpty())
|
||||||
|
{
|
||||||
|
tls.utls = new Utls4Sbox()
|
||||||
|
{
|
||||||
|
enabled = true,
|
||||||
|
fingerprint = node.Fingerprint.IsNullOrEmpty() ? _config.CoreBasicItem.DefFingerprint : node.Fingerprint
|
||||||
|
};
|
||||||
|
}
|
||||||
|
if (node.StreamSecurity == Global.StreamSecurity)
|
||||||
|
{
|
||||||
|
var certs = CertPemManager.ParsePemChain(node.Cert);
|
||||||
|
if (certs.Count > 0)
|
||||||
|
{
|
||||||
|
tls.certificate = certs;
|
||||||
|
tls.insecure = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (node.StreamSecurity == Global.StreamSecurityReality)
|
||||||
|
{
|
||||||
|
tls.reality = new Reality4Sbox()
|
||||||
|
{
|
||||||
|
enabled = true,
|
||||||
|
public_key = node.PublicKey,
|
||||||
|
short_id = node.ShortId
|
||||||
|
};
|
||||||
|
tls.insecure = false;
|
||||||
|
}
|
||||||
|
outbound.tls = tls;
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
Logging.SaveLog(_tag, ex);
|
||||||
|
}
|
||||||
|
return await Task.FromResult(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task<int> GenOutboundTransport(ProfileItem node, Outbound4Sbox outbound)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var transport = new Transport4Sbox();
|
||||||
|
|
||||||
|
switch (node.GetNetwork())
|
||||||
|
{
|
||||||
|
case nameof(ETransport.h2):
|
||||||
|
transport.type = nameof(ETransport.http);
|
||||||
|
transport.host = node.RequestHost.IsNullOrEmpty() ? null : Utils.String2List(node.RequestHost);
|
||||||
|
transport.path = node.Path.IsNullOrEmpty() ? null : node.Path;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case nameof(ETransport.tcp): //http
|
||||||
|
if (node.HeaderType == Global.TcpHeaderHttp)
|
||||||
|
{
|
||||||
|
transport.type = nameof(ETransport.http);
|
||||||
|
transport.host = node.RequestHost.IsNullOrEmpty() ? null : Utils.String2List(node.RequestHost);
|
||||||
|
transport.path = node.Path.IsNullOrEmpty() ? null : node.Path;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case nameof(ETransport.ws):
|
||||||
|
transport.type = nameof(ETransport.ws);
|
||||||
|
var wsPath = node.Path;
|
||||||
|
|
||||||
|
// Parse eh and ed parameters from path using regex
|
||||||
|
if (!wsPath.IsNullOrEmpty())
|
||||||
|
{
|
||||||
|
var edRegex = new Regex(@"[?&]ed=(\d+)");
|
||||||
|
var edMatch = edRegex.Match(wsPath);
|
||||||
|
if (edMatch.Success && int.TryParse(edMatch.Groups[1].Value, out var edValue))
|
||||||
|
{
|
||||||
|
transport.max_early_data = edValue;
|
||||||
|
transport.early_data_header_name = "Sec-WebSocket-Protocol";
|
||||||
|
|
||||||
|
wsPath = edRegex.Replace(wsPath, "");
|
||||||
|
wsPath = wsPath.Replace("?&", "?");
|
||||||
|
if (wsPath.EndsWith('?'))
|
||||||
|
{
|
||||||
|
wsPath = wsPath.TrimEnd('?');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var ehRegex = new Regex(@"[?&]eh=([^&]+)");
|
||||||
|
var ehMatch = ehRegex.Match(wsPath);
|
||||||
|
if (ehMatch.Success)
|
||||||
|
{
|
||||||
|
transport.early_data_header_name = Uri.UnescapeDataString(ehMatch.Groups[1].Value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
transport.path = wsPath.IsNullOrEmpty() ? null : wsPath;
|
||||||
|
if (node.RequestHost.IsNotEmpty())
|
||||||
|
{
|
||||||
|
transport.headers = new()
|
||||||
|
{
|
||||||
|
Host = node.RequestHost
|
||||||
|
};
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case nameof(ETransport.httpupgrade):
|
||||||
|
transport.type = nameof(ETransport.httpupgrade);
|
||||||
|
transport.path = node.Path.IsNullOrEmpty() ? null : node.Path;
|
||||||
|
transport.host = node.RequestHost.IsNullOrEmpty() ? null : node.RequestHost;
|
||||||
|
|
||||||
|
break;
|
||||||
|
|
||||||
|
case nameof(ETransport.quic):
|
||||||
|
transport.type = nameof(ETransport.quic);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case nameof(ETransport.grpc):
|
||||||
|
transport.type = nameof(ETransport.grpc);
|
||||||
|
transport.service_name = node.Path;
|
||||||
|
transport.idle_timeout = _config.GrpcItem.IdleTimeout?.ToString("##s");
|
||||||
|
transport.ping_timeout = _config.GrpcItem.HealthCheckTimeout?.ToString("##s");
|
||||||
|
transport.permit_without_stream = _config.GrpcItem.PermitWithoutStream;
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (transport.type != null)
|
||||||
|
{
|
||||||
|
outbound.transport = transport;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
Logging.SaveLog(_tag, ex);
|
||||||
|
}
|
||||||
|
return await Task.FromResult(0);
|
||||||
|
}
|
||||||
|
|
||||||
private async Task<int> GenGroupOutbound(ProfileItem node, SingboxConfig singboxConfig, string baseTagName = Global.ProxyTag, bool ignoreOriginChain = false)
|
private async Task<int> GenGroupOutbound(ProfileItem node, SingboxConfig singboxConfig, string baseTagName = Global.ProxyTag, bool ignoreOriginChain = false)
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
@@ -252,158 +484,6 @@ public partial class CoreConfigSingboxService
|
|||||||
return await Task.FromResult(0);
|
return await Task.FromResult(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
private async Task<int> GenOutboundMux(ProfileItem node, Outbound4Sbox outbound)
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
var muxEnabled = node.MuxEnabled ?? _config.CoreBasicItem.MuxEnabled;
|
|
||||||
if (muxEnabled && _config.Mux4SboxItem.Protocol.IsNotEmpty())
|
|
||||||
{
|
|
||||||
var mux = new Multiplex4Sbox()
|
|
||||||
{
|
|
||||||
enabled = true,
|
|
||||||
protocol = _config.Mux4SboxItem.Protocol,
|
|
||||||
max_connections = _config.Mux4SboxItem.MaxConnections,
|
|
||||||
padding = _config.Mux4SboxItem.Padding,
|
|
||||||
};
|
|
||||||
outbound.multiplex = mux;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
Logging.SaveLog(_tag, ex);
|
|
||||||
}
|
|
||||||
return await Task.FromResult(0);
|
|
||||||
}
|
|
||||||
|
|
||||||
private async Task<int> GenOutboundTls(ProfileItem node, Outbound4Sbox outbound)
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
if (node.StreamSecurity == Global.StreamSecurityReality || node.StreamSecurity == Global.StreamSecurity)
|
|
||||||
{
|
|
||||||
var server_name = string.Empty;
|
|
||||||
if (node.Sni.IsNotEmpty())
|
|
||||||
{
|
|
||||||
server_name = node.Sni;
|
|
||||||
}
|
|
||||||
else if (node.RequestHost.IsNotEmpty())
|
|
||||||
{
|
|
||||||
server_name = Utils.String2List(node.RequestHost)?.First();
|
|
||||||
}
|
|
||||||
var tls = new Tls4Sbox()
|
|
||||||
{
|
|
||||||
enabled = true,
|
|
||||||
record_fragment = _config.CoreBasicItem.EnableFragment,
|
|
||||||
server_name = server_name,
|
|
||||||
insecure = Utils.ToBool(node.AllowInsecure.IsNullOrEmpty() ? _config.CoreBasicItem.DefAllowInsecure.ToString().ToLower() : node.AllowInsecure),
|
|
||||||
alpn = node.GetAlpn(),
|
|
||||||
};
|
|
||||||
if (node.Fingerprint.IsNotEmpty())
|
|
||||||
{
|
|
||||||
tls.utls = new Utls4Sbox()
|
|
||||||
{
|
|
||||||
enabled = true,
|
|
||||||
fingerprint = node.Fingerprint.IsNullOrEmpty() ? _config.CoreBasicItem.DefFingerprint : node.Fingerprint
|
|
||||||
};
|
|
||||||
}
|
|
||||||
if (node.StreamSecurity == Global.StreamSecurityReality)
|
|
||||||
{
|
|
||||||
tls.reality = new Reality4Sbox()
|
|
||||||
{
|
|
||||||
enabled = true,
|
|
||||||
public_key = node.PublicKey,
|
|
||||||
short_id = node.ShortId
|
|
||||||
};
|
|
||||||
tls.insecure = false;
|
|
||||||
}
|
|
||||||
outbound.tls = tls;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
Logging.SaveLog(_tag, ex);
|
|
||||||
}
|
|
||||||
return await Task.FromResult(0);
|
|
||||||
}
|
|
||||||
|
|
||||||
private async Task<int> GenOutboundTransport(ProfileItem node, Outbound4Sbox outbound)
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
var transport = new Transport4Sbox();
|
|
||||||
|
|
||||||
switch (node.GetNetwork())
|
|
||||||
{
|
|
||||||
case nameof(ETransport.h2):
|
|
||||||
transport.type = nameof(ETransport.http);
|
|
||||||
transport.host = node.RequestHost.IsNullOrEmpty() ? null : Utils.String2List(node.RequestHost);
|
|
||||||
transport.path = node.Path.IsNullOrEmpty() ? null : node.Path;
|
|
||||||
break;
|
|
||||||
|
|
||||||
case nameof(ETransport.tcp): //http
|
|
||||||
if (node.HeaderType == Global.TcpHeaderHttp)
|
|
||||||
{
|
|
||||||
if (node.ConfigType == EConfigType.Shadowsocks)
|
|
||||||
{
|
|
||||||
outbound.plugin = "obfs-local";
|
|
||||||
outbound.plugin_opts = $"obfs=http;obfs-host={node.RequestHost};";
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
transport.type = nameof(ETransport.http);
|
|
||||||
transport.host = node.RequestHost.IsNullOrEmpty() ? null : Utils.String2List(node.RequestHost);
|
|
||||||
transport.path = node.Path.IsNullOrEmpty() ? null : node.Path;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
|
|
||||||
case nameof(ETransport.ws):
|
|
||||||
transport.type = nameof(ETransport.ws);
|
|
||||||
transport.path = node.Path.IsNullOrEmpty() ? null : node.Path;
|
|
||||||
if (node.RequestHost.IsNotEmpty())
|
|
||||||
{
|
|
||||||
transport.headers = new()
|
|
||||||
{
|
|
||||||
Host = node.RequestHost
|
|
||||||
};
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
|
|
||||||
case nameof(ETransport.httpupgrade):
|
|
||||||
transport.type = nameof(ETransport.httpupgrade);
|
|
||||||
transport.path = node.Path.IsNullOrEmpty() ? null : node.Path;
|
|
||||||
transport.host = node.RequestHost.IsNullOrEmpty() ? null : node.RequestHost;
|
|
||||||
|
|
||||||
break;
|
|
||||||
|
|
||||||
case nameof(ETransport.quic):
|
|
||||||
transport.type = nameof(ETransport.quic);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case nameof(ETransport.grpc):
|
|
||||||
transport.type = nameof(ETransport.grpc);
|
|
||||||
transport.service_name = node.Path;
|
|
||||||
transport.idle_timeout = _config.GrpcItem.IdleTimeout?.ToString("##s");
|
|
||||||
transport.ping_timeout = _config.GrpcItem.HealthCheckTimeout?.ToString("##s");
|
|
||||||
transport.permit_without_stream = _config.GrpcItem.PermitWithoutStream;
|
|
||||||
break;
|
|
||||||
|
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
if (transport.type != null)
|
|
||||||
{
|
|
||||||
outbound.transport = transport;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
Logging.SaveLog(_tag, ex);
|
|
||||||
}
|
|
||||||
return await Task.FromResult(0);
|
|
||||||
}
|
|
||||||
|
|
||||||
private async Task<int> GenMoreOutbounds(ProfileItem node, SingboxConfig singboxConfig)
|
private async Task<int> GenMoreOutbounds(ProfileItem node, SingboxConfig singboxConfig)
|
||||||
{
|
{
|
||||||
if (node.Subid.IsNullOrEmpty())
|
if (node.Subid.IsNullOrEmpty())
|
||||||
@@ -668,7 +748,10 @@ public partial class CoreConfigSingboxService
|
|||||||
{
|
{
|
||||||
var node = nodes[i];
|
var node = nodes[i];
|
||||||
if (node == null)
|
if (node == null)
|
||||||
|
{
|
||||||
continue;
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
if (node.ConfigType.IsGroupType())
|
if (node.ConfigType.IsGroupType())
|
||||||
{
|
{
|
||||||
var (childProfiles, profileGroupItem) = await ProfileGroupItemManager.GetChildProfileItems(node.IndexId);
|
var (childProfiles, profileGroupItem) = await ProfileGroupItemManager.GetChildProfileItems(node.IndexId);
|
||||||
|
|||||||
@@ -250,7 +250,9 @@ public partial class CoreConfigSingboxService
|
|||||||
foreach (var it in item.Domain)
|
foreach (var it in item.Domain)
|
||||||
{
|
{
|
||||||
if (ParseV2Domain(it, rule1))
|
if (ParseV2Domain(it, rule1))
|
||||||
|
{
|
||||||
countDomain++;
|
countDomain++;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if (countDomain > 0)
|
if (countDomain > 0)
|
||||||
{
|
{
|
||||||
@@ -265,7 +267,9 @@ public partial class CoreConfigSingboxService
|
|||||||
foreach (var it in item.Ip)
|
foreach (var it in item.Ip)
|
||||||
{
|
{
|
||||||
if (ParseV2Address(it, rule2))
|
if (ParseV2Address(it, rule2))
|
||||||
|
{
|
||||||
countIp++;
|
countIp++;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if (countIp > 0)
|
if (countIp > 0)
|
||||||
{
|
{
|
||||||
@@ -296,7 +300,7 @@ public partial class CoreConfigSingboxService
|
|||||||
|
|
||||||
private bool ParseV2Domain(string domain, Rule4Sbox rule)
|
private bool ParseV2Domain(string domain, Rule4Sbox rule)
|
||||||
{
|
{
|
||||||
if (domain.StartsWith("#") || domain.StartsWith("ext:") || domain.StartsWith("ext-domain:"))
|
if (domain.StartsWith('#') || domain.StartsWith("ext:") || domain.StartsWith("ext-domain:"))
|
||||||
{
|
{
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@@ -312,10 +316,8 @@ public partial class CoreConfigSingboxService
|
|||||||
}
|
}
|
||||||
else if (domain.StartsWith("domain:"))
|
else if (domain.StartsWith("domain:"))
|
||||||
{
|
{
|
||||||
rule.domain ??= [];
|
|
||||||
rule.domain_suffix ??= [];
|
rule.domain_suffix ??= [];
|
||||||
rule.domain?.Add(domain.Substring(7));
|
rule.domain_suffix?.Add(domain.Substring(7));
|
||||||
rule.domain_suffix?.Add("." + domain.Substring(7));
|
|
||||||
}
|
}
|
||||||
else if (domain.StartsWith("full:"))
|
else if (domain.StartsWith("full:"))
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -7,7 +7,9 @@ public partial class CoreConfigSingboxService
|
|||||||
static void AddRuleSets(List<string> ruleSets, List<string>? rule_set)
|
static void AddRuleSets(List<string> ruleSets, List<string>? rule_set)
|
||||||
{
|
{
|
||||||
if (rule_set != null)
|
if (rule_set != null)
|
||||||
|
{
|
||||||
ruleSets.AddRange(rule_set);
|
ruleSets.AddRange(rule_set);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
var geosite = "geosite";
|
var geosite = "geosite";
|
||||||
var geoip = "geoip";
|
var geoip = "geoip";
|
||||||
|
|||||||
@@ -94,8 +94,8 @@ public partial class CoreConfigV2rayService(Config config)
|
|||||||
|
|
||||||
ret.Msg = ResUI.InitialConfiguration;
|
ret.Msg = ResUI.InitialConfiguration;
|
||||||
|
|
||||||
string result = EmbedUtils.GetEmbedText(Global.V2raySampleClient);
|
var result = EmbedUtils.GetEmbedText(Global.V2raySampleClient);
|
||||||
string txtOutbound = EmbedUtils.GetEmbedText(Global.V2raySampleOutbound);
|
var txtOutbound = EmbedUtils.GetEmbedText(Global.V2raySampleOutbound);
|
||||||
if (result.IsNullOrEmpty() || txtOutbound.IsNullOrEmpty())
|
if (result.IsNullOrEmpty() || txtOutbound.IsNullOrEmpty())
|
||||||
{
|
{
|
||||||
ret.Msg = ResUI.FailedGetDefaultConfiguration;
|
ret.Msg = ResUI.FailedGetDefaultConfiguration;
|
||||||
@@ -137,7 +137,9 @@ public partial class CoreConfigV2rayService(Config config)
|
|||||||
foreach (var rule in rules)
|
foreach (var rule in rules)
|
||||||
{
|
{
|
||||||
if (rule.outboundTag == null)
|
if (rule.outboundTag == null)
|
||||||
|
{
|
||||||
continue;
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
if (balancerTagSet.Contains(rule.outboundTag))
|
if (balancerTagSet.Contains(rule.outboundTag))
|
||||||
{
|
{
|
||||||
@@ -200,8 +202,8 @@ public partial class CoreConfigV2rayService(Config config)
|
|||||||
|
|
||||||
ret.Msg = ResUI.InitialConfiguration;
|
ret.Msg = ResUI.InitialConfiguration;
|
||||||
|
|
||||||
string result = EmbedUtils.GetEmbedText(Global.V2raySampleClient);
|
var result = EmbedUtils.GetEmbedText(Global.V2raySampleClient);
|
||||||
string txtOutbound = EmbedUtils.GetEmbedText(Global.V2raySampleOutbound);
|
var txtOutbound = EmbedUtils.GetEmbedText(Global.V2raySampleOutbound);
|
||||||
if (result.IsNullOrEmpty() || txtOutbound.IsNullOrEmpty())
|
if (result.IsNullOrEmpty() || txtOutbound.IsNullOrEmpty())
|
||||||
{
|
{
|
||||||
ret.Msg = ResUI.FailedGetDefaultConfiguration;
|
ret.Msg = ResUI.FailedGetDefaultConfiguration;
|
||||||
|
|||||||
@@ -11,7 +11,9 @@ public partial class CoreConfigV2rayService
|
|||||||
|
|
||||||
// Case 1: exact match already exists -> nothing to do
|
// Case 1: exact match already exists -> nothing to do
|
||||||
if (subjectSelectors.Any(baseTagName.StartsWith))
|
if (subjectSelectors.Any(baseTagName.StartsWith))
|
||||||
|
{
|
||||||
return await Task.FromResult(0);
|
return await Task.FromResult(0);
|
||||||
|
}
|
||||||
|
|
||||||
// Case 2: prefix match exists -> reuse it and move to the first position
|
// Case 2: prefix match exists -> reuse it and move to the first position
|
||||||
var matched = subjectSelectors.FirstOrDefault(s => s.StartsWith(baseTagName));
|
var matched = subjectSelectors.FirstOrDefault(s => s.StartsWith(baseTagName));
|
||||||
|
|||||||
@@ -86,8 +86,8 @@ public partial class CoreConfigV2rayService
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Handle outbounds - append instead of override
|
var customOutboundsNode = new JsonArray();
|
||||||
var customOutboundsNode = fullConfigTemplateNode["outbounds"] is JsonArray outbounds ? outbounds : new JsonArray();
|
|
||||||
foreach (var outbound in v2rayConfig.outbounds)
|
foreach (var outbound in v2rayConfig.outbounds)
|
||||||
{
|
{
|
||||||
if (outbound.protocol.ToLower() is "blackhole" or "dns" or "freedom")
|
if (outbound.protocol.ToLower() is "blackhole" or "dns" or "freedom")
|
||||||
@@ -97,14 +97,30 @@ public partial class CoreConfigV2rayService
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if ((outbound.streamSettings?.sockopt?.dialerProxy.IsNullOrEmpty() == true) && (!fullConfigTemplate.ProxyDetour.IsNullOrEmpty()) && !(Utils.IsPrivateNetwork(outbound.settings?.servers?.FirstOrDefault()?.address ?? string.Empty) || Utils.IsPrivateNetwork(outbound.settings?.vnext?.FirstOrDefault()?.address ?? string.Empty)))
|
else if ((!fullConfigTemplate.ProxyDetour.IsNullOrEmpty())
|
||||||
|
&& ((outbound.streamSettings?.sockopt?.dialerProxy.IsNullOrEmpty() ?? true) == true))
|
||||||
{
|
{
|
||||||
outbound.streamSettings ??= new StreamSettings4Ray();
|
var outboundAddress = outbound.settings?.servers?.FirstOrDefault()?.address
|
||||||
outbound.streamSettings.sockopt ??= new Sockopt4Ray();
|
?? outbound.settings?.vnext?.FirstOrDefault()?.address
|
||||||
outbound.streamSettings.sockopt.dialerProxy = fullConfigTemplate.ProxyDetour;
|
?? string.Empty;
|
||||||
|
if (!Utils.IsPrivateNetwork(outboundAddress))
|
||||||
|
{
|
||||||
|
outbound.streamSettings ??= new StreamSettings4Ray();
|
||||||
|
outbound.streamSettings.sockopt ??= new Sockopt4Ray();
|
||||||
|
outbound.streamSettings.sockopt.dialerProxy = fullConfigTemplate.ProxyDetour;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
customOutboundsNode.Add(JsonUtils.DeepCopy(outbound));
|
customOutboundsNode.Add(JsonUtils.DeepCopy(outbound));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (fullConfigTemplateNode["outbounds"] is JsonArray templateOutbounds)
|
||||||
|
{
|
||||||
|
foreach (var outbound in templateOutbounds)
|
||||||
|
{
|
||||||
|
customOutboundsNode.Add(outbound?.DeepClone());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
fullConfigTemplateNode["outbounds"] = customOutboundsNode;
|
fullConfigTemplateNode["outbounds"] = customOutboundsNode;
|
||||||
|
|
||||||
return await Task.FromResult(JsonUtils.Serialize(fullConfigTemplateNode));
|
return await Task.FromResult(JsonUtils.Serialize(fullConfigTemplateNode));
|
||||||
|
|||||||
@@ -79,9 +79,23 @@ public partial class CoreConfigV2rayService
|
|||||||
|
|
||||||
static object CreateDnsServer(string dnsAddress, List<string> domains, List<string>? expectedIPs = null)
|
static object CreateDnsServer(string dnsAddress, List<string> domains, List<string>? expectedIPs = null)
|
||||||
{
|
{
|
||||||
|
var (domain, scheme, port, path) = Utils.ParseUrl(dnsAddress);
|
||||||
|
var domainFinal = dnsAddress;
|
||||||
|
int? portFinal = null;
|
||||||
|
if (scheme.IsNullOrEmpty() || scheme.StartsWith("udp", StringComparison.OrdinalIgnoreCase))
|
||||||
|
{
|
||||||
|
domainFinal = domain;
|
||||||
|
portFinal = port > 0 ? port : null;
|
||||||
|
}
|
||||||
|
else if (scheme.StartsWith("tcp", StringComparison.OrdinalIgnoreCase))
|
||||||
|
{
|
||||||
|
domainFinal = scheme + "://" + domain;
|
||||||
|
portFinal = port > 0 ? port : null;
|
||||||
|
}
|
||||||
var dnsServer = new DnsServer4Ray
|
var dnsServer = new DnsServer4Ray
|
||||||
{
|
{
|
||||||
address = dnsAddress,
|
address = domainFinal,
|
||||||
|
port = portFinal,
|
||||||
skipFallback = true,
|
skipFallback = true,
|
||||||
domains = domains.Count > 0 ? domains : null,
|
domains = domains.Count > 0 ? domains : null,
|
||||||
expectedIPs = expectedIPs?.Count > 0 ? expectedIPs : null
|
expectedIPs = expectedIPs?.Count > 0 ? expectedIPs : null
|
||||||
@@ -175,7 +189,10 @@ public partial class CoreConfigV2rayService
|
|||||||
foreach (var domain in item.Domain)
|
foreach (var domain in item.Domain)
|
||||||
{
|
{
|
||||||
if (domain.StartsWith('#'))
|
if (domain.StartsWith('#'))
|
||||||
|
{
|
||||||
continue;
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
var normalizedDomain = domain.Replace(Global.RoutingRuleComma, ",");
|
var normalizedDomain = domain.Replace(Global.RoutingRuleComma, ",");
|
||||||
|
|
||||||
if (item.OutboundTag == Global.DirectTag)
|
if (item.OutboundTag == Global.DirectTag)
|
||||||
@@ -333,8 +350,8 @@ public partial class CoreConfigV2rayService
|
|||||||
if (obj is null)
|
if (obj is null)
|
||||||
{
|
{
|
||||||
List<string> servers = [];
|
List<string> servers = [];
|
||||||
string[] arrDNS = normalDNS.Split(',');
|
var arrDNS = normalDNS.Split(',');
|
||||||
foreach (string str in arrDNS)
|
foreach (var str in arrDNS)
|
||||||
{
|
{
|
||||||
servers.Add(str);
|
servers.Add(str);
|
||||||
}
|
}
|
||||||
@@ -354,7 +371,10 @@ public partial class CoreConfigV2rayService
|
|||||||
foreach (var host in systemHosts)
|
foreach (var host in systemHosts)
|
||||||
{
|
{
|
||||||
if (normalHost1[host.Key] != null)
|
if (normalHost1[host.Key] != null)
|
||||||
|
{
|
||||||
continue;
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
normalHost1[host.Key] = host.Value;
|
normalHost1[host.Key] = host.Value;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -48,7 +48,7 @@ public partial class CoreConfigV2rayService
|
|||||||
|
|
||||||
private Inbounds4Ray GetInbound(InItem inItem, EInboundProtocol protocol, bool bSocks)
|
private Inbounds4Ray GetInbound(InItem inItem, EInboundProtocol protocol, bool bSocks)
|
||||||
{
|
{
|
||||||
string result = EmbedUtils.GetEmbedText(Global.V2raySampleInbound);
|
var result = EmbedUtils.GetEmbedText(Global.V2raySampleInbound);
|
||||||
if (result.IsNullOrEmpty())
|
if (result.IsNullOrEmpty())
|
||||||
{
|
{
|
||||||
return new();
|
return new();
|
||||||
|
|||||||
@@ -180,10 +180,15 @@ public partial class CoreConfigV2rayService
|
|||||||
}
|
}
|
||||||
case EConfigType.WireGuard:
|
case EConfigType.WireGuard:
|
||||||
{
|
{
|
||||||
|
var address = node.Address;
|
||||||
|
if (Utils.IsIpv6(address))
|
||||||
|
{
|
||||||
|
address = $"[{address}]";
|
||||||
|
}
|
||||||
var peer = new WireguardPeer4Ray
|
var peer = new WireguardPeer4Ray
|
||||||
{
|
{
|
||||||
publicKey = node.PublicKey,
|
publicKey = node.PublicKey,
|
||||||
endpoint = node.Address + ":" + node.Port.ToString()
|
endpoint = address + ":" + node.Port.ToString()
|
||||||
};
|
};
|
||||||
var setting = new Outboundsettings4Ray
|
var setting = new Outboundsettings4Ray
|
||||||
{
|
{
|
||||||
@@ -277,6 +282,23 @@ public partial class CoreConfigV2rayService
|
|||||||
{
|
{
|
||||||
tlsSettings.serverName = Utils.String2List(host)?.First();
|
tlsSettings.serverName = Utils.String2List(host)?.First();
|
||||||
}
|
}
|
||||||
|
var certs = CertPemManager.ParsePemChain(node.Cert);
|
||||||
|
if (certs.Count > 0)
|
||||||
|
{
|
||||||
|
var certsettings = new List<CertificateSettings4Ray>();
|
||||||
|
foreach (var cert in certs)
|
||||||
|
{
|
||||||
|
var certPerLine = cert.Split("\n").ToList();
|
||||||
|
certsettings.Add(new CertificateSettings4Ray
|
||||||
|
{
|
||||||
|
certificate = certPerLine,
|
||||||
|
usage = "verify",
|
||||||
|
});
|
||||||
|
}
|
||||||
|
tlsSettings.certificates = certsettings;
|
||||||
|
tlsSettings.disableSystemRoot = true;
|
||||||
|
tlsSettings.allowInsecure = false;
|
||||||
|
}
|
||||||
streamSettings.tlsSettings = tlsSettings;
|
streamSettings.tlsSettings = tlsSettings;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -334,7 +356,6 @@ public partial class CoreConfigV2rayService
|
|||||||
if (host.IsNotEmpty())
|
if (host.IsNotEmpty())
|
||||||
{
|
{
|
||||||
wsSettings.host = host;
|
wsSettings.host = host;
|
||||||
wsSettings.headers.Host = host;
|
|
||||||
}
|
}
|
||||||
if (path.IsNotEmpty())
|
if (path.IsNotEmpty())
|
||||||
{
|
{
|
||||||
@@ -453,16 +474,16 @@ public partial class CoreConfigV2rayService
|
|||||||
};
|
};
|
||||||
|
|
||||||
//request Host
|
//request Host
|
||||||
string request = EmbedUtils.GetEmbedText(Global.V2raySampleHttpRequestFileName);
|
var request = EmbedUtils.GetEmbedText(Global.V2raySampleHttpRequestFileName);
|
||||||
string[] arrHost = host.Split(',');
|
var arrHost = host.Split(',');
|
||||||
string host2 = string.Join(",".AppendQuotes(), arrHost);
|
var host2 = string.Join(",".AppendQuotes(), arrHost);
|
||||||
request = request.Replace("$requestHost$", $"{host2.AppendQuotes()}");
|
request = request.Replace("$requestHost$", $"{host2.AppendQuotes()}");
|
||||||
request = request.Replace("$requestUserAgent$", $"{useragent.AppendQuotes()}");
|
request = request.Replace("$requestUserAgent$", $"{useragent.AppendQuotes()}");
|
||||||
//Path
|
//Path
|
||||||
string pathHttp = @"/";
|
var pathHttp = @"/";
|
||||||
if (path.IsNotEmpty())
|
if (path.IsNotEmpty())
|
||||||
{
|
{
|
||||||
string[] arrPath = path.Split(',');
|
var arrPath = path.Split(',');
|
||||||
pathHttp = string.Join(",".AppendQuotes(), arrPath);
|
pathHttp = string.Join(",".AppendQuotes(), arrPath);
|
||||||
}
|
}
|
||||||
request = request.Replace("$requestPath$", $"{pathHttp.AppendQuotes()}");
|
request = request.Replace("$requestPath$", $"{pathHttp.AppendQuotes()}");
|
||||||
@@ -623,10 +644,10 @@ public partial class CoreConfigV2rayService
|
|||||||
// Cache for chain proxies to avoid duplicate generation
|
// Cache for chain proxies to avoid duplicate generation
|
||||||
var nextProxyCache = new Dictionary<string, Outbounds4Ray?>();
|
var nextProxyCache = new Dictionary<string, Outbounds4Ray?>();
|
||||||
var prevProxyTags = new Dictionary<string, string?>(); // Map from profile name to tag
|
var prevProxyTags = new Dictionary<string, string?>(); // Map from profile name to tag
|
||||||
int prevIndex = 0; // Index for prev outbounds
|
var prevIndex = 0; // Index for prev outbounds
|
||||||
|
|
||||||
// Process nodes
|
// Process nodes
|
||||||
int index = 0;
|
var index = 0;
|
||||||
foreach (var node in nodes)
|
foreach (var node in nodes)
|
||||||
{
|
{
|
||||||
index++;
|
index++;
|
||||||
@@ -781,7 +802,10 @@ public partial class CoreConfigV2rayService
|
|||||||
{
|
{
|
||||||
var node = nodes[i];
|
var node = nodes[i];
|
||||||
if (node == null)
|
if (node == null)
|
||||||
|
{
|
||||||
continue;
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
if (node.ConfigType.IsGroupType())
|
if (node.ConfigType.IsGroupType())
|
||||||
{
|
{
|
||||||
var (childProfiles, _) = await ProfileGroupItemManager.GetChildProfileItems(node.IndexId);
|
var (childProfiles, _) = await ProfileGroupItemManager.GetChildProfileItems(node.IndexId);
|
||||||
|
|||||||
@@ -6,7 +6,7 @@ public partial class CoreConfigV2rayService
|
|||||||
{
|
{
|
||||||
if (_config.GuiItem.EnableStatistics || _config.GuiItem.DisplayRealTimeSpeed)
|
if (_config.GuiItem.EnableStatistics || _config.GuiItem.DisplayRealTimeSpeed)
|
||||||
{
|
{
|
||||||
string tag = EInboundProtocol.api.ToString();
|
var tag = EInboundProtocol.api.ToString();
|
||||||
Metrics4Ray apiObj = new();
|
Metrics4Ray apiObj = new();
|
||||||
Policy4Ray policyObj = new();
|
Policy4Ray policyObj = new();
|
||||||
SystemPolicy4Ray policySystemSetting = new();
|
SystemPolicy4Ray policySystemSetting = new();
|
||||||
|
|||||||
@@ -7,7 +7,7 @@ namespace ServiceLib.Services;
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
public class DownloadService
|
public class DownloadService
|
||||||
{
|
{
|
||||||
public event EventHandler<RetResult>? UpdateCompleted;
|
public event EventHandler<UpdateResult>? UpdateCompleted;
|
||||||
|
|
||||||
public event ErrorEventHandler? Error;
|
public event ErrorEventHandler? Error;
|
||||||
|
|
||||||
@@ -40,10 +40,10 @@ public class DownloadService
|
|||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
UpdateCompleted?.Invoke(this, new RetResult(false, $"{ResUI.Downloading} {url}"));
|
UpdateCompleted?.Invoke(this, new UpdateResult(false, $"{ResUI.Downloading} {url}"));
|
||||||
|
|
||||||
var progress = new Progress<double>();
|
var progress = new Progress<double>();
|
||||||
progress.ProgressChanged += (sender, value) => UpdateCompleted?.Invoke(this, new RetResult(value > 100, $"...{value}%"));
|
progress.ProgressChanged += (sender, value) => UpdateCompleted?.Invoke(this, new UpdateResult(value > 100, $"...{value}%"));
|
||||||
|
|
||||||
var webProxy = await GetWebProxy(blProxy);
|
var webProxy = await GetWebProxy(blProxy);
|
||||||
await DownloaderHelper.Instance.DownloadFileAsync(webProxy,
|
await DownloaderHelper.Instance.DownloadFileAsync(webProxy,
|
||||||
@@ -71,7 +71,7 @@ public class DownloadService
|
|||||||
AllowAutoRedirect = false,
|
AllowAutoRedirect = false,
|
||||||
Proxy = await GetWebProxy(blProxy)
|
Proxy = await GetWebProxy(blProxy)
|
||||||
};
|
};
|
||||||
HttpClient client = new(webRequestHandler);
|
var client = new HttpClient(webRequestHandler);
|
||||||
|
|
||||||
var response = await client.GetAsync(url);
|
var response = await client.GetAsync(url);
|
||||||
if (response.StatusCode == HttpStatusCode.Redirect && response.Headers.Location is not null)
|
if (response.StatusCode == HttpStatusCode.Redirect && response.Headers.Location is not null)
|
||||||
@@ -156,7 +156,7 @@ public class DownloadService
|
|||||||
}
|
}
|
||||||
|
|
||||||
using var cts = new CancellationTokenSource();
|
using var cts = new CancellationTokenSource();
|
||||||
var result = await HttpClientHelper.Instance.GetAsync(client, url, cts.Token).WaitAsync(TimeSpan.FromSeconds(timeout), cts.Token);
|
var result = await client.GetStringAsync(url, cts.Token).WaitAsync(TimeSpan.FromSeconds(timeout), cts.Token);
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
|
|||||||
@@ -19,14 +19,19 @@ public class SpeedtestService(Config config, Func<SpeedTestResult, Task> updateF
|
|||||||
|
|
||||||
public void ExitLoop()
|
public void ExitLoop()
|
||||||
{
|
{
|
||||||
if (_lstExitLoop.Count > 0)
|
if (!_lstExitLoop.IsEmpty)
|
||||||
{
|
{
|
||||||
UpdateFunc("", ResUI.SpeedtestingStop);
|
_ = UpdateFunc("", ResUI.SpeedtestingStop);
|
||||||
|
|
||||||
_lstExitLoop.Clear();
|
_lstExitLoop.Clear();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static bool ShouldStopTest(string exitLoopKey)
|
||||||
|
{
|
||||||
|
return !_lstExitLoop.Any(p => p == exitLoopKey);
|
||||||
|
}
|
||||||
|
|
||||||
private async Task RunAsync(ESpeedActionType actionType, List<ProfileItem> selecteds)
|
private async Task RunAsync(ESpeedActionType actionType, List<ProfileItem> selecteds)
|
||||||
{
|
{
|
||||||
var exitLoopKey = Utils.GetGuid(false);
|
var exitLoopKey = Utils.GetGuid(false);
|
||||||
@@ -103,6 +108,11 @@ public class SpeedtestService(Config config, Func<SpeedTestResult, Task> updateF
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (lstSelected.Count > 1 && (actionType == ESpeedActionType.Speedtest || actionType == ESpeedActionType.Mixedtest))
|
||||||
|
{
|
||||||
|
NoticeManager.Instance.Enqueue(ResUI.SpeedtestingPressEscToExit);
|
||||||
|
}
|
||||||
|
|
||||||
return lstSelected;
|
return lstSelected;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -152,7 +162,7 @@ public class SpeedtestService(Config config, Func<SpeedTestResult, Task> updateF
|
|||||||
var pageSizeNext = pageSize / 2;
|
var pageSizeNext = pageSize / 2;
|
||||||
if (lstFailed.Count > 0 && pageSizeNext > 0)
|
if (lstFailed.Count > 0 && pageSizeNext > 0)
|
||||||
{
|
{
|
||||||
if (_lstExitLoop.Any(p => p == exitLoopKey) == false)
|
if (ShouldStopTest(exitLoopKey))
|
||||||
{
|
{
|
||||||
await UpdateFunc("", ResUI.SpeedtestingSkip);
|
await UpdateFunc("", ResUI.SpeedtestingSkip);
|
||||||
return;
|
return;
|
||||||
@@ -190,6 +200,12 @@ public class SpeedtestService(Config config, Func<SpeedTestResult, Task> updateF
|
|||||||
{
|
{
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (ShouldStopTest(exitLoopKey))
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
tasks.Add(Task.Run(async () =>
|
tasks.Add(Task.Run(async () =>
|
||||||
{
|
{
|
||||||
await DoRealPing(it);
|
await DoRealPing(it);
|
||||||
@@ -218,7 +234,7 @@ public class SpeedtestService(Config config, Func<SpeedTestResult, Task> updateF
|
|||||||
List<Task> tasks = new();
|
List<Task> tasks = new();
|
||||||
foreach (var it in selecteds)
|
foreach (var it in selecteds)
|
||||||
{
|
{
|
||||||
if (_lstExitLoop.Any(p => p == exitLoopKey) == false)
|
if (ShouldStopTest(exitLoopKey))
|
||||||
{
|
{
|
||||||
await UpdateFunc(it.IndexId, "", ResUI.SpeedtestingSkip);
|
await UpdateFunc(it.IndexId, "", ResUI.SpeedtestingSkip);
|
||||||
continue;
|
continue;
|
||||||
@@ -234,21 +250,27 @@ public class SpeedtestService(Config config, Func<SpeedTestResult, Task> updateF
|
|||||||
if (processService is null)
|
if (processService is null)
|
||||||
{
|
{
|
||||||
await UpdateFunc(it.IndexId, "", ResUI.FailedToRunCore);
|
await UpdateFunc(it.IndexId, "", ResUI.FailedToRunCore);
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
else
|
|
||||||
|
await Task.Delay(1000);
|
||||||
|
|
||||||
|
var delay = await DoRealPing(it);
|
||||||
|
if (blSpeedTest)
|
||||||
{
|
{
|
||||||
await Task.Delay(1000);
|
if (ShouldStopTest(exitLoopKey))
|
||||||
var delay = await DoRealPing(it);
|
|
||||||
if (blSpeedTest)
|
|
||||||
{
|
{
|
||||||
if (delay > 0)
|
await UpdateFunc(it.IndexId, "", ResUI.SpeedtestingSkip);
|
||||||
{
|
return;
|
||||||
await DoSpeedTest(downloadHandle, it);
|
}
|
||||||
}
|
|
||||||
else
|
if (delay > 0)
|
||||||
{
|
{
|
||||||
await UpdateFunc(it.IndexId, "", ResUI.SpeedtestingSkip);
|
await DoSpeedTest(downloadHandle, it);
|
||||||
}
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
await UpdateFunc(it.IndexId, "", ResUI.SpeedtestingSkip);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -272,7 +294,7 @@ public class SpeedtestService(Config config, Func<SpeedTestResult, Task> updateF
|
|||||||
private async Task<int> DoRealPing(ServerTestItem it)
|
private async Task<int> DoRealPing(ServerTestItem it)
|
||||||
{
|
{
|
||||||
var webProxy = new WebProxy($"socks5://{Global.Loopback}:{it.Port}");
|
var webProxy = new WebProxy($"socks5://{Global.Loopback}:{it.Port}");
|
||||||
var responseTime = await HttpClientHelper.Instance.GetRealPingTime(_config.SpeedTestItem.SpeedPingTestUrl, webProxy, 10);
|
var responseTime = await ConnectionHandler.GetRealPingTime(_config.SpeedTestItem.SpeedPingTestUrl, webProxy, 10);
|
||||||
|
|
||||||
ProfileExManager.Instance.SetTestDelay(it.IndexId, responseTime);
|
ProfileExManager.Instance.SetTestDelay(it.IndexId, responseTime);
|
||||||
await UpdateFunc(it.IndexId, responseTime.ToString());
|
await UpdateFunc(it.IndexId, responseTime.ToString());
|
||||||
@@ -301,31 +323,28 @@ public class SpeedtestService(Config config, Func<SpeedTestResult, Task> updateF
|
|||||||
{
|
{
|
||||||
var responseTime = -1;
|
var responseTime = -1;
|
||||||
|
|
||||||
|
if (!IPAddress.TryParse(url, out var ipAddress))
|
||||||
|
{
|
||||||
|
var ipHostInfo = await Dns.GetHostEntryAsync(url);
|
||||||
|
ipAddress = ipHostInfo.AddressList.First();
|
||||||
|
}
|
||||||
|
|
||||||
|
IPEndPoint endPoint = new(ipAddress, port);
|
||||||
|
using Socket clientSocket = new(endPoint.AddressFamily, SocketType.Stream, ProtocolType.Tcp);
|
||||||
|
|
||||||
|
var timer = Stopwatch.StartNew();
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
if (!IPAddress.TryParse(url, out var ipAddress))
|
using var cts = new CancellationTokenSource(TimeSpan.FromSeconds(5));
|
||||||
{
|
await clientSocket.ConnectAsync(endPoint, cts.Token).ConfigureAwait(false);
|
||||||
var ipHostInfo = await Dns.GetHostEntryAsync(url);
|
responseTime = (int)timer.ElapsedMilliseconds;
|
||||||
ipAddress = ipHostInfo.AddressList.First();
|
|
||||||
}
|
|
||||||
|
|
||||||
IPEndPoint endPoint = new(ipAddress, port);
|
|
||||||
using Socket clientSocket = new(endPoint.AddressFamily, SocketType.Stream, ProtocolType.Tcp);
|
|
||||||
|
|
||||||
var timer = Stopwatch.StartNew();
|
|
||||||
var result = clientSocket.BeginConnect(endPoint, null, null);
|
|
||||||
if (!result.AsyncWaitHandle.WaitOne(TimeSpan.FromSeconds(5)))
|
|
||||||
{
|
|
||||||
throw new TimeoutException("connect timeout (5s): " + url);
|
|
||||||
}
|
|
||||||
timer.Stop();
|
|
||||||
responseTime = (int)timer.Elapsed.TotalMilliseconds;
|
|
||||||
|
|
||||||
clientSocket.EndConnect(result);
|
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (OperationCanceledException)
|
||||||
{
|
{
|
||||||
Logging.SaveLog(_tag, ex);
|
}
|
||||||
|
finally
|
||||||
|
{
|
||||||
|
timer.Stop();
|
||||||
}
|
}
|
||||||
return responseTime;
|
return responseTime;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,14 +1,14 @@
|
|||||||
namespace ServiceLib.Services;
|
namespace ServiceLib.Services;
|
||||||
|
|
||||||
public class UpdateService
|
public class UpdateService(Config config, Func<bool, string, Task> updateFunc)
|
||||||
{
|
{
|
||||||
private Func<bool, string, Task>? _updateFunc;
|
private readonly Config? _config = config;
|
||||||
|
private readonly Func<bool, string, Task>? _updateFunc = updateFunc;
|
||||||
private readonly int _timeout = 30;
|
private readonly int _timeout = 30;
|
||||||
private static readonly string _tag = "UpdateService";
|
private static readonly string _tag = "UpdateService";
|
||||||
|
|
||||||
public async Task CheckUpdateGuiN(Config config, Func<bool, string, Task> updateFunc, bool preRelease)
|
public async Task CheckUpdateGuiN(bool preRelease)
|
||||||
{
|
{
|
||||||
_updateFunc = updateFunc;
|
|
||||||
var url = string.Empty;
|
var url = string.Empty;
|
||||||
var fileName = string.Empty;
|
var fileName = string.Empty;
|
||||||
|
|
||||||
@@ -17,17 +17,17 @@ public class UpdateService
|
|||||||
{
|
{
|
||||||
if (args.Success)
|
if (args.Success)
|
||||||
{
|
{
|
||||||
UpdateFunc(false, ResUI.MsgDownloadV2rayCoreSuccessfully);
|
_ = UpdateFunc(false, ResUI.MsgDownloadV2rayCoreSuccessfully);
|
||||||
UpdateFunc(true, Utils.UrlEncode(fileName));
|
_ = UpdateFunc(true, Utils.UrlEncode(fileName));
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
UpdateFunc(false, args.Msg);
|
_ = UpdateFunc(false, args.Msg);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
downloadHandle.Error += (sender2, args) =>
|
downloadHandle.Error += (sender2, args) =>
|
||||||
{
|
{
|
||||||
UpdateFunc(false, args.GetException().Message);
|
_ = UpdateFunc(false, args.GetException().Message);
|
||||||
};
|
};
|
||||||
|
|
||||||
await UpdateFunc(false, string.Format(ResUI.MsgStartUpdating, ECoreType.v2rayN));
|
await UpdateFunc(false, string.Format(ResUI.MsgStartUpdating, ECoreType.v2rayN));
|
||||||
@@ -37,7 +37,7 @@ public class UpdateService
|
|||||||
await UpdateFunc(false, string.Format(ResUI.MsgParsingSuccessfully, ECoreType.v2rayN));
|
await UpdateFunc(false, string.Format(ResUI.MsgParsingSuccessfully, ECoreType.v2rayN));
|
||||||
await UpdateFunc(false, result.Msg);
|
await UpdateFunc(false, result.Msg);
|
||||||
|
|
||||||
url = result.Data?.ToString();
|
url = result.Url.ToString();
|
||||||
fileName = Utils.GetTempPath(Utils.GetGuid());
|
fileName = Utils.GetTempPath(Utils.GetGuid());
|
||||||
await downloadHandle.DownloadFileAsync(url, fileName, true, _timeout);
|
await downloadHandle.DownloadFileAsync(url, fileName, true, _timeout);
|
||||||
}
|
}
|
||||||
@@ -47,9 +47,8 @@ public class UpdateService
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task CheckUpdateCore(ECoreType type, Config config, Func<bool, string, Task> updateFunc, bool preRelease)
|
public async Task CheckUpdateCore(ECoreType type, bool preRelease)
|
||||||
{
|
{
|
||||||
_updateFunc = updateFunc;
|
|
||||||
var url = string.Empty;
|
var url = string.Empty;
|
||||||
var fileName = string.Empty;
|
var fileName = string.Empty;
|
||||||
|
|
||||||
@@ -58,26 +57,26 @@ public class UpdateService
|
|||||||
{
|
{
|
||||||
if (args.Success)
|
if (args.Success)
|
||||||
{
|
{
|
||||||
UpdateFunc(false, ResUI.MsgDownloadV2rayCoreSuccessfully);
|
_ = UpdateFunc(false, ResUI.MsgDownloadV2rayCoreSuccessfully);
|
||||||
UpdateFunc(false, ResUI.MsgUnpacking);
|
_ = UpdateFunc(false, ResUI.MsgUnpacking);
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
UpdateFunc(true, fileName);
|
_ = UpdateFunc(true, fileName);
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
UpdateFunc(false, ex.Message);
|
_ = UpdateFunc(false, ex.Message);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
UpdateFunc(false, args.Msg);
|
_ = UpdateFunc(false, args.Msg);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
downloadHandle.Error += (sender2, args) =>
|
downloadHandle.Error += (sender2, args) =>
|
||||||
{
|
{
|
||||||
UpdateFunc(false, args.GetException().Message);
|
_ = UpdateFunc(false, args.GetException().Message);
|
||||||
};
|
};
|
||||||
|
|
||||||
await UpdateFunc(false, string.Format(ResUI.MsgStartUpdating, type));
|
await UpdateFunc(false, string.Format(ResUI.MsgStartUpdating, type));
|
||||||
@@ -87,7 +86,7 @@ public class UpdateService
|
|||||||
await UpdateFunc(false, string.Format(ResUI.MsgParsingSuccessfully, type));
|
await UpdateFunc(false, string.Format(ResUI.MsgParsingSuccessfully, type));
|
||||||
await UpdateFunc(false, result.Msg);
|
await UpdateFunc(false, result.Msg);
|
||||||
|
|
||||||
url = result.Data?.ToString();
|
url = result.Url.ToString();
|
||||||
var ext = url.Contains(".tar.gz") ? ".tar.gz" : Path.GetExtension(url);
|
var ext = url.Contains(".tar.gz") ? ".tar.gz" : Path.GetExtension(url);
|
||||||
fileName = Utils.GetTempPath(Utils.GetGuid() + ext);
|
fileName = Utils.GetTempPath(Utils.GetGuid() + ext);
|
||||||
await downloadHandle.DownloadFileAsync(url, fileName, true, _timeout);
|
await downloadHandle.DownloadFileAsync(url, fileName, true, _timeout);
|
||||||
@@ -101,36 +100,36 @@ public class UpdateService
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task UpdateGeoFileAll(Config config, Func<bool, string, Task> updateFunc)
|
public async Task UpdateGeoFileAll()
|
||||||
{
|
{
|
||||||
await UpdateGeoFiles(config, updateFunc);
|
await UpdateGeoFiles();
|
||||||
await UpdateOtherFiles(config, updateFunc);
|
await UpdateOtherFiles();
|
||||||
await UpdateSrsFileAll(config, updateFunc);
|
await UpdateSrsFileAll();
|
||||||
await UpdateFunc(true, string.Format(ResUI.MsgDownloadGeoFileSuccessfully, "geo"));
|
await UpdateFunc(true, string.Format(ResUI.MsgDownloadGeoFileSuccessfully, "geo"));
|
||||||
}
|
}
|
||||||
|
|
||||||
#region CheckUpdate private
|
#region CheckUpdate private
|
||||||
|
|
||||||
private async Task<RetResult> CheckUpdateAsync(DownloadService downloadHandle, ECoreType type, bool preRelease)
|
private async Task<UpdateResult> CheckUpdateAsync(DownloadService downloadHandle, ECoreType type, bool preRelease)
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
var result = await GetRemoteVersion(downloadHandle, type, preRelease);
|
var result = await GetRemoteVersion(downloadHandle, type, preRelease);
|
||||||
if (!result.Success || result.Data is null)
|
if (!result.Success || result.Version is null)
|
||||||
{
|
{
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
return await ParseDownloadUrl(type, (SemanticVersion)result.Data);
|
return await ParseDownloadUrl(type, result);
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
Logging.SaveLog(_tag, ex);
|
Logging.SaveLog(_tag, ex);
|
||||||
await UpdateFunc(false, ex.Message);
|
await UpdateFunc(false, ex.Message);
|
||||||
return new RetResult(false, ex.Message);
|
return new UpdateResult(false, ex.Message);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private async Task<RetResult> GetRemoteVersion(DownloadService downloadHandle, ECoreType type, bool preRelease)
|
private async Task<UpdateResult> GetRemoteVersion(DownloadService downloadHandle, ECoreType type, bool preRelease)
|
||||||
{
|
{
|
||||||
var coreInfo = CoreInfoManager.Instance.GetCoreInfo(type);
|
var coreInfo = CoreInfoManager.Instance.GetCoreInfo(type);
|
||||||
var tagName = string.Empty;
|
var tagName = string.Empty;
|
||||||
@@ -140,7 +139,7 @@ public class UpdateService
|
|||||||
var result = await downloadHandle.TryDownloadString(url, true, Global.AppName);
|
var result = await downloadHandle.TryDownloadString(url, true, Global.AppName);
|
||||||
if (result.IsNullOrEmpty())
|
if (result.IsNullOrEmpty())
|
||||||
{
|
{
|
||||||
return new RetResult(false, "");
|
return new UpdateResult(false, "");
|
||||||
}
|
}
|
||||||
|
|
||||||
var gitHubReleases = JsonUtils.Deserialize<List<GitHubRelease>>(result);
|
var gitHubReleases = JsonUtils.Deserialize<List<GitHubRelease>>(result);
|
||||||
@@ -154,12 +153,12 @@ public class UpdateService
|
|||||||
var lastUrl = await downloadHandle.UrlRedirectAsync(url, true);
|
var lastUrl = await downloadHandle.UrlRedirectAsync(url, true);
|
||||||
if (lastUrl == null)
|
if (lastUrl == null)
|
||||||
{
|
{
|
||||||
return new RetResult(false, "");
|
return new UpdateResult(false, "");
|
||||||
}
|
}
|
||||||
|
|
||||||
tagName = lastUrl?.Split("/tag/").LastOrDefault();
|
tagName = lastUrl?.Split("/tag/").LastOrDefault();
|
||||||
}
|
}
|
||||||
return new RetResult(true, "", new SemanticVersion(tagName));
|
return new UpdateResult(true, new SemanticVersion(tagName));
|
||||||
}
|
}
|
||||||
|
|
||||||
private async Task<SemanticVersion> GetCoreVersion(ECoreType type)
|
private async Task<SemanticVersion> GetCoreVersion(ECoreType type)
|
||||||
@@ -167,7 +166,7 @@ public class UpdateService
|
|||||||
try
|
try
|
||||||
{
|
{
|
||||||
var coreInfo = CoreInfoManager.Instance.GetCoreInfo(type);
|
var coreInfo = CoreInfoManager.Instance.GetCoreInfo(type);
|
||||||
string filePath = string.Empty;
|
var filePath = string.Empty;
|
||||||
foreach (var name in coreInfo.CoreExes)
|
foreach (var name in coreInfo.CoreExes)
|
||||||
{
|
{
|
||||||
var vName = Utils.GetBinPath(Utils.GetExeName(name), coreInfo.CoreType.ToString());
|
var vName = Utils.GetBinPath(Utils.GetExeName(name), coreInfo.CoreType.ToString());
|
||||||
@@ -180,14 +179,14 @@ public class UpdateService
|
|||||||
|
|
||||||
if (!File.Exists(filePath))
|
if (!File.Exists(filePath))
|
||||||
{
|
{
|
||||||
string msg = string.Format(ResUI.NotFoundCore, @"", "", "");
|
var msg = string.Format(ResUI.NotFoundCore, @"", "", "");
|
||||||
//ShowMsg(true, msg);
|
//ShowMsg(true, msg);
|
||||||
return new SemanticVersion("");
|
return new SemanticVersion("");
|
||||||
}
|
}
|
||||||
|
|
||||||
var result = await Utils.GetCliWrapOutput(filePath, coreInfo.VersionArg);
|
var result = await Utils.GetCliWrapOutput(filePath, coreInfo.VersionArg);
|
||||||
var echo = result ?? "";
|
var echo = result ?? "";
|
||||||
string version = string.Empty;
|
var version = string.Empty;
|
||||||
switch (type)
|
switch (type)
|
||||||
{
|
{
|
||||||
case ECoreType.v2fly:
|
case ECoreType.v2fly:
|
||||||
@@ -214,10 +213,11 @@ public class UpdateService
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private async Task<RetResult> ParseDownloadUrl(ECoreType type, SemanticVersion version)
|
private async Task<UpdateResult> ParseDownloadUrl(ECoreType type, UpdateResult result)
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
|
var version = result.Version ?? new SemanticVersion(0, 0, 0);
|
||||||
var coreInfo = CoreInfoManager.Instance.GetCoreInfo(type);
|
var coreInfo = CoreInfoManager.Instance.GetCoreInfo(type);
|
||||||
var coreUrl = await GetUrlFromCore(coreInfo) ?? string.Empty;
|
var coreUrl = await GetUrlFromCore(coreInfo) ?? string.Empty;
|
||||||
SemanticVersion curVersion;
|
SemanticVersion curVersion;
|
||||||
@@ -261,16 +261,17 @@ public class UpdateService
|
|||||||
|
|
||||||
if (curVersion >= version && version != new SemanticVersion(0, 0, 0))
|
if (curVersion >= version && version != new SemanticVersion(0, 0, 0))
|
||||||
{
|
{
|
||||||
return new RetResult(false, message);
|
return new UpdateResult(false, message);
|
||||||
}
|
}
|
||||||
|
|
||||||
return new RetResult(true, "", url);
|
result.Url = url;
|
||||||
|
return result;
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
Logging.SaveLog(_tag, ex);
|
Logging.SaveLog(_tag, ex);
|
||||||
await UpdateFunc(false, ex.Message);
|
await UpdateFunc(false, ex.Message);
|
||||||
return new RetResult(false, ex.Message);
|
return new UpdateResult(false, ex.Message);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -314,7 +315,7 @@ public class UpdateService
|
|||||||
_ => null,
|
_ => null,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
else if (Utils.IsOSX())
|
else if (Utils.IsMacOS())
|
||||||
{
|
{
|
||||||
return RuntimeInformation.ProcessArchitecture switch
|
return RuntimeInformation.ProcessArchitecture switch
|
||||||
{
|
{
|
||||||
@@ -330,13 +331,11 @@ public class UpdateService
|
|||||||
|
|
||||||
#region Geo private
|
#region Geo private
|
||||||
|
|
||||||
private async Task UpdateGeoFiles(Config config, Func<bool, string, Task> updateFunc)
|
private async Task UpdateGeoFiles()
|
||||||
{
|
{
|
||||||
_updateFunc = updateFunc;
|
var geoUrl = string.IsNullOrEmpty(_config?.ConstItem.GeoSourceUrl)
|
||||||
|
|
||||||
var geoUrl = string.IsNullOrEmpty(config?.ConstItem.GeoSourceUrl)
|
|
||||||
? Global.GeoUrl
|
? Global.GeoUrl
|
||||||
: config.ConstItem.GeoSourceUrl;
|
: _config.ConstItem.GeoSourceUrl;
|
||||||
|
|
||||||
List<string> files = ["geosite", "geoip"];
|
List<string> files = ["geosite", "geoip"];
|
||||||
foreach (var geoName in files)
|
foreach (var geoName in files)
|
||||||
@@ -345,33 +344,29 @@ public class UpdateService
|
|||||||
var targetPath = Utils.GetBinPath($"{fileName}");
|
var targetPath = Utils.GetBinPath($"{fileName}");
|
||||||
var url = string.Format(geoUrl, geoName);
|
var url = string.Format(geoUrl, geoName);
|
||||||
|
|
||||||
await DownloadGeoFile(url, fileName, targetPath, updateFunc);
|
await DownloadGeoFile(url, fileName, targetPath);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private async Task UpdateOtherFiles(Config config, Func<bool, string, Task> updateFunc)
|
private async Task UpdateOtherFiles()
|
||||||
{
|
{
|
||||||
//If it is not in China area, no update is required
|
//If it is not in China area, no update is required
|
||||||
if (config.ConstItem.GeoSourceUrl.IsNotEmpty())
|
if (_config.ConstItem.GeoSourceUrl.IsNotEmpty())
|
||||||
{
|
{
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
_updateFunc = updateFunc;
|
|
||||||
|
|
||||||
foreach (var url in Global.OtherGeoUrls)
|
foreach (var url in Global.OtherGeoUrls)
|
||||||
{
|
{
|
||||||
var fileName = Path.GetFileName(url);
|
var fileName = Path.GetFileName(url);
|
||||||
var targetPath = Utils.GetBinPath($"{fileName}");
|
var targetPath = Utils.GetBinPath($"{fileName}");
|
||||||
|
|
||||||
await DownloadGeoFile(url, fileName, targetPath, updateFunc);
|
await DownloadGeoFile(url, fileName, targetPath);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private async Task UpdateSrsFileAll(Config config, Func<bool, string, Task> updateFunc)
|
private async Task UpdateSrsFileAll()
|
||||||
{
|
{
|
||||||
_updateFunc = updateFunc;
|
|
||||||
|
|
||||||
var geoipFiles = new List<string>();
|
var geoipFiles = new List<string>();
|
||||||
var geoSiteFiles = new List<string>();
|
var geoSiteFiles = new List<string>();
|
||||||
|
|
||||||
@@ -403,6 +398,7 @@ public class UpdateService
|
|||||||
}
|
}
|
||||||
|
|
||||||
//append dns items TODO
|
//append dns items TODO
|
||||||
|
geoSiteFiles.Add("google");
|
||||||
geoSiteFiles.Add("cn");
|
geoSiteFiles.Add("cn");
|
||||||
geoSiteFiles.Add("geolocation-cn");
|
geoSiteFiles.Add("geolocation-cn");
|
||||||
geoSiteFiles.Add("category-ads-all");
|
geoSiteFiles.Add("category-ads-all");
|
||||||
@@ -414,29 +410,29 @@ public class UpdateService
|
|||||||
}
|
}
|
||||||
foreach (var item in geoipFiles.Distinct())
|
foreach (var item in geoipFiles.Distinct())
|
||||||
{
|
{
|
||||||
await UpdateSrsFile("geoip", item, config, updateFunc);
|
await UpdateSrsFile("geoip", item);
|
||||||
}
|
}
|
||||||
|
|
||||||
foreach (var item in geoSiteFiles.Distinct())
|
foreach (var item in geoSiteFiles.Distinct())
|
||||||
{
|
{
|
||||||
await UpdateSrsFile("geosite", item, config, updateFunc);
|
await UpdateSrsFile("geosite", item);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private async Task UpdateSrsFile(string type, string srsName, Config config, Func<bool, string, Task> updateFunc)
|
private async Task UpdateSrsFile(string type, string srsName)
|
||||||
{
|
{
|
||||||
var srsUrl = string.IsNullOrEmpty(config.ConstItem.SrsSourceUrl)
|
var srsUrl = string.IsNullOrEmpty(_config.ConstItem.SrsSourceUrl)
|
||||||
? Global.SingboxRulesetUrl
|
? Global.SingboxRulesetUrl
|
||||||
: config.ConstItem.SrsSourceUrl;
|
: _config.ConstItem.SrsSourceUrl;
|
||||||
|
|
||||||
var fileName = $"{type}-{srsName}.srs";
|
var fileName = $"{type}-{srsName}.srs";
|
||||||
var targetPath = Path.Combine(Utils.GetBinPath("srss"), fileName);
|
var targetPath = Path.Combine(Utils.GetBinPath("srss"), fileName);
|
||||||
var url = string.Format(srsUrl, type, $"{type}-{srsName}", srsName);
|
var url = string.Format(srsUrl, type, $"{type}-{srsName}", srsName);
|
||||||
|
|
||||||
await DownloadGeoFile(url, fileName, targetPath, updateFunc);
|
await DownloadGeoFile(url, fileName, targetPath);
|
||||||
}
|
}
|
||||||
|
|
||||||
private async Task DownloadGeoFile(string url, string fileName, string targetPath, Func<bool, string, Task> updateFunc)
|
private async Task DownloadGeoFile(string url, string fileName, string targetPath)
|
||||||
{
|
{
|
||||||
var tmpFileName = Utils.GetTempPath(Utils.GetGuid());
|
var tmpFileName = Utils.GetTempPath(Utils.GetGuid());
|
||||||
|
|
||||||
@@ -445,7 +441,7 @@ public class UpdateService
|
|||||||
{
|
{
|
||||||
if (args.Success)
|
if (args.Success)
|
||||||
{
|
{
|
||||||
UpdateFunc(false, string.Format(ResUI.MsgDownloadGeoFileSuccessfully, fileName));
|
_ = UpdateFunc(false, string.Format(ResUI.MsgDownloadGeoFileSuccessfully, fileName));
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
@@ -459,17 +455,17 @@ public class UpdateService
|
|||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
UpdateFunc(false, ex.Message);
|
_ = UpdateFunc(false, ex.Message);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
UpdateFunc(false, args.Msg);
|
_ = UpdateFunc(false, args.Msg);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
downloadHandle.Error += (sender2, args) =>
|
downloadHandle.Error += (sender2, args) =>
|
||||||
{
|
{
|
||||||
UpdateFunc(false, args.GetException().Message);
|
_ = UpdateFunc(false, args.GetException().Message);
|
||||||
};
|
};
|
||||||
|
|
||||||
await downloadHandle.DownloadFileAsync(url, tmpFileName, true, _timeout);
|
await downloadHandle.DownloadFileAsync(url, tmpFileName, true, _timeout);
|
||||||
|
|||||||
171
v2rayN/ServiceLib/Services/WindowsJobService.cs
Normal file
171
v2rayN/ServiceLib/Services/WindowsJobService.cs
Normal file
@@ -0,0 +1,171 @@
|
|||||||
|
namespace ServiceLib.Services;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// http://stackoverflow.com/questions/6266820/working-example-of-createjobobject-setinformationjobobject-pinvoke-in-net
|
||||||
|
/// </summary>
|
||||||
|
public sealed class WindowsJobService : IDisposable
|
||||||
|
{
|
||||||
|
private nint handle = nint.Zero;
|
||||||
|
|
||||||
|
public WindowsJobService()
|
||||||
|
{
|
||||||
|
handle = CreateJobObject(nint.Zero, null);
|
||||||
|
var extendedInfoPtr = nint.Zero;
|
||||||
|
var info = new JOBOBJECT_BASIC_LIMIT_INFORMATION
|
||||||
|
{
|
||||||
|
LimitFlags = 0x2000
|
||||||
|
};
|
||||||
|
|
||||||
|
var extendedInfo = new JOBOBJECT_EXTENDED_LIMIT_INFORMATION
|
||||||
|
{
|
||||||
|
BasicLimitInformation = info
|
||||||
|
};
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var length = Marshal.SizeOf(typeof(JOBOBJECT_EXTENDED_LIMIT_INFORMATION));
|
||||||
|
extendedInfoPtr = Marshal.AllocHGlobal(length);
|
||||||
|
Marshal.StructureToPtr(extendedInfo, extendedInfoPtr, false);
|
||||||
|
|
||||||
|
if (!SetInformationJobObject(handle, JobObjectInfoType.ExtendedLimitInformation, extendedInfoPtr,
|
||||||
|
(uint)length))
|
||||||
|
{
|
||||||
|
throw new Exception(string.Format("Unable to set information. Error: {0}",
|
||||||
|
Marshal.GetLastWin32Error()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
finally
|
||||||
|
{
|
||||||
|
if (extendedInfoPtr != nint.Zero)
|
||||||
|
{
|
||||||
|
Marshal.FreeHGlobal(extendedInfoPtr);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool AddProcess(nint processHandle)
|
||||||
|
{
|
||||||
|
var succ = AssignProcessToJobObject(handle, processHandle);
|
||||||
|
|
||||||
|
if (!succ)
|
||||||
|
{
|
||||||
|
Logging.SaveLog("Failed to call AssignProcessToJobObject! GetLastError=" + Marshal.GetLastWin32Error());
|
||||||
|
}
|
||||||
|
|
||||||
|
return succ;
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool AddProcess(int processId)
|
||||||
|
{
|
||||||
|
return AddProcess(Process.GetProcessById(processId).Handle);
|
||||||
|
}
|
||||||
|
|
||||||
|
#region IDisposable
|
||||||
|
|
||||||
|
private bool disposed;
|
||||||
|
|
||||||
|
public void Dispose()
|
||||||
|
{
|
||||||
|
Dispose(true);
|
||||||
|
GC.SuppressFinalize(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void Dispose(bool disposing)
|
||||||
|
{
|
||||||
|
if (disposed)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
disposed = true;
|
||||||
|
|
||||||
|
if (disposing)
|
||||||
|
{
|
||||||
|
// no managed objects to free
|
||||||
|
}
|
||||||
|
|
||||||
|
if (handle != nint.Zero)
|
||||||
|
{
|
||||||
|
CloseHandle(handle);
|
||||||
|
handle = nint.Zero;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
~WindowsJobService()
|
||||||
|
{
|
||||||
|
Dispose(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
#endregion IDisposable
|
||||||
|
|
||||||
|
#region Interop
|
||||||
|
|
||||||
|
[DllImport("kernel32.dll", CharSet = CharSet.Unicode)]
|
||||||
|
private static extern nint CreateJobObject(nint a, string? lpName);
|
||||||
|
|
||||||
|
[DllImport("kernel32.dll", SetLastError = true)]
|
||||||
|
private static extern bool SetInformationJobObject(nint hJob, JobObjectInfoType infoType, nint lpJobObjectInfo, uint cbJobObjectInfoLength);
|
||||||
|
|
||||||
|
[DllImport("kernel32.dll", SetLastError = true)]
|
||||||
|
private static extern bool AssignProcessToJobObject(nint job, nint process);
|
||||||
|
|
||||||
|
[DllImport("kernel32.dll", SetLastError = true)]
|
||||||
|
[return: MarshalAs(UnmanagedType.Bool)]
|
||||||
|
private static extern bool CloseHandle(nint hObject);
|
||||||
|
|
||||||
|
#endregion Interop
|
||||||
|
}
|
||||||
|
|
||||||
|
[StructLayout(LayoutKind.Sequential)]
|
||||||
|
internal struct IO_COUNTERS
|
||||||
|
{
|
||||||
|
public ulong ReadOperationCount;
|
||||||
|
public ulong WriteOperationCount;
|
||||||
|
public ulong OtherOperationCount;
|
||||||
|
public ulong ReadTransferCount;
|
||||||
|
public ulong WriteTransferCount;
|
||||||
|
public ulong OtherTransferCount;
|
||||||
|
}
|
||||||
|
|
||||||
|
[StructLayout(LayoutKind.Sequential)]
|
||||||
|
internal struct JOBOBJECT_BASIC_LIMIT_INFORMATION
|
||||||
|
{
|
||||||
|
public long PerProcessUserTimeLimit;
|
||||||
|
public long PerJobUserTimeLimit;
|
||||||
|
public uint LimitFlags;
|
||||||
|
public nuint MinimumWorkingSetSize;
|
||||||
|
public nuint MaximumWorkingSetSize;
|
||||||
|
public uint ActiveProcessLimit;
|
||||||
|
public nuint Affinity;
|
||||||
|
public uint PriorityClass;
|
||||||
|
public uint SchedulingClass;
|
||||||
|
}
|
||||||
|
|
||||||
|
[StructLayout(LayoutKind.Sequential)]
|
||||||
|
public struct SECURITY_ATTRIBUTES
|
||||||
|
{
|
||||||
|
public uint nLength;
|
||||||
|
public nint lpSecurityDescriptor;
|
||||||
|
public int bInheritHandle;
|
||||||
|
}
|
||||||
|
|
||||||
|
[StructLayout(LayoutKind.Sequential)]
|
||||||
|
internal struct JOBOBJECT_EXTENDED_LIMIT_INFORMATION
|
||||||
|
{
|
||||||
|
public JOBOBJECT_BASIC_LIMIT_INFORMATION BasicLimitInformation;
|
||||||
|
public IO_COUNTERS IoInfo;
|
||||||
|
public nuint ProcessMemoryLimit;
|
||||||
|
public nuint JobMemoryLimit;
|
||||||
|
public nuint PeakProcessMemoryUsed;
|
||||||
|
public nuint PeakJobMemoryUsed;
|
||||||
|
}
|
||||||
|
|
||||||
|
public enum JobObjectInfoType
|
||||||
|
{
|
||||||
|
AssociateCompletionPortInformation = 7,
|
||||||
|
BasicLimitInformation = 2,
|
||||||
|
BasicUIRestrictions = 4,
|
||||||
|
EndOfJobTimeInformation = 6,
|
||||||
|
ExtendedLimitInformation = 9,
|
||||||
|
SecurityLimitInformation = 5,
|
||||||
|
GroupInformation = 11
|
||||||
|
}
|
||||||
@@ -17,6 +17,14 @@ public class AddGroupServerViewModel : MyReactiveObject
|
|||||||
[Reactive]
|
[Reactive]
|
||||||
public string? PolicyGroupType { get; set; }
|
public string? PolicyGroupType { get; set; }
|
||||||
|
|
||||||
|
[Reactive]
|
||||||
|
public SubItem? SelectedSubItem { get; set; }
|
||||||
|
|
||||||
|
[Reactive]
|
||||||
|
public string? Filter { get; set; }
|
||||||
|
|
||||||
|
public IObservableCollection<SubItem> SubItems { get; } = new ObservableCollectionExtended<SubItem>();
|
||||||
|
|
||||||
public IObservableCollection<ProfileItem> ChildItemsObs { get; } = new ObservableCollectionExtended<ProfileItem>();
|
public IObservableCollection<ProfileItem> ChildItemsObs { get; } = new ObservableCollectionExtended<ProfileItem>();
|
||||||
|
|
||||||
//public ReactiveCommand<Unit, Unit> AddCmd { get; }
|
//public ReactiveCommand<Unit, Unit> AddCmd { get; }
|
||||||
@@ -64,10 +72,14 @@ public class AddGroupServerViewModel : MyReactiveObject
|
|||||||
});
|
});
|
||||||
|
|
||||||
SelectedSource = profileItem.IndexId.IsNullOrEmpty() ? profileItem : JsonUtils.DeepCopy(profileItem);
|
SelectedSource = profileItem.IndexId.IsNullOrEmpty() ? profileItem : JsonUtils.DeepCopy(profileItem);
|
||||||
|
|
||||||
CoreType = (SelectedSource?.CoreType ?? ECoreType.Xray).ToString();
|
CoreType = (SelectedSource?.CoreType ?? ECoreType.Xray).ToString();
|
||||||
|
|
||||||
ProfileGroupItemManager.Instance.TryGet(profileItem.IndexId, out var profileGroup);
|
_ = Init();
|
||||||
|
}
|
||||||
|
|
||||||
|
public async Task Init()
|
||||||
|
{
|
||||||
|
ProfileGroupItemManager.Instance.TryGet(SelectedSource.IndexId, out var profileGroup);
|
||||||
PolicyGroupType = (profileGroup?.MultipleLoad ?? EMultipleLoad.LeastPing) switch
|
PolicyGroupType = (profileGroup?.MultipleLoad ?? EMultipleLoad.LeastPing) switch
|
||||||
{
|
{
|
||||||
EMultipleLoad.LeastPing => ResUI.TbLeastPing,
|
EMultipleLoad.LeastPing => ResUI.TbLeastPing,
|
||||||
@@ -78,15 +90,16 @@ public class AddGroupServerViewModel : MyReactiveObject
|
|||||||
_ => ResUI.TbLeastPing,
|
_ => ResUI.TbLeastPing,
|
||||||
};
|
};
|
||||||
|
|
||||||
_ = Init();
|
var subs = await AppManager.Instance.SubItems();
|
||||||
}
|
subs.Add(new SubItem());
|
||||||
|
SubItems.AddRange(subs);
|
||||||
|
SelectedSubItem = SubItems.Where(s => s.Id == profileGroup?.SubChildItems).FirstOrDefault();
|
||||||
|
Filter = profileGroup?.Filter;
|
||||||
|
|
||||||
public async Task Init()
|
|
||||||
{
|
|
||||||
var childItemMulti = ProfileGroupItemManager.Instance.GetOrCreateAndMarkDirty(SelectedSource?.IndexId);
|
var childItemMulti = ProfileGroupItemManager.Instance.GetOrCreateAndMarkDirty(SelectedSource?.IndexId);
|
||||||
if (childItemMulti != null)
|
if (childItemMulti != null)
|
||||||
{
|
{
|
||||||
var childIndexIds = childItemMulti.ChildItems.IsNullOrEmpty() ? new List<string>() : Utils.String2List(childItemMulti.ChildItems);
|
var childIndexIds = Utils.String2List(childItemMulti.ChildItems) ?? [];
|
||||||
foreach (var item in childIndexIds)
|
foreach (var item in childIndexIds)
|
||||||
{
|
{
|
||||||
var child = await AppManager.Instance.GetProfileItem(item);
|
var child = await AppManager.Instance.GetProfileItem(item);
|
||||||
@@ -181,7 +194,7 @@ public class AddGroupServerViewModel : MyReactiveObject
|
|||||||
NoticeManager.Instance.Enqueue(ResUI.PleaseFillRemarks);
|
NoticeManager.Instance.Enqueue(ResUI.PleaseFillRemarks);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (ChildItemsObs.Count == 0)
|
if (ChildItemsObs.Count == 0 && SelectedSubItem?.Id.IsNullOrEmpty() == true)
|
||||||
{
|
{
|
||||||
NoticeManager.Instance.Enqueue(ResUI.PleaseAddAtLeastOneServer);
|
NoticeManager.Instance.Enqueue(ResUI.PleaseAddAtLeastOneServer);
|
||||||
return;
|
return;
|
||||||
@@ -213,6 +226,9 @@ public class AddGroupServerViewModel : MyReactiveObject
|
|||||||
_ => EMultipleLoad.LeastPing,
|
_ => EMultipleLoad.LeastPing,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
profileGroup.SubChildItems = SelectedSubItem?.Id;
|
||||||
|
profileGroup.Filter = Filter;
|
||||||
|
|
||||||
var hasCycle = ProfileGroupItemManager.HasCycle(profileGroup.IndexId);
|
var hasCycle = ProfileGroupItemManager.HasCycle(profileGroup.IndexId);
|
||||||
if (hasCycle)
|
if (hasCycle)
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -8,6 +8,14 @@ public class AddServerViewModel : MyReactiveObject
|
|||||||
[Reactive]
|
[Reactive]
|
||||||
public string? CoreType { get; set; }
|
public string? CoreType { get; set; }
|
||||||
|
|
||||||
|
[Reactive]
|
||||||
|
public string Cert { get; set; }
|
||||||
|
|
||||||
|
[Reactive]
|
||||||
|
public string CertTip { get; set; }
|
||||||
|
|
||||||
|
public ReactiveCommand<Unit, Unit> FetchCertCmd { get; }
|
||||||
|
public ReactiveCommand<Unit, Unit> FetchCertChainCmd { get; }
|
||||||
public ReactiveCommand<Unit, Unit> SaveCmd { get; }
|
public ReactiveCommand<Unit, Unit> SaveCmd { get; }
|
||||||
|
|
||||||
public AddServerViewModel(ProfileItem profileItem, Func<EViewAction, object?, Task<bool>>? updateView)
|
public AddServerViewModel(ProfileItem profileItem, Func<EViewAction, object?, Task<bool>>? updateView)
|
||||||
@@ -15,11 +23,22 @@ public class AddServerViewModel : MyReactiveObject
|
|||||||
_config = AppManager.Instance.Config;
|
_config = AppManager.Instance.Config;
|
||||||
_updateView = updateView;
|
_updateView = updateView;
|
||||||
|
|
||||||
|
FetchCertCmd = ReactiveCommand.CreateFromTask(async () =>
|
||||||
|
{
|
||||||
|
await FetchCert();
|
||||||
|
});
|
||||||
|
FetchCertChainCmd = ReactiveCommand.CreateFromTask(async () =>
|
||||||
|
{
|
||||||
|
await FetchCertChain();
|
||||||
|
});
|
||||||
SaveCmd = ReactiveCommand.CreateFromTask(async () =>
|
SaveCmd = ReactiveCommand.CreateFromTask(async () =>
|
||||||
{
|
{
|
||||||
await SaveServerAsync();
|
await SaveServerAsync();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
this.WhenAnyValue(x => x.Cert)
|
||||||
|
.Subscribe(_ => UpdateCertTip());
|
||||||
|
|
||||||
if (profileItem.IndexId.IsNullOrEmpty())
|
if (profileItem.IndexId.IsNullOrEmpty())
|
||||||
{
|
{
|
||||||
profileItem.Network = Global.DefaultNetwork;
|
profileItem.Network = Global.DefaultNetwork;
|
||||||
@@ -33,6 +52,7 @@ public class AddServerViewModel : MyReactiveObject
|
|||||||
SelectedSource = JsonUtils.DeepCopy(profileItem);
|
SelectedSource = JsonUtils.DeepCopy(profileItem);
|
||||||
}
|
}
|
||||||
CoreType = SelectedSource?.CoreType?.ToString();
|
CoreType = SelectedSource?.CoreType?.ToString();
|
||||||
|
Cert = SelectedSource?.Cert?.ToString() ?? string.Empty;
|
||||||
}
|
}
|
||||||
|
|
||||||
private async Task SaveServerAsync()
|
private async Task SaveServerAsync()
|
||||||
@@ -77,6 +97,7 @@ public class AddServerViewModel : MyReactiveObject
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
SelectedSource.CoreType = CoreType.IsNullOrEmpty() ? null : (ECoreType)Enum.Parse(typeof(ECoreType), CoreType);
|
SelectedSource.CoreType = CoreType.IsNullOrEmpty() ? null : (ECoreType)Enum.Parse(typeof(ECoreType), CoreType);
|
||||||
|
SelectedSource.Cert = Cert.IsNullOrEmpty() ? null : Cert;
|
||||||
|
|
||||||
if (await ConfigHandler.AddServer(_config, SelectedSource) == 0)
|
if (await ConfigHandler.AddServer(_config, SelectedSource) == 0)
|
||||||
{
|
{
|
||||||
@@ -88,4 +109,72 @@ public class AddServerViewModel : MyReactiveObject
|
|||||||
NoticeManager.Instance.Enqueue(ResUI.OperationFailed);
|
NoticeManager.Instance.Enqueue(ResUI.OperationFailed);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void UpdateCertTip(string? errorMessage = null)
|
||||||
|
{
|
||||||
|
CertTip = errorMessage.IsNullOrEmpty()
|
||||||
|
? (Cert.IsNullOrEmpty() ? ResUI.CertNotSet : ResUI.CertSet)
|
||||||
|
: errorMessage;
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task FetchCert()
|
||||||
|
{
|
||||||
|
if (SelectedSource.StreamSecurity != Global.StreamSecurity)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
var domain = SelectedSource.Address;
|
||||||
|
var serverName = SelectedSource.Sni;
|
||||||
|
if (serverName.IsNullOrEmpty())
|
||||||
|
{
|
||||||
|
serverName = SelectedSource.RequestHost;
|
||||||
|
}
|
||||||
|
if (serverName.IsNullOrEmpty())
|
||||||
|
{
|
||||||
|
serverName = SelectedSource.Address;
|
||||||
|
}
|
||||||
|
if (!Utils.IsDomain(serverName))
|
||||||
|
{
|
||||||
|
UpdateCertTip(ResUI.ServerNameMustBeValidDomain);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (SelectedSource.Port > 0)
|
||||||
|
{
|
||||||
|
domain += $":{SelectedSource.Port}";
|
||||||
|
}
|
||||||
|
string certError;
|
||||||
|
(Cert, certError) = await CertPemManager.Instance.GetCertPemAsync(domain, serverName);
|
||||||
|
UpdateCertTip(certError);
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task FetchCertChain()
|
||||||
|
{
|
||||||
|
if (SelectedSource.StreamSecurity != Global.StreamSecurity)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
var domain = SelectedSource.Address;
|
||||||
|
var serverName = SelectedSource.Sni;
|
||||||
|
if (serverName.IsNullOrEmpty())
|
||||||
|
{
|
||||||
|
serverName = SelectedSource.RequestHost;
|
||||||
|
}
|
||||||
|
if (serverName.IsNullOrEmpty())
|
||||||
|
{
|
||||||
|
serverName = SelectedSource.Address;
|
||||||
|
}
|
||||||
|
if (!Utils.IsDomain(serverName))
|
||||||
|
{
|
||||||
|
UpdateCertTip(ResUI.ServerNameMustBeValidDomain);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (SelectedSource.Port > 0)
|
||||||
|
{
|
||||||
|
domain += $":{SelectedSource.Port}";
|
||||||
|
}
|
||||||
|
string certError;
|
||||||
|
(var certs, certError) = await CertPemManager.Instance.GetCertChainPemAsync(domain, serverName);
|
||||||
|
Cert = CertPemManager.ConcatenatePemChain(certs);
|
||||||
|
UpdateCertTip(certError);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -119,7 +119,7 @@ public class BackupAndRestoreViewModel : MyReactiveObject
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
//check
|
//check
|
||||||
var lstFiles = FileManager.GetFilesFromZip(fileName);
|
var lstFiles = FileUtils.GetFilesFromZip(fileName);
|
||||||
if (lstFiles is null || !lstFiles.Any(t => t.Contains(_guiConfigs)))
|
if (lstFiles is null || !lstFiles.Any(t => t.Contains(_guiConfigs)))
|
||||||
{
|
{
|
||||||
DisplayOperationMsg(ResUI.LocalRestoreInvalidZipTips);
|
DisplayOperationMsg(ResUI.LocalRestoreInvalidZipTips);
|
||||||
@@ -135,7 +135,7 @@ public class BackupAndRestoreViewModel : MyReactiveObject
|
|||||||
await SQLiteHelper.Instance.DisposeDbConnectionAsync();
|
await SQLiteHelper.Instance.DisposeDbConnectionAsync();
|
||||||
|
|
||||||
var toPath = Utils.GetConfigPath();
|
var toPath = Utils.GetConfigPath();
|
||||||
FileManager.ZipExtractToFile(fileName, toPath, "");
|
FileUtils.ZipExtractToFile(fileName, toPath, "");
|
||||||
|
|
||||||
if (Utils.IsWindows())
|
if (Utils.IsWindows())
|
||||||
{
|
{
|
||||||
@@ -167,8 +167,8 @@ public class BackupAndRestoreViewModel : MyReactiveObject
|
|||||||
var configDirZipTemp = Utils.GetTempPath($"v2rayN_{DateTime.Now:yyyyMMddHHmmss}");
|
var configDirZipTemp = Utils.GetTempPath($"v2rayN_{DateTime.Now:yyyyMMddHHmmss}");
|
||||||
var configDirTemp = Path.Combine(configDirZipTemp, _guiConfigs);
|
var configDirTemp = Path.Combine(configDirZipTemp, _guiConfigs);
|
||||||
|
|
||||||
FileManager.CopyDirectory(configDir, configDirTemp, false, true, "");
|
FileUtils.CopyDirectory(configDir, configDirTemp, false, true, "");
|
||||||
var ret = FileManager.CreateFromDirectory(configDirZipTemp, fileName);
|
var ret = FileUtils.CreateFromDirectory(configDirZipTemp, fileName);
|
||||||
Directory.Delete(configDirZipTemp, true);
|
Directory.Delete(configDirZipTemp, true);
|
||||||
return await Task.FromResult(ret);
|
return await Task.FromResult(ret);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -148,7 +148,7 @@ public class CheckUpdateViewModel : MyReactiveObject
|
|||||||
UpdatedPlusPlus(_geo, "");
|
UpdatedPlusPlus(_geo, "");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
await new UpdateService().UpdateGeoFileAll(_config, _updateUI)
|
await new UpdateService(_config, _updateUI).UpdateGeoFileAll()
|
||||||
.ContinueWith(t => UpdatedPlusPlus(_geo, ""));
|
.ContinueWith(t => UpdatedPlusPlus(_geo, ""));
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -163,7 +163,7 @@ public class CheckUpdateViewModel : MyReactiveObject
|
|||||||
UpdatedPlusPlus(_v2rayN, msg);
|
UpdatedPlusPlus(_v2rayN, msg);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
await new UpdateService().CheckUpdateGuiN(_config, _updateUI, preRelease)
|
await new UpdateService(_config, _updateUI).CheckUpdateGuiN(preRelease)
|
||||||
.ContinueWith(t => UpdatedPlusPlus(_v2rayN, ""));
|
.ContinueWith(t => UpdatedPlusPlus(_v2rayN, ""));
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -180,7 +180,7 @@ public class CheckUpdateViewModel : MyReactiveObject
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
var type = (ECoreType)Enum.Parse(typeof(ECoreType), model.CoreType);
|
var type = (ECoreType)Enum.Parse(typeof(ECoreType), model.CoreType);
|
||||||
await new UpdateService().CheckUpdateCore(type, _config, _updateUI, preRelease)
|
await new UpdateService(_config, _updateUI).CheckUpdateCore(type, preRelease)
|
||||||
.ContinueWith(t => UpdatedPlusPlus(model.CoreType, ""));
|
.ContinueWith(t => UpdatedPlusPlus(model.CoreType, ""));
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -209,6 +209,7 @@ public class CheckUpdateViewModel : MyReactiveObject
|
|||||||
_ = UpdateFinishedResult(blReload);
|
_ = UpdateFinishedResult(blReload);
|
||||||
return Disposable.Empty;
|
return Disposable.Empty;
|
||||||
});
|
});
|
||||||
|
await Task.CompletedTask;
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task UpdateFinishedResult(bool blReload)
|
public async Task UpdateFinishedResult(bool blReload)
|
||||||
@@ -270,24 +271,24 @@ public class CheckUpdateViewModel : MyReactiveObject
|
|||||||
|
|
||||||
if (fileName.Contains(".tar.gz"))
|
if (fileName.Contains(".tar.gz"))
|
||||||
{
|
{
|
||||||
FileManager.DecompressTarFile(fileName, toPath);
|
FileUtils.DecompressTarFile(fileName, toPath);
|
||||||
var dir = new DirectoryInfo(toPath);
|
var dir = new DirectoryInfo(toPath);
|
||||||
if (dir.Exists)
|
if (dir.Exists)
|
||||||
{
|
{
|
||||||
foreach (var subDir in dir.GetDirectories())
|
foreach (var subDir in dir.GetDirectories())
|
||||||
{
|
{
|
||||||
FileManager.CopyDirectory(subDir.FullName, toPath, false, true);
|
FileUtils.CopyDirectory(subDir.FullName, toPath, false, true);
|
||||||
subDir.Delete(true);
|
subDir.Delete(true);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if (fileName.Contains(".gz"))
|
else if (fileName.Contains(".gz"))
|
||||||
{
|
{
|
||||||
FileManager.DecompressFile(fileName, toPath, item.CoreType);
|
FileUtils.DecompressFile(fileName, toPath, item.CoreType);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
FileManager.ZipExtractToFile(fileName, toPath, "geo");
|
FileUtils.ZipExtractToFile(fileName, toPath, "geo");
|
||||||
}
|
}
|
||||||
|
|
||||||
if (Utils.IsNonWindows())
|
if (Utils.IsNonWindows())
|
||||||
@@ -321,6 +322,7 @@ public class CheckUpdateViewModel : MyReactiveObject
|
|||||||
_ = UpdateViewResult(model);
|
_ = UpdateViewResult(model);
|
||||||
return Disposable.Empty;
|
return Disposable.Empty;
|
||||||
});
|
});
|
||||||
|
await Task.CompletedTask;
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task UpdateViewResult(CheckUpdateModel model)
|
public async Task UpdateViewResult(CheckUpdateModel model)
|
||||||
@@ -331,5 +333,6 @@ public class CheckUpdateViewModel : MyReactiveObject
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
found.Remarks = model.Remarks;
|
found.Remarks = model.Remarks;
|
||||||
|
await Task.CompletedTask;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -96,6 +96,7 @@ public class ClashConnectionsViewModel : MyReactiveObject
|
|||||||
}
|
}
|
||||||
|
|
||||||
ConnectionItems.AddRange(lstModel);
|
ConnectionItems.AddRange(lstModel);
|
||||||
|
await Task.CompletedTask;
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task ClashConnectionClose(bool all)
|
public async Task ClashConnectionClose(bool all)
|
||||||
|
|||||||
@@ -211,7 +211,7 @@ public class ClashProxiesViewModel : MyReactiveObject
|
|||||||
}
|
}
|
||||||
|
|
||||||
//from api
|
//from api
|
||||||
foreach (KeyValuePair<string, ProxiesItem> kv in _proxies)
|
foreach (var kv in _proxies)
|
||||||
{
|
{
|
||||||
if (!Global.allowSelectType.Contains(kv.Value.type.ToLower()))
|
if (!Global.allowSelectType.Contains(kv.Value.type.ToLower()))
|
||||||
{
|
{
|
||||||
@@ -245,6 +245,7 @@ public class ClashProxiesViewModel : MyReactiveObject
|
|||||||
{
|
{
|
||||||
SelectedGroup = new();
|
SelectedGroup = new();
|
||||||
}
|
}
|
||||||
|
await Task.CompletedTask;
|
||||||
}
|
}
|
||||||
|
|
||||||
private void RefreshProxyDetails(bool c)
|
private void RefreshProxyDetails(bool c)
|
||||||
@@ -319,7 +320,7 @@ public class ClashProxiesViewModel : MyReactiveObject
|
|||||||
//from providers
|
//from providers
|
||||||
if (_providers != null)
|
if (_providers != null)
|
||||||
{
|
{
|
||||||
foreach (KeyValuePair<string, ProvidersItem> kv in _providers)
|
foreach (var kv in _providers)
|
||||||
{
|
{
|
||||||
if (Global.proxyVehicleType.Contains(kv.Value.vehicleType.ToLower()))
|
if (Global.proxyVehicleType.Contains(kv.Value.vehicleType.ToLower()))
|
||||||
{
|
{
|
||||||
@@ -391,6 +392,7 @@ public class ClashProxiesViewModel : MyReactiveObject
|
|||||||
_ = ProxiesDelayTestResult(model);
|
_ = ProxiesDelayTestResult(model);
|
||||||
return Disposable.Empty;
|
return Disposable.Empty;
|
||||||
});
|
});
|
||||||
|
await Task.CompletedTask;
|
||||||
});
|
});
|
||||||
await Task.CompletedTask;
|
await Task.CompletedTask;
|
||||||
}
|
}
|
||||||
@@ -419,6 +421,7 @@ public class ClashProxiesViewModel : MyReactiveObject
|
|||||||
detail.Delay = _delayTimeout;
|
detail.Delay = _delayTimeout;
|
||||||
detail.DelayName = string.Empty;
|
detail.DelayName = string.Empty;
|
||||||
}
|
}
|
||||||
|
await Task.CompletedTask;
|
||||||
}
|
}
|
||||||
|
|
||||||
#endregion proxy function
|
#endregion proxy function
|
||||||
|
|||||||
@@ -66,10 +66,14 @@ public class FullConfigTemplateViewModel : MyReactiveObject
|
|||||||
private async Task SaveSettingAsync()
|
private async Task SaveSettingAsync()
|
||||||
{
|
{
|
||||||
if (!await SaveXrayConfigAsync())
|
if (!await SaveXrayConfigAsync())
|
||||||
|
{
|
||||||
return;
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if (!await SaveSingboxConfigAsync())
|
if (!await SaveSingboxConfigAsync())
|
||||||
|
{
|
||||||
return;
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
NoticeManager.Instance.Enqueue(ResUI.OperationSuccess);
|
NoticeManager.Instance.Enqueue(ResUI.OperationSuccess);
|
||||||
_ = _updateView?.Invoke(EViewAction.CloseWindow, null);
|
_ = _updateView?.Invoke(EViewAction.CloseWindow, null);
|
||||||
|
|||||||
@@ -62,9 +62,9 @@ public class MainWindowViewModel : MyReactiveObject
|
|||||||
[Reactive]
|
[Reactive]
|
||||||
public int TabMainSelectedIndex { get; set; }
|
public int TabMainSelectedIndex { get; set; }
|
||||||
|
|
||||||
#endregion Menu
|
[Reactive] public bool BlIsWindows { get; set; }
|
||||||
|
|
||||||
private bool _hasNextReloadJob = false;
|
#endregion Menu
|
||||||
|
|
||||||
#region Init
|
#region Init
|
||||||
|
|
||||||
@@ -72,61 +72,62 @@ public class MainWindowViewModel : MyReactiveObject
|
|||||||
{
|
{
|
||||||
_config = AppManager.Instance.Config;
|
_config = AppManager.Instance.Config;
|
||||||
_updateView = updateView;
|
_updateView = updateView;
|
||||||
|
BlIsWindows = Utils.IsWindows();
|
||||||
|
|
||||||
#region WhenAnyValue && ReactiveCommand
|
#region WhenAnyValue && ReactiveCommand
|
||||||
|
|
||||||
//servers
|
//servers
|
||||||
AddVmessServerCmd = ReactiveCommand.CreateFromTask(async () =>
|
AddVmessServerCmd = ReactiveCommand.CreateFromTask(async () =>
|
||||||
{
|
{
|
||||||
await AddServerAsync(true, EConfigType.VMess);
|
await AddServerAsync(EConfigType.VMess);
|
||||||
});
|
});
|
||||||
AddVlessServerCmd = ReactiveCommand.CreateFromTask(async () =>
|
AddVlessServerCmd = ReactiveCommand.CreateFromTask(async () =>
|
||||||
{
|
{
|
||||||
await AddServerAsync(true, EConfigType.VLESS);
|
await AddServerAsync(EConfigType.VLESS);
|
||||||
});
|
});
|
||||||
AddShadowsocksServerCmd = ReactiveCommand.CreateFromTask(async () =>
|
AddShadowsocksServerCmd = ReactiveCommand.CreateFromTask(async () =>
|
||||||
{
|
{
|
||||||
await AddServerAsync(true, EConfigType.Shadowsocks);
|
await AddServerAsync(EConfigType.Shadowsocks);
|
||||||
});
|
});
|
||||||
AddSocksServerCmd = ReactiveCommand.CreateFromTask(async () =>
|
AddSocksServerCmd = ReactiveCommand.CreateFromTask(async () =>
|
||||||
{
|
{
|
||||||
await AddServerAsync(true, EConfigType.SOCKS);
|
await AddServerAsync(EConfigType.SOCKS);
|
||||||
});
|
});
|
||||||
AddHttpServerCmd = ReactiveCommand.CreateFromTask(async () =>
|
AddHttpServerCmd = ReactiveCommand.CreateFromTask(async () =>
|
||||||
{
|
{
|
||||||
await AddServerAsync(true, EConfigType.HTTP);
|
await AddServerAsync(EConfigType.HTTP);
|
||||||
});
|
});
|
||||||
AddTrojanServerCmd = ReactiveCommand.CreateFromTask(async () =>
|
AddTrojanServerCmd = ReactiveCommand.CreateFromTask(async () =>
|
||||||
{
|
{
|
||||||
await AddServerAsync(true, EConfigType.Trojan);
|
await AddServerAsync(EConfigType.Trojan);
|
||||||
});
|
});
|
||||||
AddHysteria2ServerCmd = ReactiveCommand.CreateFromTask(async () =>
|
AddHysteria2ServerCmd = ReactiveCommand.CreateFromTask(async () =>
|
||||||
{
|
{
|
||||||
await AddServerAsync(true, EConfigType.Hysteria2);
|
await AddServerAsync(EConfigType.Hysteria2);
|
||||||
});
|
});
|
||||||
AddTuicServerCmd = ReactiveCommand.CreateFromTask(async () =>
|
AddTuicServerCmd = ReactiveCommand.CreateFromTask(async () =>
|
||||||
{
|
{
|
||||||
await AddServerAsync(true, EConfigType.TUIC);
|
await AddServerAsync(EConfigType.TUIC);
|
||||||
});
|
});
|
||||||
AddWireguardServerCmd = ReactiveCommand.CreateFromTask(async () =>
|
AddWireguardServerCmd = ReactiveCommand.CreateFromTask(async () =>
|
||||||
{
|
{
|
||||||
await AddServerAsync(true, EConfigType.WireGuard);
|
await AddServerAsync(EConfigType.WireGuard);
|
||||||
});
|
});
|
||||||
AddAnytlsServerCmd = ReactiveCommand.CreateFromTask(async () =>
|
AddAnytlsServerCmd = ReactiveCommand.CreateFromTask(async () =>
|
||||||
{
|
{
|
||||||
await AddServerAsync(true, EConfigType.Anytls);
|
await AddServerAsync(EConfigType.Anytls);
|
||||||
});
|
});
|
||||||
AddCustomServerCmd = ReactiveCommand.CreateFromTask(async () =>
|
AddCustomServerCmd = ReactiveCommand.CreateFromTask(async () =>
|
||||||
{
|
{
|
||||||
await AddServerAsync(true, EConfigType.Custom);
|
await AddServerAsync(EConfigType.Custom);
|
||||||
});
|
});
|
||||||
AddPolicyGroupServerCmd = ReactiveCommand.CreateFromTask(async () =>
|
AddPolicyGroupServerCmd = ReactiveCommand.CreateFromTask(async () =>
|
||||||
{
|
{
|
||||||
await AddServerAsync(true, EConfigType.PolicyGroup);
|
await AddServerAsync(EConfigType.PolicyGroup);
|
||||||
});
|
});
|
||||||
AddProxyChainServerCmd = ReactiveCommand.CreateFromTask(async () =>
|
AddProxyChainServerCmd = ReactiveCommand.CreateFromTask(async () =>
|
||||||
{
|
{
|
||||||
await AddServerAsync(true, EConfigType.ProxyChain);
|
await AddServerAsync(EConfigType.ProxyChain);
|
||||||
});
|
});
|
||||||
AddServerViaClipboardCmd = ReactiveCommand.CreateFromTask(async () =>
|
AddServerViaClipboardCmd = ReactiveCommand.CreateFromTask(async () =>
|
||||||
{
|
{
|
||||||
@@ -268,7 +269,6 @@ public class MainWindowViewModel : MyReactiveObject
|
|||||||
}
|
}
|
||||||
await RefreshServers();
|
await RefreshServers();
|
||||||
|
|
||||||
BlReloadEnabled = true;
|
|
||||||
await Reload();
|
await Reload();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -283,6 +283,7 @@ public class MainWindowViewModel : MyReactiveObject
|
|||||||
{
|
{
|
||||||
NoticeManager.Instance.Enqueue(msg);
|
NoticeManager.Instance.Enqueue(msg);
|
||||||
}
|
}
|
||||||
|
await Task.CompletedTask;
|
||||||
}
|
}
|
||||||
|
|
||||||
private async Task UpdateTaskHandler(bool success, string msg)
|
private async Task UpdateTaskHandler(bool success, string msg)
|
||||||
@@ -310,6 +311,7 @@ public class MainWindowViewModel : MyReactiveObject
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
AppEvents.DispatcherStatisticsRequested.Publish(update);
|
AppEvents.DispatcherStatisticsRequested.Publish(update);
|
||||||
|
await Task.CompletedTask;
|
||||||
}
|
}
|
||||||
|
|
||||||
#endregion Actions
|
#endregion Actions
|
||||||
@@ -332,7 +334,7 @@ public class MainWindowViewModel : MyReactiveObject
|
|||||||
|
|
||||||
#region Add Servers
|
#region Add Servers
|
||||||
|
|
||||||
public async Task AddServerAsync(bool blNew, EConfigType eConfigType)
|
public async Task AddServerAsync(EConfigType eConfigType)
|
||||||
{
|
{
|
||||||
ProfileItem item = new()
|
ProfileItem item = new()
|
||||||
{
|
{
|
||||||
@@ -512,7 +514,7 @@ public class MainWindowViewModel : MyReactiveObject
|
|||||||
{
|
{
|
||||||
ProcUtils.ProcessStart("xdg-open", path);
|
ProcUtils.ProcessStart("xdg-open", path);
|
||||||
}
|
}
|
||||||
else if (Utils.IsOSX())
|
else if (Utils.IsMacOS())
|
||||||
{
|
{
|
||||||
ProcUtils.ProcessStart("open", path);
|
ProcUtils.ProcessStart("open", path);
|
||||||
}
|
}
|
||||||
@@ -523,58 +525,74 @@ public class MainWindowViewModel : MyReactiveObject
|
|||||||
|
|
||||||
#region core job
|
#region core job
|
||||||
|
|
||||||
|
private bool _hasNextReloadJob = false;
|
||||||
|
private readonly SemaphoreSlim _reloadSemaphore = new(1, 1);
|
||||||
|
|
||||||
public async Task Reload()
|
public async Task Reload()
|
||||||
{
|
{
|
||||||
//If there are unfinished reload job, marked with next job.
|
//If there are unfinished reload job, marked with next job.
|
||||||
if (!BlReloadEnabled)
|
if (!await _reloadSemaphore.WaitAsync(0))
|
||||||
{
|
{
|
||||||
_hasNextReloadJob = true;
|
_hasNextReloadJob = true;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
BlReloadEnabled = false;
|
try
|
||||||
|
|
||||||
var msgs = await ActionPrecheckManager.Instance.Check(_config.IndexId);
|
|
||||||
if (msgs.Count > 0)
|
|
||||||
{
|
{
|
||||||
foreach (var msg in msgs)
|
SetReloadEnabled(false);
|
||||||
|
|
||||||
|
var msgs = await ActionPrecheckManager.Instance.Check(_config.IndexId);
|
||||||
|
if (msgs.Count > 0)
|
||||||
{
|
{
|
||||||
NoticeManager.Instance.SendMessage(msg);
|
foreach (var msg in msgs)
|
||||||
|
{
|
||||||
|
NoticeManager.Instance.SendMessage(msg);
|
||||||
|
}
|
||||||
|
NoticeManager.Instance.Enqueue(Utils.List2String(msgs.Take(10).ToList(), true));
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
NoticeManager.Instance.Enqueue(Utils.List2String(msgs.Take(10).ToList(), true));
|
|
||||||
BlReloadEnabled = true;
|
await Task.Run(async () =>
|
||||||
return;
|
{
|
||||||
|
await LoadCore();
|
||||||
|
await SysProxyHandler.UpdateSysProxy(_config, false);
|
||||||
|
await Task.Delay(1000);
|
||||||
|
});
|
||||||
|
AppEvents.TestServerRequested.Publish();
|
||||||
|
|
||||||
|
var showClashUI = _config.IsRunningCore(ECoreType.sing_box);
|
||||||
|
if (showClashUI)
|
||||||
|
{
|
||||||
|
AppEvents.ProxiesReloadRequested.Publish();
|
||||||
|
}
|
||||||
|
|
||||||
|
ReloadResult(showClashUI);
|
||||||
}
|
}
|
||||||
|
finally
|
||||||
await Task.Run(async () =>
|
|
||||||
{
|
{
|
||||||
await LoadCore();
|
SetReloadEnabled(true);
|
||||||
await SysProxyHandler.UpdateSysProxy(_config, false);
|
_reloadSemaphore.Release();
|
||||||
await Task.Delay(1000);
|
//If there is a next reload job, execute it.
|
||||||
});
|
if (_hasNextReloadJob)
|
||||||
AppEvents.TestServerRequested.Publish();
|
{
|
||||||
|
_hasNextReloadJob = false;
|
||||||
var showClashUI = _config.IsRunningCore(ECoreType.sing_box);
|
await Reload();
|
||||||
if (showClashUI)
|
}
|
||||||
{
|
|
||||||
AppEvents.ProxiesReloadRequested.Publish();
|
|
||||||
}
|
|
||||||
|
|
||||||
RxApp.MainThreadScheduler.Schedule(() => ReloadResult(showClashUI));
|
|
||||||
|
|
||||||
BlReloadEnabled = true;
|
|
||||||
if (_hasNextReloadJob)
|
|
||||||
{
|
|
||||||
_hasNextReloadJob = false;
|
|
||||||
await Reload();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void ReloadResult(bool showClashUI)
|
private void ReloadResult(bool showClashUI)
|
||||||
{
|
{
|
||||||
// BlReloadEnabled = true;
|
RxApp.MainThreadScheduler.Schedule(() =>
|
||||||
ShowClashUI = showClashUI;
|
{
|
||||||
TabMainSelectedIndex = showClashUI ? TabMainSelectedIndex : 0;
|
ShowClashUI = showClashUI;
|
||||||
|
TabMainSelectedIndex = showClashUI ? TabMainSelectedIndex : 0;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private void SetReloadEnabled(bool enabled)
|
||||||
|
{
|
||||||
|
RxApp.MainThreadScheduler.Schedule(() => BlReloadEnabled = enabled);
|
||||||
}
|
}
|
||||||
|
|
||||||
private async Task LoadCore()
|
private async Task LoadCore()
|
||||||
@@ -594,7 +612,7 @@ public class MainWindowViewModel : MyReactiveObject
|
|||||||
AppEvents.RoutingsMenuRefreshRequested.Publish();
|
AppEvents.RoutingsMenuRefreshRequested.Publish();
|
||||||
|
|
||||||
await ConfigHandler.SaveConfig(_config);
|
await ConfigHandler.SaveConfig(_config);
|
||||||
await new UpdateService().UpdateGeoFileAll(_config, UpdateTaskHandler);
|
await new UpdateService(_config, UpdateTaskHandler).UpdateGeoFileAll();
|
||||||
await Reload();
|
await Reload();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -50,6 +50,7 @@ public class OptionSettingViewModel : MyReactiveObject
|
|||||||
[Reactive] public bool EnableUpdateSubOnlyRemarksExist { get; set; }
|
[Reactive] public bool EnableUpdateSubOnlyRemarksExist { get; set; }
|
||||||
[Reactive] public bool AutoHideStartup { get; set; }
|
[Reactive] public bool AutoHideStartup { get; set; }
|
||||||
[Reactive] public bool Hide2TrayWhenClose { get; set; }
|
[Reactive] public bool Hide2TrayWhenClose { get; set; }
|
||||||
|
[Reactive] public bool MacOSShowInDock { get; set; }
|
||||||
[Reactive] public bool EnableDragDropSort { get; set; }
|
[Reactive] public bool EnableDragDropSort { get; set; }
|
||||||
[Reactive] public bool DoubleClick2Activate { get; set; }
|
[Reactive] public bool DoubleClick2Activate { get; set; }
|
||||||
[Reactive] public int AutoUpdateInterval { get; set; }
|
[Reactive] public int AutoUpdateInterval { get; set; }
|
||||||
@@ -69,11 +70,22 @@ public class OptionSettingViewModel : MyReactiveObject
|
|||||||
|
|
||||||
#endregion UI
|
#endregion UI
|
||||||
|
|
||||||
|
#region UI visibility
|
||||||
|
|
||||||
|
[Reactive] public bool BlIsWindows { get; set; }
|
||||||
|
[Reactive] public bool BlIsLinux { get; set; }
|
||||||
|
[Reactive] public bool BlIsIsMacOS { get; set; }
|
||||||
|
[Reactive] public bool BlIsNonWindows { get; set; }
|
||||||
|
|
||||||
|
#endregion UI visibility
|
||||||
|
|
||||||
#region System proxy
|
#region System proxy
|
||||||
|
|
||||||
[Reactive] public bool notProxyLocalAddress { get; set; }
|
[Reactive] public bool notProxyLocalAddress { get; set; }
|
||||||
[Reactive] public string systemProxyAdvancedProtocol { get; set; }
|
[Reactive] public string systemProxyAdvancedProtocol { get; set; }
|
||||||
[Reactive] public string systemProxyExceptions { get; set; }
|
[Reactive] public string systemProxyExceptions { get; set; }
|
||||||
|
[Reactive] public string CustomSystemProxyPacPath { get; set; }
|
||||||
|
[Reactive] public string CustomSystemProxyScriptPath { get; set; }
|
||||||
|
|
||||||
#endregion System proxy
|
#endregion System proxy
|
||||||
|
|
||||||
@@ -106,6 +118,10 @@ public class OptionSettingViewModel : MyReactiveObject
|
|||||||
{
|
{
|
||||||
_config = AppManager.Instance.Config;
|
_config = AppManager.Instance.Config;
|
||||||
_updateView = updateView;
|
_updateView = updateView;
|
||||||
|
BlIsWindows = Utils.IsWindows();
|
||||||
|
BlIsLinux = Utils.IsLinux();
|
||||||
|
BlIsIsMacOS = Utils.IsMacOS();
|
||||||
|
BlIsNonWindows = Utils.IsNonWindows();
|
||||||
|
|
||||||
SaveCmd = ReactiveCommand.CreateFromTask(async () =>
|
SaveCmd = ReactiveCommand.CreateFromTask(async () =>
|
||||||
{
|
{
|
||||||
@@ -167,6 +183,7 @@ public class OptionSettingViewModel : MyReactiveObject
|
|||||||
EnableUpdateSubOnlyRemarksExist = _config.UiItem.EnableUpdateSubOnlyRemarksExist;
|
EnableUpdateSubOnlyRemarksExist = _config.UiItem.EnableUpdateSubOnlyRemarksExist;
|
||||||
AutoHideStartup = _config.UiItem.AutoHideStartup;
|
AutoHideStartup = _config.UiItem.AutoHideStartup;
|
||||||
Hide2TrayWhenClose = _config.UiItem.Hide2TrayWhenClose;
|
Hide2TrayWhenClose = _config.UiItem.Hide2TrayWhenClose;
|
||||||
|
MacOSShowInDock = _config.UiItem.MacOSShowInDock;
|
||||||
EnableDragDropSort = _config.UiItem.EnableDragDropSort;
|
EnableDragDropSort = _config.UiItem.EnableDragDropSort;
|
||||||
DoubleClick2Activate = _config.UiItem.DoubleClick2Activate;
|
DoubleClick2Activate = _config.UiItem.DoubleClick2Activate;
|
||||||
AutoUpdateInterval = _config.GuiItem.AutoUpdateInterval;
|
AutoUpdateInterval = _config.GuiItem.AutoUpdateInterval;
|
||||||
@@ -191,6 +208,8 @@ public class OptionSettingViewModel : MyReactiveObject
|
|||||||
notProxyLocalAddress = _config.SystemProxyItem.NotProxyLocalAddress;
|
notProxyLocalAddress = _config.SystemProxyItem.NotProxyLocalAddress;
|
||||||
systemProxyAdvancedProtocol = _config.SystemProxyItem.SystemProxyAdvancedProtocol;
|
systemProxyAdvancedProtocol = _config.SystemProxyItem.SystemProxyAdvancedProtocol;
|
||||||
systemProxyExceptions = _config.SystemProxyItem.SystemProxyExceptions;
|
systemProxyExceptions = _config.SystemProxyItem.SystemProxyExceptions;
|
||||||
|
CustomSystemProxyPacPath = _config.SystemProxyItem.CustomSystemProxyPacPath;
|
||||||
|
CustomSystemProxyScriptPath = _config.SystemProxyItem.CustomSystemProxyScriptPath;
|
||||||
|
|
||||||
#endregion System proxy
|
#endregion System proxy
|
||||||
|
|
||||||
@@ -273,12 +292,12 @@ public class OptionSettingViewModel : MyReactiveObject
|
|||||||
NoticeManager.Instance.Enqueue(ResUI.FillLocalListeningPort);
|
NoticeManager.Instance.Enqueue(ResUI.FillLocalListeningPort);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
var needReboot = (EnableStatistics != _config.GuiItem.EnableStatistics
|
var needReboot = EnableStatistics != _config.GuiItem.EnableStatistics
|
||||||
|| DisplayRealTimeSpeed != _config.GuiItem.DisplayRealTimeSpeed
|
|| DisplayRealTimeSpeed != _config.GuiItem.DisplayRealTimeSpeed
|
||||||
|| EnableDragDropSort != _config.UiItem.EnableDragDropSort
|
|| EnableDragDropSort != _config.UiItem.EnableDragDropSort
|
||||||
|| EnableHWA != _config.GuiItem.EnableHWA
|
|| EnableHWA != _config.GuiItem.EnableHWA
|
||||||
|| CurrentFontFamily != _config.UiItem.CurrentFontFamily
|
|| CurrentFontFamily != _config.UiItem.CurrentFontFamily
|
||||||
|| MainGirdOrientation != (int)_config.UiItem.MainGirdOrientation);
|
|| MainGirdOrientation != (int)_config.UiItem.MainGirdOrientation;
|
||||||
|
|
||||||
//if (Utile.IsNullOrEmpty(Kcpmtu.ToString()) || !Utile.IsNumeric(Kcpmtu.ToString())
|
//if (Utile.IsNullOrEmpty(Kcpmtu.ToString()) || !Utile.IsNumeric(Kcpmtu.ToString())
|
||||||
// || Utile.IsNullOrEmpty(Kcptti.ToString()) || !Utile.IsNumeric(Kcptti.ToString())
|
// || Utile.IsNullOrEmpty(Kcptti.ToString()) || !Utile.IsNumeric(Kcptti.ToString())
|
||||||
@@ -326,6 +345,7 @@ public class OptionSettingViewModel : MyReactiveObject
|
|||||||
_config.UiItem.EnableUpdateSubOnlyRemarksExist = EnableUpdateSubOnlyRemarksExist;
|
_config.UiItem.EnableUpdateSubOnlyRemarksExist = EnableUpdateSubOnlyRemarksExist;
|
||||||
_config.UiItem.AutoHideStartup = AutoHideStartup;
|
_config.UiItem.AutoHideStartup = AutoHideStartup;
|
||||||
_config.UiItem.Hide2TrayWhenClose = Hide2TrayWhenClose;
|
_config.UiItem.Hide2TrayWhenClose = Hide2TrayWhenClose;
|
||||||
|
_config.UiItem.MacOSShowInDock = MacOSShowInDock;
|
||||||
_config.GuiItem.AutoUpdateInterval = AutoUpdateInterval;
|
_config.GuiItem.AutoUpdateInterval = AutoUpdateInterval;
|
||||||
_config.UiItem.EnableDragDropSort = EnableDragDropSort;
|
_config.UiItem.EnableDragDropSort = EnableDragDropSort;
|
||||||
_config.UiItem.DoubleClick2Activate = DoubleClick2Activate;
|
_config.UiItem.DoubleClick2Activate = DoubleClick2Activate;
|
||||||
@@ -347,6 +367,8 @@ public class OptionSettingViewModel : MyReactiveObject
|
|||||||
_config.SystemProxyItem.SystemProxyExceptions = systemProxyExceptions;
|
_config.SystemProxyItem.SystemProxyExceptions = systemProxyExceptions;
|
||||||
_config.SystemProxyItem.NotProxyLocalAddress = notProxyLocalAddress;
|
_config.SystemProxyItem.NotProxyLocalAddress = notProxyLocalAddress;
|
||||||
_config.SystemProxyItem.SystemProxyAdvancedProtocol = systemProxyAdvancedProtocol;
|
_config.SystemProxyItem.SystemProxyAdvancedProtocol = systemProxyAdvancedProtocol;
|
||||||
|
_config.SystemProxyItem.CustomSystemProxyPacPath = CustomSystemProxyPacPath;
|
||||||
|
_config.SystemProxyItem.CustomSystemProxyScriptPath = CustomSystemProxyScriptPath;
|
||||||
|
|
||||||
//tun mode
|
//tun mode
|
||||||
_config.TunModeItem.AutoRoute = TunAutoRoute;
|
_config.TunModeItem.AutoRoute = TunAutoRoute;
|
||||||
@@ -375,7 +397,7 @@ public class OptionSettingViewModel : MyReactiveObject
|
|||||||
|
|
||||||
private async Task SaveCoreType()
|
private async Task SaveCoreType()
|
||||||
{
|
{
|
||||||
for (int k = 1; k <= _config.CoreTypeItem.Count; k++)
|
for (var k = 1; k <= _config.CoreTypeItem.Count; k++)
|
||||||
{
|
{
|
||||||
var item = _config.CoreTypeItem[k - 1];
|
var item = _config.CoreTypeItem[k - 1];
|
||||||
var type = string.Empty;
|
var type = string.Empty;
|
||||||
|
|||||||
@@ -66,6 +66,7 @@ public class ProfilesViewModel : MyReactiveObject
|
|||||||
public ReactiveCommand<Unit, Unit> SpeedServerCmd { get; }
|
public ReactiveCommand<Unit, Unit> SpeedServerCmd { get; }
|
||||||
public ReactiveCommand<Unit, Unit> SortServerResultCmd { get; }
|
public ReactiveCommand<Unit, Unit> SortServerResultCmd { get; }
|
||||||
public ReactiveCommand<Unit, Unit> RemoveInvalidServerResultCmd { get; }
|
public ReactiveCommand<Unit, Unit> RemoveInvalidServerResultCmd { get; }
|
||||||
|
public ReactiveCommand<Unit, Unit> FastRealPingCmd { get; }
|
||||||
|
|
||||||
//servers export
|
//servers export
|
||||||
public ReactiveCommand<Unit, Unit> Export2ClientConfigCmd { get; }
|
public ReactiveCommand<Unit, Unit> Export2ClientConfigCmd { get; }
|
||||||
@@ -76,6 +77,7 @@ public class ProfilesViewModel : MyReactiveObject
|
|||||||
|
|
||||||
public ReactiveCommand<Unit, Unit> AddSubCmd { get; }
|
public ReactiveCommand<Unit, Unit> AddSubCmd { get; }
|
||||||
public ReactiveCommand<Unit, Unit> EditSubCmd { get; }
|
public ReactiveCommand<Unit, Unit> EditSubCmd { get; }
|
||||||
|
public ReactiveCommand<Unit, Unit> DeleteSubCmd { get; }
|
||||||
|
|
||||||
#endregion Menu
|
#endregion Menu
|
||||||
|
|
||||||
@@ -109,7 +111,7 @@ public class ProfilesViewModel : MyReactiveObject
|
|||||||
//servers delete
|
//servers delete
|
||||||
EditServerCmd = ReactiveCommand.CreateFromTask(async () =>
|
EditServerCmd = ReactiveCommand.CreateFromTask(async () =>
|
||||||
{
|
{
|
||||||
await EditServerAsync(EConfigType.Custom);
|
await EditServerAsync();
|
||||||
}, canEditRemove);
|
}, canEditRemove);
|
||||||
RemoveServerCmd = ReactiveCommand.CreateFromTask(async () =>
|
RemoveServerCmd = ReactiveCommand.CreateFromTask(async () =>
|
||||||
{
|
{
|
||||||
@@ -179,6 +181,10 @@ public class ProfilesViewModel : MyReactiveObject
|
|||||||
}, canEditRemove);
|
}, canEditRemove);
|
||||||
|
|
||||||
//servers ping
|
//servers ping
|
||||||
|
FastRealPingCmd = ReactiveCommand.CreateFromTask(async () =>
|
||||||
|
{
|
||||||
|
await ServerSpeedtest(ESpeedActionType.FastRealping);
|
||||||
|
});
|
||||||
MixedTestServerCmd = ReactiveCommand.CreateFromTask(async () =>
|
MixedTestServerCmd = ReactiveCommand.CreateFromTask(async () =>
|
||||||
{
|
{
|
||||||
await ServerSpeedtest(ESpeedActionType.Mixedtest);
|
await ServerSpeedtest(ESpeedActionType.Mixedtest);
|
||||||
@@ -230,6 +236,10 @@ public class ProfilesViewModel : MyReactiveObject
|
|||||||
{
|
{
|
||||||
await EditSubAsync(false);
|
await EditSubAsync(false);
|
||||||
});
|
});
|
||||||
|
DeleteSubCmd = ReactiveCommand.CreateFromTask(async () =>
|
||||||
|
{
|
||||||
|
await DeleteSubAsync();
|
||||||
|
});
|
||||||
|
|
||||||
#endregion WhenAnyValue && ReactiveCommand
|
#endregion WhenAnyValue && ReactiveCommand
|
||||||
|
|
||||||
@@ -295,14 +305,14 @@ public class ProfilesViewModel : MyReactiveObject
|
|||||||
|
|
||||||
if (result.Delay.IsNotEmpty())
|
if (result.Delay.IsNotEmpty())
|
||||||
{
|
{
|
||||||
int.TryParse(result.Delay, out var temp);
|
item.Delay = result.Delay.ToInt();
|
||||||
item.Delay = temp;
|
|
||||||
item.DelayVal = result.Delay ?? string.Empty;
|
item.DelayVal = result.Delay ?? string.Empty;
|
||||||
}
|
}
|
||||||
if (result.Speed.IsNotEmpty())
|
if (result.Speed.IsNotEmpty())
|
||||||
{
|
{
|
||||||
item.SpeedVal = result.Speed ?? string.Empty;
|
item.SpeedVal = result.Speed ?? string.Empty;
|
||||||
}
|
}
|
||||||
|
await Task.CompletedTask;
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task UpdateStatistics(ServerSpeedItem update)
|
public async Task UpdateStatistics(ServerSpeedItem update)
|
||||||
@@ -328,6 +338,7 @@ public class ProfilesViewModel : MyReactiveObject
|
|||||||
catch
|
catch
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
await Task.CompletedTask;
|
||||||
}
|
}
|
||||||
|
|
||||||
#endregion Actions
|
#endregion Actions
|
||||||
@@ -482,7 +493,7 @@ public class ProfilesViewModel : MyReactiveObject
|
|||||||
return lstSelected;
|
return lstSelected;
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task EditServerAsync(EConfigType eConfigType)
|
public async Task EditServerAsync()
|
||||||
{
|
{
|
||||||
if (string.IsNullOrEmpty(SelectedProfile?.IndexId))
|
if (string.IsNullOrEmpty(SelectedProfile?.IndexId))
|
||||||
{
|
{
|
||||||
@@ -494,7 +505,7 @@ public class ProfilesViewModel : MyReactiveObject
|
|||||||
NoticeManager.Instance.Enqueue(ResUI.PleaseSelectServer);
|
NoticeManager.Instance.Enqueue(ResUI.PleaseSelectServer);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
eConfigType = item.ConfigType;
|
var eConfigType = item.ConfigType;
|
||||||
|
|
||||||
bool? ret = false;
|
bool? ret = false;
|
||||||
if (eConfigType == EConfigType.Custom)
|
if (eConfigType == EConfigType.Custom)
|
||||||
@@ -547,6 +558,11 @@ public class ProfilesViewModel : MyReactiveObject
|
|||||||
|
|
||||||
private async Task RemoveDuplicateServer()
|
private async Task RemoveDuplicateServer()
|
||||||
{
|
{
|
||||||
|
if (await _updateView?.Invoke(EViewAction.ShowYesNo, null) == false)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
var tuple = await ConfigHandler.DedupServerList(_config, _config.SubIndexId);
|
var tuple = await ConfigHandler.DedupServerList(_config, _config.SubIndexId);
|
||||||
if (tuple.Item1 > 0 || tuple.Item2 > 0)
|
if (tuple.Item1 > 0 || tuple.Item2 > 0)
|
||||||
{
|
{
|
||||||
@@ -653,7 +669,7 @@ public class ProfilesViewModel : MyReactiveObject
|
|||||||
}
|
}
|
||||||
|
|
||||||
_dicHeaderSort.TryAdd(colName, true);
|
_dicHeaderSort.TryAdd(colName, true);
|
||||||
_dicHeaderSort.TryGetValue(colName, out bool asc);
|
_dicHeaderSort.TryGetValue(colName, out var asc);
|
||||||
if (await ConfigHandler.SortServers(_config, _config.SubIndexId, colName, asc) != 0)
|
if (await ConfigHandler.SortServers(_config, _config.SubIndexId, colName, asc) != 0)
|
||||||
{
|
{
|
||||||
return;
|
return;
|
||||||
@@ -729,6 +745,12 @@ public class ProfilesViewModel : MyReactiveObject
|
|||||||
{
|
{
|
||||||
SelectedProfiles = ProfileItems;
|
SelectedProfiles = ProfileItems;
|
||||||
}
|
}
|
||||||
|
else if (actionType == ESpeedActionType.FastRealping)
|
||||||
|
{
|
||||||
|
SelectedProfiles = ProfileItems;
|
||||||
|
actionType = ESpeedActionType.Realping;
|
||||||
|
}
|
||||||
|
|
||||||
var lstSelected = await GetProfileItems(false);
|
var lstSelected = await GetProfileItems(false);
|
||||||
if (lstSelected == null)
|
if (lstSelected == null)
|
||||||
{
|
{
|
||||||
@@ -742,6 +764,7 @@ public class ProfilesViewModel : MyReactiveObject
|
|||||||
_ = SetSpeedTestResult(result);
|
_ = SetSpeedTestResult(result);
|
||||||
return Disposable.Empty;
|
return Disposable.Empty;
|
||||||
});
|
});
|
||||||
|
await Task.CompletedTask;
|
||||||
});
|
});
|
||||||
_speedtestService?.RunLoop(actionType, lstSelected);
|
_speedtestService?.RunLoop(actionType, lstSelected);
|
||||||
}
|
}
|
||||||
@@ -866,5 +889,23 @@ public class ProfilesViewModel : MyReactiveObject
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private async Task DeleteSubAsync()
|
||||||
|
{
|
||||||
|
var item = await AppManager.Instance.GetSubItem(_config.SubIndexId);
|
||||||
|
if (item is null)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (await _updateView?.Invoke(EViewAction.ShowYesNo, null) == false)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
await ConfigHandler.DeleteSubItem(_config, item.Id);
|
||||||
|
|
||||||
|
await RefreshSubscriptions();
|
||||||
|
await SubSelectedChangedAsync(true);
|
||||||
|
}
|
||||||
|
|
||||||
#endregion Subscription
|
#endregion Subscription
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -215,7 +215,7 @@ public class RoutingRuleSettingViewModel : MyReactiveObject
|
|||||||
|
|
||||||
private async Task SaveRoutingAsync()
|
private async Task SaveRoutingAsync()
|
||||||
{
|
{
|
||||||
string remarks = SelectedRouting.Remarks;
|
var remarks = SelectedRouting.Remarks;
|
||||||
if (remarks.IsNullOrEmpty())
|
if (remarks.IsNullOrEmpty())
|
||||||
{
|
{
|
||||||
NoticeManager.Instance.Enqueue(ResUI.PleaseFillRemarks);
|
NoticeManager.Instance.Enqueue(ResUI.PleaseFillRemarks);
|
||||||
@@ -286,7 +286,7 @@ public class RoutingRuleSettingViewModel : MyReactiveObject
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
DownloadService downloadHandle = new DownloadService();
|
var downloadHandle = new DownloadService();
|
||||||
var result = await downloadHandle.TryDownloadString(url, true, "");
|
var result = await downloadHandle.TryDownloadString(url, true, "");
|
||||||
var ret = await AddBatchRoutingRulesAsync(SelectedRouting, result);
|
var ret = await AddBatchRoutingRulesAsync(SelectedRouting, result);
|
||||||
if (ret == 0)
|
if (ret == 0)
|
||||||
@@ -298,7 +298,7 @@ public class RoutingRuleSettingViewModel : MyReactiveObject
|
|||||||
|
|
||||||
private async Task<int> AddBatchRoutingRulesAsync(RoutingItem routingItem, string? clipboardData)
|
private async Task<int> AddBatchRoutingRulesAsync(RoutingItem routingItem, string? clipboardData)
|
||||||
{
|
{
|
||||||
bool blReplace = false;
|
var blReplace = false;
|
||||||
if (await _updateView?.Invoke(EViewAction.AddBatchRoutingRulesYesNo, null) == false)
|
if (await _updateView?.Invoke(EViewAction.AddBatchRoutingRulesYesNo, null) == false)
|
||||||
{
|
{
|
||||||
blReplace = true;
|
blReplace = true;
|
||||||
|
|||||||
@@ -120,7 +120,7 @@ public class StatusBarViewModel : MyReactiveObject
|
|||||||
this.WhenAnyValue(
|
this.WhenAnyValue(
|
||||||
x => x.SelectedServer,
|
x => x.SelectedServer,
|
||||||
y => y != null && !y.Text.IsNullOrEmpty())
|
y => y != null && !y.Text.IsNullOrEmpty())
|
||||||
.Subscribe(c => ServerSelectedChanged(c));
|
.Subscribe(ServerSelectedChanged);
|
||||||
|
|
||||||
SystemProxySelected = (int)_config.SystemProxyItem.SysProxyType;
|
SystemProxySelected = (int)_config.SystemProxyItem.SysProxyType;
|
||||||
this.WhenAnyValue(
|
this.WhenAnyValue(
|
||||||
@@ -313,10 +313,10 @@ public class StatusBarViewModel : MyReactiveObject
|
|||||||
}
|
}
|
||||||
|
|
||||||
BlServers = true;
|
BlServers = true;
|
||||||
for (int k = 0; k < lstModel.Count; k++)
|
for (var k = 0; k < lstModel.Count; k++)
|
||||||
{
|
{
|
||||||
ProfileItem it = lstModel[k];
|
ProfileItem it = lstModel[k];
|
||||||
string name = it.GetSummary();
|
var name = it.GetSummary();
|
||||||
|
|
||||||
var item = new ComboItem() { ID = it.IndexId, Text = name };
|
var item = new ComboItem() { ID = it.IndexId, Text = name };
|
||||||
Servers.Add(item);
|
Servers.Add(item);
|
||||||
@@ -367,11 +367,13 @@ public class StatusBarViewModel : MyReactiveObject
|
|||||||
_ = TestServerAvailabilityResult(msg);
|
_ = TestServerAvailabilityResult(msg);
|
||||||
return Disposable.Empty;
|
return Disposable.Empty;
|
||||||
});
|
});
|
||||||
|
await Task.CompletedTask;
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task TestServerAvailabilityResult(string msg)
|
public async Task TestServerAvailabilityResult(string msg)
|
||||||
{
|
{
|
||||||
RunningInfoDisplay = msg;
|
RunningInfoDisplay = msg;
|
||||||
|
await Task.CompletedTask;
|
||||||
}
|
}
|
||||||
|
|
||||||
#region System proxy and Routings
|
#region System proxy and Routings
|
||||||
@@ -384,7 +386,7 @@ public class StatusBarViewModel : MyReactiveObject
|
|||||||
}
|
}
|
||||||
_config.SystemProxyItem.SysProxyType = type;
|
_config.SystemProxyItem.SysProxyType = type;
|
||||||
await ChangeSystemProxyAsync(type, true);
|
await ChangeSystemProxyAsync(type, true);
|
||||||
NoticeManager.Instance.SendMessageEx($"{ResUI.TipChangeSystemProxy} - {_config.SystemProxyItem.SysProxyType.ToString()}");
|
NoticeManager.Instance.SendMessageEx($"{ResUI.TipChangeSystemProxy} - {_config.SystemProxyItem.SysProxyType}");
|
||||||
|
|
||||||
SystemProxySelected = (int)_config.SystemProxyItem.SysProxyType;
|
SystemProxySelected = (int)_config.SystemProxyItem.SysProxyType;
|
||||||
await ConfigHandler.SaveConfig(_config);
|
await ConfigHandler.SaveConfig(_config);
|
||||||
@@ -394,10 +396,10 @@ public class StatusBarViewModel : MyReactiveObject
|
|||||||
{
|
{
|
||||||
await SysProxyHandler.UpdateSysProxy(_config, false);
|
await SysProxyHandler.UpdateSysProxy(_config, false);
|
||||||
|
|
||||||
BlSystemProxyClear = (type == ESysProxyType.ForcedClear);
|
BlSystemProxyClear = type == ESysProxyType.ForcedClear;
|
||||||
BlSystemProxySet = (type == ESysProxyType.ForcedChange);
|
BlSystemProxySet = type == ESysProxyType.ForcedChange;
|
||||||
BlSystemProxyNothing = (type == ESysProxyType.Unchanged);
|
BlSystemProxyNothing = type == ESysProxyType.Unchanged;
|
||||||
BlSystemProxyPac = (type == ESysProxyType.Pac);
|
BlSystemProxyPac = type == ESysProxyType.Pac;
|
||||||
|
|
||||||
if (blChange)
|
if (blChange)
|
||||||
{
|
{
|
||||||
@@ -502,7 +504,7 @@ public class StatusBarViewModel : MyReactiveObject
|
|||||||
{
|
{
|
||||||
return AppManager.Instance.LinuxSudoPwd.IsNotEmpty();
|
return AppManager.Instance.LinuxSudoPwd.IsNotEmpty();
|
||||||
}
|
}
|
||||||
else if (Utils.IsOSX())
|
else if (Utils.IsMacOS())
|
||||||
{
|
{
|
||||||
return AppManager.Instance.LinuxSudoPwd.IsNotEmpty();
|
return AppManager.Instance.LinuxSudoPwd.IsNotEmpty();
|
||||||
}
|
}
|
||||||
@@ -561,6 +563,7 @@ public class StatusBarViewModel : MyReactiveObject
|
|||||||
catch
|
catch
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
await Task.CompletedTask;
|
||||||
}
|
}
|
||||||
|
|
||||||
#endregion UI
|
#endregion UI
|
||||||
|
|||||||
@@ -10,15 +10,17 @@ public partial class App : Application
|
|||||||
|
|
||||||
AppDomain.CurrentDomain.UnhandledException += CurrentDomain_UnhandledException;
|
AppDomain.CurrentDomain.UnhandledException += CurrentDomain_UnhandledException;
|
||||||
TaskScheduler.UnobservedTaskException += TaskScheduler_UnobservedTaskException;
|
TaskScheduler.UnobservedTaskException += TaskScheduler_UnobservedTaskException;
|
||||||
|
|
||||||
DataContext = StatusBarViewModel.Instance;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public override void OnFrameworkInitializationCompleted()
|
public override void OnFrameworkInitializationCompleted()
|
||||||
{
|
{
|
||||||
if (ApplicationLifetime is IClassicDesktopStyleApplicationLifetime desktop)
|
if (ApplicationLifetime is IClassicDesktopStyleApplicationLifetime desktop)
|
||||||
{
|
{
|
||||||
AppManager.Instance.InitComponents();
|
if (!Design.IsDesignMode)
|
||||||
|
{
|
||||||
|
AppManager.Instance.InitComponents();
|
||||||
|
DataContext = StatusBarViewModel.Instance;
|
||||||
|
}
|
||||||
|
|
||||||
desktop.Exit += OnExit;
|
desktop.Exit += OnExit;
|
||||||
desktop.MainWindow = new MainWindow();
|
desktop.MainWindow = new MainWindow();
|
||||||
|
|||||||
@@ -26,7 +26,7 @@ public class WindowBase<TViewModel> : ReactiveWindow<TViewModel> where TViewMode
|
|||||||
Height = sizeItem.Height;
|
Height = sizeItem.Height;
|
||||||
|
|
||||||
var workingArea = (Screens.ScreenFromWindow(this) ?? Screens.Primary).WorkingArea;
|
var workingArea = (Screens.ScreenFromWindow(this) ?? Screens.Primary).WorkingArea;
|
||||||
var scaling = (Utils.IsOSX() ? null : VisualRoot?.RenderScaling) ?? 1.0;
|
var scaling = (Utils.IsMacOS() ? null : VisualRoot?.RenderScaling) ?? 1.0;
|
||||||
|
|
||||||
var x = workingArea.X + ((workingArea.Width - (Width * scaling)) / 2);
|
var x = workingArea.X + ((workingArea.Width - (Width * scaling)) / 2);
|
||||||
var y = workingArea.Y + ((workingArea.Height - (Height * scaling)) / 2);
|
var y = workingArea.Y + ((workingArea.Height - (Height * scaling)) / 2);
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user