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

tari-project / tari / 17033178607

18 Aug 2025 06:45AM UTC coverage: 54.49% (-0.007%) from 54.497%
17033178607

push

github

stringhandler
Merge branch 'development' of github.com:tari-project/tari into odev

971 of 2923 new or added lines in 369 files covered. (33.22%)

5804 existing lines in 173 files now uncovered.

76688 of 140739 relevant lines covered (54.49%)

193850.18 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
#![allow(clippy::indexing_slicing)]
24
use std::{cmp, sync::Arc};
25

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

1✔
582
    mmr_position = 5;
1✔
583

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

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

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

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

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

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

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

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

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

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

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

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

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

717
        Some(Ok(self.chunk.remove(0)))
14✔
718
    }
17✔
719
}
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