Skip to main content

ts_keys/
lib.rs

1#![doc = include_str!("../README.md")]
2#![no_std]
3
4extern crate alloc;
5
6mod keystate;
7mod macros;
8
9#[doc(inline)]
10pub use keystate::{NodeState, PersistState};
11use macros::{
12    _create_x25519_base_key_type, create_ed25519_keypair_types, create_ed25519_private_key_type,
13    create_ed25519_public_key_type, create_x25519_keypair_types, create_x25519_private_key_type,
14    create_x25519_public_key_type,
15};
16
17/// Errors that may occur when parsing a string into a key type.
18#[derive(Debug, Copy, Clone, PartialEq, Eq, thiserror::Error)]
19pub enum ParseError {
20    /// Key string was formatted incorrectly.
21    #[error("key string was formatted incorrectly")]
22    InvalidFormat,
23
24    /// Key was the wrong length.
25    #[error("key was the wrong length")]
26    WrongLength,
27
28    /// Parsed prefix did not match the key type.
29    #[error("parsed prefix did not match the key type")]
30    BadPrefix,
31}
32
33// The client never handles challenge private keys, so we only create a public key type rather than
34// public/private/keypair types.
35create_x25519_public_key_type!(
36    /// The X25519 public key of a challenge issued by control to a Tailnet node during registration.
37    ChallengePublicKey,
38    "chalpub"
39);
40
41// The client never handles DERP server private keys, so we only create a public key type rather
42// than public/private/keypair types.
43create_x25519_public_key_type!(
44    /// The X25519 public key of a DERP server.
45    DerpServerPublicKey,
46    "derp"
47);
48create_x25519_keypair_types!(
49    /// The X25519 public key a Tailscale node uses for the Disco protocol.
50    DiscoPublicKey,
51    "discokey",
52    /// The X25519 private key a Tailscale node uses for the Disco protocol.
53    DiscoPrivateKey,
54    "privkey",
55    /// The X25519 public/private key pair a Tailscale node uses for the Disco protocol.
56    DiscoKeyPair
57);
58
59create_x25519_keypair_types!(
60    /// The X25519 public key of a unique piece of hardware running one or more Tailscale nodes.
61    /// Also the key type sent from a control server to a Tailscale node during the initial control
62    /// handshake.
63    MachinePublicKey,
64    "mkey",
65    /// The X25519 private key of a unique piece of hardware running one or more Tailscale nodes.
66    MachinePrivateKey,
67    "privkey",
68    /// The X25519 public/private key pair of a unique piece of hardware running one or more
69    /// Tailscale nodes.
70    MachineKeyPair
71);
72
73create_ed25519_keypair_types!(
74    /// The Ed25519 public key of a Tailscale node for use with Tailnet Lock.
75    NetworkLockPublicKey,
76    "nlpub",
77    /// The Ed25519 private key of a Tailscale node for use with Tailnet Lock.
78    NetworkLockPrivateKey,
79    "nlpriv",
80    /// The Ed25519 public/private key pair of a Tailscale node for use with Tailnet Lock.
81    NetworkLockKeyPair
82);
83
84create_x25519_keypair_types!(
85    /// The X25519 public key of a Tailscale node.
86    NodePublicKey,
87    "nodekey",
88    /// The X25519 private key of a Tailscale node.
89    NodePrivateKey,
90    "privkey",
91    /// The X25519 public/private key pair of a Tailscale node.
92    NodeKeyPair
93);
94
95#[cfg(test)]
96mod debug_redaction_tests {
97    use alloc::format;
98
99    use super::{
100        DiscoPrivateKey, MachinePrivateKey, NetworkLockPrivateKey, NodePrivateKey, NodePublicKey,
101    };
102
103    /// A private key's `Debug` MUST NOT contain the secret bytes (regression guard for the
104    /// log-leak fixed in tsr-9nu). We use an all-`0xAB` key so the hex `"ab"` is unmistakable.
105    #[test]
106    fn private_key_debug_is_redacted() {
107        let secret = [0xABu8; 32];
108
109        let m = MachinePrivateKey::from(secret);
110        let n = NodePrivateKey::from(secret);
111        let d = DiscoPrivateKey::from(secret);
112        let nl = NetworkLockPrivateKey::from(secret);
113
114        for (label, dbg) in [
115            ("MachinePrivateKey", format!("{m:?}")),
116            ("NodePrivateKey", format!("{n:?}")),
117            ("DiscoPrivateKey", format!("{d:?}")),
118            ("NetworkLockPrivateKey", format!("{nl:?}")),
119        ] {
120            assert!(
121                dbg.contains("<redacted>"),
122                "{label} Debug should be redacted, got {dbg:?}"
123            );
124            assert!(
125                !dbg.contains("abab"),
126                "{label} Debug leaked secret bytes: {dbg:?}"
127            );
128            // The secret is also reachable via Display/to_bytes — confirm those still expose it,
129            // so the redaction is Debug-only and didn't break the explicit serialization paths.
130            assert!(
131                format!("{m}").contains("abab"),
132                "Display must still expose the key bytes"
133            );
134        }
135    }
136
137    /// A public key's `Debug` SHOULD still print the full `prefix:hex` (public keys are not secret).
138    #[test]
139    fn public_key_debug_shows_hex() {
140        let pubk = NodePublicKey::from([0xABu8; 32]);
141        let dbg = format!("{pubk:?}");
142        assert!(
143            dbg.contains("abab"),
144            "public key Debug should show hex: {dbg:?}"
145        );
146        assert_eq!(dbg, format!("{pubk}"), "public Debug == Display");
147    }
148
149    /// Private keys wipe their secret bytes on drop (`ZeroizeOnDrop`, tsr-9nu). We can't observe a
150    /// value after it drops in safe Rust, so this drives `Zeroize::zeroize` explicitly (the same
151    /// code the drop glue runs) and confirms the buffer is zeroed — a behavioral guard that the
152    /// derive is wired up, not merely that it compiles.
153    #[test]
154    fn private_key_zeroize_wipes_bytes() {
155        use zeroize::Zeroize;
156
157        let mut k = NodePrivateKey::from([0xABu8; 32]);
158        assert_eq!(
159            k.to_bytes(),
160            [0xABu8; 32],
161            "precondition: key holds its bytes"
162        );
163        k.zeroize();
164        assert_eq!(
165            k.to_bytes(),
166            [0u8; 32],
167            "zeroize must wipe the secret bytes to zero"
168        );
169    }
170
171    /// `public_key()` borrows (`&self`) — deriving the public key must not consume the private key,
172    /// so it stays usable afterwards. This is the API shape that lets callers hold a private key
173    /// without it being moved/dropped on every derivation (mirrors Go's `key.NodePrivate.Public()`).
174    #[test]
175    fn public_key_derivation_borrows_private() {
176        let k = NodePrivateKey::from([0x11u8; 32]);
177        let p1 = k.public_key();
178        // `k` is still alive here precisely because `public_key` took `&self`.
179        let p2 = k.public_key();
180        assert_eq!(p1, p2, "repeated derivation from the same key agrees");
181        // And a clone derives the same public key (clone copies the secret faithfully).
182        assert_eq!(k.clone().public_key(), p1);
183    }
184
185    /// A key string of the right length+prefix but containing non-hex (or a non-ASCII char that
186    /// splits a 2-byte window) must parse to `Err`, NOT panic. Regression: the hex loop used
187    /// `.unwrap()` on `get(i..i+2)` and `from_str_radix`, so a malformed key in a control response
188    /// would unwind and kill the netmap decoder. Go's key parse returns an error here.
189    #[test]
190    fn malformed_hex_key_errors_not_panics() {
191        use core::str::FromStr;
192
193        // 64 non-hex ASCII chars: length + prefix pass, `from_str_radix("zz",16)` must error.
194        let non_hex = alloc::format!("nodekey:{}", "z".repeat(64));
195        assert!(
196            NodePublicKey::from_str(&non_hex).is_err(),
197            "non-hex key body must be a parse error, not a panic"
198        );
199
200        // A multi-byte UTF-8 char makes the byte length 64 while `get(i..i+2)` can land on a char
201        // boundary and return None — must also be an error, not a panic. "é" is 2 UTF-8 bytes.
202        let multibyte = alloc::format!("nodekey:{}", "é".repeat(32));
203        assert_eq!(multibyte.len() - "nodekey:".len(), 64, "body is 64 bytes");
204        assert!(
205            NodePublicKey::from_str(&multibyte).is_err(),
206            "a non-ASCII body must be a parse error, not a panic"
207        );
208    }
209}
210
211#[cfg(all(test, feature = "serde"))]
212mod nl_tests {
213    use core::str::FromStr;
214
215    use super::{NetworkLockKeyPair, NetworkLockPrivateKey, NetworkLockPublicKey};
216
217    /// A `NetworkLockKeyPair` round-trips through its `nlpriv:`/`nlpub:` string forms.
218    #[test]
219    fn nl_key_roundtrip_serde() {
220        let kp = NetworkLockKeyPair::new();
221
222        let priv_str = alloc::format!("{}", kp.private);
223        let pub_str = alloc::format!("{}", kp.public);
224        assert!(priv_str.starts_with("nlpriv:"));
225        assert!(pub_str.starts_with("nlpub:"));
226
227        let parsed_priv = NetworkLockPrivateKey::from_str(&priv_str).unwrap();
228        let parsed_pub = NetworkLockPublicKey::from_str(&pub_str).unwrap();
229        assert_eq!(parsed_priv, kp.private);
230        assert_eq!(parsed_pub, kp.public);
231    }
232
233    /// Public-key derivation is deterministic and matches the audited ed25519-dalek RFC 8032
234    /// seed->public derivation (proving we are NOT using X25519 scalar multiplication).
235    #[test]
236    fn nl_public_derivation_is_deterministic() {
237        let seed = [7u8; 32];
238        let sk = NetworkLockPrivateKey::from(seed);
239        let p1 = sk.public_key();
240        let p2 = sk.public_key();
241        assert_eq!(p1, p2);
242
243        let dalek = ed25519_dalek::SigningKey::from_bytes(&seed)
244            .verifying_key()
245            .to_bytes();
246        assert_eq!(p1.to_bytes(), dalek);
247    }
248
249    /// RFC 8032 §7.1 TEST 1 known-answer vector, exercised through the fork's own
250    /// `NetworkLockPrivateKey`: the canonical Ed25519 secret seed must derive the canonical public
251    /// key. This pins the fork's NL key to standards-conformant Ed25519 (RFC 8032) — a stronger proof
252    /// than the dalek cross-check, since it would catch a byte-swap, a wrong seed interpretation, or a
253    /// future dependency that derived a different curve. Vector verified against rfc-editor.org/rfc/rfc8032.txt.
254    #[test]
255    fn nl_public_matches_rfc8032_test1() {
256        fn unhex(s: &str) -> [u8; 32] {
257            let b = s.as_bytes();
258            let mut out = [0u8; 32];
259            let mut i = 0;
260            while i < 32 {
261                let hi = (b[2 * i] as char).to_digit(16).unwrap() as u8;
262                let lo = (b[2 * i + 1] as char).to_digit(16).unwrap() as u8;
263                out[i] = (hi << 4) | lo;
264                i += 1;
265            }
266            out
267        }
268        let seed = unhex("9d61b19deffd5a60ba844af492ec2cc44449c5697b326919703bac031cae7f60");
269        let public = unhex("d75a980182b10ab7d54bfed3c964073a0ee172f3daa62325af021a68f707511a");
270        let derived = NetworkLockPrivateKey::from(seed).public_key();
271        assert_eq!(derived.to_bytes(), public);
272
273        // Also lock in the full `nlpub:`+lowercase-hex emission (prefix + hex jointly), which is the
274        // exact text form Go sends as `RegisterRequest.NLKey` (`key.NLPublic.MarshalText`).
275        assert_eq!(
276            alloc::format!("{derived}"),
277            "nlpub:d75a980182b10ab7d54bfed3c964073a0ee172f3daa62325af021a68f707511a"
278        );
279    }
280
281    /// `KeyPair::new`, `From<private>`, and the standalone `private.public_key()` all agree on the
282    /// derived public — there is one derivation, regardless of how the pair is constructed.
283    #[test]
284    fn nl_keypair_derivation_is_consistent() {
285        let kp = NetworkLockKeyPair::new();
286        assert_eq!(kp.public, kp.private.public_key());
287        // `.clone()`: `From<private>` consumes the key (no longer `Copy`); keep `kp.private` for
288        // the equality check below.
289        let from_priv = NetworkLockKeyPair::from(kp.private.clone());
290        assert_eq!(from_priv.public, kp.public);
291        assert_eq!(from_priv.private, kp.private);
292    }
293
294    /// Regression guard: the Ed25519 derivation must NOT match the old (buggy) X25519 derivation
295    /// for the same 32-byte seed.
296    #[test]
297    fn nl_key_is_not_x25519() {
298        let seed = [7u8; 32];
299        let ed = NetworkLockPrivateKey::from(seed).public_key().to_bytes();
300        let x = x25519_dalek::PublicKey::from(&x25519_dalek::StaticSecret::from(seed)).to_bytes();
301        assert_ne!(ed, x);
302    }
303}