Skip to content

Commit

Permalink
feat: support more options for remote profile
Browse files Browse the repository at this point in the history
  • Loading branch information
zzzgydi committed Mar 9, 2022
1 parent 04c754c commit fe1fea6
Show file tree
Hide file tree
Showing 7 changed files with 135 additions and 28 deletions.
10 changes: 5 additions & 5 deletions src-tauri/src/cmds.rs
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
use crate::{
core::{ClashInfo, PrfItem, Profiles, VergeConfig},
core::{ClashInfo, PrfItem, PrfOption, Profiles, VergeConfig},
states::{ClashState, ProfilesState, VergeState},
utils::{dirs, sysopt::SysProxyConfig},
};
Expand Down Expand Up @@ -28,10 +28,10 @@ pub fn sync_profiles(profiles_state: State<'_, ProfilesState>) -> Result<(), Str
#[tauri::command]
pub async fn import_profile(
url: String,
with_proxy: bool,
option: Option<PrfOption>,
profiles_state: State<'_, ProfilesState>,
) -> Result<(), String> {
let item = wrap_err!(PrfItem::from_url(&url, None, None, with_proxy).await)?;
let item = wrap_err!(PrfItem::from_url(&url, None, None, option).await)?;

let mut profiles = profiles_state.0.lock().unwrap();
wrap_err!(profiles.append_item(item))
Expand All @@ -55,7 +55,7 @@ pub async fn create_profile(
#[tauri::command]
pub async fn update_profile(
index: String,
with_proxy: bool,
option: Option<PrfOption>,
clash_state: State<'_, ClashState>,
profiles_state: State<'_, ProfilesState>,
) -> Result<(), String> {
Expand All @@ -71,7 +71,7 @@ pub async fn update_profile(
item.url.clone().unwrap()
};

let item = wrap_err!(PrfItem::from_url(&url, None, None, with_proxy).await)?;
let item = wrap_err!(PrfItem::from_url(&url, None, None, option).await)?;

let mut profiles = profiles_state.0.lock().unwrap();
wrap_err!(profiles.update_item(index.clone(), item))?;
Expand Down
42 changes: 39 additions & 3 deletions src-tauri/src/core/profiles.rs
Original file line number Diff line number Diff line change
Expand Up @@ -31,13 +31,17 @@ pub struct PrfItem {
#[serde(skip_serializing_if = "Option::is_none")]
pub selected: Option<Vec<PrfSelected>>,

/// user info
/// subscription user info
#[serde(skip_serializing_if = "Option::is_none")]
pub extra: Option<PrfExtra>,

/// updated time
pub updated: Option<usize>,

/// some options of the item
#[serde(skip_serializing_if = "Option::is_none")]
pub option: Option<PrfOption>,

/// the file data
#[serde(skip)]
pub file_data: Option<String>,
Expand All @@ -57,6 +61,18 @@ pub struct PrfExtra {
pub expire: usize,
}

#[derive(Default, Debug, Clone, Deserialize, Serialize)]
pub struct PrfOption {
/// for `remote` profile's http request
/// see issue #13
#[serde(skip_serializing_if = "Option::is_none")]
pub user_agent: Option<String>,

/// for `remote` profile
#[serde(skip_serializing_if = "Option::is_none")]
pub with_proxy: Option<bool>,
}

impl Default for PrfItem {
fn default() -> Self {
PrfItem {
Expand All @@ -69,6 +85,7 @@ impl Default for PrfItem {
selected: None,
extra: None,
updated: None,
option: None,
file_data: None,
}
}
Expand All @@ -90,7 +107,7 @@ impl PrfItem {
let url = item.url.as_ref().unwrap().as_str();
let name = item.name;
let desc = item.desc;
PrfItem::from_url(url, name, desc, false).await
PrfItem::from_url(url, name, desc, item.option).await
}
"local" => {
let name = item.name.unwrap_or("Local File".into());
Expand Down Expand Up @@ -126,6 +143,7 @@ impl PrfItem {
url: None,
selected: None,
extra: None,
option: None,
updated: Some(help::get_now()),
file_data: Some(tmpl::ITEM_LOCAL.into()),
})
Expand All @@ -137,13 +155,25 @@ impl PrfItem {
url: &str,
name: Option<String>,
desc: Option<String>,
with_proxy: bool,
option: Option<PrfOption>,
) -> Result<PrfItem> {
let with_proxy = match option.as_ref() {
Some(opt) => opt.with_proxy.unwrap_or(false),
None => false,
};
let user_agent = match option.as_ref() {
Some(opt) => opt.user_agent.clone(),
None => None,
};

let mut builder = reqwest::ClientBuilder::new();

if !with_proxy {
builder = builder.no_proxy();
}
if let Some(user_agent) = user_agent {
builder = builder.user_agent(user_agent);
}

let resp = builder.build()?.get(url).send().await?;
let header = resp.headers();
Expand Down Expand Up @@ -177,6 +207,7 @@ impl PrfItem {
url: Some(url.into()),
selected: None,
extra,
option,
updated: Some(help::get_now()),
file_data: Some(data),
})
Expand All @@ -197,6 +228,7 @@ impl PrfItem {
url: None,
selected: None,
extra: None,
option: None,
updated: Some(help::get_now()),
file_data: Some(tmpl::ITEM_MERGE.into()),
})
Expand All @@ -217,6 +249,7 @@ impl PrfItem {
url: None,
selected: None,
extra: None,
option: None,
updated: Some(help::get_now()),
file_data: Some(tmpl::ITEM_SCRIPT.into()),
})
Expand Down Expand Up @@ -382,6 +415,9 @@ impl Profiles {
patch!(each, item, extra);
patch!(each, item, updated);

// can be removed
each.option = item.option;

self.items = Some(items);
return self.save_file();
}
Expand Down
34 changes: 31 additions & 3 deletions src/components/profile/profile-edit.tsx
Original file line number Diff line number Diff line change
@@ -1,14 +1,16 @@
import { mutate } from "swr";
import { useEffect } from "react";
import { useEffect, useState } from "react";
import { useLockFn, useSetState } from "ahooks";
import {
Button,
Dialog,
DialogActions,
DialogContent,
DialogTitle,
IconButton,
TextField,
} from "@mui/material";
import { Settings } from "@mui/icons-material";
import { CmdType } from "../../services/types";
import { patchProfile } from "../../services/cmds";
import Notice from "../base/base-notice";
Expand All @@ -20,26 +22,33 @@ interface Props {
}

// edit the profile item
// remote / local file / merge / script
const ProfileEdit = (props: Props) => {
const { open, itemData, onClose } = props;
const [form, setForm] = useSetState({ ...itemData });
const [option, setOption] = useSetState(itemData.option ?? {});
const [showOpt, setShowOpt] = useState(!!itemData.option);

useEffect(() => {
if (itemData) {
setForm({ ...itemData });
setOption(itemData.option ?? {});
setShowOpt(!!itemData.option?.user_agent);
}
}, [itemData]);

const onUpdate = useLockFn(async () => {
try {
const { uid } = itemData;
const { name, desc, url } = form;
const option_ = showOpt ? option : undefined;

if (itemData.type === "remote" && !url) {
throw new Error("Remote URL should not be null");
}

await patchProfile(uid, { uid, name, desc, url });
await patchProfile(uid, { uid, name, desc, url, option: option_ });
setShowOpt(false);
mutate("getProfiles");
onClose();
} catch (err: any) {
Expand Down Expand Up @@ -94,9 +103,28 @@ const ProfileEdit = (props: Props) => {
onChange={(e) => setForm({ url: e.target.value })}
/>
)}

{showOpt && (
<TextField
{...textFieldProps}
label="User Agent"
value={option.user_agent}
onChange={(e) => setOption({ user_agent: e.target.value })}
/>
)}
</DialogContent>

<DialogActions sx={{ px: 2, pb: 2 }}>
<DialogActions sx={{ px: 2, pb: 2, position: "relative" }}>
{form.type === "remote" && (
<IconButton
size="small"
sx={{ position: "absolute", left: 18 }}
onClick={() => setShowOpt((o) => !o)}
>
<Settings />
</IconButton>
)}

<Button onClick={onClose}>Cancel</Button>
<Button onClick={onUpdate} variant="contained">
Update
Expand Down
2 changes: 1 addition & 1 deletion src/components/profile/profile-item.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -98,7 +98,7 @@ const ProfileItem = (props: Props) => {
if (loading) return;
setLoading(true);
try {
await updateProfile(itemData.uid, withProxy);
await updateProfile(itemData.uid, { with_proxy: withProxy });
setLoading(false);
mutate("getProfiles");
} catch (err: any) {
Expand Down
57 changes: 44 additions & 13 deletions src/components/profile/profile-new.tsx
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
import { useState } from "react";
import { useSWRConfig } from "swr";
import { useLockFn, useSetState } from "ahooks";
import {
Expand All @@ -7,11 +8,13 @@ import {
DialogContent,
DialogTitle,
FormControl,
IconButton,
InputLabel,
MenuItem,
Select,
TextField,
} from "@mui/material";
import { Settings } from "@mui/icons-material";
import { createProfile } from "../../services/cmds";
import Notice from "../base/base-notice";

Expand All @@ -27,12 +30,17 @@ const ProfileNew = (props: Props) => {

const { mutate } = useSWRConfig();
const [form, setForm] = useSetState({
type: "remote",
name: "",
desc: "",
type: "remote",
url: "",
});

const [showOpt, setShowOpt] = useState(false);
const [option, setOption] = useSetState({
user_agent: "",
}); // able to add more option

const onCreate = useLockFn(async () => {
if (!form.type) {
Notice.error("`Type` should not be null");
Expand All @@ -43,11 +51,15 @@ const ProfileNew = (props: Props) => {
const name = form.name || `${form.type} file`;

if (form.type === "remote" && !form.url) {
throw new Error("Remote URL should not be null");
throw new Error("The URL should not be null");
}

await createProfile({ ...form, name });
setForm({ name: "", desc: "", type: "remote", url: "" });
const option_ = showOpt ? option : undefined;
await createProfile({ ...form, name, option: option_ });
setForm({ type: "remote", name: "", desc: "", url: "" });
setOption({ user_agent: "" });
setShowOpt(false);

mutate("getProfiles");
onClose();
} catch (err: any) {
Expand All @@ -67,17 +79,10 @@ const ProfileNew = (props: Props) => {
<DialogTitle sx={{ pb: 0.5 }}>Create Profile</DialogTitle>

<DialogContent sx={{ width: 336, pb: 1 }}>
<TextField
{...textFieldProps}
autoFocus
label="Name"
value={form.name}
onChange={(e) => setForm({ name: e.target.value })}
/>

<FormControl size="small" fullWidth sx={{ mt: 2, mb: 1 }}>
<InputLabel>Type</InputLabel>
<Select
autoFocus
label="Type"
value={form.type}
onChange={(e) => setForm({ type: e.target.value })}
Expand All @@ -89,6 +94,13 @@ const ProfileNew = (props: Props) => {
</Select>
</FormControl>

<TextField
{...textFieldProps}
label="Name"
value={form.name}
onChange={(e) => setForm({ name: e.target.value })}
/>

<TextField
{...textFieldProps}
label="Descriptions"
Expand All @@ -104,9 +116,28 @@ const ProfileNew = (props: Props) => {
onChange={(e) => setForm({ url: e.target.value })}
/>
)}

{showOpt && (
<TextField
{...textFieldProps}
label="User Agent"
value={option.user_agent}
onChange={(e) => setOption({ user_agent: e.target.value })}
/>
)}
</DialogContent>

<DialogActions sx={{ px: 2, pb: 2 }}>
<DialogActions sx={{ px: 2, pb: 2, position: "relative" }}>
{form.type === "remote" && (
<IconButton
size="small"
sx={{ position: "absolute", left: 18 }}
onClick={() => setShowOpt((o) => !o)}
>
<Settings />
</IconButton>
)}

<Button onClick={onClose}>Cancel</Button>
<Button onClick={onCreate} variant="contained">
Create
Expand Down
12 changes: 9 additions & 3 deletions src/services/cmds.ts
Original file line number Diff line number Diff line change
Expand Up @@ -22,11 +22,17 @@ export async function viewProfile(index: string) {
}

export async function importProfile(url: string) {
return invoke<void>("import_profile", { url, withProxy: true });
return invoke<void>("import_profile", {
url,
option: { with_proxy: true },
});
}

export async function updateProfile(index: string, withProxy: boolean) {
return invoke<void>("update_profile", { index, withProxy });
export async function updateProfile(
index: string,
option?: CmdType.ProfileOption
) {
return invoke<void>("update_profile", { index, option });
}

export async function deleteProfile(index: string) {
Expand Down
Loading

0 comments on commit fe1fea6

Please sign in to comment.