feat(desktop): add tauri desktop client for cs.workyai.cn
7
desktop-client/src-tauri/.gitignore
vendored
Normal file
@@ -0,0 +1,7 @@
|
||||
# Generated by Cargo
|
||||
# will have compiled files and executables
|
||||
/target/
|
||||
|
||||
# Generated by Tauri
|
||||
# will have schema files for capabilities auto-completion
|
||||
/gen/schemas
|
||||
5665
desktop-client/src-tauri/Cargo.lock
generated
Normal file
26
desktop-client/src-tauri/Cargo.toml
Normal file
@@ -0,0 +1,26 @@
|
||||
[package]
|
||||
name = "desktop-client"
|
||||
version = "0.1.0"
|
||||
description = "A Tauri App"
|
||||
authors = ["you"]
|
||||
edition = "2021"
|
||||
|
||||
# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
|
||||
|
||||
[lib]
|
||||
# The `_lib` suffix may seem redundant but it is necessary
|
||||
# to make the lib name unique and wouldn't conflict with the bin name.
|
||||
# This seems to be only an issue on Windows, see https://github.com/rust-lang/cargo/issues/8519
|
||||
name = "desktop_client_lib"
|
||||
crate-type = ["staticlib", "cdylib", "rlib"]
|
||||
|
||||
[build-dependencies]
|
||||
tauri-build = { version = "2", features = [] }
|
||||
|
||||
[dependencies]
|
||||
tauri = { version = "2", features = [] }
|
||||
tauri-plugin-opener = "2"
|
||||
serde = { version = "1", features = ["derive"] }
|
||||
serde_json = "1"
|
||||
reqwest = { version = "0.12", default-features = false, features = ["json", "cookies", "rustls-tls"] }
|
||||
urlencoding = "2.1"
|
||||
3
desktop-client/src-tauri/build.rs
Normal file
@@ -0,0 +1,3 @@
|
||||
fn main() {
|
||||
tauri_build::build()
|
||||
}
|
||||
10
desktop-client/src-tauri/capabilities/default.json
Normal file
@@ -0,0 +1,10 @@
|
||||
{
|
||||
"$schema": "../gen/schemas/desktop-schema.json",
|
||||
"identifier": "default",
|
||||
"description": "Capability for the main window",
|
||||
"windows": ["main"],
|
||||
"permissions": [
|
||||
"core:default",
|
||||
"opener:default"
|
||||
]
|
||||
}
|
||||
BIN
desktop-client/src-tauri/icons/128x128.png
Normal file
|
After Width: | Height: | Size: 3.4 KiB |
BIN
desktop-client/src-tauri/icons/128x128@2x.png
Normal file
|
After Width: | Height: | Size: 6.8 KiB |
BIN
desktop-client/src-tauri/icons/32x32.png
Normal file
|
After Width: | Height: | Size: 974 B |
BIN
desktop-client/src-tauri/icons/Square107x107Logo.png
Normal file
|
After Width: | Height: | Size: 2.8 KiB |
BIN
desktop-client/src-tauri/icons/Square142x142Logo.png
Normal file
|
After Width: | Height: | Size: 3.8 KiB |
BIN
desktop-client/src-tauri/icons/Square150x150Logo.png
Normal file
|
After Width: | Height: | Size: 3.9 KiB |
BIN
desktop-client/src-tauri/icons/Square284x284Logo.png
Normal file
|
After Width: | Height: | Size: 7.6 KiB |
BIN
desktop-client/src-tauri/icons/Square30x30Logo.png
Normal file
|
After Width: | Height: | Size: 903 B |
BIN
desktop-client/src-tauri/icons/Square310x310Logo.png
Normal file
|
After Width: | Height: | Size: 8.4 KiB |
BIN
desktop-client/src-tauri/icons/Square44x44Logo.png
Normal file
|
After Width: | Height: | Size: 1.3 KiB |
BIN
desktop-client/src-tauri/icons/Square71x71Logo.png
Normal file
|
After Width: | Height: | Size: 2.0 KiB |
BIN
desktop-client/src-tauri/icons/Square89x89Logo.png
Normal file
|
After Width: | Height: | Size: 2.4 KiB |
BIN
desktop-client/src-tauri/icons/StoreLogo.png
Normal file
|
After Width: | Height: | Size: 1.5 KiB |
BIN
desktop-client/src-tauri/icons/icon.icns
Normal file
BIN
desktop-client/src-tauri/icons/icon.ico
Normal file
|
After Width: | Height: | Size: 85 KiB |
BIN
desktop-client/src-tauri/icons/icon.png
Normal file
|
After Width: | Height: | Size: 14 KiB |
352
desktop-client/src-tauri/src/lib.rs
Normal file
@@ -0,0 +1,352 @@
|
||||
use reqwest::Method;
|
||||
use reqwest::StatusCode;
|
||||
use serde::Serialize;
|
||||
use serde_json::{Map, Value};
|
||||
use std::time::Duration;
|
||||
|
||||
struct ApiState {
|
||||
client: reqwest::Client,
|
||||
}
|
||||
|
||||
#[derive(Debug, Serialize)]
|
||||
#[serde(rename_all = "camelCase")]
|
||||
struct BridgeResponse {
|
||||
ok: bool,
|
||||
status: u16,
|
||||
data: Value,
|
||||
}
|
||||
|
||||
fn normalize_base_url(base_url: &str) -> String {
|
||||
let trimmed = base_url.trim();
|
||||
if trimmed.is_empty() {
|
||||
return String::new();
|
||||
}
|
||||
trimmed.trim_end_matches('/').to_string()
|
||||
}
|
||||
|
||||
fn join_api_url(base_url: &str, path: &str) -> String {
|
||||
format!("{}{}", normalize_base_url(base_url), path)
|
||||
}
|
||||
|
||||
fn fallback_json(status: StatusCode, text: &str) -> Value {
|
||||
let mut data = Map::new();
|
||||
data.insert("success".to_string(), Value::Bool(status.is_success()));
|
||||
data.insert(
|
||||
"message".to_string(),
|
||||
Value::String(if text.trim().is_empty() {
|
||||
format!("HTTP {}", status.as_u16())
|
||||
} else {
|
||||
text.to_string()
|
||||
}),
|
||||
);
|
||||
Value::Object(data)
|
||||
}
|
||||
|
||||
async fn request_json(
|
||||
client: &reqwest::Client,
|
||||
method: Method,
|
||||
url: String,
|
||||
body: Option<Value>,
|
||||
csrf_token: Option<String>,
|
||||
) -> Result<BridgeResponse, String> {
|
||||
if url.is_empty() {
|
||||
return Err("API 地址不能为空".to_string());
|
||||
}
|
||||
|
||||
let mut request = client
|
||||
.request(method, &url)
|
||||
.header("Accept", "application/json")
|
||||
.header("Content-Type", "application/json");
|
||||
|
||||
if let Some(csrf) = csrf_token {
|
||||
request = request.header("X-CSRF-Token", csrf);
|
||||
}
|
||||
|
||||
if let Some(payload) = body {
|
||||
request = request.json(&payload);
|
||||
}
|
||||
|
||||
let response = request
|
||||
.send()
|
||||
.await
|
||||
.map_err(|err| format!("请求失败: {}", err))?;
|
||||
|
||||
let status = response.status();
|
||||
let text = response
|
||||
.text()
|
||||
.await
|
||||
.map_err(|err| format!("读取响应失败: {}", err))?;
|
||||
let data = match serde_json::from_str::<Value>(&text) {
|
||||
Ok(parsed) => parsed,
|
||||
Err(_) => fallback_json(status, &text),
|
||||
};
|
||||
|
||||
Ok(BridgeResponse {
|
||||
ok: status.is_success(),
|
||||
status: status.as_u16(),
|
||||
data,
|
||||
})
|
||||
}
|
||||
|
||||
async fn fetch_csrf_token(client: &reqwest::Client, base_url: &str) -> Result<Option<String>, String> {
|
||||
let response = request_json(
|
||||
client,
|
||||
Method::GET,
|
||||
join_api_url(base_url, "/api/csrf-token"),
|
||||
None,
|
||||
None,
|
||||
)
|
||||
.await?;
|
||||
|
||||
let token = response
|
||||
.data
|
||||
.get("csrfToken")
|
||||
.and_then(Value::as_str)
|
||||
.map(|v| v.to_string());
|
||||
|
||||
Ok(token)
|
||||
}
|
||||
|
||||
async fn request_with_optional_csrf(
|
||||
client: &reqwest::Client,
|
||||
method: Method,
|
||||
base_url: &str,
|
||||
path: &str,
|
||||
body: Option<Value>,
|
||||
need_csrf: bool,
|
||||
) -> Result<BridgeResponse, String> {
|
||||
let csrf_token = if need_csrf {
|
||||
fetch_csrf_token(client, base_url).await?
|
||||
} else {
|
||||
None
|
||||
};
|
||||
|
||||
request_json(
|
||||
client,
|
||||
method,
|
||||
join_api_url(base_url, path),
|
||||
body,
|
||||
csrf_token,
|
||||
)
|
||||
.await
|
||||
}
|
||||
|
||||
#[tauri::command]
|
||||
async fn api_login(
|
||||
state: tauri::State<'_, ApiState>,
|
||||
base_url: String,
|
||||
username: String,
|
||||
password: String,
|
||||
captcha: Option<String>,
|
||||
) -> Result<BridgeResponse, String> {
|
||||
let mut body = Map::new();
|
||||
body.insert("username".to_string(), Value::String(username));
|
||||
body.insert("password".to_string(), Value::String(password));
|
||||
if let Some(value) = captcha {
|
||||
if !value.trim().is_empty() {
|
||||
body.insert("captcha".to_string(), Value::String(value));
|
||||
}
|
||||
}
|
||||
|
||||
request_with_optional_csrf(
|
||||
&state.client,
|
||||
Method::POST,
|
||||
&base_url,
|
||||
"/api/login",
|
||||
Some(Value::Object(body)),
|
||||
false,
|
||||
)
|
||||
.await
|
||||
}
|
||||
|
||||
#[tauri::command]
|
||||
async fn api_get_profile(
|
||||
state: tauri::State<'_, ApiState>,
|
||||
base_url: String,
|
||||
) -> Result<BridgeResponse, String> {
|
||||
request_with_optional_csrf(
|
||||
&state.client,
|
||||
Method::GET,
|
||||
&base_url,
|
||||
"/api/user/profile",
|
||||
None,
|
||||
false,
|
||||
)
|
||||
.await
|
||||
}
|
||||
|
||||
#[tauri::command]
|
||||
async fn api_list_files(
|
||||
state: tauri::State<'_, ApiState>,
|
||||
base_url: String,
|
||||
path: String,
|
||||
) -> Result<BridgeResponse, String> {
|
||||
let normalized = if path.trim().is_empty() {
|
||||
"/".to_string()
|
||||
} else {
|
||||
path
|
||||
};
|
||||
let encoded = urlencoding::encode(&normalized);
|
||||
let api_url = format!("{}?path={}", join_api_url(&base_url, "/api/files"), encoded);
|
||||
|
||||
request_json(&state.client, Method::GET, api_url, None, None).await
|
||||
}
|
||||
|
||||
#[tauri::command]
|
||||
async fn api_logout(
|
||||
state: tauri::State<'_, ApiState>,
|
||||
base_url: String,
|
||||
) -> Result<BridgeResponse, String> {
|
||||
request_with_optional_csrf(
|
||||
&state.client,
|
||||
Method::POST,
|
||||
&base_url,
|
||||
"/api/logout",
|
||||
Some(Value::Object(Map::new())),
|
||||
true,
|
||||
)
|
||||
.await
|
||||
}
|
||||
|
||||
#[tauri::command]
|
||||
async fn api_search_files(
|
||||
state: tauri::State<'_, ApiState>,
|
||||
base_url: String,
|
||||
path: String,
|
||||
keyword: String,
|
||||
search_type: Option<String>,
|
||||
limit: Option<u32>,
|
||||
) -> Result<BridgeResponse, String> {
|
||||
let normalized_path = if path.trim().is_empty() {
|
||||
"/".to_string()
|
||||
} else {
|
||||
path
|
||||
};
|
||||
let kind = search_type
|
||||
.unwrap_or_else(|| "all".to_string())
|
||||
.trim()
|
||||
.to_string();
|
||||
let max_limit = limit.unwrap_or(100).clamp(1, 500);
|
||||
|
||||
let api_url = format!(
|
||||
"{}?path={}&keyword={}&type={}&limit={}",
|
||||
join_api_url(&base_url, "/api/files/search"),
|
||||
urlencoding::encode(&normalized_path),
|
||||
urlencoding::encode(&keyword),
|
||||
urlencoding::encode(&kind),
|
||||
max_limit
|
||||
);
|
||||
|
||||
request_json(&state.client, Method::GET, api_url, None, None).await
|
||||
}
|
||||
|
||||
#[tauri::command]
|
||||
async fn api_mkdir(
|
||||
state: tauri::State<'_, ApiState>,
|
||||
base_url: String,
|
||||
path: String,
|
||||
folder_name: String,
|
||||
) -> Result<BridgeResponse, String> {
|
||||
let mut body = Map::new();
|
||||
body.insert("path".to_string(), Value::String(path));
|
||||
body.insert("folderName".to_string(), Value::String(folder_name));
|
||||
|
||||
request_with_optional_csrf(
|
||||
&state.client,
|
||||
Method::POST,
|
||||
&base_url,
|
||||
"/api/files/mkdir",
|
||||
Some(Value::Object(body)),
|
||||
true,
|
||||
)
|
||||
.await
|
||||
}
|
||||
|
||||
#[tauri::command]
|
||||
async fn api_rename_file(
|
||||
state: tauri::State<'_, ApiState>,
|
||||
base_url: String,
|
||||
path: String,
|
||||
old_name: String,
|
||||
new_name: String,
|
||||
) -> Result<BridgeResponse, String> {
|
||||
let mut body = Map::new();
|
||||
body.insert("path".to_string(), Value::String(path));
|
||||
body.insert("oldName".to_string(), Value::String(old_name));
|
||||
body.insert("newName".to_string(), Value::String(new_name));
|
||||
|
||||
request_with_optional_csrf(
|
||||
&state.client,
|
||||
Method::POST,
|
||||
&base_url,
|
||||
"/api/files/rename",
|
||||
Some(Value::Object(body)),
|
||||
true,
|
||||
)
|
||||
.await
|
||||
}
|
||||
|
||||
#[tauri::command]
|
||||
async fn api_delete_file(
|
||||
state: tauri::State<'_, ApiState>,
|
||||
base_url: String,
|
||||
path: String,
|
||||
file_name: String,
|
||||
) -> Result<BridgeResponse, String> {
|
||||
let mut body = Map::new();
|
||||
body.insert("path".to_string(), Value::String(path));
|
||||
body.insert("fileName".to_string(), Value::String(file_name));
|
||||
|
||||
request_with_optional_csrf(
|
||||
&state.client,
|
||||
Method::POST,
|
||||
&base_url,
|
||||
"/api/files/delete",
|
||||
Some(Value::Object(body)),
|
||||
true,
|
||||
)
|
||||
.await
|
||||
}
|
||||
|
||||
#[tauri::command]
|
||||
async fn api_get_download_url(
|
||||
state: tauri::State<'_, ApiState>,
|
||||
base_url: String,
|
||||
path: String,
|
||||
mode: Option<String>,
|
||||
) -> Result<BridgeResponse, String> {
|
||||
let normalized_mode = mode.unwrap_or_else(|| "download".to_string());
|
||||
let api_url = format!(
|
||||
"{}?path={}&mode={}",
|
||||
join_api_url(&base_url, "/api/files/download-url"),
|
||||
urlencoding::encode(&path),
|
||||
urlencoding::encode(&normalized_mode)
|
||||
);
|
||||
request_json(&state.client, Method::GET, api_url, None, None).await
|
||||
}
|
||||
|
||||
#[cfg_attr(mobile, tauri::mobile_entry_point)]
|
||||
pub fn run() {
|
||||
let client = reqwest::Client::builder()
|
||||
.cookie_store(true)
|
||||
.timeout(Duration::from_secs(30))
|
||||
.build()
|
||||
.expect("failed to build reqwest client");
|
||||
|
||||
tauri::Builder::default()
|
||||
.manage(ApiState { client })
|
||||
.plugin(tauri_plugin_opener::init())
|
||||
.invoke_handler(tauri::generate_handler![
|
||||
api_login,
|
||||
api_get_profile,
|
||||
api_list_files,
|
||||
api_logout,
|
||||
api_search_files,
|
||||
api_mkdir,
|
||||
api_rename_file,
|
||||
api_delete_file,
|
||||
api_get_download_url
|
||||
])
|
||||
.run(tauri::generate_context!())
|
||||
.expect("error while running tauri application");
|
||||
}
|
||||
6
desktop-client/src-tauri/src/main.rs
Normal file
@@ -0,0 +1,6 @@
|
||||
// Prevents additional console window on Windows in release, DO NOT REMOVE!!
|
||||
#![cfg_attr(not(debug_assertions), windows_subsystem = "windows")]
|
||||
|
||||
fn main() {
|
||||
desktop_client_lib::run()
|
||||
}
|
||||
37
desktop-client/src-tauri/tauri.conf.json
Normal file
@@ -0,0 +1,37 @@
|
||||
{
|
||||
"$schema": "https://schema.tauri.app/config/2",
|
||||
"productName": "wanwan-cloud-desktop",
|
||||
"version": "0.1.0",
|
||||
"identifier": "cn.workyai.wanwancloud.desktop",
|
||||
"build": {
|
||||
"beforeDevCommand": "npm run dev",
|
||||
"devUrl": "http://localhost:1420",
|
||||
"beforeBuildCommand": "npm run build",
|
||||
"frontendDist": "../dist"
|
||||
},
|
||||
"app": {
|
||||
"windows": [
|
||||
{
|
||||
"title": "Wanwan Cloud Desktop",
|
||||
"width": 1360,
|
||||
"height": 860,
|
||||
"minWidth": 1120,
|
||||
"minHeight": 720
|
||||
}
|
||||
],
|
||||
"security": {
|
||||
"csp": null
|
||||
}
|
||||
},
|
||||
"bundle": {
|
||||
"active": true,
|
||||
"targets": "all",
|
||||
"icon": [
|
||||
"icons/32x32.png",
|
||||
"icons/128x128.png",
|
||||
"icons/128x128@2x.png",
|
||||
"icons/icon.icns",
|
||||
"icons/icon.ico"
|
||||
]
|
||||
}
|
||||
}
|
||||