Dev Workflow
Standard dev session
You need three terminal windows for the full development loop.
Terminal 1 — the server
cd homeCore/core
cargo run -p homecore
HomeCore uses the current working directory as its base. Running from core/ picks up config/homecore.toml and writes data to data/ right there.
On first run, copy the admin password from the startup output.
To restart after a code change: Ctrl-C, then cargo run -p homecore again. Changed crates only — only they recompile.
Terminal 2 — virtual device (optional)
Start after the server is up:
cargo run -p virtual-device -- --broker 127.0.0.1 --port 1883 --id plugin.virtual
Terminal 3 — API calls
TOKEN=$(curl -s -X POST http://localhost:8080/api/v1/auth/login \
-H "Content-Type: application/json" \
-d '{"username":"admin","password":"YOUR_PASSWORD"}' | jq -r .token)
Optional: Terminal 4 — live event stream
websocat "ws://localhost:8080/api/v1/events/stream?token=$TOKEN"
Typical loop
- Make a code change
Ctrl-Cin Terminal 1cargo run -p homecore(only changed crates recompile)- Re-test with
curlin Terminal 3 - Check Terminal 1 for log output
Cargo commands reference
# Fastest feedback — check compiles without building
cargo check --workspace
cargo check -p hc-core # single crate
# Build
cargo build --workspace
cargo build --release # optimized binary
# Run
cargo run -p homecore
cargo run -p homecore --release
# Run virtual device
cargo run -p virtual-device -- --broker 127.0.0.1 --port 1883 --id plugin.virtual
# Test
cargo test --workspace
cargo test -p hc-auth # single crate
cargo test -p hc-core # rule engine tests
cargo test -p hc-api # API + WebSocket auth tests
cargo test -p homecore --test integration_test # full-stack integration test
cargo test -p hc-core repeat_until # specific test by name
cargo test --workspace -- --nocapture # show println! output
# Watch for changes (install once: cargo install cargo-watch)
cargo watch -x "check -p hc-core"
cargo watch -x "test -p hc-api"
Test coverage
| Crate | Tests | Coverage |
|---|---|---|
hc-auth | 11 | Password hashing, JWT issue/validate/expire/tamper/role |
hc-core | 12 | Rule engine trigger matching, executor RepeatUntil/Delay, CallService |
hc-api | 22 | Event log ring buffer, WebSocket auth, scope enforcement |
hc-topic-map | 4 | Pattern matching and transforms |
http-poller | 19 | Path extraction, field_map, JSON↔Dynamic bridge, Rhai transform |
homecore (integration) | 1 | Full stack: virtual device → MQTT → rule fires → command |
| Total | 69 |
Admin UI development
The Leptos/WASM admin UI (hc-web-leptos) has its own dev workflow that coexists with the server.
Development: Run trunk serve on port 3000 from the hc-web-leptos directory. Trunk proxies /api requests to HomeCore on port 8080 automatically. Leave [web_admin] enabled = false in homecore.toml (the default) so the server does not serve static files that conflict with trunk's dev server.
Production: Build with trunk build --release, copy the dist/ directory to the deploy location, and enable in homecore.toml:
[web_admin]
enabled = true
dist_path = "ui/dist" # relative to HOMECORE_HOME
HomeCore serves the built assets via tower-http ServeDir with SPA fallback. API routes at /api/v1 take priority.
Both can coexist: Disabling web_admin does not affect the trunk dev workflow. You can develop the UI with trunk serve while HomeCore runs with web_admin disabled, then switch to built-in serving for production.
Isolated dev environment
Use a throwaway directory to keep state separate from your main installation:
HOMECORE_HOME=/tmp/hc-dev cargo run -p homecore
# or
cargo run -p homecore -- --home /tmp/hc-dev
Reset it cleanly:
rm -rf /tmp/hc-dev/data/
HOMECORE_HOME=/tmp/hc-dev cargo run -p homecore
Debugging with RUST_LOG
# Rule engine internals
RUST_LOG=info,hc_core::engine=debug,hc_core::executor=debug cargo run -p homecore
# MQTT traffic
RUST_LOG=info,hc_mqtt_client=debug cargo run -p homecore
# State bridge
RUST_LOG=info,hc_core::state_bridge=debug cargo run -p homecore
# Everything (very noisy)
RUST_LOG=trace cargo run -p homecore
# All rule-engine related
RUST_LOG=info,hc_core=debug,hc_mqtt_client=debug,hc_broker=warn cargo run -p homecore
Common compiler errors
"cannot borrow self as mutable because it is also borrowed as immutable"
Usually means you're holding a reference across an async await point. The fix is to clone the value before the await:
// ❌ fails
let name = &self.state.get_name();
self.state.update().await?; // borrow problem
// ✅ fix
let name = self.state.get_name().clone();
self.state.update().await?;
"future cannot be sent between threads safely" / Send bound
An async fn that captures a non-Send type (e.g. Rc, RefCell) passed to tokio::spawn. Either use Arc<Mutex<>> instead, or restructure so the non-Send value is dropped before the first await.
"the trait bound is not satisfied for Box<dyn Future>"
Recursive async functions need Box::pin:
fn run_action(action: Action) -> Pin<Box<dyn Future<Output = Result<()>> + Send>> {
Box::pin(async move {
match action {
Action::Parallel { actions } => {
for a in actions {
run_action(a).await?; // ← recursive call OK inside Box::pin
}
}
// ...
}
Ok(())
})
}
Deadlock in RwLock
Never hold a write lock across an await. Take the data you need, drop the lock, then do async work:
// ❌ deadlock
let mut rules = self.rules.write().await;
rules.push(new_rule);
persist_to_db(&new_rule).await?; // deadlock — write lock still held
// ✅ fix
{
let mut rules = self.rules.write().await;
rules.push(new_rule.clone());
} // lock dropped here
persist_to_db(&new_rule).await?;
Adding a new REST endpoint
- Add handler function in
crates/hc-api/src/handlers.rs(or a new file for new resource types) - Add route in
crates/hc-api/src/lib.rsbuild_router() - Add scope extractor parameter to the handler signature
- Add
utoipa#[utoipa::path(...)]attribute for OpenAPI generation - Register in
openapi.rsif using the OpenAPI macro router - Write a test in
hc-apimodule tests
Adding a new action type
- Add the variant to
Actionenum incrates/hc-types/src/rule.rs - Add serde deserialization (usually automatic with
#[serde(tag = "type", rename_all = "snake_case")]) - Add match arm in
executor.rsrun_single_action() - Write unit tests in
hc-coremodule tests - Add to the action type reference in
docs/devNotes.md
Integration test
The integration test starts a full HomeCore instance on a random port:
cargo test -p homecore --test integration_test
It:
- Starts HomeCore with an in-memory config (temp files under
/tmp/) - Connects the virtual device plugin
- Creates a rule via the REST API
- Publishes a state change via MQTT
- Asserts the rule fired and the expected command was published
To add a new integration test scenario, add a test function in tests/integration_test.rs.