Skip to main content

ic_memory/ledger/
integrity.rs

1use super::{AllocationLedger, AllocationRecord, AllocationState, LedgerIntegrityError};
2use crate::{declaration::validate_runtime_fingerprint, key::StableKey, validation::Validate};
3use std::collections::BTreeSet;
4
5impl AllocationLedger {
6    /// Validate structural ledger invariants before recovery or commit.
7    pub fn validate_integrity(&self) -> Result<(), LedgerIntegrityError> {
8        let mut stable_keys = BTreeSet::new();
9        let mut slots = BTreeSet::new();
10
11        for record in self.allocation_history.records() {
12            if !stable_keys.insert(record.stable_key.clone()) {
13                return Err(LedgerIntegrityError::DuplicateStableKey {
14                    stable_key: record.stable_key.clone(),
15                });
16            }
17            if !slots.insert(record.slot.clone()) {
18                return Err(LedgerIntegrityError::DuplicateSlot {
19                    slot: Box::new(record.slot.clone()),
20                });
21            }
22            validate_record_integrity(self.current_generation, record)?;
23        }
24
25        let mut generations = BTreeSet::new();
26        for generation in self.allocation_history.generations() {
27            if !generations.insert(generation.generation) {
28                return Err(LedgerIntegrityError::DuplicateGeneration {
29                    generation: generation.generation,
30                });
31            }
32            if generation.generation > self.current_generation {
33                return Err(LedgerIntegrityError::FutureGeneration {
34                    generation: generation.generation,
35                    current_generation: self.current_generation,
36                });
37            }
38            if generation.parent_generation >= generation.generation {
39                return Err(LedgerIntegrityError::InvalidParentGeneration {
40                    generation: generation.generation,
41                    parent_generation: generation.parent_generation,
42                });
43            }
44        }
45
46        Ok(())
47    }
48
49    /// Validate strict committed-ledger invariants before recovery or commit.
50    ///
51    /// Public durable structs are DTOs: decoded or manually constructed values
52    /// are untrusted until this method succeeds.
53    pub fn validate_committed_integrity(&self) -> Result<(), LedgerIntegrityError> {
54        self.validate_integrity()?;
55
56        if self.current_generation != 0
57            && !self
58                .allocation_history
59                .generations()
60                .iter()
61                .any(|record| record.generation == self.current_generation)
62        {
63            return Err(LedgerIntegrityError::MissingCurrentGenerationRecord {
64                current_generation: self.current_generation,
65            });
66        }
67
68        let mut previous = None;
69        let mut known_generations = BTreeSet::new();
70        for generation in self.allocation_history.generations() {
71            validate_runtime_fingerprint(generation.runtime_fingerprint.as_deref())
72                .map_err(LedgerIntegrityError::DiagnosticMetadata)?;
73
74            let expected_generation = previous.map_or(1, |previous| previous + 1);
75            if generation.generation != expected_generation {
76                return Err(LedgerIntegrityError::NonIncreasingGenerationRecords {
77                    generation: generation.generation,
78                });
79            }
80
81            let expected_parent = previous.unwrap_or(0);
82            if generation.parent_generation != expected_parent {
83                return Err(LedgerIntegrityError::BrokenGenerationChain {
84                    generation: generation.generation,
85                    expected_parent,
86                    actual_parent: generation.parent_generation,
87                });
88            }
89
90            known_generations.insert(generation.generation);
91            previous = Some(generation.generation);
92        }
93
94        for record in self.allocation_history.records() {
95            validate_known_record_generation(
96                &known_generations,
97                &record.stable_key,
98                record.first_generation,
99            )?;
100            validate_known_record_generation(
101                &known_generations,
102                &record.stable_key,
103                record.last_seen_generation,
104            )?;
105            if let Some(retired_generation) = record.retired_generation {
106                validate_known_record_generation(
107                    &known_generations,
108                    &record.stable_key,
109                    retired_generation,
110                )?;
111            }
112            for schema in &record.schema_history {
113                validate_known_record_generation(
114                    &known_generations,
115                    &record.stable_key,
116                    schema.generation,
117                )?;
118            }
119        }
120
121        Ok(())
122    }
123}
124
125fn validate_record_integrity(
126    current_generation: u64,
127    record: &AllocationRecord,
128) -> Result<(), LedgerIntegrityError> {
129    record
130        .stable_key
131        .validate()
132        .map_err(LedgerIntegrityError::InvalidStableKey)?;
133    record
134        .slot
135        .validate()
136        .map_err(LedgerIntegrityError::InvalidSlotDescriptor)?;
137
138    if record.first_generation > record.last_seen_generation {
139        return Err(LedgerIntegrityError::InvalidRecordGenerationOrder {
140            stable_key: record.stable_key.clone(),
141            first_generation: record.first_generation,
142            last_seen_generation: record.last_seen_generation,
143        });
144    }
145    if record.last_seen_generation > current_generation {
146        return Err(LedgerIntegrityError::FutureRecordGeneration {
147            stable_key: record.stable_key.clone(),
148            generation: record.last_seen_generation,
149            current_generation,
150        });
151    }
152
153    match (record.state, record.retired_generation) {
154        (AllocationState::Retired, Some(retired_generation)) => {
155            if retired_generation < record.first_generation {
156                return Err(LedgerIntegrityError::RetiredBeforeFirstGeneration {
157                    stable_key: record.stable_key.clone(),
158                    first_generation: record.first_generation,
159                    retired_generation,
160                });
161            }
162            if retired_generation > current_generation {
163                return Err(LedgerIntegrityError::FutureRecordGeneration {
164                    stable_key: record.stable_key.clone(),
165                    generation: retired_generation,
166                    current_generation,
167                });
168            }
169        }
170        (AllocationState::Retired, None) => {
171            return Err(LedgerIntegrityError::MissingRetiredGeneration {
172                stable_key: record.stable_key.clone(),
173            });
174        }
175        (AllocationState::Reserved | AllocationState::Active, Some(_)) => {
176            return Err(LedgerIntegrityError::UnexpectedRetiredGeneration {
177                stable_key: record.stable_key.clone(),
178            });
179        }
180        (AllocationState::Reserved | AllocationState::Active, None) => {}
181    }
182
183    validate_schema_history_integrity(current_generation, record)
184}
185
186fn validate_known_record_generation(
187    known_generations: &BTreeSet<u64>,
188    stable_key: &StableKey,
189    generation: u64,
190) -> Result<(), LedgerIntegrityError> {
191    if known_generations.contains(&generation) {
192        return Ok(());
193    }
194    Err(LedgerIntegrityError::UnknownRecordGeneration {
195        stable_key: stable_key.clone(),
196        generation,
197    })
198}
199
200fn validate_schema_history_integrity(
201    current_generation: u64,
202    record: &AllocationRecord,
203) -> Result<(), LedgerIntegrityError> {
204    if record.schema_history.is_empty() {
205        return Err(LedgerIntegrityError::EmptySchemaHistory {
206            stable_key: record.stable_key.clone(),
207        });
208    }
209
210    let mut previous = None;
211    for schema in &record.schema_history {
212        schema
213            .schema
214            .validate()
215            .map_err(|error| LedgerIntegrityError::InvalidSchemaMetadata {
216                stable_key: record.stable_key.clone(),
217                generation: schema.generation,
218                error,
219            })?;
220        if previous.is_some_and(|generation| schema.generation <= generation) {
221            return Err(LedgerIntegrityError::NonIncreasingSchemaHistory {
222                stable_key: record.stable_key.clone(),
223            });
224        }
225        if schema.generation < record.first_generation || schema.generation > current_generation {
226            return Err(LedgerIntegrityError::SchemaHistoryOutOfBounds {
227                stable_key: record.stable_key.clone(),
228                generation: schema.generation,
229            });
230        }
231        previous = Some(schema.generation);
232    }
233
234    Ok(())
235}