• Home
  • Features
  • Pricing
  • Docs
  • Announcements
  • Sign In

tari-project / tari / 15213083820

23 May 2025 02:48PM UTC coverage: 73.244% (+0.2%) from 73.068%
15213083820

push

github

web-flow
chore: new release v3.0.2-pre.0 (#7099)

Description
---
new release esme

82228 of 112266 relevant lines covered (73.24%)

273889.71 hits per line

Source File
Press 'n' to go to next uncovered line, 'b' for previous

99.17
/base_layer/core/src/validation/test.rs
1
//  Copyright 2020, The Tari Project
2
//
3
//  Redistribution and use in source and binary forms, with or without modification, are permitted provided that the
4
//  following conditions are met:
5
//
6
//  1. Redistributions of source code must retain the above copyright notice, this list of conditions and the following
7
//  disclaimer.
8
//
9
//  2. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the
10
//  following disclaimer in the documentation and/or other materials provided with the distribution.
11
//
12
//  3. Neither the name of the copyright holder nor the names of its contributors may be used to endorse or promote
13
//  products derived from this software without specific prior written permission.
14
//
15
//  THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES,
16
//  INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
17
//  DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
18
//  SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
19
//  SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
20
//  WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE
21
//  USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
22

23
use std::{cmp, sync::Arc};
24

25
use jmt::{mock::MockTreeStore, JellyfishMerkleTree, KeyHash};
26
use tari_common::configuration::Network;
27
use tari_common_types::types::{CompressedCommitment, UncompressedCommitment};
28
use tari_script::TariScript;
29
use tari_test_utils::unpack_enum;
30
use tari_utilities::ByteArray;
31

32
use crate::{
33
    blocks::{BlockHeader, BlockHeaderAccumulatedData, ChainBlock, ChainHeader},
34
    chain_storage::{BlockchainBackend, BlockchainDatabase, ChainStorageError, DbTransaction, SmtHasher},
35
    consensus::{ConsensusConstantsBuilder, ConsensusManager, ConsensusManagerBuilder},
36
    covenants::Covenant,
37
    proof_of_work::AchievedTargetDifficulty,
38
    test_helpers::{blockchain::create_store_with_consensus, create_chain_header},
39
    transactions::{
40
        tari_amount::{uT, MicroMinotari},
41
        test_helpers::{create_random_signature_from_secret_key, create_utxo},
42
        transaction_components::{KernelBuilder, KernelFeatures, OutputFeatures, RangeProofType, TransactionKernel},
43
        transaction_key_manager::{create_memory_db_key_manager, TxoStage},
44
        CryptoFactories,
45
    },
46
    tx,
47
    validation::{ChainBalanceValidator, DifficultyCalculator, FinalHorizonStateValidation, ValidationError},
48
};
49

50
mod header_validators {
51
    use tari_common_types::types::FixedHash;
52
    use tari_utilities::epoch_time::EpochTime;
53

54
    use super::*;
55
    use crate::{
56
        block_specs,
57
        test_helpers::blockchain::{create_main_chain, create_new_blockchain},
58
        validation::{header::HeaderFullValidator, HeaderChainLinkedValidator},
59
    };
60

61
    #[test]
62
    fn header_iter_empty_and_invalid_height() {
1✔
63
        let consensus_manager = ConsensusManager::builder(Network::LocalNet).build().unwrap();
1✔
64
        let genesis = consensus_manager.get_genesis_block();
1✔
65
        let db = create_store_with_consensus(consensus_manager);
1✔
66

1✔
67
        let iter = HeaderIter::new(&db, 0, 10);
1✔
68
        let headers = iter.map(Result::unwrap).collect::<Vec<_>>();
1✔
69
        assert_eq!(headers.len(), 1);
1✔
70

71
        assert_eq!(genesis.header(), &headers[0]);
1✔
72

73
        // Invalid header height
74
        let iter = HeaderIter::new(&db, 1, 10);
1✔
75
        let headers = iter.collect::<Result<Vec<_>, _>>().unwrap();
1✔
76
        assert_eq!(headers.len(), 1);
1✔
77
    }
1✔
78

79
    #[test]
80
    fn header_iter_fetch_in_chunks() {
1✔
81
        let consensus_manager = ConsensusManagerBuilder::new(Network::LocalNet).build().unwrap();
1✔
82
        let db = create_store_with_consensus(consensus_manager.clone());
1✔
83
        let headers = (1..=15).fold(vec![db.fetch_chain_header(0).unwrap()], |mut acc, i| {
15✔
84
            let prev = acc.last().unwrap();
15✔
85
            let mut header = BlockHeader::new(0);
15✔
86
            header.height = i;
15✔
87
            header.prev_hash = *prev.hash();
15✔
88
            // These have to be unique
15✔
89
            header.kernel_mmr_size = 2 + i;
15✔
90
            header.output_smt_size = 4001 + i;
15✔
91

15✔
92
            let chain_header = create_chain_header(header, prev.accumulated_data());
15✔
93
            acc.push(chain_header);
15✔
94
            acc
15✔
95
        });
15✔
96
        db.insert_valid_headers(headers.into_iter().skip(1).collect()).unwrap();
1✔
97

1✔
98
        let iter = HeaderIter::new(&db, 11, 3);
1✔
99
        let headers = iter.map(Result::unwrap).collect::<Vec<_>>();
1✔
100
        assert_eq!(headers.len(), 12);
1✔
101
        let genesis = consensus_manager.get_genesis_block();
1✔
102
        assert_eq!(genesis.header(), &headers[0]);
1✔
103

104
        (1..=11).for_each(|i| {
11✔
105
            assert_eq!(headers[i].height, i as u64);
11✔
106
        })
11✔
107
    }
1✔
108

109
    #[test]
110
    fn it_validates_that_version_is_in_range() {
1✔
111
        let consensus_manager = ConsensusManagerBuilder::new(Network::LocalNet).build().unwrap();
1✔
112
        let db = create_store_with_consensus(consensus_manager.clone());
1✔
113

1✔
114
        let genesis = db.fetch_chain_header(0).unwrap();
1✔
115

1✔
116
        let mut header = BlockHeader::from_previous(genesis.header());
1✔
117
        header.version = u16::MAX;
1✔
118
        let difficulty_calculator = DifficultyCalculator::new(consensus_manager.clone(), Default::default());
1✔
119
        let validator = HeaderFullValidator::new(consensus_manager, difficulty_calculator);
1✔
120

1✔
121
        let err = validator
1✔
122
            .validate(
1✔
123
                &*db.db_read_access().unwrap(),
1✔
124
                &header,
1✔
125
                genesis.header(),
1✔
126
                &[],
1✔
127
                None,
1✔
128
                FixedHash::zero(),
1✔
129
            )
1✔
130
            .unwrap_err();
1✔
131
        assert!(matches!(err, ValidationError::InvalidBlockchainVersion {
1✔
132
            version: u16::MAX
133
        }));
134
    }
1✔
135

136
    #[tokio::test]
137
    async fn it_does_a_sanity_check_on_the_number_of_timestamps_provided() {
1✔
138
        let consensus_manager = ConsensusManagerBuilder::new(Network::LocalNet).build().unwrap();
1✔
139
        let db = create_new_blockchain();
1✔
140

1✔
141
        let (_, blocks) = create_main_chain(&db, block_specs!(["1->GB"], ["2->1"], ["3->2"])).await;
1✔
142
        let last_block = blocks.get("3").unwrap();
1✔
143

1✔
144
        let candidate_header = BlockHeader::from_previous(last_block.header());
1✔
145
        let difficulty_calculator = DifficultyCalculator::new(consensus_manager.clone(), Default::default());
1✔
146
        let validator = HeaderFullValidator::new(consensus_manager, difficulty_calculator);
1✔
147
        let mut timestamps = db.fetch_block_timestamps(*blocks.get("3").unwrap().hash()).unwrap();
1✔
148

1✔
149
        // First, lets check that everything else is valid
1✔
150
        validator
1✔
151
            .validate(
1✔
152
                &*db.db_read_access().unwrap(),
1✔
153
                &candidate_header,
1✔
154
                last_block.header(),
1✔
155
                &timestamps,
1✔
156
                None,
1✔
157
                FixedHash::zero(),
1✔
158
            )
1✔
159
            .unwrap();
1✔
160

1✔
161
        // Add an extra timestamp
1✔
162
        timestamps.push(EpochTime::now());
1✔
163
        let err = validator
1✔
164
            .validate(
1✔
165
                &*db.db_read_access().unwrap(),
1✔
166
                &candidate_header,
1✔
167
                last_block.header(),
1✔
168
                &timestamps,
1✔
169
                None,
1✔
170
                FixedHash::zero(),
1✔
171
            )
1✔
172
            .unwrap_err();
1✔
173
        assert!(matches!(err, ValidationError::IncorrectNumberOfTimestampsProvided {
1✔
174
            actual: 5,
1✔
175
            expected: 4
1✔
176
        }));
1✔
177
    }
1✔
178
}
179

180
#[tokio::test]
181
#[allow(clippy::too_many_lines)]
182
async fn chain_balance_validation() {
1✔
183
    let factories = CryptoFactories::default();
1✔
184
    let consensus_manager = ConsensusManagerBuilder::new(Network::Esmeralda).build().unwrap();
1✔
185
    let genesis = consensus_manager.get_genesis_block();
1✔
186
    let pre_mine_value = 5000 * uT;
1✔
187
    let key_manager = create_memory_db_key_manager().unwrap();
1✔
188
    let (pre_mine_utxo, pre_mine_key_id, _) = create_utxo(
1✔
189
        pre_mine_value,
1✔
190
        &key_manager,
1✔
191
        &OutputFeatures::default(),
1✔
192
        &TariScript::default(),
1✔
193
        &Covenant::default(),
1✔
194
        MicroMinotari::zero(),
1✔
195
    )
1✔
196
    .await;
1✔
197
    let (pk, sig) = create_random_signature_from_secret_key(
1✔
198
        &key_manager,
1✔
199
        pre_mine_key_id,
1✔
200
        0.into(),
1✔
201
        0,
1✔
202
        KernelFeatures::empty(),
1✔
203
        TxoStage::Output,
1✔
204
    )
1✔
205
    .await;
1✔
206
    let excess = CompressedCommitment::from_public_key(pk.to_public_key().unwrap());
1✔
207
    let kernel =
1✔
208
        TransactionKernel::new_current_version(KernelFeatures::empty(), MicroMinotari::from(0), 0, excess, sig, None);
1✔
209
    let mut gen_block = genesis.block().clone();
1✔
210
    gen_block.body.add_output(pre_mine_utxo);
1✔
211
    gen_block.body.add_kernels([kernel]);
1✔
212
    let mut utxo_sum = UncompressedCommitment::default();
1✔
213
    let mut kernel_sum = UncompressedCommitment::default();
1✔
214
    let burned_sum = UncompressedCommitment::default();
1✔
215
    let mock_store = MockTreeStore::new(true);
1✔
216
    let smt = JellyfishMerkleTree::<_, SmtHasher>::new(&mock_store);
1✔
217
    let mut batch = vec![];
1✔
218
    for output in gen_block.body.outputs() {
795✔
219
        utxo_sum = &output.commitment.to_commitment().unwrap() + &utxo_sum;
795✔
220
        let smt_key = KeyHash(output.commitment.as_bytes().try_into().expect("commitment is 32 bytes"));
795✔
221
        let smt_value = output.smt_hash(0);
795✔
222

795✔
223
        batch.push((smt_key, Some(smt_value.to_vec())));
795✔
224
    }
795✔
225
    for input in gen_block.body.inputs() {
313✔
226
        utxo_sum = &utxo_sum - &input.commitment().unwrap().to_commitment().unwrap();
313✔
227
        let smt_key = KeyHash(
313✔
228
            input
313✔
229
                .commitment()
313✔
230
                .unwrap()
313✔
231
                .as_bytes()
313✔
232
                .try_into()
313✔
233
                .expect("Commitment is 32 bytes"),
313✔
234
        );
313✔
235
        batch.push((smt_key, None));
313✔
236
    }
313✔
237
    for kernel in gen_block.body.kernels() {
315✔
238
        kernel_sum = &kernel.excess.to_commitment().unwrap() + &kernel_sum;
315✔
239
    }
315✔
240
    let root = smt.put_value_set(batch, 0).unwrap();
1✔
241

1✔
242
    gen_block.header.output_mr = root.0 .0.into();
1✔
243
    let mut accum = genesis.accumulated_data().clone();
1✔
244
    accum.hash = gen_block.header.hash();
1✔
245

1✔
246
    let genesis = ChainBlock::try_construct(Arc::new(gen_block), accum).unwrap();
1✔
247

1✔
248
    let total_pre_mine = pre_mine_value + consensus_manager.consensus_constants(0).pre_mine_value();
1✔
249
    let constants = ConsensusConstantsBuilder::new(Network::LocalNet)
1✔
250
        .with_consensus_constants(consensus_manager.consensus_constants(0).clone())
1✔
251
        .with_pre_mine_value(total_pre_mine)
1✔
252
        .build();
1✔
253
    // Create a LocalNet consensus manager that uses custom genesis block that contains an extra pre_mine utxo
1✔
254
    let consensus_manager = ConsensusManagerBuilder::new(Network::LocalNet)
1✔
255
        .with_block(genesis.clone())
1✔
256
        .add_consensus_constants(constants)
1✔
257
        .build()
1✔
258
        .unwrap();
1✔
259

1✔
260
    let db = create_store_with_consensus(consensus_manager.clone());
1✔
261

1✔
262
    let validator = ChainBalanceValidator::new(consensus_manager.clone(), factories.clone());
1✔
263
    // Validate the genesis state
1✔
264
    validator
1✔
265
        .validate(
1✔
266
            &*db.db_read_access().unwrap(),
1✔
267
            0,
1✔
268
            &CompressedCommitment::from_commitment(utxo_sum.clone()),
1✔
269
            &CompressedCommitment::from_commitment(kernel_sum.clone()),
1✔
270
            &CompressedCommitment::from_commitment(burned_sum.clone()),
1✔
271
        )
1✔
272
        .unwrap();
1✔
273

1✔
274
    //---------------------------------- Add a new coinbase and header --------------------------------------------//
1✔
275
    let mut txn = DbTransaction::new();
1✔
276
    let coinbase_value = consensus_manager.get_block_reward_at(1);
1✔
277
    let (coinbase, coinbase_key_id, _) = create_utxo(
1✔
278
        coinbase_value,
1✔
279
        &key_manager,
1✔
280
        &OutputFeatures::create_coinbase(1, None, RangeProofType::BulletProofPlus),
1✔
281
        &TariScript::default(),
1✔
282
        &Covenant::default(),
1✔
283
        MicroMinotari::zero(),
1✔
284
    )
1✔
285
    .await;
1✔
286

1✔
287
    let (pk, sig) = create_random_signature_from_secret_key(
1✔
288
        &key_manager,
1✔
289
        coinbase_key_id,
1✔
290
        0.into(),
1✔
291
        0,
1✔
292
        KernelFeatures::create_coinbase(),
1✔
293
        TxoStage::Output,
1✔
294
    )
1✔
295
    .await;
1✔
296
    let excess = CompressedCommitment::from_compressed_key(pk);
1✔
297
    let kernel = KernelBuilder::new()
1✔
298
        .with_signature(sig)
1✔
299
        .with_excess(&excess)
1✔
300
        .with_features(KernelFeatures::COINBASE_KERNEL)
1✔
301
        .build()
1✔
302
        .unwrap();
1✔
303

1✔
304
    let mut header1 = BlockHeader::from_previous(genesis.header());
1✔
305
    header1.kernel_mmr_size += 1;
1✔
306
    header1.output_smt_size += 1;
1✔
307
    let achieved_difficulty = AchievedTargetDifficulty::try_construct(
1✔
308
        genesis.header().pow_algo(),
1✔
309
        genesis.accumulated_data().target_difficulty,
1✔
310
        genesis.accumulated_data().achieved_difficulty,
1✔
311
    )
1✔
312
    .unwrap();
1✔
313
    let accumulated_data = BlockHeaderAccumulatedData::builder(genesis.accumulated_data())
1✔
314
        .with_hash(header1.hash())
1✔
315
        .with_achieved_target_difficulty(achieved_difficulty)
1✔
316
        .with_total_kernel_offset(header1.total_kernel_offset.clone())
1✔
317
        .build()
1✔
318
        .unwrap();
1✔
319
    let header1 = ChainHeader::try_construct(header1, accumulated_data).unwrap();
1✔
320
    txn.insert_chain_header(header1.clone());
1✔
321

1✔
322
    let mut mmr_position = 4;
1✔
323

1✔
324
    txn.insert_kernel(kernel.clone(), *header1.hash(), mmr_position);
1✔
325
    txn.insert_utxo(coinbase.clone(), *header1.hash(), 1, 0);
1✔
326

1✔
327
    db.commit(txn).unwrap();
1✔
328
    utxo_sum = &coinbase.commitment.to_commitment().unwrap() + &utxo_sum;
1✔
329
    kernel_sum = &kernel.excess.to_commitment().unwrap() + &kernel_sum;
1✔
330
    validator
1✔
331
        .validate(
1✔
332
            &*db.db_read_access().unwrap(),
1✔
333
            1,
1✔
334
            &CompressedCommitment::from_commitment(utxo_sum.clone()),
1✔
335
            &CompressedCommitment::from_commitment(kernel_sum.clone()),
1✔
336
            &CompressedCommitment::from_commitment(burned_sum.clone()),
1✔
337
        )
1✔
338
        .unwrap();
1✔
339

1✔
340
    //---------------------------------- Try to inflate --------------------------------------------//
1✔
341
    let mut txn = DbTransaction::new();
1✔
342

1✔
343
    let v = consensus_manager.get_block_reward_at(2) + uT;
1✔
344
    let (coinbase, spending_key_id, _) = create_utxo(
1✔
345
        v,
1✔
346
        &key_manager,
1✔
347
        &OutputFeatures::create_coinbase(1, None, RangeProofType::BulletProofPlus),
1✔
348
        &TariScript::default(),
1✔
349
        &Covenant::default(),
1✔
350
        MicroMinotari::zero(),
1✔
351
    )
1✔
352
    .await;
1✔
353
    let (pk, sig) = create_random_signature_from_secret_key(
1✔
354
        &key_manager,
1✔
355
        spending_key_id,
1✔
356
        0.into(),
1✔
357
        0,
1✔
358
        KernelFeatures::create_coinbase(),
1✔
359
        TxoStage::Output,
1✔
360
    )
1✔
361
    .await;
1✔
362
    let excess = CompressedCommitment::from_compressed_key(pk);
1✔
363
    let kernel = KernelBuilder::new()
1✔
364
        .with_signature(sig)
1✔
365
        .with_excess(&excess)
1✔
366
        .with_features(KernelFeatures::COINBASE_KERNEL)
1✔
367
        .build()
1✔
368
        .unwrap();
1✔
369

1✔
370
    let mut header2 = BlockHeader::from_previous(header1.header());
1✔
371
    header2.kernel_mmr_size += 1;
1✔
372
    header2.output_smt_size += 1;
1✔
373
    let achieved_difficulty = AchievedTargetDifficulty::try_construct(
1✔
374
        genesis.header().pow_algo(),
1✔
375
        genesis.accumulated_data().target_difficulty,
1✔
376
        genesis.accumulated_data().achieved_difficulty,
1✔
377
    )
1✔
378
    .unwrap();
1✔
379
    let accumulated_data = BlockHeaderAccumulatedData::builder(genesis.accumulated_data())
1✔
380
        .with_hash(header2.hash())
1✔
381
        .with_achieved_target_difficulty(achieved_difficulty)
1✔
382
        .with_total_kernel_offset(header2.total_kernel_offset.clone())
1✔
383
        .build()
1✔
384
        .unwrap();
1✔
385
    let header2 = ChainHeader::try_construct(header2, accumulated_data).unwrap();
1✔
386
    txn.insert_chain_header(header2.clone());
1✔
387
    utxo_sum = &coinbase.commitment.to_commitment().unwrap() + &utxo_sum;
1✔
388
    kernel_sum = &kernel.excess.to_commitment().unwrap() + &kernel_sum;
1✔
389
    txn.insert_utxo(coinbase, *header2.hash(), 2, 0);
1✔
390
    mmr_position += 1;
1✔
391
    txn.insert_kernel(kernel, *header2.hash(), mmr_position);
1✔
392

1✔
393
    db.commit(txn).unwrap();
1✔
394

1✔
395
    validator
1✔
396
        .validate(
1✔
397
            &*db.db_read_access().unwrap(),
1✔
398
            2,
1✔
399
            &CompressedCommitment::from_commitment(utxo_sum),
1✔
400
            &CompressedCommitment::from_commitment(kernel_sum),
1✔
401
            &CompressedCommitment::from_commitment(burned_sum),
1✔
402
        )
1✔
403
        .unwrap_err();
1✔
404
}
1✔
405

406
#[tokio::test]
407
#[allow(clippy::too_many_lines)]
408
async fn chain_balance_validation_burned() {
1✔
409
    let factories = CryptoFactories::default();
1✔
410
    let consensus_manager = ConsensusManagerBuilder::new(Network::Esmeralda).build().unwrap();
1✔
411
    let genesis = consensus_manager.get_genesis_block();
1✔
412
    let pre_mine_value = 5000 * uT;
1✔
413
    let key_manager = create_memory_db_key_manager().unwrap();
1✔
414
    let (pre_mine_utxo, pre_mine_key_id, _) = create_utxo(
1✔
415
        pre_mine_value,
1✔
416
        &key_manager,
1✔
417
        &OutputFeatures::default(),
1✔
418
        &TariScript::default(),
1✔
419
        &Covenant::default(),
1✔
420
        MicroMinotari::zero(),
1✔
421
    )
1✔
422
    .await;
1✔
423
    let (pk, sig) = create_random_signature_from_secret_key(
1✔
424
        &key_manager,
1✔
425
        pre_mine_key_id,
1✔
426
        0.into(),
1✔
427
        0,
1✔
428
        KernelFeatures::empty(),
1✔
429
        TxoStage::Output,
1✔
430
    )
1✔
431
    .await;
1✔
432
    let excess = CompressedCommitment::from_compressed_key(pk);
1✔
433
    let kernel =
1✔
434
        TransactionKernel::new_current_version(KernelFeatures::empty(), MicroMinotari::from(0), 0, excess, sig, None);
1✔
435
    let mut gen_block = genesis.block().clone();
1✔
436
    gen_block.body.add_output(pre_mine_utxo);
1✔
437
    gen_block.body.add_kernels([kernel]);
1✔
438
    let mut utxo_sum = UncompressedCommitment::default();
1✔
439
    let mut kernel_sum = UncompressedCommitment::default();
1✔
440
    let mut burned_sum = UncompressedCommitment::default();
1✔
441
    let mock_store = MockTreeStore::new(true);
1✔
442
    let smt = JellyfishMerkleTree::<_, SmtHasher>::new(&mock_store);
1✔
443
    let mut batch = vec![];
1✔
444
    for output in gen_block.body.outputs() {
795✔
445
        utxo_sum = &output.commitment.to_commitment().unwrap() + &utxo_sum;
795✔
446
        if !output.is_burned() {
795✔
447
            let smt_key = KeyHash(output.commitment.as_bytes().try_into().expect("commitment is 32 bytes"));
795✔
448
            let smt_value = output.smt_hash(0);
795✔
449

795✔
450
            batch.push((smt_key, Some(smt_value.to_vec())));
795✔
451
        }
795✔
452
    }
1✔
453
    for input in gen_block.body.inputs() {
313✔
454
        utxo_sum = &utxo_sum - &input.commitment().unwrap().to_commitment().unwrap();
313✔
455
        let smt_key = KeyHash(
313✔
456
            input
313✔
457
                .commitment()
313✔
458
                .unwrap()
313✔
459
                .as_bytes()
313✔
460
                .try_into()
313✔
461
                .expect("Commitment is 32 bytes"),
313✔
462
        );
313✔
463
        batch.push((smt_key, None));
313✔
464
    }
313✔
465
    for kernel in gen_block.body.kernels() {
315✔
466
        kernel_sum = &kernel.excess.to_commitment().unwrap() + &kernel_sum;
315✔
467
    }
315✔
468
    let root = smt.put_value_set(batch, 0).unwrap();
1✔
469

1✔
470
    gen_block.header.output_mr = root.0 .0.into();
1✔
471
    let mut accum = genesis.accumulated_data().clone();
1✔
472
    accum.hash = gen_block.header.hash();
1✔
473
    let genesis = ChainBlock::try_construct(Arc::new(gen_block), accum).unwrap();
1✔
474
    let total_pre_mine = pre_mine_value + consensus_manager.consensus_constants(0).pre_mine_value();
1✔
475
    let constants = ConsensusConstantsBuilder::new(Network::LocalNet)
1✔
476
        .with_consensus_constants(consensus_manager.consensus_constants(0).clone())
1✔
477
        .with_pre_mine_value(total_pre_mine)
1✔
478
        .build();
1✔
479
    // Create a LocalNet consensus manager that uses rincewind consensus constants and has a custom rincewind genesis
1✔
480
    // block that contains an extra pre-mine utxo
1✔
481
    let consensus_manager = ConsensusManagerBuilder::new(Network::LocalNet)
1✔
482
        .with_block(genesis.clone())
1✔
483
        .add_consensus_constants(constants)
1✔
484
        .build()
1✔
485
        .unwrap();
1✔
486

1✔
487
    let db = create_store_with_consensus(consensus_manager.clone());
1✔
488

1✔
489
    let validator = ChainBalanceValidator::new(consensus_manager.clone(), factories.clone());
1✔
490
    // Validate the genesis state
1✔
491
    validator
1✔
492
        .validate(
1✔
493
            &*db.db_read_access().unwrap(),
1✔
494
            0,
1✔
495
            &CompressedCommitment::from_commitment(utxo_sum.clone()),
1✔
496
            &CompressedCommitment::from_commitment(kernel_sum.clone()),
1✔
497
            &CompressedCommitment::from_commitment(burned_sum.clone()),
1✔
498
        )
1✔
499
        .unwrap();
1✔
500

1✔
501
    //---------------------------------- Add block (coinbase + burned) --------------------------------------------//
1✔
502
    let mut txn = DbTransaction::new();
1✔
503
    let coinbase_value = consensus_manager.get_block_reward_at(1) - MicroMinotari::from(100);
1✔
504
    let (coinbase, coinbase_key_id, _) = create_utxo(
1✔
505
        coinbase_value,
1✔
506
        &key_manager,
1✔
507
        &OutputFeatures::create_coinbase(1, None, RangeProofType::RevealedValue),
1✔
508
        &TariScript::default(),
1✔
509
        &Covenant::default(),
1✔
510
        coinbase_value,
1✔
511
    )
1✔
512
    .await;
1✔
513
    let (pk, sig) = create_random_signature_from_secret_key(
1✔
514
        &key_manager,
1✔
515
        coinbase_key_id,
1✔
516
        0.into(),
1✔
517
        0,
1✔
518
        KernelFeatures::create_coinbase(),
1✔
519
        TxoStage::Output,
1✔
520
    )
1✔
521
    .await;
1✔
522
    let excess = CompressedCommitment::from_compressed_key(pk);
1✔
523
    let kernel = KernelBuilder::new()
1✔
524
        .with_signature(sig)
1✔
525
        .with_excess(&excess)
1✔
526
        .with_features(KernelFeatures::COINBASE_KERNEL)
1✔
527
        .build()
1✔
528
        .unwrap();
1✔
529

1✔
530
    let (burned, burned_key_id, _) = create_utxo(
1✔
531
        100.into(),
1✔
532
        &key_manager,
1✔
533
        &OutputFeatures::create_burn_output(),
1✔
534
        &TariScript::default(),
1✔
535
        &Covenant::default(),
1✔
536
        MicroMinotari::zero(),
1✔
537
    )
1✔
538
    .await;
1✔
539

1✔
540
    let (pk2, sig2) = create_random_signature_from_secret_key(
1✔
541
        &key_manager,
1✔
542
        burned_key_id,
1✔
543
        0.into(),
1✔
544
        0,
1✔
545
        KernelFeatures::create_burn(),
1✔
546
        TxoStage::Output,
1✔
547
    )
1✔
548
    .await;
1✔
549
    let excess2 = CompressedCommitment::from_compressed_key(pk2);
1✔
550
    let kernel2 = KernelBuilder::new()
1✔
551
        .with_signature(sig2)
1✔
552
        .with_excess(&excess2)
1✔
553
        .with_features(KernelFeatures::create_burn())
1✔
554
        .with_burn_commitment(Some(burned.commitment.clone()))
1✔
555
        .build()
1✔
556
        .unwrap();
1✔
557
    burned_sum = &burned_sum + &kernel2.get_burn_commitment().unwrap().to_commitment().unwrap();
1✔
558
    let mut header1 = BlockHeader::from_previous(genesis.header());
1✔
559
    header1.kernel_mmr_size += 2;
1✔
560
    header1.output_smt_size += 2;
1✔
561
    let achieved_difficulty = AchievedTargetDifficulty::try_construct(
1✔
562
        genesis.header().pow_algo(),
1✔
563
        genesis.accumulated_data().target_difficulty,
1✔
564
        genesis.accumulated_data().achieved_difficulty,
1✔
565
    )
1✔
566
    .unwrap();
1✔
567
    let accumulated_data = BlockHeaderAccumulatedData::builder(genesis.accumulated_data())
1✔
568
        .with_hash(header1.hash())
1✔
569
        .with_achieved_target_difficulty(achieved_difficulty)
1✔
570
        .with_total_kernel_offset(header1.total_kernel_offset.clone())
1✔
571
        .build()
1✔
572
        .unwrap();
1✔
573
    let header1 = ChainHeader::try_construct(header1, accumulated_data).unwrap();
1✔
574
    txn.insert_chain_header(header1.clone());
1✔
575

1✔
576
    let mut mmr_position = 4;
1✔
577

1✔
578
    txn.insert_kernel(kernel.clone(), *header1.hash(), mmr_position);
1✔
579
    txn.insert_utxo(coinbase.clone(), *header1.hash(), 1, 0);
1✔
580

1✔
581
    mmr_position = 5;
1✔
582

1✔
583
    txn.insert_kernel(kernel2.clone(), *header1.hash(), mmr_position);
1✔
584
    // txn.insert_pruned_utxo(burned.hash(), *header1.hash(), header1.height(),  0);
1✔
585

1✔
586
    db.commit(txn).unwrap();
1✔
587
    utxo_sum = &coinbase.commitment.to_commitment().unwrap() + &utxo_sum;
1✔
588
    kernel_sum = &(&kernel.excess.to_commitment().unwrap() + &kernel_sum) + &kernel2.excess.to_commitment().unwrap();
1✔
589
    validator
1✔
590
        .validate(
1✔
591
            &*db.db_read_access().unwrap(),
1✔
592
            1,
1✔
593
            &CompressedCommitment::from_commitment(utxo_sum),
1✔
594
            &CompressedCommitment::from_commitment(kernel_sum),
1✔
595
            &CompressedCommitment::from_commitment(burned_sum),
1✔
596
        )
1✔
597
        .unwrap();
1✔
598
}
1✔
599

600
mod transaction_validator {
601
    use std::convert::TryFrom;
602

603
    use super::*;
604
    use crate::{
605
        transactions::transaction_components::{CoinBaseExtra, OutputType, TransactionError},
606
        validation::transaction::TransactionInternalConsistencyValidator,
607
    };
608

609
    #[tokio::test]
610
    async fn it_rejects_coinbase_outputs() {
1✔
611
        let key_manager = create_memory_db_key_manager().unwrap();
1✔
612
        let consensus_manager = ConsensusManagerBuilder::new(Network::LocalNet).build().unwrap();
1✔
613
        let db = create_store_with_consensus(consensus_manager.clone());
1✔
614
        let factories = CryptoFactories::default();
1✔
615
        let validator = TransactionInternalConsistencyValidator::new(true, consensus_manager, factories);
1✔
616
        let features = OutputFeatures::create_coinbase(0, None, RangeProofType::BulletProofPlus);
1✔
617
        let tx = match tx!(MicroMinotari(100_000), fee: MicroMinotari(5), inputs: 1, outputs: 1, features: features, &key_manager)
1✔
618
        {
1✔
619
            Ok((tx, _, _)) => tx,
1✔
620
            Err(e) => panic!("Error found: {}", e),
1✔
621
        };
1✔
622
        let tip = db.get_chain_metadata().unwrap();
1✔
623
        let err = validator.validate_with_current_tip(&tx, tip).unwrap_err();
1✔
624
        unpack_enum!(
1✔
625
            ValidationError::OutputTypeNotPermitted {
1✔
626
                output_type: OutputType::Coinbase
1✔
627
            } = err
1✔
628
        );
1✔
629
    }
1✔
630

631
    #[tokio::test]
632
    async fn coinbase_extra_must_be_empty() {
1✔
633
        let key_manager = create_memory_db_key_manager().unwrap();
1✔
634
        let consensus_manager = ConsensusManagerBuilder::new(Network::LocalNet).build().unwrap();
1✔
635
        let db = create_store_with_consensus(consensus_manager.clone());
1✔
636
        let factories = CryptoFactories::default();
1✔
637
        let validator = TransactionInternalConsistencyValidator::new(true, consensus_manager, factories);
1✔
638
        let mut features = OutputFeatures { ..Default::default() };
1✔
639
        features.coinbase_extra = CoinBaseExtra::try_from(b"deadbeef".to_vec()).unwrap();
1✔
640
        let tx = match tx!(MicroMinotari(100_000), fee: MicroMinotari(5), inputs: 1, outputs: 1, features: features, &key_manager)
1✔
641
        {
1✔
642
            Ok((tx, _, _)) => tx,
1✔
643
            Err(e) => panic!("Error found: {}", e),
1✔
644
        };
1✔
645
        let tip = db.get_chain_metadata().unwrap();
1✔
646
        let err = validator.validate_with_current_tip(&tx, tip).unwrap_err();
1✔
647
        assert!(matches!(
1✔
648
            err,
1✔
649
            ValidationError::TransactionError(TransactionError::NonCoinbaseHasOutputFeaturesCoinbaseExtra)
1✔
650
        ));
1✔
651
    }
1✔
652
}
653

654
/// Iterator that emits BlockHeaders until a given height. This iterator loads headers in chunks of size `chunk_size`
655
/// for a low memory footprint. The chunk buffer is allocated once and reused.
656
pub struct HeaderIter<'a, B> {
657
    chunk: Vec<BlockHeader>,
658
    chunk_size: usize,
659
    cursor: usize,
660
    is_error: bool,
661
    height: u64,
662
    db: &'a BlockchainDatabase<B>,
663
}
664

