BOLT12 Address Support (DRAFT!)
Inspired by the awesome lightningaddress.com, except for BOLT12:
- Supports BOLT12
- Allows BOLT12 vendor string authentication
- Doesn't require your wallet to query the server directly
- Required only to establish the initial node linkage
How Does it Work?
Like lightningaddress.com, you turn [email protected] into a web request:
https://domain.com/.well-known/bolt12/bitcoin/[email protected]
But you can also authenticate the entire domain:
https://domain.com/.well-known/bolt12/bitcoin/domain.com
(Instead of bitcoin you could use testnet, signet or regtest)
The Format
The format is a bolt12 TLV binary (Content-type: application/x-lightning-bolt12), containing the following fields:
tlv_stream:addressproof- types:
- type: 2 (
chains) - data:
- [
...*chain_hash:chains]
- [
- type: 10 (
description) - data:
- [
...*utf8:description]
- [
- type: 12 (
features) - data:
- [
...*byte:features]
- [
- type: 14 (
absolute_expiry) - data:
- [
tu64:seconds_from_epoch]
- [
- type: 16 (
paths) - data:
- [
...*blinded_path:paths]
- [
- type: 20 (
vendor) - data:
- [
...*utf8:vendor]
- [
- type: 60 (
node_ids) - data:
- [
...*point32:node_ids]
- [
- type: 500 (
certsignature) - data:
- [
...*byte:sig]
- [
- type: 501 (
cert) - data:
- [
...*byte:cert]
- [
- type: 503 (
certchain) - data:
- [
...*byte:chain]
- [
- type: 2 (
Only the vendor, node_ids and certsignature fields are required, the others are optional.
Requirements
The writer:
- MUST set
vendorto filename being served:- either [email protected] or simply domain.
- MUST set
node_idsto zero or more node_ids which will be used to sign offers for this vendor. - MUST set
chainsto the chains thesenode_idsare valid for, or MAY not setchainsif thenode_idsare valid for Bitcoin. - MAY set
features,absolute_expiry,descriptionandpaths(see BOLT 12). - MUST set
certsignatureto the RSA signature (using PSS padding mode maximum saltlen) of the BOLT-12 merkle root as per BOLT12 Signature Calculation using the secret key for the domain invendor. - MUST NOT set
descriptionunless it has an offer which is constructed using the other fields, and the offer'snode_idset to the first of thenode_ids. - If it is serving the
addressproofover HTTPS:- MAY set
certandcertchain
- MAY set
- Otherwise:
- MUST set both
certandcertchain
- MUST set both
- If it sets
cert:- MUST set it to the PEM-encoded certificate corresponding to the domain
- If it sets
certchain:- MUST set it to the PEM-encoded chain of certificates leading from
certto the root CA
- MUST set it to the PEM-encoded chain of certificates leading from
The reader:
-
MUST NOT accept the address proof if
vendor,node_idsorcertsignatureis not present. -
MUST NOT accept the address proof if an even unknown bit is set in
features. -
If it has NOT retrieved the
addressproofover HTTPS:- MUST NOT accept the address proof if:
certis not present, or not valid for the domain invendor.certchainis not present, or does not linkcertto a root certificate authority.certsignaturefield is not a valid signature for BOLT-12 merkle root using the key incert.
- MUST NOT accept the address proof if:
-
otherwise:
- MAY retrieve
certandcertchainfrom the HTTPS connection. - MAY NOT accept the address proof as it would in the non-HTTPS case above.
- MAY retrieve
-
If if has a previous, valid
addressprooffor this vendor:- MUST ONLY replace it with this address proof if:
absolute_expiryis set, AND- it is greater than the previous
absolute_expiryOR the previous had noabsolute_expiryfield.
- MUST ONLY replace it with this address proof if:
-
MUST consider the
addressproofno longer valid ifabsolute_expiryis set and the current number of seconds since 1970 is greater than that value. -
if
descriptionis present:- MAY use the fields of this
addressproofas an unsigned offer.
- MAY use the fields of this
-
When it encounters a
vendorfield in a BOLT12 offer or invoice:- if the vendor begins with a valid domain, up to a space character:
- SHOULD WARN the user if it cannot find a current valid address proof.
- SHOULD reject the offer or invoice if the node_id is not one of the
node_idsin the offer.
- if the vendor begins with a valid domain, up to a space character:
Text Encoding
The human-readable prefix for addressproof is lnap, if you want it encoded as a string.
What Does All This Do?
This allows domain validation for bolt 12 offers, which already have a vendor field for this purpose. e.g if the vendor in an offer is "blockstream.com Get your blocks here!" then your wallet can reach out to blockstream.com to see if it the node_id in the offer really is under their control.
It also allows node_id proofs for individual addresses.
But you don't need to reach out to blockstream.com: anyone (including your wallet vendor, or the node claiming to be blockstream.com) can collect all the addressproofs and certificates for you, as they contain a signature using the existing web certificate infrastructure. Bundling these protects your privacy more than having to request to a vendor's website before making a payment.
This format is a subset of the BOLT12 offer format, so if it has a description it is actually a valid (amountless) offer, allowing immediate tipping using it.
You can also include zero node_ids, as a way of indicating that you do not have any lightning nodes.
Examples
You will need access to your privkey.pem, cert.pem and chain.pem files on your HTTPS webserver which serves the domain.
This creates a proof that bolt12.org operates nodeid 4b9a1fa8e006f1e3937f65f66c408e6da8e1ca728ea43222a7381df1cc449605 (note we omit the 02/03 prefix):
$ ./shell/make-addressproof.sh \
--vendor=bolt12.org \
--nodeid=4b9a1fa8e006f1e3937f65f66c408e6da8e1ca728ea43222a7381df1cc449605 \
--privkeyfile=certs/privkey.pem \
--certfile=certs/cert.pem \
--chainfile=certs/chain.pem > .well-known/bolt12/bitcoin/bolt12.org
This does the same thing using the Python script:
$ ./python/bolt12address.py create \
--raw \
bolt12.org \
certs/privkey.pem \
certs/cert.pem \
certs/chain.pem \
4b9a1fa8e006f1e3937f65f66c408e6da8e1ca728ea43222a7381df1cc449605 \
> .well-known/bolt12/bitcoin/bolt12.org
This creates a signet signature for multiple nodeids, for the user [email protected], and adds a description so it can also serve as offer for sending unsolicited payments (note: see below!):
$ ./shell/make-addressproof.sh \
[email protected] \
--nodeid=4b9a1fa8e006f1e3937f65f66c408e6da8e1ca728ea43222a7381df1cc449605 \
--nodeid=994b9a1fa8e006f1e3937f65f66c408e6da8e1ca728ea43222a7381df1cc4496 \
--privkeyfile=certs/privkey.pem \
--certfile=certs/cert.pem \
--chainfile=certs/chain.pem \
--chain=signet \
--description='Unsolicited bolt12address donation' \
> .well-known/bolt12/signet/[email protected]
And in Python:
$ ./python/bolt12address.py create \
--raw \
--description='Unsolicited bolt12address donation' \
--chain=signet \
[email protected] \
certs/privkey.pem \
certs/cert.pem \
certs/chain.pem \
4b9a1fa8e006f1e3937f65f66c408e6da8e1ca728ea43222a7381df1cc449605 \
994b9a1fa8e006f1e3937f65f66c408e6da8e1ca728ea43222a7381df1cc4496 \
> .well-known/bolt12/signet/[email protected]
We can check this using python:
$ ./python/bolt12address.py check --raw-stdin < .well-known/bolt12/signet/[email protected]
chains: ['f61eee3b63a380a477a063af32b2bbc97c9ff9f01f2c4225e973988108000000']
description: Unsolicited bolt12address donation
vendor: [email protected]
node_ids: ['4b9a1fa8e006f1e3937f65f66c408e6da8e1ca728ea43222a7381df1cc449605', '994b9a1fa8e006f1e3937f65f66c408e6da8e1ca728ea43222a7381df1cc4496']
certsignature: 173e...
offer_id: 3234297fb2414b62c16ac9751ac241050199ec5e2cd83e713136cc26974f09a8
offer_id: 3139b327c9fa6637a7ef620149425a1163e19a1181c9f1cbdc7820360dd40c23
The offer_id at the end if description is populated (one for each node_id) is the offer_id anyone reading would expect to be able to send funds to. You should create this offer (with that description and no amount) on your node!
Refreshing Existing Proofs
There's a simple helper to refresh existing address proofs, such as when your certificate changes:
$ ./python/bolt12address.py refresh \
certs/privkey.pem \
certs/cert.pem \
certs/chain.pem \
.well-known/bolt12/signet/*bolt12.org
.well-known/bolt12/signet/[email protected]: REFRESHED
TODO
This is a draft: I expect it to change after feedback (especially since the certinficates and signatures are large and clunky).
The code does not check the certificate chain, and is generally could use polishing.
We also need more routines in different languages to fetch and check the bolt12address, and a method so Lightning nodes can serve their addressproof directly.
Feedback
You can reach out to me as [email protected] or join the bolt12 telegram group at https://t.me/bolt12org.
Happy hacking!