leo_passes/static_analysis/await_checker.rs
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
// Copyright (C) 2019-2025 Provable Inc.
// This file is part of the Leo library.
// The Leo library is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
// The Leo library is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
// You should have received a copy of the GNU General Public License
// along with the Leo library. If not, see <https://www.gnu.org/licenses/>.
use crate::ConditionalTreeNode;
use indexmap::IndexSet;
use leo_ast::Identifier;
use leo_errors::StaticAnalyzerWarning;
use leo_span::{Span, Symbol};
// TODO: Could optimize by removing duplicate paths (if set of futures is the same).
pub struct AwaitChecker {
/// All possible subsets of futures that must be awaited.
pub(crate) to_await: Vec<ConditionalTreeNode>,
/// Statically updated set of futures to await.
pub(crate) static_to_await: IndexSet<Symbol>,
/// Whether or not to do full tree search for await checking.
pub(crate) enabled: bool,
/// Maximum nesting depth to search for await checking.
pub(crate) max_depth: usize,
}
impl AwaitChecker {
/// Initializes a new `AwaitChecker`.
pub fn new(max_depth: usize, enabled: bool) -> Self {
Self { to_await: Vec::new(), static_to_await: IndexSet::new(), enabled, max_depth }
}
/// Remove from list.
/// Returns `true` if there was a path where the future was not awaited in the order of the input list.
pub fn remove(&mut self, id: &Identifier) -> bool {
// Can assume in finalize block.
let is_not_first = if self.enabled {
// Remove from dynamic list.
self.to_await.iter_mut().any(|node| node.remove_element(&id.name))
} else {
false
};
// Remove from static list.
self.static_to_await.shift_remove(&id.name);
is_not_first
}
/// Initialize futures.
pub fn set_futures(&mut self, futures: IndexSet<Symbol>) {
if futures.is_empty() {
self.to_await = Vec::new();
} else {
self.to_await = vec![ConditionalTreeNode::new(futures.clone())];
}
self.static_to_await = futures;
}
/// Enter scope for `then` branch of conditional.
pub fn create_then_scope(
&mut self,
is_finalize: bool,
input: Span,
) -> Result<Vec<ConditionalTreeNode>, StaticAnalyzerWarning> {
if is_finalize && self.enabled {
let mut current_nodes = Vec::new();
// Extend all paths by one node to represent the upcoming `then` branch.
for node in self.to_await.iter() {
// Error if exceed maximum depth.
if node.depth > self.max_depth {
return Err(StaticAnalyzerWarning::max_conditional_block_depth_exceeded(self.max_depth, input));
}
// Extend current path.
current_nodes.push(node.clone().create_child());
}
// Update the set of nodes to be current set.
self.to_await = current_nodes.clone();
Ok(current_nodes)
} else {
Ok(Vec::new())
}
}
/// Exit scope for `then` branch of conditional.
pub fn exit_then_scope(
&mut self,
is_finalize: bool,
parent_nodes: Vec<ConditionalTreeNode>,
) -> Vec<ConditionalTreeNode> {
// Check if a nested conditional statement signaled their existence.
if is_finalize && self.enabled { core::mem::replace(&mut self.to_await, parent_nodes) } else { Vec::new() }
}
/// Exit scope for conditional statement at current depth.
pub fn exit_statement_scope(&mut self, is_finalize: bool, then_nodes: Vec<ConditionalTreeNode>) {
if is_finalize && self.enabled {
// Merge together the current set of nodes (from `otherwise` branch) with `then` nodes.
self.to_await.extend(then_nodes);
}
}
}