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 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140
/*
This file is part of Donet.
Copyright © 2024 Max Rodriguez
Donet is free software; you can redistribute it and/or modify
it under the terms of the GNU Affero General Public License,
as published by the Free Software Foundation, either version 3
of the License, or (at your option) any later version.
Donet 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 Affero General Public License for more details.
You should have received a copy of the GNU Affero General Public
License along with Donet. If not, see <https://www.gnu.org/licenses/>.
*/
//! The DC parser outputs an [`Abstract Syntax Tree`], which is just a big
//! nested structure that defines the declarations in the DC file. At runtime,
//! the Donet daemon (and its services) need a class hierarchy structure in
//! memory to access while processing network messages.
//!
//! This source file defines the process of taking in the DC file abstract
//! syntax tree as input and generating an output of a class hierarchy structure,
//! where each class has pointers to its children, and vice versa, with methods
//! that make it easy for the Donet daemon to look up information on the DC contract
//! at runtime in order to understand the network messages it receives.
//!
//! [`Abstract Syntax Tree`]: https://en.wikipedia.org/wiki/Abstract_syntax_tree
use super::ast;
use super::error::DCReadError;
use super::PipelineData;
use crate::dcfile;
use crate::dconfig::*;
use anyhow::Result;
/// Takes in the [`Abstract Syntax Trees`] from the last stage of the pipeline
/// and outputs a [`crate::dcfile::DCFile`] immutable structure.
///
/// [`Abstract Syntax Tree`]: https://en.wikipedia.org/wiki/Abstract_syntax_tree
pub fn semantic_analyzer<'a>(pipeline: &mut PipelineData) -> Result<dcfile::DCFile<'a>, DCReadError> {
// tell the pipeline we are moving onto the next stage
pipeline.next_stage();
// create a new interim DC file struct from our pipeline's dc parser configuration
let mut dc_file = dcfile::interim::DCFile::from(pipeline.get_dc_config().clone());
// Iterate through all ASTs and add them to our DCFile intermediate object.
for ast in pipeline.syntax_trees.clone() {
for type_declaration in ast.type_declarations {
match type_declaration {
ast::TypeDeclaration::PythonImport(import) => {
dc_file.add_python_import(pipeline, import.clone());
}
ast::TypeDeclaration::KeywordType(keyword) => {
dc_file.add_keyword(pipeline, keyword);
}
ast::TypeDeclaration::StructType(_) => {}
ast::TypeDeclaration::DClassType(_) => {}
ast::TypeDeclaration::TypedefType(_) => {}
// Ignore is returned by productions that parsed certain
// grammar that may be deprecated but ignored for
// compatibility & should not be added to the DC file.
ast::TypeDeclaration::Ignore => {}
}
}
pipeline.next_file(); // tell the pipeline we are processing the next file
}
if pipeline.failing() {
Err(DCReadError::Semantic)
} else {
// Convert intermediate DC file structure to final immutable DC file structure.
Ok(dc_file.into())
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::read_dc;
use dcfile::DCPythonImport;
#[test]
fn python_imports() {
let dc_config = DCFileConfig::default();
let dc_string: &str = "
from views import *
from views import DistributedDonut
from views import Class/AI/OV
";
let dcf: dcfile::DCFile = read_dc(dc_config, dc_string.into()).expect("Failed to parse syntax.");
let num_imports: usize = dcf.get_num_imports();
assert_eq!(num_imports, 3);
let symbols: Vec<Vec<String>> = vec![
vec!["*".into()],
vec!["DistributedDonut".into()],
vec!["Class".into(), "ClassAI".into(), "ClassOV".into()],
];
for index in 0..num_imports - 1 {
let import: &DCPythonImport = dcf.get_python_import(index);
assert_eq!(import.module, "views");
let target_symbols: &Vec<String> = symbols.get(index).unwrap();
assert_eq!(*target_symbols, import.symbols);
}
}
#[test]
#[should_panic]
fn redundant_view_suffix() {
let dc_config = DCFileConfig::default();
let dc_string: &str = "
from views import Class/AI/OV/OV
";
let _ = read_dc(dc_config, dc_string.into()).expect("Should fail.");
}
#[test]
#[should_panic]
fn keyword_already_defined() {
let dc_config = DCFileConfig::default();
let dc_string: &str = "
keyword abcdef;
keyword abcdef;
";
let _ = read_dc(dc_config, dc_string.into()).expect("Should fail.");
}
}