BIP: PSET Layer: Applications Title: Partially Signed Elements Transaction Format Author: Andrew Chow <[email protected]> Comments-Summary: No comments yet. Status: Proposed Type: Standards Track Created: 2020-07-09 License: BSD-2-Clause
This document describes proprietary extensions to the BIP 174 Partially Signed Bitcoin Transaction format. These extensions are for use in Elements Sidechains to hold the information necessary for Confidential Transactions, Confidential Assets, and Peg-in transactions.
This BIP is licensed under the 2-clause BSD license.
The Partially Signed Elements Transaction (PSET) format extends the BIP 370 PSBT format. The changes are new proprietary type fields, a new magic sequence, new roles, and changes to make some PSBT fields conditionally mandatory.
The fields added for PSET are only allowed when the PSBT version is 2.
For elements, the identifier prefix string is pset. Note that previous implementations of PSET used a different prefix string and had different fields. The use of a new identifier prefix is to avoid conflicts with the previous implementations.
The following PSBT fields are made conditionally mandatory:
- PSBT_OUT_AMOUNT is not mandatory once a transaction output is blinded
- PSBT_OUT_SCRIPT is not mandatory for L-BTC fee outputs
Name | <keytype> | <keydata> | <keydata> Description | <valuedata> | <valuedata> Description | Versions Requiring Inclusion | Versions Requiring Exclusion | Versions Allowing Inclusion |
---|---|---|---|---|---|---|---|---|
Scalar Offset | PSBT_ELEMENTS_GLOBAL_SCALAR = 0x00 | <32 byte scalar> | A 32 byte Scalar Offset computed by a blinder. | None | No value. The value must have a length of zero. | 0 | 2 | |
Elements Transaction Modifiable Flag | PSBT_ELEMENTS_GLOBAL_TX_MODIFIABLE = 0x01 | None | No key data | <8-bit uint> | An 8 bit unsigned integer as a bitfield for various elements specific transaction modification flags. Bit 0 is reserved and should not be used. If it is set when reading a PSET, it should be ignored. This field defaults to 0 and should not be output if its value is 0. | 0 | 2 |
The currently defined elements per-input proprietary types are as follows:
Name | <keytype> | <keydata> | <keydata> Description | <valuedata> | <valuedata> Description | Versions Requiring Inclusion | Versions Requiring Exclusion | Versions Allowing Inclusion |
---|---|---|---|---|---|---|---|---|
Issuance Value | PSBT_ELEMENTS_IN_ISSUANCE_VALUE = 0x00 | None | No key data | <64-bit int> | The explicit little endian 64-bit integer for the value of this issuance. | 0 | 2 | |
Issuance Value Commitment | PSBT_ELEMENTS_IN_ISSUANCE_VALUE_COMMITMENT = 0x01 | None | No key data | <33 byte commitment> | The 33 byte Value Commitment. If provided, either PSBT_ELEMENTS_IN_ISSUANCE_VALUE must be removed, or PSBT_ELEMENTS_IN_ISSUANCE_BLIND_VALUE_PROOF must be provided too. | 0 | 2 | |
Issuance Value Rangeproof | PSBT_ELEMENTS_IN_ISSUANCE_VALUE_RANGEPROOF = 0x02 | None | No key data | <rangeproof> | The rangeproof | 0 | 2 | |
Issuance Inflation Keys Rangeproof | PSBT_ELEMENTS_IN_ISSUANCE_INFLATION_KEYS_RANGEPROOF = 0x03 | None | No key data | <rangeproof> | The rangeproof | 0 | 2 | |
Peg-in Transaction | PSBT_ELEMENTS_IN_PEG_IN_TX = 0x04 | None | No key data | <tx> | The Peg-in Transaction serialized without witnesses. | 0 | 2 | |
Peg-in Transaction Output Proof | PSBT_ELEMENTS_IN_PEG_IN_TXOUT_PROOF = 0x05 | None | No key data | <txoutproof> | The transaction output proof for the Peg-in Transaction. | 0 | 2 | |
Peg-in Genesis Hash | PSBT_ELEMENTS_IN_PEG_IN_GENESIS_HASH = 0x06 | None | No key data | <hash> | The 32 byte genesis hash for the Peg-in Transaction. | 0 | 2 | |
Peg-in Claim Script | PSBT_ELEMENTS_IN_PEG_IN_CLAIM_SCRIPT = 0x07 | None | No key data | <script> | The claim script for the Peg-in Transaction. | 0 | 2 | |
Peg-in Value | PSBT_ELEMENTS_IN_PEG_IN_VALUE = 0x08 | None | No key data | <64-bit int> | The little endian 64-bit value of the peg-in for the Peg-in Transaction. | 0 | 2 | |
Peg-in Witness | PSBT_ELEMENTS_IN_PEG_IN_WITNESS = 0x09 | None | No key data | <scriptWitness> | The Peg-in witness for the Peg-in Transaction. | 0 | 2 | |
Issuance Inflation Keys Amount | PSBT_ELEMENTS_IN_ISSUANCE_INFLATION_KEYS_AMOUNT = 0x0a | None | No key data | <64-bit int> | The value for the inflation keys output to set in this issuance. | 0 | 2 | |
Issuance Inflation Keys Amount Commitment | PSBT_ELEMENTS_IN_ISSUANCE_INFLATION_KEYS_COMMITMENT = 0x0b | None | No key data | <33 byte commitment> | The 33 byte commitment to the inflation keys output value in this issuance. If provided, either PSBT_ELEMENTS_IN_ISSUANCE_INFLATION_KEYS_AMOUNT must be removed or PSBT_ELEMENTS_IN_ISSUANCE_BLIND_INFLATION_KEYS_PROOF must be provided too. | 0 | 2 | |
Issuance Blinding Nonce | PSBT_ELEMENTS_IN_ISSUANCE_BLINDING_NONCE = 0x0c | None | No key data | <32 byte blinding nonce | The 32 byte asset blinding nonce. For new assets, must be 0. For reissuances, this is a revelation of the blinding factor for the input. | 0 | 2 | |
Issuance Asset Entropy | PSBT_ELEMENTS_IN_ISSUANCE_ASSET_ENTROPY = 0x0d | None | No key data | <32 byte entropy> | The 32 byte asset entropy. For new issuances, an arbitrary and optional 32 bytes of no consensus meaning combined used as additional entropy in the asset tag calculation. For reissuances, the original, final entropy used for the asset tag calculation. | 0 | 2 | |
UTXO Rangeproof | PSBT_ELEMENTS_IN_UTXO_RANGEPROOF = 0x0e | None | No key data | <rangeproof> | The rangeproof for the UTXO for this input. This rangeproof is found in the output witness data for the transaction and thus is not included as part of either of the UTXOs (as witness data is not included in either case). However the rangeproof is needed in order for stateless blinders to learn the blinding factors for the UTXOs that they are involved in. | 0 | 2 | |
Issuance Blind Value Proof | PSBT_ELEMENTS_IN_ISSUANCE_BLIND_VALUE_PROOF = 0x0f | None | No key data | <rangeproof> | An explicit value rangeproof that proves that the value commitment in PSBT_ELEMENTS_IN_ISSUANCE_VALUE_COMMITMENT matches the explicit value in PSBT_ELEMENTS_IN_ISSUANCE_VALUE. If provided, PSBT_ELEMENTS_IN_ISSUANCE_VALUE_COMMITMENT must be provided too. | 0 | 2 | |
Issuance Inflation Keys Blind Value Proof | PSBT_ELEMENTS_IN_ISSUANCE_BLIND_INFLATION_KEYS_PROOF = 0x10 | None | No key data | <rangeproof> | An explicit value rangeproof that proves that the value commitment in PSBT_ELEMENTS_IN_ISSUANCE_INFLATION_KEYS_COMMITMENT matches the explicit value in PSBT_ELEMENTS_IN_ISSUANCE_INFLATION_KEYS_AMOUNT. If provided, PSBT_ELEMENTS_IN_ISSUANCE_INFLATION_KEYS_COMMITMENT must be provided too. | 0 | 2 | |
Explicit Value | PSBT_ELEMENTS_IN_EXPLICIT_VALUE = 0x11 | None | No key data | <64-bit little endian int value> | The explicit value for the input being spent. If provided, PSBT_ELEMENTS_IN_VALUE_PROOF must be provided too. Must not be provided if the input's value in the UTXO is already explicit. | 0 | 2 | |
Explicit Value Proof | PSBT_ELEMENTS_IN_VALUE_PROOF = 0x12 | None | No key data | <rangeproof> | An explicit value rangeproof that proves that the value commitment in this input's UTXO matches the explicit value in PSBT_ELEMENTS_IN_EXPLICIT_VALUE. If provided, PSBT_ELEMENTS_IN_EXPLICIT_VALUE must be provided too. | 0 | 2 | |
Explicit Asset | PSBT_ELEMENTS_IN_EXPLICIT_ASSET = 0x13 | None | No key data | <32 byte asset tag> | The explicit asset for the input being spent. If provided, PSBT_ELEMENTS_IN_ASSET_PROOF must be provided too. Must not be provided if the input's asset in the UTXO is already explicit. | 0 | 2 | |
Explicit Asset Proof | PSBT_ELEMENTS_IN_ASSET_PROOF = 0x14 | None | No key data | <proof> | An asset surjection proof with this input's asset as the only asset in the input set in order to prove that the asset commitment in the UTXO matches the explicit asset in PSBT_ELEMENTS_IN_EXPLICIT_ASSET. If provided, PSBT_ELEMENTS_IN_EXPLICIT_ASSET must be provided too. | 0 | 2 | |
Blinded Issuance Flag | PSBT_ELEMENTS_IN_BLINDED_ISSUANCE = 0x15 | None | No key data | <1 byte boolean> | A boolean flag. 0x00 indicates the issuance should not be blinded, 0x01 indicates it should be. If not specified, assumed to be 0x01. Note that this does not indicate actual blinding status, but rather the expected blinding status prior to signing. | 0 | 2 |
The currently defined elements per-output proprietary types are as follows:
Name | <keytype> | <keydata> | <keydata> Description | <valuedata> | <valuedata> Description | Versions Requiring Inclusion | Versions Requiring Exclusion | Versions Allowing Inclusion |
---|---|---|---|---|---|---|---|---|
Value Commitment | PSBT_ELEMENTS_OUT_VALUE_COMMITMENT = 0x01 | None | No key data | <33 byte commitment> | The 33 byte Value Commitment for this output. If provided, either PSBT_OUT_AMOUNT must be removed or PSBT_ELEMENTS_OUT_BLIND_VALUE_PROOF must be provided too. | 0 | 2 | |
Asset Tag | PSBT_ELEMENTS_OUT_ASSET = 0x02 | None | No key data | <32 byte asset tag> | The explicit 32 byte asset tag for this output. | 0 | 2 | |
Asset Commitment | PSBT_ELEMENTS_OUT_ASSET_COMMITMENT = 0x03 | None | No key data | <33 byte commitment> | The 33 byte Asset Commitment for this output. If provided, either PSBT_ELEMENTS_OUT_ASSET must be removed or PSBT_ELEMENTS_OUT_BLIND_ASSET_PROOF must be provided too. | 0 | 2 | |
Value Rangeproof | PSBT_ELEMENTS_OUT_VALUE_RANGEPROOF = 0x04 | None | No key data | <rangeproof> | The rangeproof for the value of this output. | 0 | 2 | |
Asset Surjection Proof | PSBT_ELEMENTS_OUT_ASSET_SURJECTION_PROOF = 0x05 | None | No key data | <asset surjection proof> | The asset surjection proof for this output's asset. | 0 | 2 | |
Blinding Pubkey | PSBT_ELEMENTS_OUT_BLINDING_PUBKEY = 0x06 | None | No key data | <pubkey> | The 33 byte blinding pubkey to be used when blinding this output. | 0 | 2 | |
Ephemeral ECDH Pubkey | PSBT_ELEMENTS_OUT_ECDH_PUBKEY = 0x07 | None | No key data | <pubkey> | The 33 byte ephemeral pubkey used for ECDH in the blinding of this output. | 0 | 2 | |
Blinder Index | PSBT_ELEMENTS_OUT_BLINDER_INDEX = 0x08 | None | No key data | <32 bit uint> | The unsigned 32-bit little endian integer index of the input whose owner should blind this output. | 0 | 2 | |
Blind Value Proof | PSBT_ELEMENTS_OUT_BLIND_VALUE_PROOF = 0x09 | None | No key data | <rangeproof> | An explicit value rangeproof that proves that the value commitment in PSBT_ELEMENTS_OUT_VALUE_COMMITMENT matches the explicit value in PSBT_OUT_VALUE. If provided, PSBT_ELEMENTS_OUT_VALUE_COMMITMENT must be provided too. | 0 | ||
Blind Asset Proof | PSBT_ELEMENTS_OUT_BLIND_ASSET_PROOF = 0x0a | None | No key data | <proof> | An asset surjection proof with this output's asset as the only asset in the input set in order to prove that the asset commitment in PSBT_ELEMENTS_OUT_ASSET_COMMITMENT matches the explicit asset in PSBT_ELEMENTS_OUT_ASSET. If provided, PSBT_ELEMENTS_OUT_ASSET_COMMITMENT must be provided too. | 0 | 2 |
The PSET Magic Bytes are 0x70736574
For all Elements PSBT proprietary fields, PSETs cannot be combined when there are duplicate fields except in the case where those fields are identical.
Using the transaction format involves many different responsibilities. Multiple responsibilities can be handled by a single entity, but each responsibility is specialized in what it should be capable of doing. Some roles are shared with BIP 174 PSBT and have expanded responsibilities.
The Creator creates a new PSET in the same way that BIP 370 Creators operate.
For any output which is going to be blinded, the Creator must add the blinding pubkey in the PSBT_ELEMENTS_OUT_BLINDING_PUBKEY field.
If any input is blinded, at least one of the outputs must also be blinded, i.e. it must have a PSBT_ELEMENTS_OUT_BLINDING_PUBKEY. Such an output can be a 0 value OP_RETURN output.
The Creator decides whether and which inputs are issuance inputs. For those inputs, the fields PSBT_ELEMENTS_IN_ISSUANCE_BLINDING_NONCE and PSBT_ELEMENTS_IN_ISSUANCE_ASSET_ENTROPY are set. Such inputs must also have the issuance flag set in the outpoint of the input. The amount is added to the PSET input as PSBT_ELEMENTS_IN_ISSUANCE_VALUE and the amount for the issuance inflation keys output as PSBT_ELEMENTS_IN_ISSUANCE_INFLATION_KEYS_AMOUNT.
The Creator decides whether and which inputs are Peg-in inputs. Peg-in inputs must have the Peg-in flag set in their outpoints.
The Updater must only accept a PSET. The Updater adds information to the PSET that it has access to. If it has the UTXO for an input, it should add it to the PSET. Such UTXOs must contain all commitments if they are present. Furthermore, the Updater must add the UTXO rangeproof if it is present. The Updater should also add redeemScripts, witnessScripts, and BIP 32 derivation paths to the input and output data if it knows them.
For Peg-in inputs, if available, the updater should add the Peg-in transaction, the txout proof, the genesis hash, and the claim script.
A single entity is likely to be both a Creator and Updater.
PSET requires a role not present in PSBT, the Blinder. Blinders are similar to Signers and own inputs. The Blinder adds the blinding data to a transaction.
For issuance inputs that belong to the Blinder, the Blinder should generate a random blinding factor and create a value commitment for the issuance value. It will then add the value commitment in the PSBT_ELEMENTS_IN_ISSUANCE_VALUE_COMMITMENT. The blinder will also add the issuance value rangeproof and the issuance keys rangeproof in their respective fields. The blinder will also add the issuance blind value and issuance keys blind value proofs in their respective fields. For ease of identifying the blinder for an issuance, the input the issuance is attached to must belong to the blinder for the issuance.
For the Blinder's outputs that are to be blinded (i.e. they have a blinding pubkey), the Blinder will create value and asset commitments and put them in their respective fields. The Blinder will create the Value Rangeproof and Asset Surjection Proof and put them in their respective fields. The Blinder will create the blind value and blind asset proofs and put them in their respective fields. It will also add the ephemeral pubkey used for ECDH of the nonce for the rangeproof to the PSBT_ELEMENTS_OUT_ECDH_PUBKEY field.
The blinder will then compute a scalar offset that will be added as a PSBT_ELEMENTS_GLOBAL_SCALAR. For each input and output owned/blinded by this blinder, the following formula is computed: asset_blinding_factor * amount + amount_blinding_factor (mod n). The scalars for the inputs are summed, and then that sum is subtracted from the sum of the scalars for the outputs. The result is the scalar offset added as a PSBT_ELEMENTS_GLOBAL_SCALAR.
The Blinder can identify which outputs it should blind by checking the PSBT_ELEMENTS_OUT_BLINDER_INDEX. Using that index, the Blinder should inspect the input at that same index. If that input belongs to the Blinder, then the output is to be blinded by it.
For the Blinder who blinds the last output to be blinded, it will generate uniformly random asset and value blinding factors as normal. It will then compute a final scalar offset. Then it will subtract all of the scalar offsets from the value blinding factor for the last output and the result is the value blinding factor to be used for that last output. The creation of the commitments, proofs, and other fields proceeds as usual. Once all outputs are blinded, all PSBT_ELEMENTS_GLOBAL_SCALAR fields must be removed from the PSET.
A single entity is likely to be a Creator, Updater, and Blinder.
In addition to the BIP 370 PSBT Signer behavior, PSET specifies some additional constraints. Before signing, the Signer must check whether blinding is complete. If any output contains a blinding pubkey but no commitments or proofs, then it must not sign.
In addition to the BIP 370 PSBT Combiner behavior, PSET specifies some additional constraints. For all PSET output fields, the combiner must fail to combine if these fields are not identical. If the PSETs to be combined have incomplete blinded outputs but the combined result would only complete blinded outputs, then the Combiner must fail unless it is also a Blinder and can reblind its outputs.
In addition to the BIP 370 PSBT Input Finalizer behavior, PSET specifies some additional constraints. The Input Finalizer should also create the Peg-in witness.
In addition to the BIP 370 PSBT Transaction Extractor behavior, PSET specifies some additional constraints. The Transaction Extractor also places the commitments, proofs, and other elements data in their correct place in the transaction.
TBD
The reference implementation of the PSET format is available at https://github.com/achow101/elements/tree/pset.