AI assistant for communities — React + Tauri v2 desktop app with a Rust core (JSON-RPC / CLI) embedded in-process.
Architecture docs: gitbooks/developing/architecture.md | Frontend | Tauri shell | Agent harness
| Path | Role |
|---|---|
app/ |
pnpm workspace openhuman-app: Vite + React (app/src/), Tauri desktop host (app/src-tauri/), Vitest tests |
src/ (root) |
Rust lib crate openhuman + openhuman-core CLI binary (src/main.rs) — src/core/ (transport), src/openhuman/* domains |
Cargo.toml (root) |
Core crate; cargo build --bin openhuman-core. Also slack-backfill and gmail-backfill-3d in src/bin/. |
docs/ |
Deep internals. Public contributor docs in gitbooks/developing/. |
Commands assume repo root. Root package.json is openhuman-repo (private, pnpm-enforced).
- Shipped product: desktop — Windows, macOS, Linux. No Android/iOS in the Tauri host.
- Core runs in-process as a tokio task (sidecar removed PR #1061). Lifecycle:
core_process::CoreProcessHandleinapp/src-tauri/src/core_process.rs. Frontend RPC →http://127.0.0.1:<port>/rpcwith per-launch hex bearer handed in-memory viarun_server_embedded_with_ready(rpc_token: Some(_)). Renderer reads bearer viacore_rpc_tokenTauri command.OPENHUMAN_CORE_TOKENstill honoured for CLI/docker/cloud. SetOPENHUMAN_CORE_REUSE_EXISTING=1for external core debugging.
Where logic lives:
- Rust core (
src/): business logic, execution, domains, RPC, persistence, CLI. Authoritative. - Tauri + React (
app/): UX, screens, navigation, bridging. Presents and orchestrates only.
Connects to desktop core via ConnectionProfile transport strategies in app/src/services/transport/: LanHttpTransport, TunnelTransport (E2E encrypted XChaCha20-Poly1305), CloudHttpTransport. Key paths: PTT plugin packages/tauri-plugin-ptt/, iOS screens app/src/pages/ios/, devices domain src/openhuman/devices/, tunnel crypto app/src/lib/tunnel/. Build: pnpm tauri:ios:dev (stock @tauri-apps/cli, not vendored CEF). Backend dep: tinyhumansai/backend#709.
pnpm dev # Vite dev server only
pnpm dev:app # Full Tauri desktop dev (CEF, loads env via scripts/load-dotenv.sh)
pnpm build # Production UI build
pnpm typecheck # tsc --noEmit (alias: compile)
pnpm lint # ESLint --cache
pnpm format # Prettier write + cargo fmt
pnpm format:check # Prettier check + cargo fmt --check
# Rust
cargo check --manifest-path Cargo.toml
cargo build --manifest-path Cargo.toml --bin openhuman-core
cargo check --manifest-path app/src-tauri/Cargo.toml # or: pnpm rust:check
# macOS Apple Silicon workaround (whisper-rs / llama.cpp)
GGML_NATIVE=OFF cargo check --manifest-path Cargo.tomlpnpm core:stage is a no-op (sidecar removed).
Tests: pnpm test (Vitest) · pnpm test:coverage · pnpm test:rust (scripts/test-rust-with-mock.sh).
Quality: ESLint + Prettier + Husky. Pre-push hook runs pnpm rust:check.
Summary-sized stdout; full output teed to target/debug-logs/. Add --verbose to stream raw.
pnpm debug unit # full Vitest suite
pnpm debug unit src/components/Foo.test.tsx # one file
pnpm debug unit -t "renders empty state" # filter by name
pnpm debug e2e test/e2e/specs/smoke.spec.ts # WDIO E2E
pnpm debug rust # cargo tests
pnpm debug rust json_rpc_e2e # targeted
pnpm debug logs # list recent
pnpm debug logs last # print most recentPRs need ≥ 80% coverage on changed lines via diff-cover over Vitest + cargo-llvm-cov lcov. Enforced by .github/workflows/coverage.yml.
.env.example— Rust core, Tauri shell, backend URL, logging. Load:source scripts/load-dotenv.sh.app/.env.example—VITE_*vars. Copy toapp/.env.local.- Frontend config centralized in
app/src/utils/config.ts— never readimport.meta.envdirectly elsewhere. - Rust config: TOML
Configstruct (src/openhuman/config/schema/types.rs) with env overrides (load.rs).
The [autonomy] block (src/openhuman/config/schema/autonomy.rs) drives SecurityPolicy (src/openhuman/security/policy.rs). Tiers: readonly / supervised / full × workspace_only × trusted_roots × allow_tool_install. Edit via config.update_autonomy_settings RPC or Settings → Agent access.
Two path roots (src/openhuman/config/schema/types.rs):
action_dir— agent's read/write root. Acting tools resolve relative paths here. Default:~/OpenHuman/projects(OPENHUMAN_ACTION_DIR).workspace_dir— internal state (~/.openhuman/users/<id>/workspace). Agent tools cannot write here — enforced byis_workspace_internal_pathfail-closed regardless of tier/trusted_roots.
Command permission model: classify_command → CommandClass (Read/Write/Network/Install/Destructive); unrecognized = Write. gate_decision(class, tier) → Allow/Prompt/Block. System/credential dirs unconditionally blocked (is_always_forbidden).
Approval gate ON by default (opt out: OPENHUMAN_APPROVAL_GATE=0). Parks interactive chat turns only; background/cron allowed through. Frontend surfaces via ApprovalRequestCard. 10-min TTL → Deny.
Sandbox backends (opt-in per agent via sandbox_mode = "sandboxed"): Docker (remote/cron), Local OS jail (Landlock/Seatbelt/AppContainer, desktop), Noop fallback. In-Rust path hardening applies regardless.
- Co-locate as
*.test.ts(x)underapp/src/**. Config:app/test/vitest.config.ts. - Run:
pnpm testorpnpm test:coverage. Prefer behavior over implementation. No real network, no time flakes.
- Core:
scripts/mock-api-core.mjs· Server:scripts/mock-api-server.mjs· E2E:app/test/e2e/mock-server.ts. - Admin:
GET /__admin/health,POST /__admin/reset,POST /__admin/behavior,GET /__admin/requests. - Manual:
pnpm mock:api.
Full guide: gitbooks/developing/e2e-testing.md.
- Linux (CI):
tauri-driver(WebDriver :4444). macOS (local): Appium Mac2 (XCUITest :4723). - Specs:
app/test/e2e/specs/*.spec.ts. Useelement-helpers.tshelpers, never rawXCUIElementType*. e2e-run-spec.shcreates/cleans tempOPENHUMAN_WORKSPACEby default.
pnpm test:rust
bash scripts/test-rust-with-mock.sh --test json_rpc_e2eProvider chain (App.tsx): Sentry.ErrorBoundary → Redux Provider → PersistGate → BootCheckGate → CoreStateProvider → SocketProvider → ChatRuntimeProvider → HashRouter → CommandProvider → ServiceBlockingGate → AppShell.
No UserProvider/AIProvider/SkillProvider — auth lives in CoreStateProvider via fetchCoreAppSnapshot() RPC.
State (store/): Redux Toolkit slices — accounts, channelConnections, chatRuntime, coreMode, deepLinkAuth, mascot, notification, providerSurface, socket, thread. Prefer Redux over ad-hoc localStorage.
Services (services/): apiClient, socketService, coreRpcClient, coreCommandClient, chatService, analytics, notificationService, webviewAccountService, daemonHealthService, plus domain api/* clients. Always use invoke('core_rpc_relay', ...) for core RPC.
Routing (AppRoutes.tsx, HashRouter): / (Welcome), /onboarding/*, /home, /human, /intelligence, /skills, /chat, /channels, /invites, /notifications, /rewards, /settings/*. No /login, /mnemonic, /agents, /conversations.
AI config: bundled prompts in src/openhuman/agent/prompts/ (also via tauri.conf.json resources). Loaders in app/src/lib/ai/ with ?raw imports.
Thin desktop host. Key modules: core_process, core_rpc, cdp, cef_preflight, cef_profile, dictation_hotkeys, file_logging, mascot_native_window, screen_capture, window_state, per-provider scanners (discord_scanner, slack_scanner, telegram_scanner, whatsapp_scanner, etc.), meet_audio/meet_call/meet_video, fake_camera, webview_accounts, webview_apis.
IPC commands: greet, write_ai_config_file, ai_get_config, ai_refresh_config, core_rpc_relay, core_rpc_token, start_core_process, restart_core_process, window commands, openhuman_* daemon helpers.
Embedded provider webviews must not grow new JS injection. No new .js under webview_accounts/, no new build_init_script/RUNTIME_JS blocks, no CDP Page.addScriptToEvaluateOnNewDocument. New behavior lives in CEF handlers, CDP from scanner modules, or Rust-side IPC hooks. Legacy injection (gmail, linkedin, google-meet) is grandfathered but should shrink. Audit new Tauri plugins for js_init_script calls.
Domains: about_app, accessibility, agent, app_state, approval, autocomplete, billing, channels, composio, config, context, cost, credentials, cron, doctor, embeddings, encryption, health, heartbeat, integrations, learning, local_ai, meet, meet_agent, memory, migration, node_runtime, notifications, overlay, people, prompt_injection, provider_surfaces, providers, redirect_links, referral, routing, scheduler_gate, screen_intelligence, security, service, skills, socket, subconscious, team, text_input, threads, tokenjuice, tool_timeout, tools, tree_summarizer, update, voice, wallet, webhooks, webview_accounts, webview_apis, webview_notifications.
Skills runtime removed: QuickJS gone. src/openhuman/skills/ is metadata-only now.
Rules:
- New functionality → dedicated subdirectory (
openhuman/<domain>/mod.rs+ siblings). No new root-level*.rsfiles. - Tool ownership: domain tools live in that domain's
tools.rs, re-exported viasrc/openhuman/tools/mod.rs. Only cross-cutting families stay intools/impl/. - Memory source identity: per-item IDs are dedupe keys only; set
metadata.path_scopeto stable collection scope. - Controller-only exposure: use the registry, not branches in
cli.rs/jsonrpc.rs.
| File | When | Role |
|---|---|---|
mod.rs |
always | Export-focused only: mod/pub mod + pub use + controller schema pair. No business logic. |
types.rs |
domain has types | Serde domain types. |
store.rs |
domain persists | Persistence layer. |
ops.rs |
domain has logic | Business logic + handlers returning RpcOutcome<T>. |
schemas.rs |
RPC-facing | Controller schemas + handle_* fns delegating to ops.rs. |
tools.rs |
domain owns agent tools | Tool implementations. |
bus.rs |
domain has event subscribers | EventHandler impls. |
| tests | new/changed behavior | Inline #[cfg(test)] mod tests or sibling *_tests.rs. |
mod.rs: addmod schemas;, re-exportall_controller_schemas/all_registered_controllers.schemas.rs: define schemas, handlers delegating toops.rs.- Wire into
src/core/all.rs. Remove fromsrc/core/dispatch.rs.
Modules: all, auth, cli, dispatch, event_bus/, jsonrpc, logging, observability, types, etc. No business logic here.
Typed pub/sub + native request/response. Both singletons — use module-level functions.
- Broadcast (
publish_global/subscribe_global): fire-and-forget, many subscribers. - Native request/response (
register_native_global/request_native_global): one-to-one typed dispatch, zero serialization, internal-only.
Core types: DomainEvent (events.rs), EventBus (bus.rs), NativeRegistry (native_request.rs), EventHandler/SubscriptionHandle (subscriber.rs).
Domains: agent, memory, channel, cron, skill, tool, webhook, system.
Each domain owns bus.rs with handlers. Convention: <Purpose>Subscriber, name() → "<domain>::<purpose>".
Adding events: add to DomainEvent, extend domain() match, create <domain>/bus.rs, register at startup, publish via publish_global.
Adding native handlers: define req/resp types (Send + 'static, not Serialize), register at startup keyed by "<domain>.<verb>", dispatch via request_native_global.
Visual: ocean primary #4A83DD, sage/amber/coral semantics, Inter + Cabinet Grotesk + JetBrains Mono. Tokens in app/tailwind.config.js.
Key rules:
- File size: prefer ≤ ~500 lines.
- No dynamic imports in production
app/src— staticimport/import typeonly. Guard heavy paths with try/catch. Exceptions: test files,.d.ts, config files. - i18n: all UI text through
useT()fromapp/src/lib/i18n/I18nContext. Add key toen.tsand real translations to all locale files (ar,bn,de,es,fr,hi,id,it,ko,pl,pt,ru,zh-CN). CI enforces parity (pnpm i18n:check) and detects English placeholders (pnpm i18n:english:check). - Dual socket sync: keep
socketService/MCP transport aligned with core socket behavior. - Tauri guard: use
isTauri()or wrapinvoke(...)in try/catch — never checkwindow.__TAURI__directly.
- Default to verbose diagnostics on new/changed flows.
- Log entry/exit, branches, external calls, retries/timeouts, state transitions, errors.
- Stable grep-friendly prefixes (
[domain],[rpc]), correlation fields (request IDs, method names). - Rust:
log/tracingatdebug/trace. App: namespaceddebug. - Never log secrets or full PII.
- Changes lacking logging are incomplete.
Specify → prove in Rust → prove over RPC → surface in UI → test.
- Specify — ground in existing domains, controller patterns, JSON-RPC naming (
openhuman.<namespace>_<function>). - Implement in Rust — domain logic + unit tests.
- JSON-RPC E2E — extend
tests/json_rpc_e2e.rs/scripts/test-rust-with-mock.sh. - UI — React +
core_rpc_relay/coreRpcClient. Keep rules in core. - App unit tests — Vitest.
- App E2E — desktop specs.
Update src/openhuman/about_app/ when adding/removing/renaming user-facing features. Define E2E scenarios up front covering happy paths, failures, auth gates.
Contribute via your fork. Recommended remotes:
origin git@github.com:<your-username>/openhuman.git (push here)
upstream git@github.com:tinyhumansai/openhuman.git (fetch-only)
- Never write code on
main. Branch offupstream/mainfor all work. - Issues and PRs on upstream
tinyhumansai/openhuman. - Push to
origin(fork), neverupstream. PRs with--head <your-username>:<branch>. - Use issue/PR templates verbatim.
- On push blockers: fix your own hook failures; bypass with
--no-verifyonly for unrelated pre-existing breakage (call out in PR body).
- Vendored CEF-aware
tauri-cli: only the vendored CLI atapp/src-tauri/vendor/tauri-cef/crates/tauri-clibundles Chromium correctly. Stock@tauri-apps/cliproduces broken bundles. Reinstall:cargo install --locked --path app/src-tauri/vendor/tauri-cef/crates/tauri-cli. - macOS deep links: require built
.appbundle, not justtauri dev. - Windows deep links:
openhuman://registered viatauri-plugin-deep-link::register_all. Check inapp/src-tauri/src/deep_link_registration_check.rs. - Core standalone debugging:
./target/debug/openhuman-core serve(token at{workspace}/core.token). Public endpoints:GET /health,GET /schema,GET /events.
- Unix-style modules: small, single-responsibility, composed through clear boundaries.
- Tests before the next layer: untested code is incomplete.
- Docs with code: update AGENTS.md or architecture docs when rules or behavior change.