1use crate::types::{EdgeIdx, Generation, NodeId};
4
5#[derive(Debug, thiserror::Error)]
8pub enum M1ndError {
9 #[error("dangling edge: edge {edge:?} references non-existent node {node:?}")]
12 DanglingEdge { edge: EdgeIdx, node: NodeId },
13
14 #[error("graph generation mismatch: expected {expected:?}, actual {actual:?}")]
16 GraphGenerationMismatch {
17 expected: Generation,
18 actual: Generation,
19 },
20
21 #[error("duplicate node: interned ID {0:?}")]
23 DuplicateNode(NodeId),
24
25 #[error("graph not finalised: call Graph::finalize() before queries")]
27 GraphNotFinalized,
28
29 #[error("graph is empty")]
31 EmptyGraph,
32
33 #[error("non-finite value at firewall: node={node:?}, value={value}")]
36 NonFiniteActivation { node: NodeId, value: f32 },
37
38 #[error("parameter out of range: {name} = {value} (expected {range})")]
40 ParameterOutOfRange {
41 name: &'static str,
42 value: f64,
43 range: &'static str,
44 },
45
46 #[error("non-positive resonance parameter: {name} = {value}")]
48 NonPositiveResonanceParam { name: &'static str, value: f32 },
49
50 #[error("pulse budget exhausted: {budget} pulses processed")]
53 PulseBudgetExhausted { budget: u64 },
54
55 #[error("chain budget exhausted: {budget} chains generated")]
57 ChainBudgetExhausted { budget: u64 },
58
59 #[error("matrix entry budget exhausted: {budget} entries")]
61 MatrixBudgetExhausted { budget: u64 },
62
63 #[error("ingestion timeout after {elapsed_s:.1}s")]
65 IngestionTimeout { elapsed_s: f64 },
66
67 #[error("ingestion node budget exhausted: {budget} nodes")]
69 IngestionNodeBudget { budget: u64 },
70
71 #[error("fingerprint pair budget exhausted: {budget} pairs")]
73 FingerprintPairBudget { budget: u64 },
74
75 #[error("XLR over-cancellation: all signal cancelled")]
78 XlrOverCancellation,
79
80 #[error("Louvain non-convergence after {passes} passes")]
82 LouvainNonConvergence { passes: u32 },
83
84 #[error("spectral analysis: power iteration divergence suspected")]
86 SpectralDivergence,
87
88 #[error("resonance normalization: max amplitude is zero")]
90 ResonanceZeroAmplitude,
91
92 #[error("CAS retry limit ({limit}) exceeded at edge {edge:?}")]
94 CasRetryExhausted { edge: EdgeIdx, limit: u32 },
95
96 #[error("encoding detection failed for {path} (confidence={confidence:.2})")]
99 EncodingDetectionFailed { path: String, confidence: f32 },
100
101 #[error("binary file skipped: {path}")]
103 BinaryFileSkipped { path: String },
104
105 #[error("label collision: {label} maps to {count} nodes")]
107 LabelCollision { label: String, count: usize },
108
109 #[error("corrupt persistence state: {reason}")]
112 CorruptState { reason: String },
113
114 #[error("schema drift on import: {reason}")]
116 SchemaDrift { reason: String },
117
118 #[error("counterfactual seed overlap: seed {node:?} is in the removal set")]
121 CounterfactualSeedOverlap { node: NodeId },
122
123 #[error("unknown tool: {name}")]
126 UnknownTool { name: String },
127
128 #[error("invalid params for {tool}: {detail}")]
130 InvalidParams { tool: String, detail: String },
131
132 #[error("perspective not found: {perspective_id} for agent {agent_id}")]
134 PerspectiveNotFound {
135 perspective_id: String,
136 agent_id: String,
137 },
138
139 #[error(
141 "perspective stale: {perspective_id} expected gen {expected_gen}, actual {actual_gen}"
142 )]
143 PerspectiveStale {
144 perspective_id: String,
145 expected_gen: u64,
146 actual_gen: u64,
147 },
148
149 #[error("perspective limit exceeded for agent {agent_id}: {current}/{limit}")]
151 PerspectiveLimitExceeded {
152 agent_id: String,
153 current: usize,
154 limit: usize,
155 },
156
157 #[error("route set stale: version {route_set_version}, current {current_version}")]
159 RouteSetStale {
160 route_set_version: u64,
161 current_version: u64,
162 },
163
164 #[error("route not found: {route_id} in perspective {perspective_id}")]
166 RouteNotFound {
167 route_id: String,
168 perspective_id: String,
169 },
170
171 #[error("navigation at root: perspective {perspective_id}")]
173 NavigationAtRoot { perspective_id: String },
174
175 #[error("branch depth exceeded in {perspective_id}: depth {depth}/{limit}")]
177 BranchDepthExceeded {
178 perspective_id: String,
179 depth: usize,
180 limit: usize,
181 },
182
183 #[error("lock not found: {lock_id}")]
185 LockNotFound { lock_id: String },
186
187 #[error("lock ownership violation: {lock_id} owned by {owner}, called by {caller}")]
189 LockOwnership {
190 lock_id: String,
191 owner: String,
192 caller: String,
193 },
194
195 #[error("lock scope too large: {node_count} nodes exceeds cap of {cap}")]
197 LockScopeTooLarge { node_count: usize, cap: usize },
198
199 #[error("lock limit exceeded for agent {agent_id}: {current}/{limit}")]
201 LockLimitExceeded {
202 agent_id: String,
203 current: usize,
204 limit: usize,
205 },
206
207 #[error("watch strategy not supported: {strategy}")]
209 WatchStrategyNotSupported { strategy: String },
210
211 #[error("affinity timeout: {elapsed_ms:.1}ms exceeded budget of {budget_ms:.1}ms")]
213 AffinityTimeout { elapsed_ms: f64, budget_ms: f64 },
214
215 #[error("pattern too broad: specificity {specificity:.2} below minimum {minimum:.2}")]
218 PatternTooBroad { specificity: f32, minimum: f32 },
219
220 #[error("antibody not found: {id}")]
222 AntibodyNotFound { id: String },
223
224 #[error("antibody limit exceeded: {current}/{limit}")]
226 AntibodyLimitExceeded { current: usize, limit: usize },
227
228 #[error("epidemic burnout: {infected_pct:.1}% infected in {iteration} iterations")]
231 EpidemicBurnout { infected_pct: f32, iteration: u32 },
232
233 #[error("no valid infected nodes")]
235 NoValidInfectedNodes,
236
237 #[error("no entry points found for flow simulation")]
240 NoEntryPoints,
241
242 #[error("layer not found: level {level}")]
245 LayerNotFound { level: u8 },
246
247 #[error("ingest error: {0}")]
250 IngestError(String),
251
252 #[error("I/O error: {0}")]
254 Io(#[from] std::io::Error),
255
256 #[error("serialization error: {0}")]
257 Serde(#[from] serde_json::Error),
258
259 #[error("persistence failed: {0}")]
260 PersistenceFailed(String),
261}
262
263pub type M1ndResult<T> = Result<T, M1ndError>;