Compare commits

..

No commits in common. "main" and "v1.3.7" have entirely different histories.
main ... v1.3.7

32 changed files with 59 additions and 271 deletions

View file

@ -21,7 +21,7 @@ jobs:
if: startsWith(github.repository, 'zzzgydi') if: startsWith(github.repository, 'zzzgydi')
steps: steps:
- name: Checkout repository - name: Checkout repository
uses: actions/checkout@v4 uses: actions/checkout@v3
- name: install Rust stable - name: install Rust stable
uses: dtolnay/rust-toolchain@stable uses: dtolnay/rust-toolchain@stable
@ -32,7 +32,7 @@ jobs:
workspaces: src-tauri workspaces: src-tauri
- name: Install Node - name: Install Node
uses: actions/setup-node@v4 uses: actions/setup-node@v3
with: with:
node-version: "16" node-version: "16"
cache: "yarn" cache: "yarn"
@ -53,7 +53,6 @@ jobs:
*.dmg *.dmg
*.msi *.msi
*.sig *.sig
*.exe
- name: Install Dependencies (ubuntu only) - name: Install Dependencies (ubuntu only)
if: startsWith(matrix.os, 'ubuntu-') if: startsWith(matrix.os, 'ubuntu-')
@ -64,7 +63,7 @@ jobs:
- name: Yarn install and check - name: Yarn install and check
run: | run: |
yarn install --network-timeout 1000000 --frozen-lockfile yarn install --network-timeout 1000000 --frozen-lockfile
yarn run check --force yarn run check
- name: Tauri build - name: Tauri build
uses: tauri-apps/tauri-action@v0 uses: tauri-apps/tauri-action@v0

View file

@ -19,7 +19,7 @@ jobs:
if: startsWith(github.repository, 'zzzgydi') if: startsWith(github.repository, 'zzzgydi')
steps: steps:
- name: Checkout repository - name: Checkout repository
uses: actions/checkout@v4 uses: actions/checkout@v3
- name: install Rust stable - name: install Rust stable
uses: dtolnay/rust-toolchain@stable uses: dtolnay/rust-toolchain@stable
@ -30,7 +30,7 @@ jobs:
workspaces: src-tauri workspaces: src-tauri
- name: Install Node - name: Install Node
uses: actions/setup-node@v4 uses: actions/setup-node@v3
with: with:
node-version: "16" node-version: "16"
cache: "yarn" cache: "yarn"
@ -81,10 +81,10 @@ jobs:
startsWith(github.ref, 'refs/tags/v') startsWith(github.ref, 'refs/tags/v')
steps: steps:
- name: Checkout repository - name: Checkout repository
uses: actions/checkout@v4 uses: actions/checkout@v3
- name: Install Node - name: Install Node
uses: actions/setup-node@v4 uses: actions/setup-node@v3
with: with:
node-version: "16" node-version: "16"
cache: "yarn" cache: "yarn"

View file

@ -35,7 +35,7 @@ jobs:
echo ${{ github.event.inputs.os }} echo ${{ github.event.inputs.os }}
- name: Checkout repository - name: Checkout repository
uses: actions/checkout@v4 uses: actions/checkout@v3
- name: install Rust stable - name: install Rust stable
uses: dtolnay/rust-toolchain@stable uses: dtolnay/rust-toolchain@stable
@ -46,7 +46,7 @@ jobs:
workspaces: src-tauri workspaces: src-tauri
- name: Install Node - name: Install Node
uses: actions/setup-node@v4 uses: actions/setup-node@v3
with: with:
node-version: "16" node-version: "16"
cache: "yarn" cache: "yarn"

View file

@ -8,10 +8,10 @@ jobs:
if: startsWith(github.repository, 'zzzgydi') if: startsWith(github.repository, 'zzzgydi')
steps: steps:
- name: Checkout repository - name: Checkout repository
uses: actions/checkout@v4 uses: actions/checkout@v3
- name: Install Node - name: Install Node
uses: actions/setup-node@v4 uses: actions/setup-node@v3
with: with:
node-version: "16" node-version: "16"
cache: "yarn" cache: "yarn"

View file