665
impl<'a, B> HeaderIter<'a, B> {
666
    #[allow(dead_code)]
667
    pub fn new(db: &'a BlockchainDatabase<B>, height: u64, chunk_size: usize) -> Self {
3✔
668
        Self {
3✔
669
            db,
3✔
670
            chunk_size,
3✔
671
            cursor: 0,
3✔
672
            is_error: false,
3✔
673
            height,
3✔
674
            chunk: Vec::with_capacity(chunk_size),
3✔
675
        }
3✔
676
    }
3✔
677

678
    fn get_next_chunk(&self) -> (u64, u64) {
8✔
679
        #[allow(clippy::cast_possible_truncation)]
8✔
680
        let upper_bound = cmp::min(self.cursor + self.chunk_size, self.height as usize);
8✔
681
        (self.cursor as u64, upper_bound as u64)
8✔
682
    }
8✔
683
}
684

685
impl<B: BlockchainBackend> Iterator for HeaderIter<'_, B> {
686
    type Item = Result<BlockHeader, ChainStorageError>;
687

688
    fn next(&mut self) -> Option<Self::Item> {
17✔
689
        if self.is_error {
17✔
690
            return None;
×
691
        }
17✔
692

17✔
693
        if self.chunk.is_empty() {
17✔
694
            let (start, end) = self.get_next_chunk();
8✔
695
            // We're done: No more block headers to fetch
8✔
696
            if start > end {
8✔
697
                return None;
2✔
698
            }
6✔
699

6✔
700
            match self.db.fetch_headers(start..=end) {
6✔
701
                Ok(headers) => {
6✔
702
                    if headers.is_empty() {
6✔
703
                        return None;
1✔
704
                    }
5✔
705
                    self.cursor += headers.len();
5✔
706
                    self.chunk.extend(headers);
5✔
707
                },
708
                Err(err) => {
×
709
                    // On the next call, the iterator will end
×
710
                    self.is_error = true;
×
711
                    return Some(Err(err));
×
712
                },
713
            }
714
        }
9✔
715

716
        Some(Ok(self.chunk.remove(0)))
14✔
717
    }
17✔
718
}
STATUS · Troubleshooting · Open an Issue · Sales · Support · CAREERS · ENTERPRISE · START FREE · SCHEDULE DEMO
ANNOUNCEMENTS · TWITTER · TOS & SLA · Supported CI Services · What's a CI service? · Automated Testing

© 2026 Coveralls, Inc