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
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
use alloc::vec::Vec;
use fuel_types::Bytes32;

use crate::ConsensusParameters;
#[cfg(feature = "std")]
use crate::{field, UniqueIdentifier};

/// Entity support metadata computation to cache results.
pub trait Cacheable {
    /// The cache is already computed.
    ///
    /// # Note: `true` doesn't mean that the cache is actual.
    fn is_computed(&self) -> bool;

    /// Computes the cache for the entity.
    fn precompute(&mut self, parameters: &ConsensusParameters);
}

#[cfg(feature = "std")]
impl Cacheable for super::Transaction {
    fn is_computed(&self) -> bool {
        match self {
            Self::Script(script) => script.is_computed(),
            Self::Create(create) => create.is_computed(),
            Self::Mint(mint) => mint.is_computed(),
        }
    }

    fn precompute(&mut self, parameters: &ConsensusParameters) {
        match self {
            Self::Script(script) => script.precompute(parameters),
            Self::Create(create) => create.precompute(parameters),
            Self::Mint(mint) => mint.precompute(parameters),
        }
    }
}

/// Common metadata for `Script` and `Create` transactions.
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub(crate) struct CommonMetadata {
    pub id: Bytes32,
    pub inputs_offset: usize,
    pub inputs_offset_at: Vec<usize>,
    pub inputs_predicate_offset_at: Vec<Option<(usize, usize)>>,
    pub outputs_offset: usize,
    pub outputs_offset_at: Vec<usize>,
    pub witnesses_offset: usize,
    pub witnesses_offset_at: Vec<usize>,
}

#[cfg(feature = "std")]
impl CommonMetadata {
    /// Computes the `Metadata` for the `tx` transaction.
    pub fn compute<Tx>(tx: &Tx, params: &ConsensusParameters) -> Self
    where
        Tx: UniqueIdentifier,
        Tx: field::Inputs,
        Tx: field::Outputs,
        Tx: field::Witnesses,
        Tx: fuel_types::bytes::SizedBytes,
    {
        use fuel_types::bytes::SizedBytes;
        use itertools::Itertools;

        let id = tx.id(params);

        let inputs_predicate_offset_at = tx
            .inputs()
            .iter()
            .enumerate()
            .map(|(i, _)| tx.inputs_predicate_offset_at(i))
            .collect_vec();

        let mut offset = tx.inputs_offset();
        let inputs_offset = offset;
        let inputs_offset_at = tx
            .inputs()
            .iter()
            .map(|input| {
                let i = offset;
                offset += input.serialized_size();
                i
            })
            .collect_vec();

        let outputs_offset = offset;
        #[cfg(feature = "internals")]
        assert_eq!(outputs_offset, tx.outputs_offset());

        let outputs_offset_at = tx
            .outputs()
            .iter()
            .map(|output| {
                let i = offset;
                offset += output.serialized_size();
                i
            })
            .collect_vec();

        let witnesses_offset = offset;
        #[cfg(feature = "internals")]
        assert_eq!(witnesses_offset, tx.witnesses_offset());

        let witnesses_offset_at = tx
            .witnesses()
            .iter()
            .map(|witness| {
                let i = offset;
                offset += witness.serialized_size();
                i
            })
            .collect_vec();

        Self {
            id,
            inputs_offset,
            inputs_offset_at,
            inputs_predicate_offset_at,
            outputs_offset,
            outputs_offset_at,
            witnesses_offset,
            witnesses_offset_at,
        }
    }
}