@ -23,11 +23,11 @@ A <a href="https://github.com/Dreamacro/clash">Clash</a> GUI based on <a href="h
- High-performance overseas VPN, free trial, discounted packages, unlock streaming media, the world's first to support Hysteria protocol. - High-performance overseas VPN, free trial, discounted packages, unlock streaming media, the world's first to support Hysteria protocol.
- 高性能海外机场,免费试用,优惠套餐,解锁流媒体,全球首家支持 Hysteria 协议。 - 高性能海外机场,免费试用,优惠套餐,解锁流媒体,全球首家支持 Hysteria 协议。
- 使用 Clash Verge 专属邀请链接注册送 15 天,每天 1G 流量免费试用https://panel.dg1.top/#/register?code=sFCDayZf
<details> <details>
<summary>Promotion Detail</summary> <summary>Promotion Detail</summary>
- 使用 Clash Verge 专属邀请链接注册送 15 天,每天 1G 流量免费试用https://panel.dg1.top/#/register?code=sFCDayZf
- Clash Verge 专属 8 折优惠码: verge20 (仅有 500 份) - Clash Verge 专属 8 折优惠码: verge20 (仅有 500 份)
- 优惠套餐每月仅需 15.8 元160G 流量,年付 8 折 - 优惠套餐每月仅需 15.8 元160G 流量,年付 8 折
- 海外团队,无跑路风险,高达 50% 返佣 - 海外团队,无跑路风险,高达 50% 返佣
@ -38,34 +38,15 @@ A <a href="https://github.com/Dreamacro/clash">Clash</a> GUI based on <a href="h
</details> </details>
<br />
[EEVPN —— 海外运营机场 ※ 支持 ChatGPT](https://www.eejsq.net/#/register?code=yRr6qBO3)
- 年付低至 9.99 元,价格低,速度不减
<details>
<summary>Promotion Detail</summary>
- 中国大陆 BGP 网络接入
- IEPL 专线网络
- 最高 2500Mbps 速率可用
- 不限制在线客户端
- 解锁流媒体及 ChatGPT
- 海外运营 数据安全
</details>
## Install ## Install
Download from [release](https://github.com/zzzgydi/clash-verge/releases). Supports Windows x64, Linux x86_64 and macOS 11+ Download from [release](https://github.com/zzzgydi/clash-verge/releases). Supports Windows x64, Linux x86_64 and macOS 11+
- [Windows x64](https://github.com/zzzgydi/clash-verge/releases/download/v1.3.8/Clash.Verge_1.3.8_x64_en-US.msi) - [Windows x64](https://github.com/zzzgydi/clash-verge/releases/download/v1.3.6/Clash.Verge_1.3.6_x64_en-US.msi)
- [macOS intel](https://github.com/zzzgydi/clash-verge/releases/download/v1.3.8/Clash.Verge_1.3.8_x64.dmg) - [macOS intel](https://github.com/zzzgydi/clash-verge/releases/download/v1.3.6/Clash.Verge_1.3.6_x64.dmg)
- [macOS arm](https://github.com/zzzgydi/clash-verge/releases/download/v1.3.8/Clash.Verge_1.3.8_aarch64.dmg) - [macOS arm](https://github.com/zzzgydi/clash-verge/releases/download/v1.3.6/Clash.Verge_1.3.6_aarch64.dmg)
- [Linux AppImage](https://github.com/zzzgydi/clash-verge/releases/download/v1.3.8/clash-verge_1.3.8_amd64.AppImage) - [Linux AppImage](https://github.com/zzzgydi/clash-verge/releases/download/v1.3.6/clash-verge_1.3.6_amd64.AppImage)
- [Linux deb](https://github.com/zzzgydi/clash-verge/releases/download/v1.3.8/clash-verge_1.3.8_amd64.deb) - [Linux deb](https://github.com/zzzgydi/clash-verge/releases/download/v1.3.6/clash-verge_1.3.6_amd64.deb)
- [Fedora Linux](https://github.com/zzzgydi/clash-verge/issues/352)
Or you can build it yourself. Supports Windows, Linux and macOS 10.15+ Or you can build it yourself. Supports Windows, Linux and macOS 10.15+

View file

@ -1,17 +1,3 @@
## v1.3.8
### Features
- update clash meta core
- add default valid keys
- adjust the delay display interval and color
### Bug Fixes
- fix connections page undefined exception
---
## v1.3.7 ## v1.3.7
### Features ### Features

View file

@ -1,6 +1,6 @@
{ {
"name": "clash-verge", "name": "clash-verge",
"version": "1.3.8", "version": "1.3.7",
"license": "GPL-3.0", "license": "GPL-3.0",
"scripts": { "scripts": {
"dev": "tauri dev", "dev": "tauri dev",

View file

@ -18,19 +18,19 @@ const SIDECAR_HOST = execSync("rustc -vV")
const CLASH_STORAGE_PREFIX = "https://release.dreamacro.workers.dev/"; const CLASH_STORAGE_PREFIX = "https://release.dreamacro.workers.dev/";
const CLASH_URL_PREFIX = const CLASH_URL_PREFIX =
"https://github.com/Dreamacro/clash/releases/download/premium/"; "https://github.com/Dreamacro/clash/releases/download/premium/";
const CLASH_LATEST_DATE = "latest"; const CLASH_LATEST_DATE = "2023.08.17";
const CLASH_MAP = { const CLASH_MAP = {
"win32-x64": "clash-windows-amd64", "win32-x64": "clash-windows-amd64",
"darwin-x64": "clash-darwin-amd64", "darwin-x64": "clash-darwin-amd64",
"darwin-arm64": "clash-darwin-arm64", "darwin-arm64": "clash-darwin-arm64",
"linux-x64": "clash-linux-amd64", "linux-x64": "clash-linux-amd64",
"linux-arm64": "clash-linux-arm64", "linux-arm64": "clash-linux-armv8",
}; };
/* ======= clash meta ======= */ /* ======= clash meta ======= */
const META_URL_PREFIX = `https://github.com/MetaCubeX/Clash.Meta/releases/download/`; const META_URL_PREFIX = `https://github.com/MetaCubeX/Clash.Meta/releases/download/`;
const META_VERSION = "v1.16.0"; const META_VERSION = "v1.15.1";
const META_MAP = { const META_MAP = {
"win32-x64": "clash.meta-windows-amd64-compatible", "win32-x64": "clash.meta-windows-amd64-compatible",
@ -300,7 +300,7 @@ const resolveGeoIP = () =>
}); });
const tasks = [ const tasks = [
{ name: "clash", func: () => resolveSidecar(clashS3()), retry: 5 }, { name: "clash", func: resolveClash, retry: 5 },
{ name: "clash-meta", func: () => resolveSidecar(clashMeta()), retry: 5 }, { name: "clash-meta", func: () => resolveSidecar(clashMeta()), retry: 5 },
{ name: "wintun", func: resolveWintun, retry: 5, winOnly: true }, { name: "wintun", func: resolveWintun, retry: 5, winOnly: true },
{ name: "service", func: resolveService, retry: 5, winOnly: true }, { name: "service", func: resolveService, retry: 5, winOnly: true },

View file

@ -43,7 +43,6 @@ async function resolveUpdater() {
darwin: { signature: "", url: "" }, // compatible with older formats darwin: { signature: "", url: "" }, // compatible with older formats
"darwin-aarch64": { signature: "", url: "" }, "darwin-aarch64": { signature: "", url: "" },
"darwin-intel": { signature: "", url: "" }, "darwin-intel": { signature: "", url: "" },
"darwin-x86_64": { signature: "", url: "" },
"linux-x86_64": { signature: "", url: "" }, "linux-x86_64": { signature: "", url: "" },
"windows-x86_64": { signature: "", url: "" }, "windows-x86_64": { signature: "", url: "" },
"windows-i686": { signature: "", url: "" }, // no supported "windows-i686": { signature: "", url: "" }, // no supported
@ -69,14 +68,12 @@ async function resolveUpdater() {
if (name.endsWith(".app.tar.gz") && !name.includes("aarch")) { if (name.endsWith(".app.tar.gz") && !name.includes("aarch")) {
updateData.platforms.darwin.url = browser_download_url; updateData.platforms.darwin.url = browser_download_url;
updateData.platforms["darwin-intel"].url = browser_download_url; updateData.platforms["darwin-intel"].url = browser_download_url;
updateData.platforms["darwin-x86_64"].url = browser_download_url;
} }
// darwin signature (intel) // darwin signature (intel)
if (name.endsWith(".app.tar.gz.sig") && !name.includes("aarch")) { if (name.endsWith(".app.tar.gz.sig") && !name.includes("aarch")) {
const sig = await getSignature(browser_download_url); const sig = await getSignature(browser_download_url);
updateData.platforms.darwin.signature = sig; updateData.platforms.darwin.signature = sig;
updateData.platforms["darwin-intel"].signature = sig; updateData.platforms["darwin-intel"].signature = sig;
updateData.platforms["darwin-x86_64"].signature = sig;
} }
// darwin url (aarch) // darwin url (aarch)

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.3 KiB

After

Width:  |  Height:  |  Size: 3.1 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.5 KiB

After

Width:  |  Height:  |  Size: 1.7 KiB

View file

@ -229,17 +229,6 @@ pub fn open_web_url(url: String) -> CmdResult<()> {
wrap_err!(open::that(url)) wrap_err!(open::that(url))
} }
#[tauri::command]
pub async fn clash_api_get_proxy_delay(
name: String,
url: Option<String>,
) -> CmdResult<clash_api::DelayRes> {
match clash_api::get_proxy_delay(name, url).await {
Ok(res) => Ok(res),
Err(err) => Err(format!("{}", err.to_string())),
}
}
#[cfg(windows)] #[cfg(windows)]
pub mod service { pub mod service {
use super::*; use super::*;

View file

@ -24,23 +24,11 @@ impl IClashTemp {
pub fn template() -> Self { pub fn template() -> Self {
let mut map = Mapping::new(); let mut map = Mapping::new();
map.insert( map.insert("mixed-port".into(), 7890.into());
"mixed-port".into(),
match cfg!(feature = "default-meta") {
false => 7890.into(),
true => 7898.into(),
},
);
map.insert("log-level".into(), "info".into()); map.insert("log-level".into(), "info".into());
map.insert("allow-lan".into(), false.into()); map.insert("allow-lan".into(), false.into());
map.insert("mode".into(), "rule".into()); map.insert("mode".into(), "rule".into());
map.insert( map.insert("external-controller".into(), "127.0.0.1:9090".into());
"external-controller".into(),
match cfg!(feature = "default-meta") {
false => "127.0.0.1:9090".into(),
true => "127.0.0.1:9098".into(),
},
);
map.insert("secret".into(), "".into()); map.insert("secret".into(), "".into());
Self(map) Self(map)

View file

@ -26,7 +26,7 @@ macro_rules! draft_define {
} }
pub fn draft(&self) -> MappedMutexGuard<$id> { pub fn draft(&self) -> MappedMutexGuard<$id> {
MutexGuard::map(self.inner.lock(), |inner| { MutexGuard::map(self.inner.lock(), |mut inner| {
if inner.1.is_none() { if inner.1.is_none() {
inner.1 = Some(inner.0.clone()); inner.1 = Some(inner.0.clone());
} }

View file

@ -38,7 +38,7 @@ impl IProfiles {
} }
// compatible with the old old old version // compatible with the old old old version
profiles.items.as_mut().map(|items| { profiles.items.as_mut().map(|items| {
for item in items.iter_mut() { for mut item in items.iter_mut() {
if item.uid.is_none() { if item.uid.is_none() {
item.uid = Some(help::get_uid("d")); item.uid = Some(help::get_uid("d"));
} }
@ -55,7 +55,7 @@ impl IProfiles {
pub fn template() -> Self { pub fn template() -> Self {
Self { Self {
valid: Some(vec!["dns".into(), "sub-rules".into(), "unified-delay".into()]), valid: Some(vec!["dns".into()]),
items: Some(vec![]), items: Some(vec![]),
..Self::default() ..Self::default()
} }
@ -155,7 +155,7 @@ impl IProfiles {
pub fn patch_item(&mut self, uid: String, item: PrfItem) -> Result<()> { pub fn patch_item(&mut self, uid: String, item: PrfItem) -> Result<()> {
let mut items = self.items.take().unwrap_or(vec![]); let mut items = self.items.take().unwrap_or(vec![]);
for each in items.iter_mut() { for mut each in items.iter_mut() {
if each.uid == Some(uid.clone()) { if each.uid == Some(uid.clone()) {
patch!(each, item, itype); patch!(each, item, itype);
patch!(each, item, name); patch!(each, item, name);
@ -189,7 +189,7 @@ impl IProfiles {
if let Some(items) = self.items.as_mut() { if let Some(items) = self.items.as_mut() {
let some_uid = Some(uid.clone()); let some_uid = Some(uid.clone());
for each in items.iter_mut() { for mut each in items.iter_mut() {
if each.uid == some_uid { if each.uid == some_uid {
each.extra = item.extra; each.extra = item.extra;
each.updated = item.updated; each.updated = item.updated;

View file

@ -83,10 +83,6 @@ pub struct IVerge {
/// proxy 页面布局 列数 /// proxy 页面布局 列数
pub proxy_layout_column: Option<i32>, pub proxy_layout_column: Option<i32>,
/// 日志清理
/// 0: 不清理; 1: 7天; 2: 30天; 3: 90天
pub auto_log_clean: Option<i32>,
/// window size and position /// window size and position
#[serde(skip_serializing_if = "Option::is_none")] #[serde(skip_serializing_if = "Option::is_none")]
pub window_size_position: Option<Vec<f64>>, pub window_size_position: Option<Vec<f64>>,
@ -141,7 +137,6 @@ impl IVerge {
auto_close_connection: Some(true), auto_close_connection: Some(true),
enable_builtin_enhanced: Some(true), enable_builtin_enhanced: Some(true),
enable_clash_fields: Some(true), enable_clash_fields: Some(true),
auto_log_clean: Some(3),
..Self::default() ..Self::default()
} }
} }
@ -188,7 +183,7 @@ impl IVerge {
patch!(enable_builtin_enhanced); patch!(enable_builtin_enhanced);
patch!(proxy_layout_column); patch!(proxy_layout_column);
patch!(enable_clash_fields); patch!(enable_clash_fields);
patch!(auto_log_clean);
patch!(window_size_position); patch!(window_size_position);
} }

View file

@ -1,7 +1,6 @@
use crate::config::Config; use crate::config::Config;
use anyhow::{bail, Result}; use anyhow::{bail, Result};
use reqwest::header::HeaderMap; use reqwest::header::HeaderMap;
use serde::{Deserialize, Serialize};
use serde_yaml::Mapping; use serde_yaml::Mapping;
use std::collections::HashMap; use std::collections::HashMap;
@ -37,32 +36,6 @@ pub async fn patch_configs(config: &Mapping) -> Result<()> {
Ok(()) Ok(())
} }
#[derive(Default, Debug, Clone, Deserialize, Serialize)]
pub struct DelayRes {
delay: u64,
}
/// GET /proxies/{name}/delay
/// 获取代理延迟
pub async fn get_proxy_delay(name: String, test_url: Option<String>) -> Result<DelayRes> {
let (url, headers) = clash_client_info()?;
let url = format!("{url}/proxies/{name}/delay");
let default_url = "http://www.gstatic.com/generate_204";
let test_url = test_url
.map(|s| if s.is_empty() { default_url.into() } else { s })
.unwrap_or(default_url.into());
let client = reqwest::ClientBuilder::new().no_proxy().build()?;
let builder = client
.get(&url)
.headers(headers)
.query(&[("timeout", "10000"), ("url", &test_url)]);
let response = builder.send().await?;
Ok(response.json::<DelayRes>().await?)
}
/// 根据clash info获取clash服务地址和请求头 /// 根据clash info获取clash服务地址和请求头
fn clash_client_info() -> Result<(String, HeaderMap)> { fn clash_client_info() -> Result<(String, HeaderMap)> {
let client = { Config::clash().data().get_client_info() }; let client = { Config::clash().data().get_client_info() };

View file

@ -66,8 +66,6 @@ fn main() -> std::io::Result<()> {
cmds::service::check_service, cmds::service::check_service,
cmds::service::install_service, cmds::service::install_service,
cmds::service::uninstall_service, cmds::service::uninstall_service,
// clash api
cmds::clash_api_get_proxy_delay
]); ]);
#[cfg(target_os = "macos")] #[cfg(target_os = "macos")]

View file

@ -1,14 +1,13 @@
use crate::config::*; use crate::config::*;
use crate::utils::{dirs, help}; use crate::utils::{dirs, help};
use anyhow::Result; use anyhow::Result;
use chrono::{DateTime, Local}; use chrono::Local;
use log::LevelFilter; use log::LevelFilter;
use log4rs::append::console::ConsoleAppender; use log4rs::append::console::ConsoleAppender;
use log4rs::append::file::FileAppender; use log4rs::append::file::FileAppender;
use log4rs::config::{Appender, Logger, Root}; use log4rs::config::{Appender, Logger, Root};
use log4rs::encode::pattern::PatternEncoder; use log4rs::encode::pattern::PatternEncoder;
use std::fs::{self, DirEntry}; use std::fs;
use std::str::FromStr;
use tauri::PackageInfo; use tauri::PackageInfo;
/// initialize this instance's log file /// initialize this instance's log file
@ -70,72 +69,6 @@ fn init_log() -> Result<()> {
Ok(()) Ok(())
} }
/// 删除log文件
pub fn delete_log() -> Result<()> {
let log_dir = dirs::app_logs_dir()?;
if !log_dir.exists() {
return Ok(());
}
let auto_log_clean = {
let verge = Config::verge();
let verge = verge.data();
verge.auto_log_clean.clone().unwrap_or(0)
};
let day = match auto_log_clean {
1 => 7,
2 => 30,
3 => 90,
_ => return Ok(()),
};
log::debug!(target: "app", "try to delete log files, day: {day}");
// %Y-%m-%d to NaiveDateTime
let parse_time_str = |s: &str| {
let sa: Vec<&str> = s.split('-').collect();
if sa.len() != 4 {
return Err(anyhow::anyhow!("invalid time str"));
}
let year = i32::from_str(sa[0])?;
let month = u32::from_str(sa[1])?;
let day = u32::from_str(sa[2])?;
let time = chrono::NaiveDate::from_ymd_opt(year, month, day)
.ok_or(anyhow::anyhow!("invalid time str"))?
.and_hms_opt(0, 0, 0)
.ok_or(anyhow::anyhow!("invalid time str"))?;
Ok(time)
};
let process_file = |file: DirEntry| -> Result<()> {
let file_name = file.file_name();
let file_name = file_name.to_str().unwrap_or_default();
if file_name.ends_with(".log") {
let now = Local::now();
let created_time = parse_time_str(&file_name[0..file_name.len() - 4])?;
let file_time = DateTime::<Local>::from_local(created_time, now.offset().clone());
let duration = now.signed_duration_since(file_time);
if duration.num_days() > day {
let file_path = file.path();
let _ = fs::remove_file(file_path);
log::info!(target: "app", "delete log file: {file_name}");
}
}
Ok(())
};
for file in fs::read_dir(&log_dir)? {
if let Ok(file) = file {
let _ = process_file(file);
}
}
Ok(())
}
/// Initialize all the config files /// Initialize all the config files
/// before tauri setup /// before tauri setup
pub fn init_config() -> Result<()> { pub fn init_config() -> Result<()> {
@ -145,7 +78,6 @@ pub fn init_config() -> Result<()> {
} }
let _ = init_log(); let _ = init_log();
let _ = delete_log();
crate::log_err!(dirs::app_home_dir().map(|app_dir| { crate::log_err!(dirs::app_home_dir().map(|app_dir| {
if !app_dir.exists() { if !app_dir.exists() {

View file

@ -1,7 +1,7 @@
{ {
"package": { "package": {
"productName": "Clash Verge", "productName": "Clash Verge",
"version": "1.3.8" "version": "1.3.7"
}, },
"build": { "build": {
"distDir": "../dist", "distDir": "../dist",

View file

@ -58,7 +58,7 @@ export const ConnectionItem = (props: Props) => {
{!!metadata.process && <Tag>{metadata.process}</Tag>} {!!metadata.process && <Tag>{metadata.process}</Tag>}
{chains?.length > 0 && <Tag>{chains[value.chains.length - 1]}</Tag>} {chains.length > 0 && <Tag>{chains[value.chains.length - 1]}</Tag>}
<Tag>{dayjs(start).fromNow()}</Tag> <Tag>{dayjs(start).fromNow()}</Tag>

View file

@ -143,7 +143,7 @@ export const ProfileViewer = forwardRef<ProfileViewerRef, Props>(
<BaseDialog <BaseDialog
open={open} open={open}
title={openType === "new" ? t("Create Profile") : t("Edit Profile")} title={openType === "new" ? t("Create Profile") : t("Edit Profile")}
contentSx={{ width: 375, pb: 0, maxHeight: "80%" }} contentSx={{ width: 375, pb: 0, maxHeight: 320 }}
okBtn={t("Save")} okBtn={t("Save")}
cancelBtn={t("Cancel")} cancelBtn={t("Cancel")}
onClose={handleClose} onClose={handleClose}

View file

@ -137,14 +137,20 @@ export const ProxyItemMini = (props: Props) => {
e.stopPropagation(); e.stopPropagation();
onDelay(); onDelay();
}} }}
color={delayManager.formatDelayColor(delay)} color={
delay > 500
? "error.main"
: delay < 100
? "success.main"
: "text.secondary"
}
sx={({ palette }) => sx={({ palette }) =>
!proxy.provider !proxy.provider
? { ":hover": { bgcolor: alpha(palette.primary.main, 0.15) } } ? { ":hover": { bgcolor: alpha(palette.primary.main, 0.15) } }
: {} : {}
} }
> >
{delayManager.formatDelay(delay)} {delay > 1e5 ? "Error" : delay > 3000 ? "Timeout" : `${delay}`}
</Widget> </Widget>
)} )}

View file

@ -145,14 +145,20 @@ export const ProxyItem = (props: Props) => {
e.stopPropagation(); e.stopPropagation();
onDelay(); onDelay();
}} }}
color={delayManager.formatDelayColor(delay)} color={
delay > 500
? "error.main"
: delay < 100
? "success.main"
: "text.secondary"
}
sx={({ palette }) => sx={({ palette }) =>
!proxy.provider !proxy.provider
? { ":hover": { bgcolor: alpha(palette.primary.main, 0.15) } } ? { ":hover": { bgcolor: alpha(palette.primary.main, 0.15) } }
: {} : {}
} }
> >
{delayManager.formatDelay(delay)} {delay > 1e5 ? "Error" : delay > 3000 ? "Timeout" : `${delay}ms`}
</Widget> </Widget>
)} )}

View file

@ -25,7 +25,6 @@ export const MiscViewer = forwardRef<DialogRef>((props, ref) => {
enableBuiltinEnhanced: true, enableBuiltinEnhanced: true,
proxyLayoutColumn: 6, proxyLayoutColumn: 6,
defaultLatencyTest: "", defaultLatencyTest: "",
autoLogClean: 0,
}); });
useImperativeHandle(ref, () => ({ useImperativeHandle(ref, () => ({
@ -38,7 +37,6 @@ export const MiscViewer = forwardRef<DialogRef>((props, ref) => {
enableBuiltinEnhanced: verge?.enable_builtin_enhanced ?? true, enableBuiltinEnhanced: verge?.enable_builtin_enhanced ?? true,
proxyLayoutColumn: verge?.proxy_layout_column || 6, proxyLayoutColumn: verge?.proxy_layout_column || 6,
defaultLatencyTest: verge?.default_latency_test || "", defaultLatencyTest: verge?.default_latency_test || "",
autoLogClean: verge?.auto_log_clean || 0,
}); });
}, },
close: () => setOpen(false), close: () => setOpen(false),
@ -53,7 +51,6 @@ export const MiscViewer = forwardRef<DialogRef>((props, ref) => {
enable_builtin_enhanced: values.enableBuiltinEnhanced, enable_builtin_enhanced: values.enableBuiltinEnhanced,
proxy_layout_column: values.proxyLayoutColumn, proxy_layout_column: values.proxyLayoutColumn,
default_latency_test: values.defaultLatencyTest, default_latency_test: values.defaultLatencyTest,
auto_log_clean: values.autoLogClean as any,
}); });
setOpen(false); setOpen(false);
} catch (err: any) { } catch (err: any) {
@ -131,7 +128,7 @@ export const MiscViewer = forwardRef<DialogRef>((props, ref) => {
<ListItemText primary={t("Proxy Layout Column")} /> <ListItemText primary={t("Proxy Layout Column")} />
<Select <Select
size="small" size="small"
sx={{ width: 135, "> div": { py: "7.5px" } }} sx={{ width: 100, "> div": { py: "7.5px" } }}
value={values.proxyLayoutColumn} value={values.proxyLayoutColumn}
onChange={(e) => { onChange={(e) => {
setValues((v) => ({ setValues((v) => ({
@ -151,32 +148,6 @@ export const MiscViewer = forwardRef<DialogRef>((props, ref) => {
</Select> </Select>
</ListItem> </ListItem>
<ListItem sx={{ padding: "5px 2px" }}>
<ListItemText primary={t("Auto Log Clean")} />
<Select
size="small"
sx={{ width: 135, "> div": { py: "7.5px" } }}
value={values.autoLogClean}
onChange={(e) => {
setValues((v) => ({
...v,
autoLogClean: e.target.value as number,
}));
}}
>
{[
{ key: "Never Clean", value: 0 },
{ key: "Retain 7 Days", value: 1 },
{ key: "Retain 30 Days", value: 2 },
{ key: "Retain 90 Days", value: 3 },
].map((i) => (
<MenuItem key={i.value} value={i.value}>
{t(i.key)}
</MenuItem>
))}
</Select>
</ListItem>
<ListItem sx={{ padding: "5px 2px" }}> <ListItem sx={{ padding: "5px 2px" }}>
<ListItemText primary={t("Default Latency Test")} /> <ListItemText primary={t("Default Latency Test")} />
<TextField <TextField

View file

@ -125,11 +125,5 @@
"Enable Clash Fields Filter": "Enable Clash Fields Filter", "Enable Clash Fields Filter": "Enable Clash Fields Filter",
"Enable Builtin Enhanced": "Enable Builtin Enhanced", "Enable Builtin Enhanced": "Enable Builtin Enhanced",
"Proxy Layout Column": "Proxy Layout Column", "Proxy Layout Column": "Proxy Layout Column",
"Default Latency Test": "Default Latency Test", "Default Latency Test": "Default Latency Test"
"Auto Log Clean": "Auto Log Clean",
"Never Clean": "Never Clean",
"Retain 7 Days": "Retain 7 Days",
"Retain 30 Days": "Retain 30 Days",
"Retain 90 Days": "Retain 90 Days"
} }

View file

@ -125,11 +125,5 @@
"Enable Clash Fields Filter": "开启Clash字段过滤", "Enable Clash Fields Filter": "开启Clash字段过滤",
"Enable Builtin Enhanced": "开启内建增强功能", "Enable Builtin Enhanced": "开启内建增强功能",
"Proxy Layout Column": "代理页布局列数", "Proxy Layout Column": "代理页布局列数",
"Default Latency Test": "默认测试链接", "Default Latency Test": "默认测试链接"
"Auto Log Clean": "自动清理日志",
"Never Clean": "不清理",
"Retain 7 Days": "保留7天",
"Retain 30 Days": "保留30天",
"Retain 90 Days": "保留90天"
} }

View file

@ -69,7 +69,7 @@ const ConnectionsPage = () => {
const connections: typeof oldConn = []; const connections: typeof oldConn = [];
const rest = (data.connections || []).filter((each) => { const rest = data.connections?.filter((each) => {
const index = oldConn.findIndex((o) => o.id === each.id); const index = oldConn.findIndex((o) => o.id === each.id);
if (index >= 0 && index < maxLen) { if (index >= 0 && index < maxLen) {

View file

@ -65,7 +65,7 @@ export const getRules = async () => {
/// Get Proxy delay /// Get Proxy delay
export const getProxyDelay = async (name: string, url?: string) => { export const getProxyDelay = async (name: string, url?: string) => {
const params = { const params = {
timeout: 10000, timeout: 5000,
url: url || "http://www.gstatic.com/generate_204", url: url || "http://www.gstatic.com/generate_204",
}; };
const instance = await getAxios(); const instance = await getAxios();

View file

@ -153,11 +153,6 @@ export async function openWebUrl(url: string) {
return invoke<void>("open_web_url", { url }); return invoke<void>("open_web_url", { url });
} }
export async function cmdGetProxyDelay(name: string, url?: string) {
name = encodeURIComponent(name);
return invoke<{ delay: number }>("clash_api_get_proxy_delay", { name, url });
}
/// service mode /// service mode
export async function checkService() { export async function checkService() {

View file

@ -1,4 +1,4 @@
import { cmdGetProxyDelay } from "./cmds"; import { getProxyDelay } from "./api";
const hashKey = (name: string, group: string) => `${group ?? ""}::${name}`; const hashKey = (name: string, group: string) => `${group ?? ""}::${name}`;
@ -74,7 +74,7 @@ class DelayManager {
try { try {
const url = this.getUrl(group); const url = this.getUrl(group);
const result = await cmdGetProxyDelay(name, url); const result = await getProxyDelay(name, url);
delay = result.delay; delay = result.delay;
} catch { } catch {
delay = 1e6; // error delay = 1e6; // error
@ -84,7 +84,7 @@ class DelayManager {
return delay; return delay;
} }
async checkListDelay(nameList: string[], group: string, concurrency = 36) { async checkListDelay(nameList: string[], group: string, concurrency = 6) {
const names = nameList.filter(Boolean); const names = nameList.filter(Boolean);
// 设置正在延迟测试中 // 设置正在延迟测试中
names.forEach((name) => this.setDelay(name, group, -2)); names.forEach((name) => this.setDelay(name, group, -2));
@ -107,21 +107,6 @@ class DelayManager {
for (let i = 0; i < concurrency; ++i) help(); for (let i = 0; i < concurrency; ++i) help();
}); });
} }
formatDelay(delay: number) {
if (delay < 0) return "-";
if (delay > 1e5) return "Error";
if (delay >= 10000) return "Timeout"; // 10s
return `${delay}`;
}
formatDelayColor(delay: number) {
if (delay <= 0) return "text.secondary";
if (delay >= 10000) return "error.main";
if (delay > 500) return "warning.main";
if (delay > 100) return "text.secondary";
return "success.main";
}
} }
export default new DelayManager(); export default new DelayManager();

View file

@ -186,7 +186,6 @@ interface IVergeConfig {
default_latency_test?: string; default_latency_test?: string;
enable_clash_fields?: boolean; enable_clash_fields?: boolean;
enable_builtin_enhanced?: boolean; enable_builtin_enhanced?: boolean;
auto_log_clean?: 0 | 1 | 2 | 3;
proxy_layout_column?: number; proxy_layout_column?: number;
} }