1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
use ckb_types::core::EpochNumberWithFraction;

use crate::constants::{LOCK_TYPE_FLAG, METRIC_TYPE_FLAG_MASK, REMAIN_FLAGS_BITS, VALUE_MASK};

#[derive(Clone, Copy, Debug, Eq, PartialEq)]
pub enum SinceType {
    BlockNumber,
    EpochNumberWithFraction,
    Timestamp,
}

#[derive(Clone, Copy, Debug, Eq, PartialEq)]
pub struct Since(u64);

impl Since {
    pub fn new(ty: SinceType, value: u64, is_relative: bool) -> Since {
        let value = match ty {
            SinceType::BlockNumber => value,
            SinceType::EpochNumberWithFraction => 0x2000_0000_0000_0000 | value,
            SinceType::Timestamp => 0x4000_0000_0000_0000 | value,
        };
        if is_relative {
            Since(LOCK_TYPE_FLAG | value)
        } else {
            Since(value)
        }
    }

    pub fn new_absolute_epoch(epoch_number: u64) -> Since {
        let epoch = EpochNumberWithFraction::new(epoch_number, 0, 1);
        Self::new(
            SinceType::EpochNumberWithFraction,
            epoch.full_value(),
            false,
        )
    }

    pub fn from_raw_value(value: u64) -> Since {
        Since(value)
    }

    pub fn value(self) -> u64 {
        self.0
    }

    pub fn is_absolute(self) -> bool {
        self.0 & LOCK_TYPE_FLAG == 0
    }

    pub fn is_relative(self) -> bool {
        !self.is_absolute()
    }

    pub fn flags_is_valid(self) -> bool {
        (self.0 & REMAIN_FLAGS_BITS == 0)
            && ((self.0 & METRIC_TYPE_FLAG_MASK) != METRIC_TYPE_FLAG_MASK)
    }

    pub fn extract_metric(self) -> Option<(SinceType, u64)> {
        let value = self.0 & VALUE_MASK;
        let ty_opt = match self.0 & METRIC_TYPE_FLAG_MASK {
            //0b0000_0000
            0x0000_0000_0000_0000 => Some(SinceType::BlockNumber),
            //0b0010_0000
            0x2000_0000_0000_0000 => Some(SinceType::EpochNumberWithFraction),
            //0b0100_0000
            0x4000_0000_0000_0000 => Some(SinceType::Timestamp),
            _ => None,
        };
        ty_opt.map(|ty| (ty, value))
    }
}