Create the alknet-secret crate with BIP39 mnemonic generation, SLIP-0010 Ed25519 HD key derivation, AES-256-GCM encryption, and SecretProtocol irpc service definition. This is Phase 3.1 from the integration plan. Architecture changes: - Promote secret-service.md to reviewed status with full spec format (crate structure, public API, security model, phase progression, ADR/OQ cross-references, wire format compatibility section) - Add ADR-038 (seed lifecycle and memory security): zeroize for v1, mlock deferred to Phase B - Add OQ-SEC-01 (mlock/VirtualLock for seed RAM) to open-questions.md - Update README.md with ADR-038 and secret-service status Crate structure: - src/mnemonic.rs: BIP39 phrase generation, validation, seed derivation - src/derivation.rs: SLIP-0010 HD key derivation, path constants (74') - src/encryption.rs: AES-256-GCM encrypt/decrypt, EncryptedData type - src/protocol.rs: SecretProtocol irpc enum, DerivedKey, KeyType - src/service.rs: SecretServiceHandle with Unlock/Lock lifecycle - 40 passing tests (unit + integration + doc)
283 lines
11 KiB
Markdown
283 lines
11 KiB
Markdown
---
|
|
status: reviewed
|
|
last_updated: 2026-06-09
|
|
---
|
|
|
|
# Secret Service (alknet-secret)
|
|
|
|
## What
|
|
|
|
The `alknet-secret` crate provides BIP39 mnemonic generation, SLIP-0010 Ed25519
|
|
HD key derivation, AES-256-GCM encryption for external credentials, and the
|
|
`SecretProtocol` irpc service. It is the only component that holds the master
|
|
seed phrase.
|
|
|
|
## Why
|
|
|
|
Operations like SSH key generation, API key storage, and Ethereum transaction
|
|
signing all need deterministic key derivation from a single root of trust. The
|
|
seed phrase is the single recovery mechanism — from it, all self-generated
|
|
secrets can be derived on demand. External credentials (third-party API keys,
|
|
OAuth tokens) cannot be derived and must be stored encrypted, with the
|
|
encryption key itself derived from the seed.
|
|
|
|
The secret service isolates this responsibility: no other crate sees the seed,
|
|
and derived keys are provided on demand through an irpc service interface. This
|
|
follows ADR-027 (crate decomposition) — alknet-secret is fully independent of
|
|
alknet-core and alknet-storage.
|
|
|
|
## Architecture
|
|
|
|
### Crate Structure
|
|
|
|
```
|
|
alknet-secret/
|
|
├── Cargo.toml
|
|
├── src/
|
|
│ ├── lib.rs # Crate root, re-exports
|
|
│ ├── mnemonic.rs # BIP39: phrase generation, validation, seed derivation
|
|
│ ├── derivation.rs # SLIP-0010: HD key derivation, path constants
|
|
│ ├── encryption.rs # AES-256-GCM: encrypt/decrypt, EncryptedData type
|
|
│ ├── protocol.rs # SecretProtocol irpc service enum, DerivedKey, KeyType
|
|
│ └── service.rs # SecretServiceImpl: in-memory seed, Unlock/Lock lifecycle
|
|
└── tests/
|
|
├── derivation_tests.rs # Path derivation, coin type 74' consistency
|
|
├── encryption_tests.rs # Round-trip encrypt/decrypt, key version
|
|
└── service_tests.rs # Unlock/Lock lifecycle, derive on locked = error
|
|
```
|
|
|
|
### Dependencies
|
|
|
|
```toml
|
|
[dependencies]
|
|
bip39 = "2"
|
|
ed25519-bip32 = "0.x" # IOHK SLIP-0010 Ed25519 HD derivation
|
|
aes-gcm = "0.10" # AES-256-GCM
|
|
sha2 = "0.10" # SHA-256
|
|
serde = { version = "1", features = ["derive"] }
|
|
serde_json = "1"
|
|
thiserror = "2"
|
|
irpc = "0.x" # Always-on, not feature-gated (ADR-027)
|
|
zeroize = { version = "1", features = ["derive"] } # Secure memory wiping (ADR-038)
|
|
```
|
|
|
|
irpc is always a dependency (not behind a feature flag). Per ADR-027, irpc
|
|
in alknet-secret and alknet-storage is not feature-gated because these crates
|
|
are used in production deployments where the service layer is always active.
|
|
|
|
### Crate Interface (Public API)
|
|
|
|
The crate exposes these types as its stable public interface:
|
|
|
|
```rust
|
|
// Core types (always available)
|
|
pub use mnemonic::{Mnemonic, Language, Seed};
|
|
pub use derivation::{ExtendedPrivKey, DerivationPath, PATHS};
|
|
pub use encryption::{EncryptedData, EncryptionError};
|
|
pub use protocol::{SecretProtocol, DerivedKey, KeyType, SecretMessage};
|
|
pub use service::{SecretService, SecretServiceHandle, SecretServiceError};
|
|
```
|
|
|
|
Other crates consume this interface:
|
|
- **alknet-storage** references `EncryptedData` for wire format compatibility
|
|
(type-level, not a crate dependency)
|
|
- **alknet** (CLI binary) assembles `SecretService` and wires it to the
|
|
`OperationEnv`
|
|
- **alknet-core** never depends on alknet-secret; `CredentialProvider` stub
|
|
returns `None` until Phase A wiring
|
|
|
|
### Security Model
|
|
|
|
Per ADR-038 (seed lifecycle and memory security):
|
|
|
|
| State | What's in memory | What's on disk |
|
|
|-------|-----------------|---------------|
|
|
| Locked | Nothing | Encrypted database, derivation path metadata |
|
|
| Unlocked | Master seed in zeroize-protected RAM | Same (seed is never persisted) |
|
|
| After use | Derived keys cached in zeroize-protected RAM | Derivation paths only |
|
|
|
|
The seed phrase is entered once (at node startup or via `Unlock`), held only in
|
|
RAM, and never written to disk. `Lock` calls `zeroize()` on the seed and all
|
|
cached derived keys. The `SecretService` uses `Zeroize`-derived types for all
|
|
sensitive material.
|
|
|
|
### Key Derivation
|
|
|
|
#### BIP39 Mnemonic and Seed Derivation
|
|
|
|
```rust
|
|
let mnemonic = Mnemonic::from_phrase(&phrase, Language::English)?;
|
|
let seed = mnemonic.to_seed(Some(&passphrase));
|
|
let master_key = ExtendedPrivKey::new_master(Network::Alknet, &seed)?;
|
|
```
|
|
|
|
#### SLIP-0010 Ed25519 HD Key Derivation
|
|
|
|
The `74'` coin type is unallocated per SLIP-0044 and reserved for alknet.
|
|
|
|
#### Derivation Path Constants
|
|
|
|
| Path | Purpose | Curve/Algorithm |
|
|
|------|---------|----------------|
|
|
| `m/74'/0'/0'/0'` | Primary identity keypair | Ed25519 (alknet auth) |
|
|
| `m/74'/0'/0'/{n}'` | Worker/device identity | Ed25519 |
|
|
| `m/74'/0'/1'/0'` | SSH host key | Ed25519 |
|
|
| `m/74'/1'/0'/{hash}'` | Site-specific password | Deterministic |
|
|
| `m/74'/2'/0'/0'` | Encryption key for external credentials | AES-256-GCM |
|
|
| `m/44'/60'/0'/0/0` | Ethereum signing key | secp256k1 |
|
|
|
|
These constants are defined in `derivation::PATHS` for programmatic access.
|
|
|
|
### AES-256-GCM Encryption for External Credentials
|
|
|
|
External credentials (API keys, OAuth tokens) that cannot be derived are
|
|
encrypted using a key derived from the seed at path `m/74'/2'/0'/0'`. The
|
|
`EncryptedData` type stores the key version, salt, IV, and ciphertext.
|
|
|
|
1. The secret service derives an AES-256-GCM key via path `m/74'/2'/0'/0'`
|
|
2. External credentials are encrypted with this key
|
|
3. The encrypted data is stored as a `SecretNode` in the metagraph
|
|
4. Only the derivation path and key version are stored in plain attributes
|
|
5. The seed phrase (or derived encryption key) is held only by the secret
|
|
service — never in the database
|
|
|
|
### SecretProtocol irpc Service
|
|
|
|
```rust
|
|
#[rpc_requests(message = SecretMessage)]
|
|
#[derive(Debug, Serialize, Deserialize)]
|
|
enum SecretProtocol {
|
|
#[rpc(tx=oneshot::Sender<DerivedKey>)]
|
|
#[wrap(DeriveEd25519)]
|
|
DeriveEd25519 { path: String },
|
|
|
|
#[rpc(tx=oneshot::Sender<DerivedKey>)]
|
|
#[wrap(DeriveEncryptionKey)]
|
|
DeriveEncryptionKey { path: String },
|
|
|
|
#[rpc(tx=oneshot::Sender<DerivedKey>)]
|
|
#[wrap(DeriveEthereumKey)]
|
|
DeriveEthereumKey { path: String },
|
|
|
|
#[rpc(tx=oneshot::Sender<Vec<u8>>)]
|
|
#[wrap(DerivePassword)]
|
|
DerivePassword { path: String, length: usize },
|
|
|
|
#[rpc(tx=oneshot::Sender<EncryptedData>)]
|
|
#[wrap(Encrypt)]
|
|
Encrypt { plaintext: String, key_version: u32 },
|
|
|
|
#[rpc(tx=oneshot::Sender<String>)]
|
|
#[wrap(Decrypt)]
|
|
Decrypt { encrypted: EncryptedData },
|
|
|
|
#[rpc(tx=oneshot::Sender<()>)]
|
|
#[wrap(Lock)]
|
|
Lock,
|
|
|
|
#[rpc(tx=oneshot::Sender<()>)]
|
|
#[wrap(Unlock)]
|
|
Unlock { passphrase: String },
|
|
}
|
|
|
|
#[derive(Debug, Serialize, Deserialize)]
|
|
struct DerivedKey {
|
|
key_type: KeyType,
|
|
private_key: Vec<u8>,
|
|
public_key: Vec<u8>,
|
|
}
|
|
|
|
#[derive(Debug, Serialize, Deserialize)]
|
|
enum KeyType {
|
|
Ed25519,
|
|
Aes256Gcm,
|
|
Secp256k1,
|
|
}
|
|
|
|
#[derive(Debug, Serialize, Deserialize)]
|
|
struct EncryptedData {
|
|
key_version: u32,
|
|
salt: String, // Base64-encoded
|
|
iv: String, // Base64-encoded
|
|
data: String, // Base64-encoded
|
|
}
|
|
```
|
|
|
|
### Wire Format Compatibility with alknet-storage
|
|
|
|
The `EncryptedData` type (`key_version`, `salt`, `iv`, `data`) is the stable
|
|
wire format shared with alknet-storage. This is type-level compatibility — not a
|
|
crate dependency. alknet-storage stores encrypted nodes using this format;
|
|
alknet-secret encrypts and decrypts using this format.
|
|
|
|
The Rust `EncryptedData` struct in alknet-secret is a superset of the TypeScript
|
|
`EncryptedDataSchema` from `@alkdev/storage`. Migration path: re-encrypt
|
|
TypeScript-encrypted data using the Rust secret service with a new key version.
|
|
See OQ-SVC-03.
|
|
|
|
### Deployment Topologies
|
|
|
|
**Minimal (single node, CLI)**: Secret service runs in the same process. Seed
|
|
phrase entered at startup. All keys derived locally. No irpc overhead.
|
|
|
|
**Production (head node)**: Secret service runs on a dedicated node or as a
|
|
local irpc service. Workers request derived keys via irpc over QUIC. The seed
|
|
never leaves the secret service node.
|
|
|
|
## Constraints
|
|
|
|
- The seed phrase is never persisted to disk. It is entered at startup or via
|
|
`Unlock` and held only in `Zeroize`-protected RAM (ADR-038).
|
|
- `Lock` calls `zeroize()` on the seed and all cached derived keys.
|
|
- alknet-secret does not depend on alknet-core or alknet-storage. It is fully
|
|
independent (ADR-027).
|
|
- The `EncryptedData` wire format is shared with alknet-storage for type-level
|
|
compatibility, not a crate dependency.
|
|
- Per ADR-032, secret service domain events (key derivation notifications) stay
|
|
within the service boundary. External consumers use irpc calls or call
|
|
protocol operations projected to integration events.
|
|
- irpc is always a dependency (not feature-gated) per ADR-027.
|
|
- `SecretProtocol` defines the wire format for in-cluster communication
|
|
(postcard serialization). For call protocol exposure (e.g.,
|
|
`/head/secrets/derive`), the service is wrapped in an operation that
|
|
serializes to JSON.
|
|
|
|
## Phase Progression
|
|
|
|
| Phase | Scope | Notes |
|
|
|-------|-------|-------|
|
|
| Phase 3 (now) | Basic crate: mnemonic, derivation, encryption, irpc protocol, service lifecycle | Core key management |
|
|
| Phase A | Integration with alknet-storage via `EncryptedData` wire format. CLI commands for unlock/lock/derive. `SecretStoreCredentialProvider` wiring. | Full service integration |
|
|
| Phase B | Memory hardening: `mlock`/`VirtualLock` for seed RAM, constant-time comparison, audit logging of derivation requests. | Security hardening |
|
|
| Phase C | Multi-seed support (tenant isolation): indexed `Unlock` with tenant ID. | Multi-tenancy |
|
|
|
|
## Open Questions
|
|
|
|
- **OQ-SVC-01**: Should the secret service support multiple seed phrases (one
|
|
per tenant)? See [open-questions.md](open-questions.md).
|
|
|
|
- **OQ-SVC-03**: How does the secret service integrate with the existing
|
|
`EncryptedDataSchema` from `@alkdev/storage`? See [open-questions.md](open-questions.md).
|
|
|
|
- **OQ-SVC-04**: Should workers cache derived keys locally? See
|
|
[open-questions.md](open-questions.md).
|
|
|
|
- **OQ-SEC-01**: Should alknet-secret use `mlock`/`VirtualLock` to prevent seed
|
|
RAM from being paged to disk? See [open-questions.md](open-questions.md).
|
|
|
|
## Design Decisions
|
|
|
|
| ADR | Decision | Summary |
|
|
|-----|----------|---------|
|
|
| [027](decisions/027-crate-decomposition.md) | Crate decomposition | alknet-secret is independent of core and storage |
|
|
| [032](decisions/032-event-boundary-discipline.md) | Event boundary | Secret service domain events stay internal |
|
|
| [038](decisions/038-seed-lifecycle-memory-security.md) | Seed lifecycle and memory security | Zeroize for sensitive material, mlock deferred to Phase B |
|
|
|
|
## References
|
|
|
|
- [research/services.md](../research/services.md) — SecretProtocol definition, DerivedKey, KeyType
|
|
- [research/storage.md](../research/storage.md) — Secrets section, derivation paths, EncryptedData
|
|
- [research/integration-plan.md](../research/integration-plan.md) — Phase 3.1
|
|
- [credentials.md](credentials.md) — CredentialProvider (outbound auth, consumes SecretProtocol::Decrypt)
|
|
- SLIP-0010 — https://github.com/satoshilabs/slips/blob/master/slip-0010.md
|
|
- BIP39 — https://github.com/bitcoin/bips/blob/master/bip-0039.mediawiki |