ZIP: 248
Title: Extensible Transaction Format
Owners: Jack Grigg <jack@znewco.com>
Kris Nuttycombe <kris@nutty.land>
Daira-Emma Hopwood <daira@jacaranda.org>
Schell Scivally <efsubenovex@gmail.com>
Status: Draft
Category: Consensus / Wallet
Created: 2025-12-17
License: MIT
Discussions-To: <https://github.com/zcash/zips/pull/1163>
Pull-Request: <https://github.com/zcash/zips/pull/1156>
{Edit this to reflect the key words that are actually used.} The key words "MUST", "REQUIRED", "MUST NOT", "SHOULD", and "MAY" in this document are to be interpreted as described in BCP 14 [^BCP14] when, and only when, they appear in all capitals.
The character § is used when referring to sections of the Zcash Protocol Specification. 2
The terms "Mainnet" and "Testnet" are to be interpreted as described in § 3.12 ‘Mainnet and Testnet’. 4
The term "full validator" in this document is to be interpreted as defined in § 3.3 ‘The Block Chain’. 3
The terms below are to be interpreted as follows:
This ZIP proposes an encoding for V6 Zcash transactions that is intended to reduce the impact of future changes to the Zcash transaction format on the Zcash ecosystem. It defines a new typecode-length-value encoding for a sequence of protocol bundles, and a "value balance" map that describes the effect of each bundle on the transparent transaction value pool for each asset. The entries of this map serve the same purpose as the Sapling and Orchard value balance fields have done in the past.
In the past, Zcash network upgrades that changed the transaction format have resulted in substantial disruption for wallets and other third-party clients in the Zcash ecosystem. In order to continue functioning after a network upgrade, clients were required to upgrade their Zcash transaction parsers to read the new format, even if the context in which those parsers were being used didn't need or couldn't make use of newly added transaction data. An example of this is that transparent-only wallets were forced to update their parsers to understand the Sapling and Orchard parts of transactions, even if they would never read or act upon those parts. This has led on occasion to significant problems in the Zcash ecosystem, including situations where funds have been locked and rendered unspendable from transparent-only wallets.
For some kinds of changes to consensus features, it's imperative that every wallet be aware of and be adapted to those changes, and in those cases making a major (breaking) transaction version update as we've done in the past is appropriate. For many new features, however, it is possible for a wallet to continue functioning correctly without having to fully understand a transaction using that feature.
For example, if TZEs were to be added to the protocol, it wouild be possible for wallets to continue operating with transparent/Sapling/Orchard functionality, ignoring TZE parts. There is substantial precedent for this sort of behavior; transparent-only hardware wallets are currently still important in the Zcash ecosystem, and many wallets didn't begin interacting with Orchard transaction parts until quite a while after Orchard activation.
After this change to transaction encoding, wallets and other third parties will not be required to update their transaction parsers in advance of a network upgrade for the introduction of many (and perhaps most) types of new protocol features. This will enable the Zcash ecosystem to make smaller and more incremental network upgrades without breaking existing wallets.
This change alters the encoding of transactions, but does not alter the information content of the transaction. As such, the only implication of this change is that the use of this transaction format acts as a 1-bit distinguisher that reveals that the wallet that generated the transaction has been updated to be aware of the new format. This information leakage is unavoidable for any transaction format change.
In the future, this change may reduce the amount of information leakage, since transactions created using the proposed TLV format will include bundles only for those protocols for which the transaction modifies chain state. For example, if this transaction format change is deployed in NU7 and NU8 defines a bundle type for TZE components, it will not be possible for a chain observer to distinguish whether or not the wallet that produced an Orchard-only transaction is one that has been updated to understand the TZE component. Under prior practices for changing the transaction format, this would have been distinguishable.
In summary, this proposal provides a net improvement in user privacy in addition to its other benefits.
This ZIP refines and codifies the concept of "protocol bundles" that emerged from the implementation of the ZIP 225 9 transaction format. It makes bundles first-class objects and defines a registry of bundle type and variant identifiers. Within the period that a given transaction format version is used on the Zcash network, the semantics of the bundle associated with a given (bundleType, bundleVariant) pair are fixed.
A protocol bundle is a self-contained component of a transaction that implements a specific piece of protocol functionality. Each bundle is identified by a (bundleType, bundleVariant) pair, where:
A transaction MUST NOT contain more than one variant of any given bundle type. This constraint is enforced by the map structure of the transaction encoding: the maps mEffectBundles, and mAuthBundles are each keyed by bundleType alone; mValuePoolDeltas is keyed by bundle type and asset ID.
Each bundle type/variant pair defines:
The transparent transaction value pool is an ephemeral concept that exists only within the scope of processing a single transaction. It serves as a balancing mechanism through which value flows between bundles.
Each bundle may contribute a value pool delta — a signed value indicating how much the bundle adds to or removes from the transparent transaction value pool for a given asset. A positive delta means the bundle is adding value to the pool (e.g., a shielded spend releasing value), while a negative delta means the bundle is consuming value from the pool (e.g., a shielded output absorbing value, or a transaction fee).
For a valid non-coinbase transaction, the sum of all value pool deltas for each asset MUST equal zero. This ensures that value is neither created nor destroyed — it is only transferred between bundles within the transaction.
For a coinbase transaction, the sum of value pool deltas for ZEC equals the negative of the block subsidy, reflecting that the block subsidy implicitly adds value to the transparent transaction value pool.
When a ZIP introduces a new bundle type or a new variant of an existing bundle type, it MUST:
(bundleType, bundleVariant) pair in the registry defined below. The bundleType and bundleVariant are each non-negative integers. If the ZIP introduces a new bundle type, it requests a new bundleType value with bundleVariant 0. If it introduces a new variant of an existing bundle type, it requests a new bundleVariant value for the existing bundleType.mValuePoolDeltas (the value pool delta map).mEffectBundles (the effecting data map).mAuthBundles (the authorizing data map).A bundle type MUST NOT permit entries in mAuthBundles unless it also permits entries in mEffectBundles. That is, authorizing data cannot exist without corresponding effecting data for a given bundle type.
Once a (bundleType, bundleVariant) pair is assigned for a given transaction version, its semantics are fixed for the lifetime of that transaction version. A subsequent network upgrade may define a new transaction version that reassigns identifiers or changes bundle semantics, but within a single transaction version, bundle type and variant identifiers have stable, unchanging meanings.
A new variant of an existing bundle type MUST affect the same value pool(s) as the original bundle type. This ensures that a client that does not recognize a particular bundleVariant can still determine which pool(s) are affected by the bundle and inform its user accordingly.
The following (bundleType, bundleVariant) pairs are registered for the V6 transaction format. All currently-defined values are encoded as single-byte compactSize values where they appear in the transaction format.
The mValuePoolDeltas column indicates whether or not an entry for this bundle type MAY appear in mValuePoolDeltas. For rows where an ❌ is present, the value pool delta for every pool is guaranteed to be zero, and so entries in mValuePoolDeltas MUST NOT be present.
The mEffectBundles column indicates whether or not an entry for this bundle type MAY appear in mEffectBundles. For rows where an ❌ is present, the bundle has no effecting data, and so an entry in mEffectBundles MUST NOT be present.
The mAuthBundles column indicates whether or not an entry for this bundle type MAY appear in mAuthBundles. For rows where an ❌ is present, the bundle has no authorizing data, and so an entry in mAuthBundles MUST NOT be present.
| BundleType | BundleVariant | mValuePoolDeltas |
mEffectBundles |
mAuthBundles |
Defining ZIP | Bundle kind |
|---|---|---|---|---|---|---|
| 0 | 0 | ✅ | ✅ | ✅ | This ZIP | Transparent |
| 1 | Reserved | |||||
| 2 | 0 | ✅ | ✅ | ✅ | This ZIP | Sapling |
| 2 | ✅ | ✅ | ✅ | ZIP 231 | Sapling-post-ZIP 231 | |
| 3 | 0 | ✅ | ✅ | ✅ | This ZIP | Orchard |
| 3 | ✅ | ✅ | ✅ | ZIP 231 | Orchard-post-ZIP 231 | |
| 3 | ✅ | ✅ | ✅ | ZIP 226 | OrchardZSA | |
| 4 | 0 | ✅ | ❌ | ❌ | ZIP 2002 | Transaction fee |
| 5 | 0 | ✅ | ❌ | ❌ | ZIP 233 | ZIP 233 NSM field |
| 6 | 0 | ❌ | ✅ | ✅ | ZIP 270 | Key rotation |
| 7 | 0 | ✅ | ✅ | ✅ | TBD | Lockbox disbursement |
| 0 | ❌ | ✅ | ❌ | ZIP 231 | Memos | |
| 0 | ✅ | ✅ | ✅ | ZIP 227 | ZSA Issuance |
Additional bundle types or variants MAY be added to this registry via modifications to this ZIP specified in other ZIPs. Such modifications MUST specify all of the information required by the Bundle Type Registration section above.
The following entries are provided to illustrate how potential future upgrades might affect the bundle registry:
| BundleType | BundleVariant | mValuePoolDeltas |
mEffectBundles |
mAuthBundles |
Bundle kind |
|---|---|---|---|---|---|
| 0 | ✅ | ✅ | ✅ | TZEs | |
| 0 | ✅ | ✅ | ✅ | Pool that only has a long-term storage protocol (PQ, very simple thus insulated from counterfeiting fears, can be used for payments but higher latency for that purpose) | |
| 0 | ✅ | ✅ | ✅ | Tachyon | |
| 0 | ✅ | ✅ | ❌ | Staking | |
| 0 | ✅ | ✅ | ✅ | Unstaking (if it can't be combined with the Staking bundle) | |
| 0 | ✅ | ✅ | ✅ | Post-quantum fast payment protocol |
| Bytes | Name | Data Type | Description |
|---|---|---|---|
| Common Transaction Fields | |||
| 4 | header |
uint32 |
Contains:
|
| 4 | nVersionGroupId |
uint32 |
Version group ID (nonzero). |
| 4 | nConsensusBranchId |
uint32 |
Consensus branch ID (nonzero). |
| 4 | lock_time |
uint32 |
Unix-epoch UTC time or block height, encoded as in Bitcoin. |
| 4 | nExpiryHeight |
uint32 |
A block height in the range {1 .. 499999999} after which the transaction will expire, or 0 to disable expiry. 7 |
| Transaction transparent value pool balance map | |||
| varies | nValuePoolDeltas |
compactSize |
Number of entries in the mValuePoolDeltas map. |
| varies | mValuePoolDeltas |
ValuePoolDelta[nValuePoolDeltas] |
A map describing the change to the transparent value pool produced by each bundle. Only bundles that produces changes to the transparent value balance will have corresponding entries in this map. For bundles that have no data except for a value, such as the ZIP 233 amount, no additional bundle data will be present in the Bundles section. |
| Bundles | |||
| varies | nEffectBundles |
compactSize |
Number of bundles in the transaction that have per-bundle data. |
| varies | mEffectBundles |
BundleData[nEffectBundles] |
A map from bundle identifier to the effecting data of a bundle. |
| varies | nAuthBundles |
compactSize |
Number of bundles in the transaction that have per-bundle data. |
| varies | mAuthBundles |
BundleData[nAuthBundles] |
A map from bundle identifier to the authorizing data of a bundle. |
mValuePoolDeltas is interpreted as a map keyed by the tuple
\((\mathsf{BundleType}, \mathsf{AssetUuid})\!\)
; this logical map MUST NOT contain more than a single entry for a given key. The ValuePoolDelta records that represent the entries this map MUST be encoded in increasing order of (bundleType, assetClass, assetUuid) Lookups in this map are denoted with the syntax
\(\mathsf{mValuePoolDeltas}[(\mathsf{BundleType}, \mathsf{AssetUuid})].\)
mEffectBundles and mAuthBundles are interpreted as maps keyed by bundleType. The records that represent each map entry map MUST be encoded in increasing order of bundleType. Each map MUST NOT contain more than a single entry for a given bundleType. For each bundleType that exists in mAuthBundles, a corresponding entry with the same bundleVariant must exist in mEffectBundles.
| Bytes | Name | Data Type | Description |
|---|---|---|---|
| varies | bundleType |
compactSize |
An encoding of the bundle type identifier. |
| varies | bundleVariant |
compactSize |
An encoding of the bundle variant identifier. |
| 1 | assetClass |
uint8 |
An asset class identifier. 0x00 for the ZEC asset, 0x01 for other assets. |
| one of {0, 64} | assetUuid |
byte[0] or byte[64] |
If assetClass == 0, the zero-length byte array, otherwise a byte array containing a universally unique 64-byte identifier for the asset. |
| 8 | value |
nonzero int64 |
The net change to the transparent transaction value pool for the given asset, produced by the bundle with this bundle type identifier. This value MUST be nonzero; if a ValuePoolDelta record would have zero value, it MUST be elided from the encoding of mValuePoolDeltas instead. |
mValuePoolDeltas is interpreted as a map keyed by the tuple
\((\mathsf{BundleType}, \mathsf{AssetUuid}).\)
The map MUST NOT contain more than a single entry for a given key. Lookups in this map are denoted with the syntax
\(\mathsf{mValuePoolDeltas}[(\mathsf{BundleType}, \mathsf{AssetUuid})].\)
All entries in mValuePoolDeltas having the same bundleType MUST have the same bundleVariant.
Let \(\mathsf{Zec}\) be a distinguished value representing the ZEC asset. It is used as the asset identifier when \(\mathsf{assetClass} = 0\!\) .
Let
\(\mathsf{AssetUuid}(\mathsf{d})\)
be the asset indicated by the mValuePoolDeltas entry
\(\mathsf{d}.\)
| Bytes | Name | Data Type | Description |
|---|---|---|---|
| varies | bundleType |
compactSize |
An encoding of the bundle type identifier. |
| varies | bundleVariant |
compactSize |
An encoding of the bundle variant identifier. |
| varies | nBundleDataLen |
compactSize |
The length of the vBundleData byte array. |
| varies | vBundleData |
byte[nBundleDataLen] |
The effecting or authorizing data for the bundle, dependent upon whether the BundleData occurs in mEffectBundles or mAuthBundles. |
The effecting data for the transparent bundle describes the transparent inputs being spent and the transparent outputs being created.
| Bytes | Name | Data Type | Description |
|---|---|---|---|
varies |
tx_in_count |
compactSize |
Number of transparent inputs. |
varies |
tx_in_effecting |
TransparentInputEffecting[tx_in_count] |
Effecting data for each transparent input. |
varies |
tx_out_count |
compactSize |
Number of transparent outputs. |
varies |
tx_out |
TransparentOutput[tx_out_count] |
Transparent outputs. |
| Bytes | Name | Data Type | Description |
|---|---|---|---|
32 |
prevout_hash |
byte[32] |
The transaction ID of the output being spent. |
4 |
prevout_index |
uint32 |
The index of the output being spent within that transaction. |
4 |
nSequence |
uint32 |
Sequence number, encoded as in Bitcoin. |
| Bytes | Name | Data Type | Description |
|---|---|---|---|
8 |
value |
int64 |
The value of the output in zatoshi. |
varies |
scriptPubKeyLen |
compactSize |
Length of the scriptPubKey. |
scriptPubKeyLen |
scriptPubKey |
byte[scriptPubKeyLen] |
The script that must be satisfied to spend this output. |
The effecting data for the Sapling bundle describes the Sapling spends and outputs. Unlike the V5 transaction format defined in ZIP 225 9, the value balance is not included here; it appears in mValuePoolDeltas instead.
| Bytes | Name | Data Type | Description |
|---|---|---|---|
varies |
nSpendsSapling |
compactSize |
Number of Sapling Spend descriptions. |
96 * nSpendsSapling |
vSpendsSapling |
SaplingSpendEffecting[nSpendsSapling] |
Effecting data for each Sapling Spend. |
varies |
nOutputsSapling |
compactSize |
Number of Sapling Output descriptions. |
756 * nOutputsSapling |
vOutputsSapling |
SaplingOutput[nOutputsSapling] |
Sapling Output descriptions. |
32 |
anchorSapling |
byte[32] |
A root of the Sapling note commitment tree at some block height in the past. |
anchorSapling is present if and only if
\(\mathtt{nSpendsSapling} > 0\!\)
.| Bytes | Name | Data Type | Description |
|---|---|---|---|
32 |
cv |
byte[32] |
A value commitment to the net value of the input note. |
32 |
nullifier |
byte[32] |
The nullifier of the input note. |
32 |
rk |
byte[32] |
The randomized validating key for this Spend. |
This is identical to OutputDescriptionV5 as defined in ZIP 225 9.
| Bytes | Name | Data Type | Description |
|---|---|---|---|
32 |
cv |
byte[32] |
A value commitment to the net value of the output note. |
32 |
cmu |
byte[32] |
The \(u\!\) -coordinate of the note commitment for the output note. |
32 |
ephemeralKey |
byte[32] |
An encoding of an ephemeral Jubjub public key. |
580 |
encCiphertext |
byte[580] |
The encrypted contents of the note plaintext. |
80 |
outCiphertext |
byte[80] |
The encrypted contents of the byte string created by concatenation of the transmission key with the ephemeral secret key. |
The effecting data for the Orchard bundle describes the Orchard actions. Unlike the V5 transaction format defined in ZIP 225 9, the value balance is not included here; it appears in mValuePoolDeltas instead.
| Bytes | Name | Data Type | Description |
|---|---|---|---|
varies |
nActionsOrchard |
compactSize |
The number of Orchard Action descriptions. |
820 * nActionsOrchard |
vActionsOrchard |
OrchardActionEffecting[nActionsOrchard] |
Effecting data for each Orchard Action. |
1 |
flagsOrchard |
byte |
An 8-bit value representing a set of flags. Ordered from LSB to MSB:
|
32 |
anchorOrchard |
byte[32] |
A root of the Orchard note commitment tree at some block height in the past. |
flagsOrchard and anchorOrchard are present if and only if
\(\mathtt{nActionsOrchard} > 0\!\)
.enableSpendsOrchard bit MUST be set to
\(0\!\)
.| Bytes | Name | Data Type | Description |
|---|---|---|---|
32 |
cv |
byte[32] |
A value commitment to the net value of the input note minus the output note. |
32 |
nullifier |
byte[32] |
The nullifier of the input note. |
32 |
rk |
byte[32] |
The randomized validating key for this Action. |
32 |
cmx |
byte[32] |
The \(x\!\) -coordinate of the note commitment for the output note. |
32 |
ephemeralKey |
byte[32] |
An encoding of an ephemeral Pallas public key. |
580 |
encCiphertext |
byte[580] |
The encrypted contents of the note plaintext. |
80 |
outCiphertext |
byte[80] |
The encrypted contents of the byte string created by concatenation of the transmission key with the ephemeral secret key. |
This ZIP requires the following modifications to the consensus rules in the Zcash Protocol Specification.
Let FeeBundleId be the identifier of the fee bundle. In V6 transactions,
\(\mathsf{FeeBundleId} = 4\)
as defined in the table above.
The following transaction validity rules are added:
assetClass value for any entry in mValuePoolDeltas having bundleType = FeeBundleId is 0 (fee amounts are denominated in ZEC and no other asset.)bundleType that appears in any of mValuePoolDeltas, mEffectBundles, or mAuthBundles, all entries for that bundleType across all three maps MUST have the same bundleVariant. (This, combined with the constraint that each map is keyed by bundleType, ensures that a transaction uses at most one variant of any given bundle type.)where \(\mathsf{BlockSubsidy}\) is defined in § 7.8 'Block Subsidy and Founders' Reward'. 5
All digests are personalized BLAKE2b-256 hashes. In cases where no elements are available for hashing (for example, if there are no transparent transaction inputs), a personalized hash of the empty byte array will be used. The personalization string therefore provides domain separation for the hashes of even empty data fields.
The notation BLAKE2b-256(personalization_string, []) is used to refer to hashes constructed in this manner.
A new transaction digest algorithm is defined that constructs the identifier for a V6 transaction from a tree of hashes. The overall structure of the hash is as follows:
txid_digest
├── header_digest
├── value_pool_deltas_digest
└── effects_bundles_digest
├─ (bundle_type_id || bundle_variant || transparent_effects_digest)
├─ (bundle_type_id || bundle_variant || sapling_effects_digest)
│ ├── sapling_spends_digest
│ │ ├── sapling_spends_compact_digest
│ │ └── sapling_spends_noncompact_digest
│ └── sapling_outputs_digest
│ ├── sapling_outputs_compact_digest
│ ├── sapling_outputs_memos_digest
│ └── sapling_outputs_noncompact_digest
├─ (bundle_type_id || bundle_variant || orchard_effects_digest)
│ ├── orchard_actions_compact_digest
│ ├── orchard_actions_memos_digest
│ └── orchard_actions_noncompact_digest
└─ (bundle_type_id || bundle_variant || unknown_bundle_effects_digest) ...
Each node written as snake_case in this tree is a BLAKE2b-256 hash of its children, initialized with a personalization string specific to that branch of the tree. Nodes that are not themselves digests are written in camelCase. In the specification below, nodes of the tree are presented in depth-first order.
A BLAKE2b-256 hash of the following values:
T.1: header_digest (32-byte hash output) T.2: value_pool_deltas_digest (32-byte hash output) T.3: effects_bundles_digest (32-byte hash output)
The personalization field of this hash is set to:
"ZcashTxHash_" || CONSENSUS_BRANCH_ID
ZcashTxHash_ has 1 underscore character.
As in ZIP 143 6, CONSENSUS_BRANCH_ID is the 4-byte little-endian encoding of the consensus branch ID for the epoch of the block containing the transaction.
A BLAKE2b-256 hash of the following values:
T.1a: version (4-byte little-endian version identifier including overwintered flag) T.1b: nVersionGroupId (4-byte little-endian version group identifier) T.1c: nConsensusBranchId (4-byte little-endian consensus branch id) T.1d: lock_time (4-byte little-endian nLockTime value) T.1e: nExpiryHeight (4-byte little-endian block height)
The personalization field of this hash is set to:
"ZTxIdHeadersHash"
A BLAKE2b-256 hash of the concatenated encodings of all entries in mValuePoolDeltas, in transaction order. For each entry, the following values are concatenated:
T.2a: bundleType (compactSize encoding) T.2b: bundleVariant (compactSize encoding) T.2c: assetClass (1 byte) T.2d: assetUuid (0 or 64 bytes, depending on assetClass) T.2e: value (8-byte signed little-endian)
The personalization field of this hash is set to:
"ZTxIdVPDeltaHash"
In the case that the transaction has no value pool delta entries (which would only occur for transactions that have no effect on any value pool), value_pool_deltas_digest is:
BLAKE2b-256("ZTxIdVPDeltaHash", [])
A BLAKE2b-256 hash of the concatenated tagged bundle effect digests for all bundles present in mEffectBundles, in transaction order. For each bundle, the following values are concatenated:
T.3a: bundleType (compactSize encoding) T.3b: bundleVariant (compactSize encoding) T.3c: bundle_effects_digest (32-byte hash output)
where bundle_effects_digest is the root hash of the bundle's effecting data tree, as defined below for each known bundle type.
The personalization field of this hash is set to:
"ZTxIdEffBndHash"
In the case that the transaction has no effect bundles, effects_bundles_digest is:
BLAKE2b-256("ZTxIdEffBndHash", [])
In the case that transparent inputs or outputs are present, the transparent effects digest is a BLAKE2b-256 hash of the following values:
T.3.0a: prevouts_digest (32-byte hash) T.3.0b: sequence_digest (32-byte hash) T.3.0c: outputs_digest (32-byte hash)
The personalization field of this hash is set to:
"ZTxIdTranspaHash"
In the case that the transaction has no transparent components, transparent_effects_digest is:
BLAKE2b-256("ZTxIdTranspaHash", [])
A BLAKE2b-256 hash of the field encoding of all (prevout_hash, prevout_index) pairs from the transparent effecting data.
The personalization field of this hash is set to:
"ZTxIdPrevoutHash"
In the case that the transaction has transparent outputs but no transparent inputs, prevouts_digest is:
BLAKE2b-256("ZTxIdPrevoutHash", [])
A BLAKE2b-256 hash of the 32-bit little-endian representation of all nSequence field values from the transparent effecting data.
The personalization field of this hash is set to:
"ZTxIdSequencHash"
In the case that the transaction has transparent outputs but no transparent inputs, sequence_digest is:
BLAKE2b-256("ZTxIdSequencHash", [])
A BLAKE2b-256 hash of the concatenated field encodings of all transparent outputs. The field encoding of each output consists of the encoded output value (8-byte little endian) followed by the scriptPubKey byte array (with leading compactSize length).
The personalization field of this hash is set to:
"ZTxIdOutputsHash"
In the case that the transaction has transparent inputs but no transparent outputs, outputs_digest is:
BLAKE2b-256("ZTxIdOutputsHash", [])
In the case that Sapling spends or outputs are present, the Sapling effects digest is a BLAKE2b-256 hash of the following values:
T.3.2a: sapling_spends_digest (32-byte hash) T.3.2b: sapling_outputs_digest (32-byte hash) T.3.2c: anchorSapling (32 bytes)
The personalization field of this hash is set to:
"ZTxIdSaplingHash"
Note that unlike ZIP 244, the value balance is not included here; it is committed via value_pool_deltas_digest instead.
In the case that the transaction has no Sapling spends or outputs, sapling_effects_digest is:
BLAKE2b-256("ZTxIdSaplingHash", [])
In the case that Sapling spends are present, this digest is a BLAKE2b-256 hash of the following values:
T.3.2a.i: sapling_spends_compact_digest (32-byte hash) T.3.2a.ii: sapling_spends_noncompact_digest (32-byte hash)
The personalization field of this hash is set to:
"ZTxIdSSpendsHash"
In the case that the transaction has Sapling outputs but no Sapling spends, sapling_spends_digest is:
BLAKE2b-256("ZTxIdSSpendsHash", [])
A BLAKE2b-256 hash of the field encoding of all nullifier field values of Sapling spends belonging to the transaction.
The personalization field of this hash is set to:
"ZTxIdSSpendCHash"
A BLAKE2b-256 hash of the non-nullifier information for all Sapling spends belonging to the transaction. For each spend, the following elements are included in the hash:
T.3.2a.ii.1: cv (32 bytes) T.3.2a.ii.2: anchor (32 bytes) T.3.2a.ii.3: rk (32 bytes)
The anchor is hashed for each spend (even though it is shared in the encoding).
The personalization field of this hash is set to:
"ZTxIdSSpendNHash"
In the case that Sapling outputs are present, this digest is a BLAKE2b-256 hash of the following values:
T.3.2b.i: sapling_outputs_compact_digest (32-byte hash) T.3.2b.ii: sapling_outputs_memos_digest (32-byte hash) T.3.2b.iii: sapling_outputs_noncompact_digest (32-byte hash)
The personalization field of this hash is set to:
"ZTxIdSOutputHash"
In the case that the transaction has Sapling spends but no Sapling outputs, sapling_outputs_digest is:
BLAKE2b-256("ZTxIdSOutputHash", [])
A BLAKE2b-256 hash of the subset of Sapling output information included in the ZIP 307 12 CompactBlock format for all Sapling outputs belonging to the transaction. For each output, the following elements are included:
T.3.2b.i.1: cmu (32 bytes) T.3.2b.i.2: ephemeralKey (32 bytes) T.3.2b.i.3: encCiphertext[..52] (first 52 bytes)
The personalization field of this hash is set to:
"ZTxIdSOutC__Hash" (2 underscore characters)
A BLAKE2b-256 hash of the memo field data for all Sapling outputs belonging to the transaction. For each output:
T.3.2b.ii.1: encCiphertext[52..564] (512 bytes, encrypted memo)
The personalization field of this hash is set to:
"ZTxIdSOutM__Hash" (2 underscore characters)
A BLAKE2b-256 hash of the remaining Sapling output information not included in the CompactBlock format. For each output:
T.3.2b.iii.1: cv (32 bytes) T.3.2b.iii.2: encCiphertext[564..] (post-memo AEAD tag, 16 bytes) T.3.2b.iii.3: outCiphertext (80 bytes)
The personalization field of this hash is set to:
"ZTxIdSOutN__Hash" (2 underscore characters)
In the case that Orchard actions are present, the Orchard effects digest is a BLAKE2b-256 hash of the following values:
T.3.3a: orchard_actions_compact_digest (32-byte hash) T.3.3b: orchard_actions_memos_digest (32-byte hash) T.3.3c: orchard_actions_noncompact_digest (32-byte hash) T.3.3d: flagsOrchard (1 byte) T.3.3e: anchorOrchard (32 bytes)
The personalization field of this hash is set to:
"ZTxIdOrchardHash"
Note that unlike ZIP 244, the value balance is not included here; it is committed via value_pool_deltas_digest instead.
In the case that the transaction has no Orchard actions, orchard_effects_digest is:
BLAKE2b-256("ZTxIdOrchardHash", [])
A BLAKE2b-256 hash of the subset of Orchard action information intended for inclusion in the CompactBlock format. For each action:
T.3.3a.i: nullifier (32 bytes) T.3.3a.ii: cmx (32 bytes) T.3.3a.iii: ephemeralKey (32 bytes) T.3.3a.iv: encCiphertext[..52] (first 52 bytes)
The personalization field of this hash is set to:
"ZTxIdOrcActCHash"
A BLAKE2b-256 hash of the memo field data for all Orchard actions. For each action:
T.3.3b.i: encCiphertext[52..564] (512 bytes, encrypted memo)
The personalization field of this hash is set to:
"ZTxIdOrcActMHash"
A BLAKE2b-256 hash of the remaining Orchard action information not intended for inclusion in the CompactBlock format. For each action:
T.3.3c.i: cv (32 bytes) T.3.3c.ii: rk (32 bytes) T.3.3c.iii: encCiphertext[564..] (post-memo AEAD tag, 16 bytes) T.3.3c.iv: outCiphertext (80 bytes)
The personalization field of this hash is set to:
"ZTxIdOrcActNHash"
A new per-input transaction digest algorithm is defined that constructs a hash that may be signed by a transaction creator to commit to the effects of the transaction. This follows closely the algorithm from ZIP 244 11.
For transactions that have no transparent inputs, the signature digest is identical to the transaction identifier digest.
For transactions with transparent inputs, the signature digest replaces effects_bundles_digest with a signature_bundles_digest that incorporates hash_type-dependent transparent signing data:
signature_digest ├── header_digest ├── value_pool_deltas_digest └── signature_bundles_digest
A BLAKE2b-256 hash of the following values:
S.1: header_digest (32-byte hash output) S.2: value_pool_deltas_digest (32-byte hash output) S.3: signature_bundles_digest (32-byte hash output)
The personalization field of this hash is set to:
"ZcashTxHash_" || CONSENSUS_BRANCH_ID
This value has the same personalization as the transaction identifier digest, so that what is being signed in the case that there are no transparent inputs is exactly the transaction id.
If the transaction has no transparent inputs, signature_bundles_digest is identical to effects_bundles_digest.
Otherwise, signature_bundles_digest is constructed the same as effects_bundles_digest, except that transparent_effects_digest is replaced with transparent_sig_digest.
This digest is a BLAKE2b-256 hash of the following values:
S.3.0a: hash_type (1 byte) S.3.0b: prevouts_sig_digest (32-byte hash) S.3.0c: amounts_sig_digest (32-byte hash) S.3.0d: scriptpubkeys_sig_digest (32-byte hash) S.3.0e: sequence_sig_digest (32-byte hash) S.3.0f: outputs_sig_digest (32-byte hash) S.3.0g: txin_sig_digest (32-byte hash)
The personalization field of this hash is set to:
"ZTxIdTranspaHash"
An 8-bit unsigned value. The SIGHASH encodings from the legacy script system are used: one of SIGHASH_ALL (0x01), SIGHASH_NONE (0x02), or SIGHASH_SINGLE (0x03), optionally combined with SIGHASH_ANYONECANPAY (0x80).
The following restrictions apply:
hash_type (not 0x01, 0x02, 0x03, 0x81, 0x82, or 0x83) causes validation failure.SIGHASH_SINGLE without a corresponding output at the same index causes validation failure.For signatures over Sapling Spends or Orchard Actions, hash_type is set to SIGHASH_ALL (0x01).
If the SIGHASH_ANYONECANPAY flag is not set, identical to prevouts_digest (T.3.0a).
Otherwise:
BLAKE2b-256("ZTxIdPrevoutHash", [])
If the SIGHASH_ANYONECANPAY flag is not set, a BLAKE2b-256 hash of the concatenation of the 8-byte signed little-endian representations of all value fields for the coins spent by the transparent inputs to the transaction.
The personalization field of this hash is set to:
"ZTxTrAmountsHash"
If the SIGHASH_ANYONECANPAY flag is set:
BLAKE2b-256("ZTxTrAmountsHash", [])
If the SIGHASH_ANYONECANPAY flag is not set, a BLAKE2b-256 hash of the concatenation of the field encodings (each including a leading compactSize) of all scriptPubKey fields for the coins spent by the transparent inputs.
The personalization field of this hash is set to:
"ZTxTrScriptsHash"
If the SIGHASH_ANYONECANPAY flag is set:
BLAKE2b-256("ZTxTrScriptsHash", [])
Identical to sequence_digest (T.3.0b) regardless of hash_type.
If the sighash type is neither SIGHASH_SINGLE nor SIGHASH_NONE, identical to outputs_digest (T.3.0c).
If the sighash type is SIGHASH_SINGLE and a transparent output exists at the same index as the input being signed, a hash of that output's encoding.
Otherwise:
BLAKE2b-256("ZTxIdOutputsHash", [])
For signatures over a transparent input, a BLAKE2b-256 hash of:
S.3.0g.i: prevout (36 bytes: 32-byte hash + 4-byte index) S.3.0g.ii: value (8-byte signed little-endian) S.3.0g.iii: scriptPubKey (with compactSize length prefix) S.3.0g.iv: nSequence (4-byte unsigned little-endian)
The personalization field of this hash is set to:
"Zcash___TxInHash" (3 underscores)
For signatures over a Sapling Spend or Orchard Action:
BLAKE2b-256("Zcash___TxInHash", [])
All Zcash wallets SHOULD, without undue delay, switch to sending only v6 transactions once they are allowed on the network. This applies to all transactions regardless of whether they use new v6 features.
Zcash wallets MUST support parsing v6 transactions by the time they are allowed on the network.
Because the v6 transaction format uses a type-length-value encoding for bundles, a wallet is not required to understand the internal encoding of every bundle in order to parse a transaction. However, a wallet that encounters a bundle with an unrecognized bundleType SHOULD alert the user that the transaction contains components it does not understand. A wallet that encounters a bundle with a recognized bundleType but unrecognized bundleVariant SHOULD alert the user that the transaction affects a pool the wallet is aware of, but in a way the wallet does not fully understand.
For bundle types not understood by a wallet, the wallet can compute the transaction identifier so long as it has been provided with the 32-byte bundle_effects_digest value for each bundle that it does not understand. This enables partial verification of transactions containing unknown bundle types.
A wallet MUST NOT construct or sign a transaction containing a bundle type or variant that it does not fully understand.
Effecting data bundles and authorizing data bundles are stored separately in the transaction format so that the authorizing data may be pruned by straightforward truncation of the encoded representation of the transaction.
This section should be removed as soon as all the considerations described here are accounted for in ZIP.
Wallets or consensus-dependent applications that send transactions, might do something wrong that compromises user funds or privacy if they do not take into account consensus changes in an upgrade; therefore, only a subset of consensus changes can be safely adapted to using this mechanism.
In particular, consensus rules may change in such a way that a wallet doing what it has done in the past causes risk of loss of funds.
An example of this was ZIP 212 8. In that case the existing mechanisms failed to prevent loss of funds because in practice, wallets updated the consensus branch ID without updating note encryption. We made the mistake of requiring wallets to change their behaviour for an existing transaction version. Except for certain cases involving severe security flaws, we should avoid doing that again.
If a wallet needs to actively do something differently (for example, advertizing addresses in a new format or creating an output with a TZE precondition) in order to be affected by a new feature, then it is reasonably safe for it to ignore the feature as long as it can still parse transactions and, and create and sign transactions that don't make use of those features.
It is okay that such a wallet might not be able to see funds that depend on new features, as long as they do not create such funds themselves.
Loss of funds is unacceptable. Temporary inaccessibility of funds in certain circumstances can be okay -- provided that this potential inaccessiblity and the circumstances where it can occur is documented and an explicit design decision.
Modify how we approach transaction format evolution, such that (after one more change to transaction encoding) it is possible for a wallet that has not adopted a parser for a given transaction format to continue to function after an additive change to the transaction format. Another way to state this is that we should make it possible to make "semver-compatible" transaction format changes.
Treat bundles as individually versioned.
Sketch of the format:
Rename assetDigest to assetUuid in ZIP 227
| 1 | Information on BCP 14 — "RFC 2119: Key words for use in RFCs to Indicate Requirement Levels" and "RFC 8174: Ambiguity of Uppercase vs Lowercase in RFC 2119 Key Words |
|---|
| 2 | Zcash Protocol Specification, Version 2025.6.3 [NU6.1] or later |
|---|
| 3 | Zcash Protocol Specification, Version 2025.6.3 [NU6.1]. Section 3.3: The Block Chain |
|---|
| 4 | Zcash Protocol Specification, Version 2025.6.3 [NU6.1]. Section 3.12: Mainnet and Testnet |
|---|
| 5 | Zcash Protocol Specification, Version 2025.6.3 [NU6.1]. Section 7.8: Block Subsidy and Founders' Reward |
|---|
| 6 | ZIP 143: Transaction Signature Validation for Overwinter |
|---|
| 7 | ZIP 203: Transaction Expiry |
|---|
| 8 | ZIP 212: Allow Recipient to Derive Ephemeral Secret from Note Plaintext |
|---|
| 9 | ZIP 225: Version 5 Transaction Format |
|---|
| 10 | ZIP 239: Relay of Version 5 Transactions |
|---|
| 11 | ZIP 244: Transaction Identifier Non-Malleability |
|---|
| 12 | ZIP 307: Light Client Protocol for Payment Detection |
|---|