forked from kcl-lang/kcl
-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathoverride.rs
414 lines (395 loc) · 15.8 KB
/
override.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
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
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
use std::collections::HashSet;
use anyhow::{anyhow, Result};
use kclvm_ast::config::try_get_config_expr_mut;
use kclvm_ast::path::{get_attr_paths_from_config_expr, get_key_path};
use kclvm_ast::walker::MutSelfMutWalker;
use kclvm_ast::{ast, walk_if_mut};
use kclvm_parser::parse_expr;
use kclvm_sema::pre_process::{fix_config_expr_nest_attr, transform_multi_assign};
use crate::printer::print_ast_module;
use super::util::{invalid_spec_error, split_field_path};
/// Import statement column offset always start with 1.
/// todo: The (1-based) column offset needs to be constrained by specifications.
const IMPORT_STMT_COLUMN_OFFSET: u64 = 1;
/// Apply overrides on the AST program with the override specifications.
///
/// Please note that this a low level internal API used by compiler itself,
/// The parameters of the method are all compiler internal concepts such as
/// AST, etc.
///
/// # Examples
///
/// ```no_check
/// use kclvm_parser::load_program;
/// use kclvm_tools::query::r#override::apply_overrides;
///
/// let mut prog = load_program(&["config.k"], None).unwrap();
/// let overrides = vec![parse_override_spec("config.id=1").unwrap()];
/// let import_paths = vec!["path.to.pkg".to_string()];
/// let result = apply_overrides(&mut prog, &overrides, &import_paths, true).unwrap();
/// ```
pub fn apply_overrides(
prog: &mut ast::Program,
overrides: &[ast::OverrideSpec],
import_paths: &[String],
print_ast: bool,
) -> Result<()> {
for o in overrides {
let pkgpath = if o.pkgpath.is_empty() {
&prog.main
} else {
&o.pkgpath
};
if let Some(modules) = prog.pkgs.get_mut(pkgpath) {
for m in modules.iter_mut() {
if apply_override_on_module(m, o, import_paths)? && print_ast {
let code_str = print_ast_module(m);
std::fs::write(&m.filename, &code_str)?
}
}
}
}
Ok(())
}
/// Build a expression from string.
fn build_expr_from_string(value: &str) -> Option<ast::NodeRef<ast::Expr>> {
let expr: Option<ast::NodeRef<ast::Expr>> = parse_expr(value);
match &expr {
Some(e) => match &e.node {
// fix attr=value to attr="value"
ast::Expr::Identifier(_) | ast::Expr::Unary(_) | ast::Expr::Binary(_) => {
Some(ast::NodeRef::new(ast::Node::node_with_pos(
ast::Expr::StringLit(ast::StringLit {
is_long_string: false,
raw_value: format!("{:?}", value),
value: value.to_string(),
}),
e.pos(),
)))
}
_ => expr,
},
None => None,
}
}
/// Apply overrides on the AST module with the override specifications.
///
/// Please note that this a low level internal API used by compiler itself,
/// The parameters of the method are all compiler internal concepts such as
/// AST, etc.
///
/// # Examples
///
/// ```no_check
/// use kclvm_parser::parse_file;
/// use kclvm_tools::query::apply_override_on_module;
///
/// let mut module = parse_file("", None).unwrap();
/// let override_spec = parse_override_spec("config.id=1").unwrap();
/// let import_paths = vec!["path.to.pkg".to_string()];
/// let result = apply_override_on_module(&mut module, override_spec, &import_paths).unwrap();
/// ```
pub fn apply_override_on_module(
m: &mut ast::Module,
o: &ast::OverrideSpec,
import_paths: &[String],
) -> Result<bool> {
// Apply import paths on AST module.
apply_import_paths_on_module(m, import_paths)?;
let ss = o.field_path.split('.').collect::<Vec<&str>>();
if ss.len() <= 1 {
return Ok(false);
}
let target_id = ss[0];
let field = ss[1..].join(".");
let value = &o.field_value;
let key = ast::Identifier {
names: field.split('.').map(|s| s.to_string()).collect(),
ctx: ast::ExprContext::Store,
pkgpath: "".to_string(),
};
// Transform config expr to simplify the config path query and override.
fix_config_expr_nest_attr(m);
// When there is a multi-target assignment statement of the form `a = b = Config {}`,
// it needs to be transformed into the following form first to prevent the configuration
// from being incorrectly modified.
// ```kcl
// a = Config {}
// b = Config {}
// ```
transform_multi_assign(m);
let mut transformer = OverrideTransformer {
target_id: target_id.to_string(),
field_path: field,
override_key: key,
override_value: build_expr_from_string(value),
override_target_count: 0,
has_override: false,
action: o.action.clone(),
};
transformer.walk_module(m);
Ok(transformer.has_override)
}
/// Parse override spec string to override structure.
///
/// parse_override_spec("alice.age=10") -> ast::OverrideSpec {
/// pkgpath: "".to_string(),
/// field_path: "alice.age".to_string(),
/// field_value: "10".to_string(),
/// action: ast::OverrideAction::CreateOrUpdate,
/// }
pub(crate) fn parse_override_spec(spec: &str) -> Result<ast::OverrideSpec> {
if spec.contains('=') {
// Create or update the override value.
let split_values = spec.splitn(2, '=').collect::<Vec<&str>>();
let path = split_values
.get(0)
.ok_or_else(|| invalid_spec_error(spec))?;
let field_value = split_values
.get(1)
.ok_or_else(|| invalid_spec_error(spec))?;
let (pkgpath, field_path) = split_field_path(path)?;
Ok(ast::OverrideSpec {
pkgpath,
field_path,
field_value: field_value.to_string(),
action: ast::OverrideAction::CreateOrUpdate,
})
} else if let Some(stripped_spec) = spec.strip_suffix('-') {
// Delete the override value.
let (pkgpath, field_path) = split_field_path(stripped_spec)?;
Ok(ast::OverrideSpec {
pkgpath,
field_path,
field_value: "".to_string(),
action: ast::OverrideAction::Delete,
})
} else {
Err(invalid_spec_error(spec))
}
}
// Transform the AST module with the import path list.
fn apply_import_paths_on_module(m: &mut ast::Module, import_paths: &[String]) -> Result<()> {
if import_paths.is_empty() {
return Ok(());
}
let mut exist_import_set: HashSet<String> = HashSet::new();
for stmt in &m.body {
if let ast::Stmt::Import(import_stmt) = &stmt.node {
if let Some(asname) = &import_stmt.asname {
exist_import_set.insert(format!("{} as {}", import_stmt.path, asname));
} else {
exist_import_set.insert(import_stmt.path.to_string());
}
}
}
for (i, path) in import_paths.iter().enumerate() {
let line: u64 = i as u64 + 1;
if exist_import_set.contains(path) {
continue;
}
let name = path
.split('.')
.last()
.ok_or_else(|| anyhow!("Invalid import path {}", path))?;
let import_node = ast::ImportStmt {
path: path.to_string(),
rawpath: "".to_string(),
name: name.to_string(),
asname: None,
};
let import_stmt = Box::new(ast::Node::new(
ast::Stmt::Import(import_node),
m.filename.clone(),
line,
IMPORT_STMT_COLUMN_OFFSET,
line,
// i denotes the space len between the `import` keyword and the path.
("import".len() + path.len() + 1) as u64,
));
m.body.insert((line - 1) as usize, import_stmt)
}
Ok(())
}
/// OverrideTransformer is used to walk AST and transform it with the override values.
struct OverrideTransformer {
pub target_id: String,
pub field_path: String,
pub override_key: ast::Identifier,
pub override_value: Option<ast::NodeRef<ast::Expr>>,
pub override_target_count: usize,
pub has_override: bool,
pub action: ast::OverrideAction,
}
impl<'ctx> MutSelfMutWalker<'ctx> for OverrideTransformer {
fn walk_unification_stmt(&mut self, unification_stmt: &'ctx mut ast::UnificationStmt) {
let name = match unification_stmt.target.node.names.get(0) {
Some(name) => name,
None => bug!(
"Invalid AST unification target names {:?}",
unification_stmt.target.node.names
),
};
if name != &self.target_id {
return;
}
self.override_target_count = 1;
self.has_override = true;
self.walk_schema_expr(&mut unification_stmt.value.node);
}
fn walk_assign_stmt(&mut self, assign_stmt: &'ctx mut ast::AssignStmt) {
if let ast::Expr::Schema(_) = &assign_stmt.value.node {
self.override_target_count = 0;
for target in &assign_stmt.targets {
if target.node.names.len() != 1 {
continue;
}
if target.node.names[0] != self.target_id {
continue;
}
self.override_target_count += 1;
}
if self.override_target_count == 0 {
return;
}
self.has_override = true;
self.walk_expr(&mut assign_stmt.value.node);
}
}
fn walk_schema_expr(&mut self, schema_expr: &'ctx mut ast::SchemaExpr) {
if self.override_target_count == 0 {
return;
}
if let ast::Expr::Config(config_expr) = &mut schema_expr.config.node {
if !self.lookup_config_and_replace(config_expr) {
// Not exist and append an override value when the action is CREATE_OR_UPDATE
if let ast::OverrideAction::CreateOrUpdate = self.action {
if let ast::Expr::Config(config_expr) = &mut schema_expr.config.node {
config_expr
.items
.push(Box::new(ast::Node::dummy_node(ast::ConfigEntry {
key: Some(Box::new(ast::Node::dummy_node(ast::Expr::Identifier(
self.override_key.clone(),
)))),
value: self.clone_override_value(),
operation: ast::ConfigEntryOperation::Override,
insert_index: -1,
})));
}
}
}
}
self.override_target_count = 0;
}
fn walk_config_expr(&mut self, config_expr: &'ctx mut ast::ConfigExpr) {
for config_entry in config_expr.items.iter_mut() {
walk_if_mut!(self, walk_expr, config_entry.node.key);
self.walk_expr(&mut config_entry.node.value.node);
}
}
fn walk_if_stmt(&mut self, _: &'ctx mut ast::IfStmt) {
// Do not override AssignStmt in IfStmt
}
fn walk_schema_stmt(&mut self, _: &'ctx mut ast::SchemaStmt) {
// Do not override AssignStmt in SchemaStmt
}
fn walk_lambda_expr(&mut self, _: &'ctx mut ast::LambdaExpr) {
// Do not override AssignStmt in LambdaExpr
}
}
impl OverrideTransformer {
/// Lookup schema config all fields and replace if it is matched with the override spec,
/// return whether is found a replaced one.
fn lookup_config_and_replace(&mut self, config_expr: &mut ast::ConfigExpr) -> bool {
// Get all entry paths from a config expression.
let paths = get_attr_paths_from_config_expr(config_expr);
// Query whether there is a matching path from the path lookup table.
match paths.iter().position(|r| r == &self.field_path) {
Some(pos) => {
let path = &paths[pos];
// Split a path into multiple parts. `a.b.c` -> ["a", "b", "c"]
let parts = path.split('.').collect::<Vec<&str>>();
self.replace_config_with_path_parts(config_expr, &parts);
true
}
None => false,
}
}
/// Replace AST config expr with one part of path. The implementation of this function
/// uses recursive matching to find the config entry need to be modified.
fn replace_config_with_path_parts(
&mut self,
config_expr: &mut ast::ConfigExpr,
parts: &[&str],
) {
// Do not replace empty path parts and out of index parts on the config expression.
if parts.is_empty() {
return;
}
// Always take the first part to match, because recursive search is required.
let part = parts[0];
let mut delete_index_set = HashSet::new();
// Loop all entries in the config expression and replace, because there may be duplicate
// configuration items in config.
for (i, item) in config_expr.items.iter_mut().enumerate() {
// Compare each field of the config structure one by one.
// - `part` denotes the path entered by the user to be modified.
// - `get_path_key` returns the real config key name.
// For example, the real config node is `a: {b: c: {}}`. The path
// that needs to be modified is `a.b.c`, and its parts are ["a", "b", "c"].
if part == get_key_path(&item.node.key) {
// When the last part of the path is successfully recursively matched,
// it indicates that the original value that needs to be overwritten
// is successfully found, and the new value is used to overwrite it.
// - `parts.len() == 1` denotes the path matches exactly.
if parts.len() == 1 {
match self.action {
ast::OverrideAction::CreateOrUpdate => {
let mut value = self.clone_override_value();
// Use position information that needs to override the expression.
value.set_pos(item.pos());
// Override the node value.
item.node.value = value;
}
ast::OverrideAction::Delete => {
// Store the config entry delete index into the delete index set.
// Because we can't delete the entry directly in the loop
delete_index_set.insert(i);
}
}
}
// Replace value recursively using the path composed by subsequent parts.
//
// The reason for using recursion instead of looping for path matching
// is that rust cannot directly hold shared references to AST nodes
// (ast::NodeRef<T> is a Box<T>), so recursive search is performed
// directly on AST nodes.
else if let Some(config_expr) = try_get_config_expr_mut(&mut item.node.value.node)
{
self.replace_config_with_path_parts(config_expr, &parts[1..]);
}
}
}
// Delete entries according delete index set.
if !delete_index_set.is_empty() {
let items: Vec<(usize, &ast::NodeRef<ast::ConfigEntry>)> = config_expr
.items
.iter()
.enumerate()
.filter(|(i, _)| !delete_index_set.contains(i))
.collect();
config_expr.items = items
.iter()
.map(|(_, item)| <&ast::NodeRef<ast::ConfigEntry>>::clone(item).clone())
.collect();
}
}
/// Clone a override value
#[inline]
fn clone_override_value(&mut self) -> ast::NodeRef<ast::Expr> {
match &self.override_value {
Some(v) => v.clone(),
None => bug!("Override value is None"),
}
}
}