Add `zcash_unstable` compiler flag to fully gate `orchard` functionality
This commit is contained in:
parent
20150c32a0
commit
184e3c741f
|
@ -17,6 +17,11 @@ jobs:
|
||||||
- orchard
|
- orchard
|
||||||
- unstable-nu6
|
- unstable-nu6
|
||||||
- zfuture
|
- zfuture
|
||||||
|
include:
|
||||||
|
- extra_flags: orchard
|
||||||
|
rustflags: '--cfg zcash_unstable="orchard"'
|
||||||
|
env:
|
||||||
|
RUSTFLAGS: ${{ matrix.rustflags }}
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v4
|
||||||
|
|
|
@ -1 +1,2 @@
|
||||||
target
|
target
|
||||||
|
.cargo
|
||||||
|
|
|
@ -61,6 +61,9 @@ pub enum Error<DataSourceError, CommitmentTreeError, SelectionError, FeeError> {
|
||||||
/// Attempted to create a spend to an unsupported pool type (currently, Orchard).
|
/// Attempted to create a spend to an unsupported pool type (currently, Orchard).
|
||||||
UnsupportedPoolType(PoolType),
|
UnsupportedPoolType(PoolType),
|
||||||
|
|
||||||
|
/// Attempted to create a spend to an unsupported Unified Address receiver
|
||||||
|
NoSupportedReceivers(Vec<u32>),
|
||||||
|
|
||||||
/// A note being spent does not correspond to either the internal or external
|
/// A note being spent does not correspond to either the internal or external
|
||||||
/// full viewing key for an account.
|
/// full viewing key for an account.
|
||||||
NoteMismatch(NoteId),
|
NoteMismatch(NoteId),
|
||||||
|
@ -117,7 +120,8 @@ where
|
||||||
Error::ScanRequired => write!(f, "Must scan blocks first"),
|
Error::ScanRequired => write!(f, "Must scan blocks first"),
|
||||||
Error::Builder(e) => write!(f, "An error occurred building the transaction: {}", e),
|
Error::Builder(e) => write!(f, "An error occurred building the transaction: {}", e),
|
||||||
Error::MemoForbidden => write!(f, "It is not possible to send a memo to a transparent address."),
|
Error::MemoForbidden => write!(f, "It is not possible to send a memo to a transparent address."),
|
||||||
Error::UnsupportedPoolType(t) => write!(f, "Attempted to create spend to an unsupported pool type: {}", t),
|
Error::UnsupportedPoolType(t) => write!(f, "Attempted to send to an unsupported pool: {}", t),
|
||||||
|
Error::NoSupportedReceivers(t) => write!(f, "Unified address contained only unsupported receiver types: {:?}", &t[..]),
|
||||||
Error::NoteMismatch(n) => write!(f, "A note being spent ({:?}) does not correspond to either the internal or external full viewing key for the provided spending key.", n),
|
Error::NoteMismatch(n) => write!(f, "A note being spent ({:?}) does not correspond to either the internal or external full viewing key for the provided spending key.", n),
|
||||||
|
|
||||||
#[cfg(feature = "transparent-inputs")]
|
#[cfg(feature = "transparent-inputs")]
|
||||||
|
|
|
@ -693,9 +693,9 @@ where
|
||||||
builder.add_transparent_output(taddr, payment.amount)?;
|
builder.add_transparent_output(taddr, payment.amount)?;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
return Err(Error::UnsupportedPoolType(PoolType::Shielded(
|
return Err(Error::NoSupportedReceivers(
|
||||||
ShieldedProtocol::Orchard,
|
ua.unknown().iter().map(|(tc, _)| *tc).collect(),
|
||||||
)));
|
));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Address::Sapling(addr) => {
|
Address::Sapling(addr) => {
|
||||||
|
@ -738,6 +738,7 @@ where
|
||||||
Some(memo),
|
Some(memo),
|
||||||
))
|
))
|
||||||
}
|
}
|
||||||
|
#[cfg(zcash_unstable = "orchard")]
|
||||||
ShieldedProtocol::Orchard => {
|
ShieldedProtocol::Orchard => {
|
||||||
#[cfg(not(feature = "orchard"))]
|
#[cfg(not(feature = "orchard"))]
|
||||||
return Err(Error::UnsupportedPoolType(PoolType::Shielded(
|
return Err(Error::UnsupportedPoolType(PoolType::Shielded(
|
||||||
|
|
|
@ -635,11 +635,16 @@ where
|
||||||
Err(other) => return Err(other.into()),
|
Err(other) => return Err(other.into()),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[cfg(not(zcash_unstable = "orchard"))]
|
||||||
|
let selectable_pools = &[ShieldedProtocol::Sapling];
|
||||||
|
#[cfg(zcash_unstable = "orchard")]
|
||||||
|
let selectable_pools = &[ShieldedProtocol::Sapling, ShieldedProtocol::Orchard];
|
||||||
|
|
||||||
shielded_inputs = wallet_db
|
shielded_inputs = wallet_db
|
||||||
.select_spendable_notes(
|
.select_spendable_notes(
|
||||||
account,
|
account,
|
||||||
amount_required.into(),
|
amount_required.into(),
|
||||||
&[ShieldedProtocol::Sapling, ShieldedProtocol::Orchard],
|
selectable_pools,
|
||||||
anchor_height,
|
anchor_height,
|
||||||
&exclude,
|
&exclude,
|
||||||
)
|
)
|
||||||
|
|
|
@ -37,12 +37,18 @@ pub use decrypt::{decrypt_transaction, DecryptedOutput, TransferType};
|
||||||
#[macro_use]
|
#[macro_use]
|
||||||
extern crate assert_matches;
|
extern crate assert_matches;
|
||||||
|
|
||||||
|
#[cfg(all(feature = "orchard", not(zcash_unstable = "orchard")))]
|
||||||
|
core::compile_error!(
|
||||||
|
"The `orchard` feature flag requires the `zcash_unstable=\"orchard\"` RUSTFLAG."
|
||||||
|
);
|
||||||
|
|
||||||
/// A shielded transfer protocol known to the wallet.
|
/// A shielded transfer protocol known to the wallet.
|
||||||
#[derive(Debug, Copy, Clone, PartialEq, Eq, PartialOrd, Ord)]
|
#[derive(Debug, Copy, Clone, PartialEq, Eq, PartialOrd, Ord)]
|
||||||
pub enum ShieldedProtocol {
|
pub enum ShieldedProtocol {
|
||||||
/// The Sapling protocol
|
/// The Sapling protocol
|
||||||
Sapling,
|
Sapling,
|
||||||
/// The Orchard protocol
|
/// The Orchard protocol
|
||||||
|
#[cfg(zcash_unstable = "orchard")]
|
||||||
Orchard,
|
Orchard,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -60,6 +66,7 @@ impl fmt::Display for PoolType {
|
||||||
match self {
|
match self {
|
||||||
PoolType::Transparent => f.write_str("Transparent"),
|
PoolType::Transparent => f.write_str("Transparent"),
|
||||||
PoolType::Shielded(ShieldedProtocol::Sapling) => f.write_str("Sapling"),
|
PoolType::Shielded(ShieldedProtocol::Sapling) => f.write_str("Sapling"),
|
||||||
|
#[cfg(zcash_unstable = "orchard")]
|
||||||
PoolType::Shielded(ShieldedProtocol::Orchard) => f.write_str("Orchard"),
|
PoolType::Shielded(ShieldedProtocol::Orchard) => f.write_str("Orchard"),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -222,7 +222,7 @@ pub enum ProposalDecodingError<DbError> {
|
||||||
/// A transaction identifier string did not decode to a valid transaction ID.
|
/// A transaction identifier string did not decode to a valid transaction ID.
|
||||||
TxIdInvalid(TryFromSliceError),
|
TxIdInvalid(TryFromSliceError),
|
||||||
/// An invalid value pool identifier was encountered.
|
/// An invalid value pool identifier was encountered.
|
||||||
ValuePoolInvalid(i32),
|
ValuePoolNotSupported(i32),
|
||||||
/// A failure occurred trying to retrieve an unspent note or UTXO from the wallet database.
|
/// A failure occurred trying to retrieve an unspent note or UTXO from the wallet database.
|
||||||
InputRetrieval(DbError),
|
InputRetrieval(DbError),
|
||||||
/// The unspent note or UTXO corresponding to a proposal input was not found in the wallet
|
/// The unspent note or UTXO corresponding to a proposal input was not found in the wallet
|
||||||
|
@ -257,7 +257,7 @@ impl<E: Display> Display for ProposalDecodingError<E> {
|
||||||
ProposalDecodingError::TxIdInvalid(err) => {
|
ProposalDecodingError::TxIdInvalid(err) => {
|
||||||
write!(f, "Invalid transaction id: {:?}", err)
|
write!(f, "Invalid transaction id: {:?}", err)
|
||||||
}
|
}
|
||||||
ProposalDecodingError::ValuePoolInvalid(id) => {
|
ProposalDecodingError::ValuePoolNotSupported(id) => {
|
||||||
write!(f, "Invalid value pool identifier: {:?}", id)
|
write!(f, "Invalid value pool identifier: {:?}", id)
|
||||||
}
|
}
|
||||||
ProposalDecodingError::InputRetrieval(err) => write!(
|
ProposalDecodingError::InputRetrieval(err) => write!(
|
||||||
|
@ -312,8 +312,9 @@ fn pool_type<T>(pool_id: i32) -> Result<PoolType, ProposalDecodingError<T>> {
|
||||||
match proposal::ValuePool::try_from(pool_id) {
|
match proposal::ValuePool::try_from(pool_id) {
|
||||||
Ok(proposal::ValuePool::Transparent) => Ok(PoolType::Transparent),
|
Ok(proposal::ValuePool::Transparent) => Ok(PoolType::Transparent),
|
||||||
Ok(proposal::ValuePool::Sapling) => Ok(PoolType::Shielded(ShieldedProtocol::Sapling)),
|
Ok(proposal::ValuePool::Sapling) => Ok(PoolType::Shielded(ShieldedProtocol::Sapling)),
|
||||||
|
#[cfg(zcash_unstable = "orchard")]
|
||||||
Ok(proposal::ValuePool::Orchard) => Ok(PoolType::Shielded(ShieldedProtocol::Orchard)),
|
Ok(proposal::ValuePool::Orchard) => Ok(PoolType::Shielded(ShieldedProtocol::Orchard)),
|
||||||
_ => Err(ProposalDecodingError::ValuePoolInvalid(pool_id)),
|
_ => Err(ProposalDecodingError::ValuePoolNotSupported(pool_id)),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -337,6 +338,7 @@ impl From<ShieldedProtocol> for proposal::ValuePool {
|
||||||
fn from(value: ShieldedProtocol) -> Self {
|
fn from(value: ShieldedProtocol) -> Self {
|
||||||
match value {
|
match value {
|
||||||
ShieldedProtocol::Sapling => proposal::ValuePool::Sapling,
|
ShieldedProtocol::Sapling => proposal::ValuePool::Sapling,
|
||||||
|
#[cfg(zcash_unstable = "orchard")]
|
||||||
ShieldedProtocol::Orchard => proposal::ValuePool::Orchard,
|
ShieldedProtocol::Orchard => proposal::ValuePool::Orchard,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -447,7 +449,7 @@ impl proposal::Proposal {
|
||||||
match input.pool_type()? {
|
match input.pool_type()? {
|
||||||
PoolType::Transparent => {
|
PoolType::Transparent => {
|
||||||
#[cfg(not(feature = "transparent-inputs"))]
|
#[cfg(not(feature = "transparent-inputs"))]
|
||||||
return Err(ProposalDecodingError::ValuePoolInvalid(1));
|
return Err(ProposalDecodingError::ValuePoolNotSupported(1));
|
||||||
|
|
||||||
#[cfg(feature = "transparent-inputs")]
|
#[cfg(feature = "transparent-inputs")]
|
||||||
{
|
{
|
||||||
|
|
|
@ -132,6 +132,7 @@ pub(crate) fn pool_code(pool_type: PoolType) -> i64 {
|
||||||
match pool_type {
|
match pool_type {
|
||||||
PoolType::Transparent => 0i64,
|
PoolType::Transparent => 0i64,
|
||||||
PoolType::Shielded(ShieldedProtocol::Sapling) => 2i64,
|
PoolType::Shielded(ShieldedProtocol::Sapling) => 2i64,
|
||||||
|
#[cfg(zcash_unstable = "orchard")]
|
||||||
PoolType::Shielded(ShieldedProtocol::Orchard) => 3i64,
|
PoolType::Shielded(ShieldedProtocol::Orchard) => 3i64,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -759,6 +760,7 @@ pub(crate) fn get_received_memo(
|
||||||
)
|
)
|
||||||
.optional()?
|
.optional()?
|
||||||
.flatten(),
|
.flatten(),
|
||||||
|
#[cfg(zcash_unstable = "orchard")]
|
||||||
_ => {
|
_ => {
|
||||||
return Err(SqliteClientError::UnsupportedPoolType(PoolType::Shielded(
|
return Err(SqliteClientError::UnsupportedPoolType(PoolType::Shielded(
|
||||||
note_id.protocol(),
|
note_id.protocol(),
|
||||||
|
|
|
@ -150,6 +150,11 @@ impl UnifiedAddress {
|
||||||
self.transparent.as_ref()
|
self.transparent.as_ref()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Returns the set of unknown receivers of the unified address.
|
||||||
|
pub fn unknown(&self) -> &[(u32, Vec<u8>)] {
|
||||||
|
&self.unknown
|
||||||
|
}
|
||||||
|
|
||||||
fn to_address(&self, net: Network) -> ZcashAddress {
|
fn to_address(&self, net: Network) -> ZcashAddress {
|
||||||
#[cfg(feature = "orchard")]
|
#[cfg(feature = "orchard")]
|
||||||
let orchard_receiver = self
|
let orchard_receiver = self
|
||||||
|
|
Loading…
Reference in New Issue