Files
openbridge-token-usage-viewer/src-tauri/AGENTS.md
imbytecat 8004bf51a8 feat: 添加 Tauri 桌面应用支持及完整开发环境配置
- 添加 Rust、Tauri 和 Even Better TOML 语言支持扩展以增强开发环境功能。
- 修改构建配置中的输出目录为 src-tauri/binaries,以适配新的构建输出路径。
- 添加 Tauri CLI 及其各平台兼容的可执行文件依赖,以支持多平台应用开发。
- 添加最新版本的 Rust 工具支持并移除旧版本的 Bun。
- 添加 Tauri CLI 工具以支持桌面应用开发
- 添加 Tauri 项目生成文件和构建产物的忽略规则
- 新增项目开发指南文档,明确 Tauri 桌面应用的构建流程、代码风格、Tauri 特定规范及最佳实践。
- 添加 Tauri 构建脚本以支持应用打包和构建流程
- 添加主窗口权限配置,允许执行名为 binaries/server 的侧车二进制文件。
- 生成新的 Cargo.lock 文件以更新项目依赖项的版本和校验和
- 添加 Tauri 应用的 Cargo 项目配置,包含 shell 插件和异步运行时支持。
- 添加32x32像素图标文件以支持应用图标显示
- 添加128x128像素图标文件以支持应用图标显示
- 添加高分辨率图标文件以支持高清显示
- 添加应用图标文件以支持 macOS 平台的图标显示
- 添加图标文件以用于应用程序的视觉标识
- 添加应用图标文件
- 添加方形30x30像素的图标文件以支持应用启动画面和系统显示
- 添加方形44x44像素应用图标文件
- 添加方形71x71像素应用图标以支持不同平台显示需求
- 添加新的方形89x89像素应用图标文件
- 添加新的正方形107x107像素应用图标文件
- 添加方形142x142像素的图标文件以支持应用启动画面和系统显示
- 添加新的方形150x150像素图标文件以用于应用程序界面显示
- 添加新的方形284x284像素应用图标文件
- 添加新的方形310x310像素应用图标文件
- 添加应用商店图标文件
- 添加原生桌面功能命令模块,包含示例问候函数以支持从 Rust 向前端传递消息。
- 添加侧车进程管理功能并注册全局状态与生命周期事件处理
- 添加启动配置以在发布模式下防止Windows出现额外控制台窗口,并启动应用主函数。
- 添加 Sidecar 服务启动与管理功能,自动查找可用端口、启动后端服务并创建主窗口,同时支持超时检测和进程清理。
- 添加 Tauri 项目配置文件并设置应用基本信息、打包选项及图标路径
2026-01-18 15:09:20 +08:00

8.6 KiB
Raw Blame History

AGENTS.md - Tauri Shell 项目开发指南

本文档为 AI 编程助手和开发者提供项目规范、构建命令和代码风格指南。

