363 lines
12 KiB
Rust
363 lines
12 KiB
Rust
// SPDX-License-Identifier: CC0-1.0
|
|
|
|
//! Provides type [`LockTime`] that implements the logic around nSequence/OP_CHECKSEQUENCEVERIFY.
|
|
//!
|
|
//! There are two types of lock time: lock-by-blockheight and lock-by-blocktime, distinguished by
|
|
//! whether bit 22 of the `u32` consensus value is set.
|
|
//!
|
|
|
|
use core::{cmp, fmt};
|
|
|
|
#[cfg(all(test, mutate))]
|
|
use mutagen::mutate;
|
|
|
|
#[cfg(doc)]
|
|
use crate::relative;
|
|
|
|
#[rustfmt::skip] // Keep public re-exports separate.
|
|
#[doc(inline)]
|
|
pub use units::locktime::relative::{Height, Time, TimeOverflowError};
|
|
|
|
/// A relative lock time value, representing either a block height or time (512 second intervals).
|
|
///
|
|
/// Used for sequence numbers (`nSequence` in Bitcoin Core and [`crate::Transaction::TxIn::sequence`]
|
|
/// in this library) and also for the argument to opcode 'OP_CHECKSEQUENCEVERIFY`.
|
|
///
|
|
/// ### Note on ordering
|
|
///
|
|
/// Because locktimes may be height- or time-based, and these metrics are incommensurate, there
|
|
/// is no total ordering on locktimes. We therefore have implemented [`PartialOrd`] but not [`Ord`].
|
|
///
|
|
/// ### Relevant BIPs
|
|
///
|
|
/// * [BIP 68 Relative lock-time using consensus-enforced sequence numbers](https://github.com/bitcoin/bips/blob/master/bip-0065.mediawiki)
|
|
/// * [BIP 112 CHECKSEQUENCEVERIFY](https://github.com/bitcoin/bips/blob/master/bip-0112.mediawiki)
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
|
#[cfg_attr(feature = "serde", derive(Serialize, Deserialize))]
|
|
#[cfg_attr(feature = "serde", serde(crate = "actual_serde"))]
|
|
pub enum LockTime {
|
|
/// A block height lock time value.
|
|
Blocks(Height),
|
|
/// A 512 second time interval value.
|
|
Time(Time),
|
|
}
|
|
|
|
impl LockTime {
|
|
/// A relative locktime of 0 is always valid, and is assumed valid for inputs that
|
|
/// are not yet confirmed.
|
|
pub const ZERO: LockTime = LockTime::Blocks(Height::ZERO);
|
|
|
|
/// The number of bytes that the locktime contributes to the size of a transaction.
|
|
pub const SIZE: usize = 4; // Serialized length of a u32.
|
|
|
|
/// Constructs a `LockTime` from `n`, expecting `n` to be a 16-bit count of blocks.
|
|
#[inline]
|
|
pub const fn from_height(n: u16) -> Self { LockTime::Blocks(Height::from_height(n)) }
|
|
|
|
/// Constructs a `LockTime` from `n`, expecting `n` to be a count of 512-second intervals.
|
|
///
|
|
/// This function is a little awkward to use, and users may wish to instead use
|
|
/// [`Self::from_seconds_floor`] or [`Self::from_seconds_ceil`].
|
|
#[inline]
|
|
pub const fn from_512_second_intervals(intervals: u16) -> Self {
|
|
LockTime::Time(Time::from_512_second_intervals(intervals))
|
|
}
|
|
|
|
/// Create a [`LockTime`] from seconds, converting the seconds into 512 second interval
|
|
/// with truncating division.
|
|
///
|
|
/// # Errors
|
|
///
|
|
/// Will return an error if the input cannot be encoded in 16 bits.
|
|
#[inline]
|
|
pub const fn from_seconds_floor(seconds: u32) -> Result<Self, TimeOverflowError> {
|
|
match Time::from_seconds_floor(seconds) {
|
|
Ok(time) => Ok(LockTime::Time(time)),
|
|
Err(e) => Err(e),
|
|
}
|
|
}
|
|
|
|
/// Create a [`LockTime`] from seconds, converting the seconds into 512 second interval
|
|
/// with ceiling division.
|
|
///
|
|
/// # Errors
|
|
///
|
|
/// Will return an error if the input cannot be encoded in 16 bits.
|
|
#[inline]
|
|
pub const fn from_seconds_ceil(seconds: u32) -> Result<Self, TimeOverflowError> {
|
|
match Time::from_seconds_ceil(seconds) {
|
|
Ok(time) => Ok(LockTime::Time(time)),
|
|
Err(e) => Err(e),
|
|
}
|
|
}
|
|
|
|
/// Returns true if this [`relative::LockTime`] is satisfied by either height or time.
|
|
///
|
|
/// # Examples
|
|
///
|
|
/// ```rust
|
|
/// # use bitcoin::Sequence;
|
|
/// # use bitcoin::locktime::relative::{LockTime, Height, Time};
|
|
///
|
|
/// # let height = 100; // 100 blocks.
|
|
/// # let intervals = 70; // Approx 10 hours.
|
|
/// # let current_height = || Height::from(height + 10);
|
|
/// # let current_time = || Time::from_512_second_intervals(intervals + 10);
|
|
/// # let lock = Sequence::from_height(height).to_relative_lock_time().expect("valid height");
|
|
///
|
|
/// // Users that have chain data can get the current height and time to check against a lock.
|
|
/// let height_and_time = (current_time(), current_height()); // tuple order does not matter.
|
|
/// assert!(lock.is_satisfied_by(current_height(), current_time()));
|
|
/// ```
|
|
#[inline]
|
|
#[cfg_attr(all(test, mutate), mutate)]
|
|
pub fn is_satisfied_by(&self, h: Height, t: Time) -> bool {
|
|
if let Ok(true) = self.is_satisfied_by_height(h) {
|
|
true
|
|
} else {
|
|
matches!(self.is_satisfied_by_time(t), Ok(true))
|
|
}
|
|
}
|
|
|
|
/// Returns true if satisfaction of `other` lock time implies satisfaction of this
|
|
/// [`relative::LockTime`].
|
|
///
|
|
/// A lock time can only be satisfied by n blocks being mined or n seconds passing. If you have
|
|
/// two lock times (same unit) then the larger lock time being satisfied implies (in a
|
|
/// mathematical sense) the smaller one being satisfied.
|
|
///
|
|
/// This function is useful when checking sequence values against a lock, first one checks the
|
|
/// sequence represents a relative lock time by converting to `LockTime` then use this function
|
|
/// to see if satisfaction of the newly created lock time would imply satisfaction of `self`.
|
|
///
|
|
/// Can also be used to remove the smaller value of two `OP_CHECKSEQUENCEVERIFY` operations
|
|
/// within one branch of the script.
|
|
///
|
|
/// # Examples
|
|
///
|
|
/// ```rust
|
|
/// # use bitcoin::Sequence;
|
|
/// # use bitcoin::locktime::relative::{LockTime, Height, Time};
|
|
///
|
|
/// # let height = 100; // 100 blocks.
|
|
/// # let lock = Sequence::from_height(height).to_relative_lock_time().expect("valid height");
|
|
/// # let test_sequence = Sequence::from_height(height + 10);
|
|
///
|
|
/// let satisfied = match test_sequence.to_relative_lock_time() {
|
|
/// None => false, // Handle non-lock-time case.
|
|
/// Some(test_lock) => lock.is_implied_by(test_lock),
|
|
/// };
|
|
/// assert!(satisfied);
|
|
/// ```
|
|
#[inline]
|
|
#[cfg_attr(all(test, mutate), mutate)]
|
|
pub fn is_implied_by(&self, other: LockTime) -> bool {
|
|
use LockTime::*;
|
|
|
|
match (*self, other) {
|
|
(Blocks(this), Blocks(other)) => this.value() <= other.value(),
|
|
(Time(this), Time(other)) => this.value() <= other.value(),
|
|
_ => false, // Not the same units.
|
|
}
|
|
}
|
|
|
|
/// Returns true if this [`relative::LockTime`] is satisfied by [`Height`].
|
|
///
|
|
/// # Errors
|
|
///
|
|
/// Returns an error if this lock is not lock-by-height.
|
|
///
|
|
/// # Examples
|
|
///
|
|
/// ```rust
|
|
/// # use bitcoin::Sequence;
|
|
/// # use bitcoin::locktime::relative::{LockTime, Height, Time};
|
|
///
|
|
/// let height: u16 = 100;
|
|
/// let lock = Sequence::from_height(height).to_relative_lock_time().expect("valid height");
|
|
/// assert!(lock.is_satisfied_by_height(Height::from(height+1)).expect("a height"));
|
|
/// ```
|
|
#[inline]
|
|
#[cfg_attr(all(test, mutate), mutate)]
|
|
pub fn is_satisfied_by_height(&self, height: Height) -> Result<bool, IncompatibleHeightError> {
|
|
use LockTime::*;
|
|
|
|
match *self {
|
|
Blocks(ref h) => Ok(h.value() <= height.value()),
|
|
Time(time) => Err(IncompatibleHeightError { height, time }),
|
|
}
|
|
}
|
|
|
|
/// Returns true if this [`relative::LockTime`] is satisfied by [`Time`].
|
|
///
|
|
/// # Errors
|
|
///
|
|
/// Returns an error if this lock is not lock-by-time.
|
|
///
|
|
/// # Examples
|
|
///
|
|
/// ```rust
|
|
/// # use bitcoin::Sequence;
|
|
/// # use bitcoin::locktime::relative::{LockTime, Height, Time};
|
|
///
|
|
/// let intervals: u16 = 70; // approx 10 hours;
|
|
/// let lock = Sequence::from_512_second_intervals(intervals).to_relative_lock_time().expect("valid time");
|
|
/// assert!(lock.is_satisfied_by_time(Time::from_512_second_intervals(intervals + 10)).expect("a time"));
|
|
/// ```
|
|
#[inline]
|
|
#[cfg_attr(all(test, mutate), mutate)]
|
|
pub fn is_satisfied_by_time(&self, time: Time) -> Result<bool, IncompatibleTimeError> {
|
|
use LockTime::*;
|
|
|
|
match *self {
|
|
Time(ref t) => Ok(t.value() <= time.value()),
|
|
Blocks(height) => Err(IncompatibleTimeError { time, height }),
|
|
}
|
|
}
|
|
}
|
|
|
|
impl From<Height> for LockTime {
|
|
#[inline]
|
|
fn from(h: Height) -> Self { LockTime::Blocks(h) }
|
|
}
|
|
|
|
impl From<Time> for LockTime {
|
|
#[inline]
|
|
fn from(t: Time) -> Self { LockTime::Time(t) }
|
|
}
|
|
|
|
impl PartialOrd for LockTime {
|
|
#[inline]
|
|
fn partial_cmp(&self, other: &LockTime) -> Option<cmp::Ordering> {
|
|
use LockTime::*;
|
|
|
|
match (*self, *other) {
|
|
(Blocks(ref a), Blocks(ref b)) => a.partial_cmp(b),
|
|
(Time(ref a), Time(ref b)) => a.partial_cmp(b),
|
|
(_, _) => None,
|
|
}
|
|
}
|
|
}
|
|
|
|
impl fmt::Display for LockTime {
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
use LockTime::*;
|
|
|
|
if f.alternate() {
|
|
match *self {
|
|
Blocks(ref h) => write!(f, "block-height {}", h),
|
|
Time(ref t) => write!(f, "block-time {} (512 second intervals)", t),
|
|
}
|
|
} else {
|
|
match *self {
|
|
Blocks(ref h) => fmt::Display::fmt(h, f),
|
|
Time(ref t) => fmt::Display::fmt(t, f),
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Tried to satisfy a lock-by-blocktime lock using a height value.
|
|
#[derive(Debug, Clone, PartialEq, Eq)]
|
|
#[non_exhaustive]
|
|
pub struct IncompatibleHeightError {
|
|
/// Attempted to satisfy a lock-by-blocktime lock with this height.
|
|
pub height: Height,
|
|
/// The inner time value of the lock-by-blocktime lock.
|
|
pub time: Time,
|
|
}
|
|
|
|
impl fmt::Display for IncompatibleHeightError {
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
write!(
|
|
f,
|
|
"tried to satisfy a lock-by-blocktime lock {} with height: {}",
|
|
self.time, self.height
|
|
)
|
|
}
|
|
}
|
|
|
|
#[cfg(feature = "std")]
|
|
impl std::error::Error for IncompatibleHeightError {}
|
|
|
|
/// Tried to satisfy a lock-by-blockheight lock using a time value.
|
|
#[derive(Debug, Clone, PartialEq, Eq)]
|
|
#[non_exhaustive]
|
|
pub struct IncompatibleTimeError {
|
|
/// Attempted to satisfy a lock-by-blockheight lock with this time.
|
|
pub time: Time,
|
|
/// The inner height value of the lock-by-blockheight lock.
|
|
pub height: Height,
|
|
}
|
|
|
|
impl fmt::Display for IncompatibleTimeError {
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
write!(
|
|
f,
|
|
"tried to satisfy a lock-by-blockheight lock {} with time: {}",
|
|
self.height, self.time
|
|
)
|
|
}
|
|
}
|
|
|
|
#[cfg(feature = "std")]
|
|
impl std::error::Error for IncompatibleTimeError {}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
use super::*;
|
|
|
|
#[test]
|
|
fn satisfied_by_height() {
|
|
let height = Height::from(10);
|
|
let time = Time::from_512_second_intervals(70);
|
|
|
|
let lock = LockTime::from(height);
|
|
|
|
assert!(!lock.is_satisfied_by(Height::from(9), time));
|
|
assert!(lock.is_satisfied_by(Height::from(10), time));
|
|
assert!(lock.is_satisfied_by(Height::from(11), time));
|
|
}
|
|
|
|
#[test]
|
|
fn satisfied_by_time() {
|
|
let height = Height::from(10);
|
|
let time = Time::from_512_second_intervals(70);
|
|
|
|
let lock = LockTime::from(time);
|
|
|
|
assert!(!lock.is_satisfied_by(height, Time::from_512_second_intervals(69)));
|
|
assert!(lock.is_satisfied_by(height, Time::from_512_second_intervals(70)));
|
|
assert!(lock.is_satisfied_by(height, Time::from_512_second_intervals(71)));
|
|
}
|
|
|
|
#[test]
|
|
fn height_correctly_implies() {
|
|
let height = Height::from(10);
|
|
let lock = LockTime::from(height);
|
|
|
|
assert!(!lock.is_implied_by(LockTime::from(Height::from(9))));
|
|
assert!(lock.is_implied_by(LockTime::from(Height::from(10))));
|
|
assert!(lock.is_implied_by(LockTime::from(Height::from(11))));
|
|
}
|
|
|
|
#[test]
|
|
fn time_correctly_implies() {
|
|
let time = Time::from_512_second_intervals(70);
|
|
let lock = LockTime::from(time);
|
|
|
|
assert!(!lock.is_implied_by(LockTime::from(Time::from_512_second_intervals(69))));
|
|
assert!(lock.is_implied_by(LockTime::from(Time::from_512_second_intervals(70))));
|
|
assert!(lock.is_implied_by(LockTime::from(Time::from_512_second_intervals(71))));
|
|
}
|
|
|
|
#[test]
|
|
fn incorrect_units_do_not_imply() {
|
|
let time = Time::from_512_second_intervals(70);
|
|
let height = Height::from(10);
|
|
|
|
let lock = LockTime::from(time);
|
|
assert!(!lock.is_implied_by(LockTime::from(height)));
|
|
}
|
|
}
|