Add ADR-026 (vault key model — HD derivation) recording the foundational HD-derivation decision, 74' coin type reservation, SLIP-0010/Ed25519 default, secp256k1 feature-gating, and AES-256-GCM cipher choice. These were previously inline rationale with no ADR (W9). Extend ADR-018 with an explicit EncryptedData wire format lock — fields, encoding, and semantics are frozen; no removal without a format-version migration (W10). Resolve the remaining guard clauses and spec decisions: - W2: Capabilities must be immutable after construction (no interior mutability). Makes the Arc vs deep-copy clone semantics genuinely two-way. - W5: Published to_* specs are compatibility contracts — best-effort mappings are two-way before first publication, one-way after. Version generated specs. - W6: Salt field clarification — v2 salt is permanently unused; a future KDF is a different derivation family, not a version-indexed path; the field saves a wire-format change only. - W7: unlock_new returns Zeroizing<String> — the mnemonic is the root of trust and must not linger in freed memory. - W17: OQ-09 WASM — server-side dispatch door is honestly closed (Connection is concrete, tokio-bound), not implicitly preserved. - W18: OQ-10 git — composability fork (raw smart protocol vs call-protocol projection) is a separate decision from ERC721 scope. - W20: from_openapi must prefix imported error codes (HTTP_404) to avoid collision with protocol-level codes (NOT_FOUND). Normative rule, not naming convention. - W21: ScopedOperationEnv field is private — construction via new()/ empty(), query via allows(). Makes the future subgraph refactor non-breaking. - C13: Connection::set_identity — the endpoint does not read identity() after handle() returns (Connection is moved into the spawned task). Observability is handler-side logging. Simplest honest answer. - W1: OperationAdapter trait is async, returns Vec<HandlerRegistration>. from_call requires async discovery; ADR-022 changed the return type. - W11: CompositionAuthority::as_identity() defined — constructs a synthetic Identity (label as id, scopes, resources) not resolvable via IdentityProvider. Second Identity construction path, acknowledged. - W14: SecretKey is iroh::SecretKey (Ed25519) — consistent with the endpoint's iroh dependency. - W19: Grandchild abort propagation is inherit-by-default (option a) — invoke() with no explicit policy inherits parent's policy. ContinueRunning auto-propagates to grandchildren unless explicitly overridden.
status, last_updated
| status | last_updated |
|---|---|
| draft | 2026-06-22-25 |
alknet-vault
Local key vault: BIP39 mnemonic generation, SLIP-0010 Ed25519 HD key derivation, BIP-0032 secp256k1 derivation (feature-gated), and AES-256-GCM encryption. Holds the master seed — the root of trust for all derived keys and encrypted credentials in the alknet system.
What This Crate Is
alknet-vault is a standalone crate with zero alknet crate dependencies
(ADR-018) and zero RPC framework dependencies (ADR-025). It provides the
cryptographic primitives and runtime API for managing the root of trust.
The CLI binary (the alknet crate) is the sole component that talks to the
vault directly (ADR-019) — handlers receive derived/decrypted material
through capabilities, never through a vault reference.
The vault is not a network service. It has no ALPN, no
ProtocolHandler implementation, no operations registered in the call
protocol (ADR-008, ADR-014), and no remote dispatch capability (ADR-025).
The vault is local-only by construction — direct method calls on
VaultServiceHandle, no actor, no message enum, no wire format. The master
seed and derived private keys never cross the network.
Documents
| Document | Status | Description |
|---|---|---|
| mnemonic-derivation.md | draft | BIP39, SLIP-0010, BIP-0032, derivation paths, key types |
| encryption.md | draft | AES-256-GCM, EncryptedData, key versioning, HD derivation (ADR-020) |
| service.md | draft | VaultServiceHandle lifecycle, direct dispatch, cache, error model |
| protocol.md | draft | DerivedKey redaction, KeyType, serialization behavior |
Applicable ADRs
| ADR | Title | Relevance |
|---|---|---|
| 003 | Crate Decomposition | alknet-vault's standalone position |
| 008 | Vault Integration Point | CLI-embedded, capability source |
| 010 | ALPN Router and Endpoint | Ed25519 as default curve for TLS raw key identity |
| 014 | Secret Material Flow and Capability Injection | Capabilities carry vault-derived material |
| 018 | Vault as Standalone Crate | Zero alknet crate dependencies |
| 019 | Vault Assembly-Layer-Only Access | The assembly layer is the sole caller |
| 020 | HD Derivation for Encryption Keys | SLIP-0010 derivation, not PBKDF2; salt unused in v2 |
| 021 | Key Rotation via Version-Indexed Paths | Version-indexed paths; rotate re-encrypts |
| 025 | Vault Local-Only Dispatch | Dropped irpc; direct method calls; local-only by construction |
| 026 | Vault Key Model — HD Derivation | HD derivation from BIP39 seed; 74' coin type; AES-256-GCM |
Relevant Open Questions
| OQ | Title | Status | Relevance |
|---|---|---|---|
| OQ-20 | Encryption key derivation | resolved (ADR-020) | HD derivation from seed; salt field unused in v2 |
| OQ-21 | Remote vault access | resolved (ADR-025) | Vault is local-only by construction; remote access requires a separate vault-server crate with its own ADR |
| OQ-22 | Key rotation mechanism | resolved (ADR-021) | Version-indexed paths; rotate method |
Key Design Principles
- Standalone: The vault depends on no alknet crate and no RPC framework. It defines its own types and errors. External crates depend on the vault; the vault depends on nothing in alknet.
- Assembly-layer only: The vault's API is consumed by the CLI binary, not by handlers. Handlers receive material through capabilities (ADR-014). The vault is not on the wire.
- Local-only by construction: The vault has no remote dispatch
capability. Direct method calls on
VaultServiceHandle— no actor, no message enum, no wire format (ADR-025). Remote access, if ever needed, requires a separate crate with its own ADR. - Zeroize everything sensitive: The mnemonic, seed, derived private
keys, encryption keys, and cached keys all implement
ZeroizeandZeroizeOnDrop. Secret material does not linger in freed heap memory. - Deterministic derivation: The same mnemonic + passphrase + path always produces the same key. Derivation is reproducible across runs and across nodes.
- OsRng for nonces: AES-GCM IVs and any cryptographic nonces use
OsRng(or equivalent CSPRNG), neverrand::random(). IV reuse under the same key is catastrophic for GCM. - No
unwrap()orexpect()outside tests: vault operations propagate errors. A poisoned lock is recovered withunwrap_or_else(|e| e.into_inner()), notunwrap(). A panic in one vault operation must not brick the vault for all other operations.
Security Constraints
These are security-critical implementation requirements, not architectural decisions (the architecture is locked by the ADRs above). They are documented here so implementation agents don't miss them. See service.md → Security Constraints for the full list.
- OsRng for IVs: AES-GCM IVs must use
OsRng, notrand::random(). The current source usesrand::random()— this is a known drift from the spec and must be corrected during implementation sync. - Zeroized drop:
Seed,Mnemonic,ExtendedPrivKey,Secp256k1ExtendedPrivKey,EncryptionKey,CachedKey, andDerivedKeyall deriveZeroizeandZeroizeOnDrop. The cache must clear on drop, not just on explicitlock(). - No
unwrap()outside tests: poisoned lock recovery usesunwrap_or_else(|e| e.into_inner())or explicit error propagation. The current source usesunwrap()inVaultServiceHandlemethods — this is a known drift and must be corrected. - DerivedKey redaction in serialization:
DerivedKeyserializes theprivate_keyas"[REDACTED]"in all formats (ADR-025 dropped the postcard/remote path that previously preserved bytes in binary formats). Deserialization rejects"[REDACTED]"with an error (resolves review #002 W8). The redaction is a defense-in-depth measure for logging safety, not the primary control — the primary control is thatDerivedKeynever crosses the call protocol wire (ADR-014).
Known Source Drift
The vault crate carries over source from the POC. The following items are known divergences between the current source and the spec. All must be corrected during implementation sync. This table is the single source of truth for drift tracking — if an item is fixed in source, update this table.
| # | Item | Current source behavior | Target behavior (per spec) | Source location | Spec reference |
|---|---|---|---|---|---|
| 1 | IV generation | rand::random() |
OsRng (CSPRNG) |
encryption.rs L133 |
encryption.md → Security Constraints, service.md → Security Constraints |
| 2 | RwLock unwrap() |
unwrap() on every RwLock acquisition (L142, 161, 182, 191, 196, 227, 264, 307, 340, 367) |
unwrap_or_else(|e| e.into_inner()) for poisoned lock recovery |
service.rs (see line numbers) |
service.md → Security Constraints |
| 3 | CURRENT_KEY_VERSION |
1 (HD-derived, but v1 is reserved for TS PBKDF2 legacy per ADR-020) |
2 (HD-derived, per ADR-020) |
encryption.rs |
encryption.md → Key Versioning, ADR-020 |
| 4 | irpc dependency | VaultProtocol enum with #[rpc_requests], VaultServiceActor, Client<VaultProtocol>, irpc/postcard deps |
Remove entirely — direct method calls on VaultServiceHandle (ADR-025) |
protocol.rs, service.rs, Cargo.toml |
ADR-025 |
| 5 | DerivedKey dual serialization |
JSON redacts, postcard preserves bytes | Always redact on serialize; reject "[REDACTED]" on deserialize with error (ADR-025, resolves W8) |
protocol.rs |
protocol.md → Serialization Redaction, ADR-025 |
| 6 | HashMap::clear zeroization |
KeyCache::clear() removes entries and relies on CachedKey's Drop impl for zeroization |
Verify HashMap::clear() actually drops values (it does, but worth a test) |
cache.rs |
service.md → Security Constraints |
| 7 | derive_password / site_password_path |
derive_password, derive_password_string, site_password_path methods exist |
Remove entirely — password-manager pattern not relevant to RPC system's vault (ADR-025, resolves C9) | service.rs, mnemonic-derivation.rs |
ADR-025 |
| 8 | unlock_new return type |
Returns String (not zeroized on drop) |
Return Zeroizing<String> — the mnemonic is the root of trust and must not linger in freed memory (resolves W7) |
service.rs |
service.md → unlock_new |
Public API
The vault re-exports its primary types from the crate root:
// Mnemonic and seed
pub use mnemonic::{Language, Mnemonic, Seed};
// Derivation
pub use derivation::{DerivationError, ExtendedPrivKey, PATHS};
// Encryption
pub use encryption::{EncryptedData, EncryptionError};
// Key types (DerivedKey, KeyType)
pub use protocol::{DerivedKey, KeyType};
// Service (runtime)
pub use service::{VaultServiceError, VaultServiceHandle};
// Cache
pub use cache::CacheConfig;
The secp256k1 feature flag gates Ethereum (BIP-0032) derivation:
#[cfg(feature = "secp256k1")]
pub mod ethereum;