项目概览

  • 项目类型: Tauri v2 桌面应用(轻量级壳子)
  • 后端: Rust (Edition 2021)
  • 架构: Sidecar 模式 - Sidecar Server 承载主要业务逻辑
  • 设计理念: Tauri 仅提供原生桌面能力文件对话框、系统通知等Web 逻辑全部由 Sidecar Server 处理
  • 异步运行时: Tokio
  • Rust 版本: 1.92.0+
  • 工具管理: 使用 mise 管理 Rust 和 Tauri CLI 版本(见 mise.toml

构建、测试、运行命令

开发运行

# 开发模式运行 (带 hot-reload)
tauri dev

# 仅运行 Rust 二进制 (不推荐,需要手动启动 Sidecar Server)
cargo run

构建

# 开发构建 (debug mode)
cargo build

# 生产构建
cargo build --release

# Tauri 应用打包 (生成安装程序)
tauri build

代码检查

# 编译检查 (不生成二进制)
cargo check

# Clippy 代码质量检查
cargo clippy

# Clippy 严格模式 (所有警告视为错误)
cargo clippy -- -D warnings

# 代码格式化检查
cargo fmt -- --check

# 自动格式化代码
cargo fmt

测试

# 运行所有测试
cargo test

# 运行单个测试 (按名称过滤)
cargo test test_function_name

# 运行特定模块的测试
cargo test module_name::

# 显示测试输出 (包括 println!)
cargo test -- --nocapture

# 运行单个测试并显示输出
cargo test test_name -- --nocapture

清理

# 清理构建产物
cargo clean

项目结构

tauri-shell/
├── src/
│   ├── main.rs          # 入口文件 (仅调用 lib::run)
│   ├── lib.rs           # 核心应用逻辑 (注册插件、命令、状态)
│   ├── commands/
│   │   └── mod.rs       # 原生桌面功能命令 (文件对话框、通知等)
│   └── sidecar.rs       # Sidecar 进程管理 (启动、端口扫描、清理)
├── binaries/            # Sidecar 二进制文件
│   └── server-*   # Sidecar Server 可执行文件 (示例: server)
├── capabilities/        # Tauri v2 权限配置
│   └── default.json
├── icons/               # 应用图标资源
├── gen/schemas/         # 自动生成的 Schema (不要手动编辑)
├── Cargo.toml          # Rust 项目配置
├── tauri.conf.json     # Tauri 应用配置
├── build.rs            # Rust 构建脚本
└── mise.toml           # 开发工具版本管理

Rust 代码风格指南

导入 (Imports)

  • 使用标准库、外部 crate、当前 crate 的顺序,用空行分隔
  • 按字母顺序排列
  • 优先使用具体导入而非通配符 *
// ✅ 推荐
use std::sync::Mutex;
use std::time::Duration;

use tauri::Manager;
use tauri_plugin_shell::ShellExt;
use tauri_plugin_shell::process::{CommandEvent, CommandChild};

// ❌ 避免
use tauri::*;

命名规范

  • 函数和变量: snake_case
  • 类型、结构体、枚举、Trait: PascalCase
  • 常量和静态变量: SCREAMING_SNAKE_CASE
  • 生命周期参数: 简短小写字母,如 'a, 'b
// ✅ 推荐
struct SidecarProcess(Mutex<Option<CommandChild>>);
const DEFAULT_PORT: u16 = 3000;
async fn find_available_port(start: u16) -> u16 { }

// ❌ 避免
struct sidecar_process { }
const defaultPort: u16 = 3000;

类型注解

  • 函数参数必须有类型注解
  • 函数返回值必须明确声明 (除非返回 ())
  • 优先使用具体类型而非 impl Trait (除非必要)
  • 使用 &str 而非 String 作为只读字符串参数
// ✅ 推荐
#[tauri::command]
fn greet(name: &str) -> String {
    format!("Hello, {}!", name)
}

async fn is_port_available(port: u16) -> bool {
    tokio::net::TcpListener::bind(format!("127.0.0.1:{}", port))
        .await
        .is_ok()
}

错误处理

  • 使用 Result<T, E> 返回可能失败的操作
  • 使用 expect() 时提供有意义的错误消息 (中文)
  • 避免 unwrap() 在生产代码中,除非逻辑上保证不会 panic
  • 使用 ? 操作符传播错误
  • 记录关键错误信息到控制台
// ✅ 推荐
let sidecar = app_handle
    .shell()
    .sidecar("server")
    .expect("无法找到 server sidecar");

let (mut rx, child) = sidecar.spawn().expect("启动 sidecar 失败");

// 日志记录
eprintln!("✗ Sidecar Server 启动失败");
println!("✓ Sidecar Server 启动成功!");

// ❌ 避免
let data = read_file().unwrap();  // 无上下文信息

异步代码

  • 使用 async/await 而非手动创建 Future
  • Tauri 内部使用 tauri::async_runtime::spawn 启动异步任务
  • 使用 Tokio 的异步 API (如 tokio::net::TcpListener)
  • 避免阻塞异步运行时 (使用 tokio::task::spawn_blocking)
// ✅ 推荐
tauri::async_runtime::spawn(async move {
    let port = find_available_port(3000).await;
    // ...
});

格式化

  • 使用 cargo fmt 自动格式化
  • 缩进: 4 空格
  • 行宽: 100 字符 (rustfmt 默认)
  • 结构体和枚举的字段每行一个 (如果超过一定长度)
  • 链式调用适当换行提高可读性

注释

  • 使用中文注释说明复杂逻辑
  • 代码块前添加简短说明注释
  • 避免显而易见的注释
// ✅ 推荐
// 全局状态:存储 Sidecar Server 进程句柄
struct SidecarProcess(Mutex<Option<CommandChild>>);

// 检查端口是否可用
async fn is_port_available(port: u16) -> bool { }

Tauri 特定规范

模块组织

  • lib.rs: 主入口,负责注册插件、命令、状态管理
  • commands/mod.rs: 所有 Tauri 命令集中定义,命令必须是 pub fn
  • sidecar.rs: Sidecar 进程管理逻辑,导出公共 APIspawn_sidecar, cleanup_sidecar_process
// lib.rs - 模块声明
mod commands;
mod sidecar;

use sidecar::SidecarProcess;

// 注册命令时使用模块路径
.invoke_handler(tauri::generate_handler![commands::greet])

命令定义

  • 使用 #[tauri::command] 宏标记命令
  • 命令函数必须是公开的或在 invoke_handler 中注册
  • 参数类型必须实现 serde::Deserialize
  • 返回类型必须实现 serde::Serialize
#[tauri::command]
fn greet(name: &str) -> String {
    format!("Hello, {}!", name)
}

// 在 Builder 中注册
.invoke_handler(tauri::generate_handler![greet])

状态管理

  • 使用 app.manage() 注册全局状态
  • 状态必须实现 Send + Sync
  • 使用 MutexRwLock 保证线程安全
struct SidecarProcess(Mutex<Option<CommandChild>>);

// 注册状态
app.manage(SidecarProcess(Mutex::new(None)));

// 访问状态
if let Some(state) = app_handle.try_state::<SidecarProcess>() {
    *state.0.lock().unwrap() = Some(child);
}

Sidecar 进程管理

  • Sidecar 二进制必须在 tauri.conf.jsonbundle.externalBin 中声明
  • 使用 app.shell().sidecar() 启动 sidecar
  • 在应用退出时清理子进程 (监听 RunEvent::ExitRequested)
// 启动 sidecar
let sidecar = app_handle
    .shell()
    .sidecar("server")
    .expect("无法找到 server sidecar")
    .env("NITRO_PORT", port.to_string());

// 清理进程
match event {
    tauri::RunEvent::ExitRequested { .. } | tauri::RunEvent::Exit => {
        if let Some(child) = process.take() {
            let _ = child.kill();
        }
    }
    _ => {}
}

依赖管理

  • Cargo.toml 中明确声明依赖版本
  • 使用语义化版本 (如 "2" 表示兼容 2.x.x)
  • 仅启用需要的 feature 以减少编译时间和二进制大小
tauri = { version = "2", features = [] }
tauri-plugin-opener = "2"
tauri-plugin-shell = "2"
serde = { version = "1", features = ["derive"] }
tokio = { version = "1", features = ["net"] }

开发工具

推荐安装以下 VSCode 扩展:

  • tauri-apps.tauri-vscode - Tauri 官方支持
  • rust-lang.rust-analyzer - Rust 语言服务器

最佳实践

  1. 进程生命周期: 始终在应用退出时清理子进程和资源
  2. 端口管理: 使用端口扫描避免硬编码端口冲突
  3. 超时处理: 异步操作设置合理的超时时间 (如 5 秒)
  4. 日志: 使用表情符号 (✓/✗) 和中文消息提供清晰的状态反馈
  5. 错误退出: 关键错误时调用 std::process::exit(1)
  6. 窗口配置: 使用 WebviewWindowBuilder 动态创建窗口

提交代码前检查清单

  • cargo fmt 格式化通过
  • cargo clippy 无警告
  • cargo check 编译通过
  • cargo test 测试通过
  • 更新相关注释和文档
  • 检查是否有 unwrap() 需要替换为 expect()
  • 验证 Tauri 应用正常启动和退出