bip-85: add example use for age key derivation #2174

pull dmonakhov wants to merge 1 commits into bitcoin:master from dmonakhov:bip-85-age-example changing 1 files +15 −0
  1. dmonakhov commented at 5:06 PM on May 23, 2026: none

    Motivation

    age is a widely-deployed file-encryption format. Both its classic (X25519, since 2021) and post-quantum (X-Wing / HPKE-MLKEM768-X25519, age v1.3.0+ Dec 2024) identity types take a 32-byte uniform seed as the secret-key material — which is precisely what BIP-85's HEX application at num_bytes=32 already returns.

    This PR adds a brief example-use subsection at the end of the HEX application section, documenting how to interpret the existing 32-byte HEX output as an age identity. No new application number, no normative MUSTs added, no changes required in existing implementations — the HEX application already produces exactly what age consumes; this PR just records the connection in the spec.

    Documenting this gives users a single backup story: the same BIP-39 mnemonic that already secures their Bitcoin wallets can deterministically produce age identities, with no additional key-rotation infrastructure on top of what they already do.

    It also unblocks downstream hardware-wallet firmware integration maintainers of firmware projects accept "implement this BIP-85 use case" much more readily than "implement a convention from a third-party repo." A spec-blessed convention turns the conversation from a bespoke contribution into a standards-conformance issue.

    What changes

    A new ====Example use: age key derivation==== subsection at the end of the ===HEX=== section in bip-0085.mediawiki.

    The subsection:

    • Describes the bech32 encoding for both classic and PQ age identities + their corresponding recipients.
    • Adds a worked test vector using the same master xprv as the existing HEX test vector, so a reader can derive both side-by-side from one master.
    • Links to a reference implementation (age-keygen-det).

    Test vectors

    Derived from the master xprv already used by the HEX test vector in the same section. Reproducible end-to-end with stock tools:

    INPUT:
    * MASTER BIP32 ROOT KEY: xprv9s21ZrQH143K2LBWUUQRFXhucrQqBpKdRRxNVq2zBqsx8HVqFk2uYo8kmbaLLHRdqtQpUm98uKfu3vca1LqdGhUtyoFnCNkfmXRyPXLjbKb
    * PATH: m/83696968'/128169'/32'/0'
    
    OUTPUT
    * DERIVED ENTROPY=ea3ceb0b02ee8e587779c63f4b7b3a21e950a213f1ec53cab608d13e8796e6dc
    
    # Classic (X25519):
    * DERIVED AGE IDENTITY=AGE-SECRET-KEY-1AG7WKZCZA689SAMECCL5K7E6Y854PGSN78K98J4KPRGNAPUKUMWQWNNT4U
    * DERIVED AGE RECIPIENT=age1m0hhzxelxsxnxm4ennvdpk75j8s7mn5w4tt3e4ntug5qx256wslqmdz8e9
    
    # Post-quantum (X-Wing):
    * DERIVED AGE PQ IDENTITY=AGE-SECRET-KEY-PQ-1AG7WKZCZA689SAMECCL5K7E6Y854PGSN78K98J4KPRGNAPUKUMWQ5AN2M5
    * DERIVED AGE PQ RECIPIENT (SHA-256)=feecdb11f82478ea4b9dd934965f974e37701d963ac2ec5d4fb120357032641a
    

    The PQ recipient is a 1959-character bech32 string (it encodes the 1216-byte X-Wing public key); only its SHA-256 is shown inline. The full string is regenerable from the seed via X-Wing.NewKeyFromSeed(seed).EncapsulationKey() and lives in the reference implementation's test data.

    End-to-end verification path:

    1. bipsea derive -a hex -n 64 -i 0 -x <MASTER> reproduces the existing HEX64 test vector (sanity check).
    2. bipsea derive -a hex -n 32 -i 0 -x <MASTER> produces the 32-byte entropy above.
    3. Bech32-encoding with the respective HRPs produces the identity strings.
    4. Stock age-keygen -y (age v1.3.0+) on each identity returns the matching recipient byte-for-byte (classic full string; PQ full string hashes to the stated SHA-256).

    Scope

    This amendment is intentionally minimal:

    • No new application number — reuses the existing HEX application (128169').
    • No normative MUSTs added beyond what BIP-85 already says about the HEX application.
    • No changes required in existing implementations (bipsea, bip85-js, ethankosakovsky/bip85).

    Possible follow-ups (not in this PR):

    • A dedicated AGE subsection paralleling the existing RSA-GPG subsection, with full worked examples for each flavour and an optional index-partition recommendation (e.g. [0, 1000) for PQ identities, [1000, 2000) for classic identities, to avoid producing the same 32 bytes under two different HRPs when the same master is used for both flavours).
    • A dedicated application number for age (e.g. 0x616765' = hex-pack of ASCII "age"), with independent paths per flavour.

    These are noted only to clarify that this PR deliberately picks the minimal form; they can be raised separately if there is appetite.

    Reference implementation

    age-keygen-det — single-binary Go tool that converts a 32-byte hex seed into an age identity file (classic or PQ). The output format mirrors stock age-keygen so existing age tooling (age-keygen -y, age -d, age -r) works unmodified.

    • BSD-3-Clause licensed.
    • Cross-validates byte-for-byte against stock age-keygen -y in CI for both flavours.
    • Reproducible static build (CGO_ENABLED=0, -trimpath, -buildid=); same source → byte-identical binary.

    A worked end-to-end example using bipsea + age-keygen-det + the spec's own master xprv is documented in the project's README under "Worked example: BIP-85 → age".

    Open question

    Should the subsection also include the index-range partition guidance ([0, 1000) for PQ identities, [1000, 2000) for classic identities) to avoid producing the same 32 bytes under two different HRPs when both flavours are derived from one master? The reference implementation does NOT enforce this partition — it is a documentation convention, not a crypto contract. Including it would be useful guidance; excluding it keeps the amendment maximally minimal. Open to reviewer preference.


    cc @ethankosakovsky @akarve — original BIP-85 authors per the BIP header. Happy to revise based on review feedback.

  2. bip-85: add example use for age key derivation
    Documents the connection between BIP-85's HEX application output
    at num_bytes=32 and the private-key seed format used by the age
    file-encryption format (https://age-encryption.org/v1) for both
    the classic X25519 identity (since age v1.0) and the
    post-quantum X-Wing identity (age v1.3.0+, Dec 2024).
    
    No new application number, no normative MUSTs added, no changes
    required in existing implementations. The amendment is purely
    documentational: a single subsection appended at the end of the
    HEX section.
    
    A test vector is added using the same master xprv as the
    existing HEX test vector in the same section, with both classic
    and post-quantum outputs. The PQ recipient (a ~1959-character
    bech32 string encoding the 1216-byte X-Wing public key) is
    included as a SHA-256 hash inline; the full string is in the
    reference implementation's test data.
    
    Reference implementation: https://github.com/dmonakhov/age-keygen-det
    - single-binary Go tool, BSD-3-Clause
    - cross-validates byte-for-byte against stock age-keygen -y in
      CI for both flavours
    6cb352dd26
  3. kwest3170-wq commented at 1:00 AM on May 24, 2026: none

    Thanks i apologies

  4. in bip-0085.mediawiki:287 in 6cb352dd26
     281 | @@ -282,6 +282,21 @@ INPUT:
     282 |  OUTPUT
     283 |  * DERIVED ENTROPY=492db4698cf3b73a5a24998aa3e9d7fa96275d85724a91e71aa2d645442f878555d078fd1f1f67e368976f04137b1f7a0d19232136ca50c44614af72b5582a5c
     284 |  
     285 | +====Example use: age key derivation====
     286 | +
     287 | +The 32-byte output at <code>m/83696968'/128169'/32'/{index}'</code> is uniformly random and is suitable as the private-key seed for the [https://age-encryption.org/v1 age] file-encryption format. For classic age identities, bech32-encode the 32 bytes with HRP <code>AGE-SECRET-KEY-</code>; the corresponding recipient is the bech32 encoding (HRP <code>age</code>) of <code>X25519(seed, BASEPOINT)</code>. For post-quantum age identities (age v1.3.0+, Dec 2024), the same 32 bytes are a valid X-Wing seed; bech32-encode with HRP <code>AGE-SECRET-KEY-PQ-</code>. X-Wing internally SHAKE256-expands the seed into ML-KEM-768 and X25519 components, and the recipient is the bech32 encoding (HRP <code>age1pq</code>) of the X-Wing encapsulation key. A reference implementation is [https://github.com/dmonakhov/age-keygen-det age-keygen-det].
    


    akarve commented at 9:13 PM on May 24, 2026:

    A few requests for this section:

    1. Please insert newlines every ~80-90 chars so this file reads well in a terminal
    2. The Dec 2024 date is probably TMI, version makes sense but dates generally don't age well
    3. You say "uniformly random" but the point is "cryptographically random"
    4. Consider breaking most of the above into a small table for readability and using standard nomenclature like || for concat, etc.

    Example: please check the math and semantics (assume what's here is wrong), but to give you the idea:

    flavor role HRP bytes encoded in data encoded string
    classic identity AGE-SECRET-KEY- seed (32 B) AGE-SECRET-KEY- || 1 || AG7WK…UKUMWQ || WNNT4U
    classic recipient age X25519(seed, G) (32 B) age || 1 || m0hhzx…256wslq || mdz8e9
    PQ (X-Wing) identity AGE-SECRET-KEY-PQ- seed (32 B, same bytes) AGE-SECRET-KEY-PQ- || 1 || AG7WK…UKUMWQ || 5AN2M5
    PQ (X-Wing) recipient age1pq XWing.pk(seed) (≈1216 B) age1pq || 1 || (~1950 chars) || (6 chars)

    And then define your terms below, G, etc.


    akarve commented at 9:29 PM on May 24, 2026:

    "The 32-byte output at...": let's just say "HEX application outputs can be extended with concatenation to function as age file encryption keys" or something like that.

  5. in bip-0085.mediawiki:293 in 6cb352dd26
     288 | +
     289 | +INPUT:
     290 | +* MASTER BIP32 ROOT KEY: xprv9s21ZrQH143K2LBWUUQRFXhucrQqBpKdRRxNVq2zBqsx8HVqFk2uYo8kmbaLLHRdqtQpUm98uKfu3vca1LqdGhUtyoFnCNkfmXRyPXLjbKb
     291 | +* PATH: m/83696968'/128169'/32'/0'
     292 | +
     293 | +OUTPUT
    


    akarve commented at 9:22 PM on May 24, 2026:
    OUTPUT:
    
  6. jonatack added the label Proposed BIP modification on May 24, 2026
  7. jonatack added the label Pending acceptance on May 24, 2026
  8. in bip-0085.mediawiki:285 in 6cb352dd26
     281 | @@ -282,6 +282,21 @@ INPUT:
     282 |  OUTPUT
     283 |  * DERIVED ENTROPY=492db4698cf3b73a5a24998aa3e9d7fa96275d85724a91e71aa2d645442f878555d078fd1f1f67e368976f04137b1f7a0d19232136ca50c44614af72b5582a5c
     284 |  
     285 | +====Example use: age key derivation====
    


    akarve commented at 9:25 PM on May 24, 2026:
    ====<code>age</code> file encryption keys====
    

    jonatack commented at 9:27 PM on May 24, 2026:

    It might be helpful to specify that "age" as used here specifically refers to a file encryption format, either in parentheses or with a footnote.


github-metadata-mirror

This is a metadata mirror of the GitHub repository bitcoin/bips. This site is not affiliated with GitHub. Content is generated from a GitHub metadata backup.
generated: 2026-05-25 11:10 UTC

This site is hosted by @0xB10C
More mirrored repositories can be found on mirror.b10c.me