This file provides guidance when working with code in this repository.
cargo run- Build and run Warp locallycargo bundle --bin warp- Bundle the main app
To connect Warp client to a local warp-server instance:
# Connect to server on default port 8080
cargo run --features with_local_server
# Connect to server on custom port (e.g., 8082)
SERVER_ROOT_URL=http://localhost:8082 WS_SERVER_URL=ws://localhost:8082/graphql/v2 cargo run --features with_local_serverEnvironment variables:
SERVER_ROOT_URL- HTTP endpoint (default:http://localhost:8080)WS_SERVER_URL- WebSocket endpoint (default:ws://localhost:8080/graphql/v2)
cargo nextest run --no-fail-fast --workspace --exclude command-signatures-v2- Run tests with nextestcargo nextest run -p warp_completer --features v2- Run completer tests with v2 featurescargo test --doc- Run doc testscargo test- Run standard tests for individual packages
./script/presubmit- Run all presubmit checks (fmt, clippy, tests)cargo fmt- Format codecargo clippy --workspace --all-targets --all-features --tests -- -D warnings- Run clippy./script/run-clang-format.py -r --extensions 'c,h,cpp,m' ./crates/warpui/src/ ./app/src/- Format C/C++/Obj-C codefind . -name "*.wgsl" -exec wgslfmt --check {} +- Check WGSL shader formatting
./script/bootstrap- Platform-specific setup (calls platform-specific bootstrap scripts)./script/install_cargo_build_deps- Install Cargo build dependencies./script/install_cargo_test_deps- Install Cargo test dependencies
This is a Rust-based terminal emulator with a custom UI framework called WarpUI.
WarpUI Framework (ui/):
- Custom UI framework with Entity-Component-Handle pattern
- Global
Appobject owns all views/models (entities) - Views hold
ViewHandle<T>references to other views AppContextprovides temporary access to handles during render/events- Elements describe visual layout (Flutter-inspired)
- Actions system for event handling
- MouseStateHandle must be created once during construction, and then referenced/cloned anywhere we're using mouse input to track mouse changes. Inline
MouseStateHandle::default()while rendering will cause no mouse interactions to work.
Main App (app/):
- Terminal emulation and shell management (
terminal/) - AI integration including Agent Mode (
ai/) - Cloud synchronization and Drive features (
drive/) - Authentication and user management (
auth/) - Settings and preferences (
settings/) - Workspace and session management (
workspace/)
Core Libraries:
crates/warp_core/- Core utilities and platform abstractionscrates/editor/- Text editing functionalitycrates/warpui/andcrates/warpui_core/- Custom UI frameworkcrates/ipc/- Inter-process communicationcrates/graphql/- GraphQL client and schema
- Entity-Handle System: Views reference other views via handles, not direct ownership
- Modular Structure: Workspace contains multiple workspace configurations, each with terminals, notebooks, etc.
- Cross-Platform: Native implementations for macOS, Windows, Linux, plus WASM target
- AI Integration: Built-in AI assistant with context awareness and codebase indexing
- Cloud Sync: Objects can be synchronized across devices via Warp Drive
Workspace Structure:
- This is a Cargo workspace with 60+ member crates
- Main binary is in
app/, UI framework incrates/warpui/ - Platform-specific code is conditionally compiled
- Integration tests are in
crates/integration/
Coding Style Preferences:
- Avoid unnecessary type annotations, especially in closure params.
- Avoid using too many Rust path qualifiers and use imports for concision. Place import statements at the top of the file as per convention. An exception to this is inside cfg-guarded code branches. In those cases, you can either embed the import into the relevant scope or just use an absolute path for one-offs.
- If a function takes a context parameter (
AppContext,ViewContext, orModelContext), it should be namedctxand go last. The one exception is for functions that take a closure parameter, in which case the closure should be last. - Always remove unused parameters completely rather than prefixing them with
_. Update the function signature and all call sites accordingly. - Prefer inline format arguments in macros like
println!,eprintln!, andformat!(for example,eprintln!("{message}")instead ofeprintln!("{}", message)) to satisfy Clippy'suninlined_format_argslint. - Do not remove existing comments when making unrelated changes. Only remove or modify a comment if the logic it describes has changed.
Terminal Model Locking:
- Be extremely careful when calling
model.lock()on the terminal model (TerminalModel). Acquiring multiple locks on the same model from different call sites can cause a deadlock, resulting in a UI freeze (beach ball on macOS). - Before adding a new
model.lock()call, verify that no caller in the current call stack already holds the lock. - Prefer passing already-locked model references down the call stack rather than acquiring new locks.
- If you must lock the model, keep the lock scope as short as possible and avoid calling other functions that might also attempt to lock.
Testing:
- Use
cargo nextestfor parallel test execution - Integration tests use custom framework in
integration/ - Tests should be run via presubmit script before submitting
- Unit tests should be placed in separate files using the naming convention
${filename}_tests.rsormod_test.rs - Test files should be included at the end of their corresponding module with:
#[cfg(test)] #[path = "filename_tests.rs"] // or "mod_test.rs" mod tests;
Pull Request Workflow:
- ALWAYS run cargo fmt and cargo clippy (the versions specified in ./script/presubmit) before opening a PR or pushing updates to an existing PR branch
- Those commands must pass completely before creating or updating a pull request
- Specifically, ensure
cargo fmtandcargo clippychecks pass - If they fail, fix all issues before proceeding with the PR
- This applies to:
- Opening new pull requests
- Pushing new commits to existing PR branches
- Any branch updates that will be reviewed
- When opening PRs, use the PR template at
.github/pull_request_template.md - Add changelog entries when appropriate using the format at the bottom of the PR template. Use the following prefixes (without the
{{}}brackets):CHANGELOG-NEW-FEATURE:for new, relatively sizable features (use sparingly - these may get marketing/docs)CHANGELOG-IMPROVEMENT:for new functionality of existing featuresCHANGELOG-BUG-FIX:for fixes related to known bugs or regressionsCHANGELOG-IMAGE:for GCP-hosted image URLs- Leave changelog lines blank or remove them if no changelog entry is needed
Database:
- Uses Diesel ORM with SQLite
- Migrations in
crates/persistence/migrations/ - Schema defined in
crates/persistence/src/schema.rs
GraphQL:
- Schema and client code generation from
crates/warp_graphql_schema/api/schema.graphql - TypeScript types generated for frontend integration
Warp uses compile-time feature flags with a small runtime plumbing layer.
How to add a feature flag:
- Add a new variant to
warp_core/src/features.rsin theFeatureFlagenum - (Optional) Enable it by default for dogfood builds by listing it in
DOGFOOD_FLAGS - Gate code paths with
FeatureFlag::YourFlag.is_enabled() - For preview or release rollout, add to
PREVIEW_FLAGSorRELEASE_FLAGSrespectively (as appropriate)
Best practices:
- Prefer runtime checks over cfg directives: Prefer
FeatureFlag::YourFlag.is_enabled()over#[cfg(...)]compile-time directives so flags can be toggled without recompilation and are easier to clean up later. Use#[cfg(...)]only when the code cannot compile without them (for example, platform-specific code or dependencies that do not exist when the feature is disabled). - Keep flags high-level and product-focused rather than per-call-site
- Remove the flag and dead branches after launch has stabilized
- For UI sections that expose a new feature, hide the UI behind the same flag
Example:
#[derive(Sequence)]
pub enum FeatureFlag {
YourNewFeature,
}
// Default-on for dogfood builds
pub const DOGFOOD_FLAGS: &[FeatureFlag] = &[
FeatureFlag::YourNewFeature,
];
// Use in code
if FeatureFlag::YourNewFeature.is_enabled() {
// gated behavior
}When adding/editing match statements, avoid using the wildcard _ when at all possible. Exhaustive matching is helpful for ensuring that all variants are handled, especially when adding new variants to enums in the future.