snail-lang 0.5.3__tar.gz → 0.6.0__tar.gz

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (56) hide show
  1. {snail_lang-0.5.3 → snail_lang-0.6.0}/Cargo.lock +7 -7
  2. {snail_lang-0.5.3 → snail_lang-0.6.0}/PKG-INFO +14 -4
  3. {snail_lang-0.5.3 → snail_lang-0.6.0}/README.md +14 -4
  4. {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-ast/Cargo.toml +1 -1
  5. {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-ast/src/awk.rs +1 -0
  6. {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-core/Cargo.toml +1 -1
  7. {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-core/src/lib.rs +5 -0
  8. {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-error/Cargo.toml +1 -1
  9. {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-lower/Cargo.toml +1 -1
  10. {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-lower/README.md +1 -1
  11. {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-lower/src/constants.rs +16 -0
  12. {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-lower/src/lib.rs +2 -0
  13. snail_lang-0.6.0/crates/snail-lower/src/map.rs +235 -0
  14. {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-parser/Cargo.toml +1 -1
  15. {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-parser/src/lib.rs +386 -1
  16. {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-parser/src/snail.pest +2 -2
  17. {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-parser/src/string.rs +17 -0
  18. {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-parser/tests/errors.rs +35 -0
  19. {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-python/Cargo.toml +1 -1
  20. {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-python/src/lib.rs +6 -2
  21. {snail_lang-0.5.3 → snail_lang-0.6.0}/pyproject.toml +1 -1
  22. {snail_lang-0.5.3 → snail_lang-0.6.0}/python/snail/cli.py +76 -12
  23. {snail_lang-0.5.3 → snail_lang-0.6.0}/python/snail/runtime/__init__.py +11 -0
  24. snail_lang-0.6.0/python/snail/runtime/lazy_text.py +50 -0
  25. {snail_lang-0.5.3 → snail_lang-0.6.0}/Cargo.toml +0 -0
  26. {snail_lang-0.5.3 → snail_lang-0.6.0}/LICENSE +0 -0
  27. {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-ast/README.md +0 -0
  28. {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-ast/src/ast.rs +0 -0
  29. {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-ast/src/lib.rs +0 -0
  30. {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-core/README.md +0 -0
  31. {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-error/README.md +0 -0
  32. {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-error/src/lib.rs +0 -0
  33. {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-lower/src/awk.rs +0 -0
  34. {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-lower/src/expr.rs +0 -0
  35. {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-lower/src/helpers.rs +0 -0
  36. {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-lower/src/operators.rs +0 -0
  37. {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-lower/src/program.rs +0 -0
  38. {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-lower/src/py_ast.rs +0 -0
  39. {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-lower/src/stmt.rs +0 -0
  40. {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-parser/README.md +0 -0
  41. {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-parser/src/awk.rs +0 -0
  42. {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-parser/src/expr.rs +0 -0
  43. {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-parser/src/literal.rs +0 -0
  44. {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-parser/src/stmt.rs +0 -0
  45. {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-parser/src/util.rs +0 -0
  46. {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-parser/tests/common.rs +0 -0
  47. {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-parser/tests/parser.rs +0 -0
  48. {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-parser/tests/statements.rs +0 -0
  49. {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-parser/tests/syntax_expressions.rs +0 -0
  50. {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-parser/tests/syntax_strings.rs +0 -0
  51. {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-python/build.rs +0 -0
  52. {snail_lang-0.5.3 → snail_lang-0.6.0}/python/snail/__init__.py +0 -0
  53. {snail_lang-0.5.3 → snail_lang-0.6.0}/python/snail/runtime/compact_try.py +0 -0
  54. {snail_lang-0.5.3 → snail_lang-0.6.0}/python/snail/runtime/regex.py +0 -0
  55. {snail_lang-0.5.3 → snail_lang-0.6.0}/python/snail/runtime/structured_accessor.py +0 -0
  56. {snail_lang-0.5.3 → snail_lang-0.6.0}/python/snail/runtime/subprocess.py +0 -0
@@ -485,11 +485,11 @@ checksum = "67b1b7a3b5fe4f1376887184045fcf45c69e92af734b7aaddc05fb777b6fbd03"
485
485
 
486
486
  [[package]]
487
487
  name = "snail-ast"
488
- version = "0.5.3"
488
+ version = "0.6.0"
489
489
 
490
490
  [[package]]
491
491
  name = "snail-core"
492
- version = "0.5.3"
492
+ version = "0.6.0"
493
493
  dependencies = [
494
494
  "pyo3",
495
495
  "snail-ast",
@@ -500,14 +500,14 @@ dependencies = [
500
500
 
501
501
  [[package]]
502
502
  name = "snail-error"
503
- version = "0.5.3"
503
+ version = "0.6.0"
504
504
  dependencies = [
505
505
  "snail-ast",
506
506
  ]
507
507
 
508
508
  [[package]]
509
509
  name = "snail-lower"
510
- version = "0.5.3"
510
+ version = "0.6.0"
511
511
  dependencies = [
512
512
  "pyo3",
513
513
  "snail-ast",
@@ -516,7 +516,7 @@ dependencies = [
516
516
 
517
517
  [[package]]
518
518
  name = "snail-parser"
519
- version = "0.5.3"
519
+ version = "0.6.0"
520
520
  dependencies = [
521
521
  "pest",
522
522
  "pest_derive",
@@ -526,7 +526,7 @@ dependencies = [
526
526
 
527
527
  [[package]]
528
528
  name = "snail-proptest"
529
- version = "0.5.3"
529
+ version = "0.6.0"
530
530
  dependencies = [
531
531
  "proptest",
532
532
  "pyo3",
@@ -540,7 +540,7 @@ dependencies = [
540
540
 
541
541
  [[package]]
542
542
  name = "snail-python"
543
- version = "0.5.3"
543
+ version = "0.6.0"
544
544
  dependencies = [
545
545
  "pyo3",
546
546
  "snail-core",
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: snail-lang
3
- Version: 0.5.3
3
+ Version: 0.6.0
4
4
  Requires-Dist: jmespath>=1.0.1
5
5
  Requires-Dist: maturin>=1.5 ; extra == 'dev'
6
6
  Requires-Dist: pytest ; extra == 'dev'
@@ -62,11 +62,21 @@ Process files line-by-line with familiar awk semantics:
62
62
  { print($1, "->", $2) }
63
63
  ```
64
64
 
65
- Built-in variables: `$0` (line), `$1`, `$2` etc (access fields), `$n` (line number), `$fn` (per-file line number), `$p` (file path), `$m` (last match).
65
+ **Built-in variables:**
66
66
 
67
- Begin/end blocks use CLI flags for setup and teardown:
67
+ | Variable | Description |
68
+ |----------|-------------|
69
+ | `$0` | Current line (with newline stripped) |
70
+ | `$1`, `$2`, ... | Individual fields (whitespace-split) |
71
+ | `$f` | All fields as a list |
72
+ | `$n` | Global line number (across all files) |
73
+ | `$fn` | Per-file line number |
74
+ | `$p` | Current file path |
75
+ | `$m` | Last regex match object |
76
+
77
+ Begin/end blocks use CLI flags (`-b`/`--begin`, `-e`/`--end`) for setup and teardown:
68
78
  ```bash
69
- echo -e "5\n4\n3\n2\n1" | snail --awk -b 'total = 0' -e 'print("Sum:", total)' '/^[0-9]+/ { total = total + int($1) }'
79
+ echo -e "5\n4\n3\n2\n1" | snail --awk --begin 'total = 0' --end 'print("Sum:", total)' '/^[0-9]+/ { total = total + int($1) }'
70
80
  ```
71
81
 
72
82
  ### Compact Error Handling
@@ -50,11 +50,21 @@ Process files line-by-line with familiar awk semantics:
50
50
  { print($1, "->", $2) }
51
51
  ```
52
52
 
53
- Built-in variables: `$0` (line), `$1`, `$2` etc (access fields), `$n` (line number), `$fn` (per-file line number), `$p` (file path), `$m` (last match).
54
-
55
- Begin/end blocks use CLI flags for setup and teardown:
53
+ **Built-in variables:**
54
+
55
+ | Variable | Description |
56
+ |----------|-------------|
57
+ | `$0` | Current line (with newline stripped) |
58
+ | `$1`, `$2`, ... | Individual fields (whitespace-split) |
59
+ | `$f` | All fields as a list |
60
+ | `$n` | Global line number (across all files) |
61
+ | `$fn` | Per-file line number |
62
+ | `$p` | Current file path |
63
+ | `$m` | Last regex match object |
64
+
65
+ Begin/end blocks use CLI flags (`-b`/`--begin`, `-e`/`--end`) for setup and teardown:
56
66
  ```bash
57
- echo -e "5\n4\n3\n2\n1" | snail --awk -b 'total = 0' -e 'print("Sum:", total)' '/^[0-9]+/ { total = total + int($1) }'
67
+ echo -e "5\n4\n3\n2\n1" | snail --awk --begin 'total = 0' --end 'print("Sum:", total)' '/^[0-9]+/ { total = total + int($1) }'
58
68
  ```
59
69
 
60
70
  ### Compact Error Handling
@@ -1,6 +1,6 @@
1
1
  [package]
2
2
  name = "snail-ast"
3
- version = "0.5.3"
3
+ version = "0.6.0"
4
4
  edition = "2024"
5
5
  readme = "README.md"
6
6
 
@@ -4,6 +4,7 @@ use crate::ast::{Expr, SourceSpan, Stmt};
4
4
  pub enum CompileMode {
5
5
  Snail,
6
6
  Awk,
7
+ Map,
7
8
  }
8
9
 
9
10
  #[derive(Debug, Clone, PartialEq)]
@@ -1,6 +1,6 @@
1
1
  [package]
2
2
  name = "snail-core"
3
- version = "0.5.3"
3
+ version = "0.6.0"
4
4
  edition.workspace = true
5
5
  readme = "README.md"
6
6
 
@@ -32,6 +32,11 @@ pub fn compile_snail_source_with_auto_print(
32
32
  let module = lower_awk_program_with_auto_print(py, &program, auto_print_last)?;
33
33
  Ok(module)
34
34
  }
35
+ CompileMode::Map => {
36
+ let program = parse_map_program(source)?;
37
+ let module = lower_map_program_with_auto_print(py, &program, auto_print_last)?;
38
+ Ok(module)
39
+ }
35
40
  }
36
41
  }
37
42
 
@@ -1,6 +1,6 @@
1
1
  [package]
2
2
  name = "snail-error"
3
- version = "0.5.3"
3
+ version = "0.6.0"
4
4
  edition = "2024"
5
5
  readme = "README.md"
6
6
 
@@ -1,6 +1,6 @@
1
1
  [package]
2
2
  name = "snail-lower"
3
- version = "0.5.3"
3
+ version = "0.6.0"
4
4
  edition = "2024"
5
5
  readme = "README.md"
6
6
 
@@ -27,7 +27,7 @@ This crate is the semantic transformation core of the Snail compiler. It takes S
27
27
  - **Regex expressions** (`/pattern/`): Transformed into `__snail_regex_compile(pattern)` call
28
28
  - **Regex matching** (`string in /pattern/`): Transformed into `__snail_regex_search(string, pattern)` call
29
29
  - **Structured accessors** (`$[query]`): Transformed into `__SnailStructuredAccessor(query)` instance
30
- - **Awk variables**: `$0`, `$<num>`, `$n`, `$fn`, `$p`, `$m` mapped to Python variable names
30
+ - **Awk variables**: `$0`, `$<num>`, `$n`, `$fn`, `$f`, `$p`, `$m` mapped to Python variable names
31
31
 
32
32
  ## Awk Mode Lowering
33
33
 
@@ -20,6 +20,7 @@ pub(crate) const SNAIL_AWK_NR: &str = "$n";
20
20
  pub(crate) const SNAIL_AWK_FNR: &str = "$fn";
21
21
  pub(crate) const SNAIL_AWK_PATH: &str = "$p";
22
22
  pub(crate) const SNAIL_AWK_MATCH: &str = "$m";
23
+ pub(crate) const SNAIL_AWK_FIELDS: &str = "$f";
23
24
  pub(crate) const SNAIL_AWK_LINE_PYVAR: &str = "__snail_line";
24
25
  pub(crate) const SNAIL_AWK_FIELDS_PYVAR: &str = "__snail_fields";
25
26
  pub(crate) const SNAIL_AWK_NR_PYVAR: &str = "__snail_nr_user";
@@ -27,12 +28,27 @@ pub(crate) const SNAIL_AWK_FNR_PYVAR: &str = "__snail_fnr_user";
27
28
  pub(crate) const SNAIL_AWK_PATH_PYVAR: &str = "__snail_path_user";
28
29
  pub(crate) const SNAIL_AWK_MATCH_PYVAR: &str = "__snail_match";
29
30
 
31
+ // Map-related constants (public within crate)
32
+ pub(crate) const SNAIL_MAP_SRC: &str = "$src";
33
+ pub(crate) const SNAIL_MAP_FD: &str = "$fd";
34
+ pub(crate) const SNAIL_MAP_TEXT: &str = "$text";
35
+ pub(crate) const SNAIL_MAP_SRC_PYVAR: &str = "__snail_src";
36
+ pub(crate) const SNAIL_MAP_FD_PYVAR: &str = "__snail_fd";
37
+ pub(crate) const SNAIL_MAP_TEXT_PYVAR: &str = "__snail_text";
38
+ pub const SNAIL_LAZY_TEXT_CLASS: &str = "__SnailLazyText";
39
+
30
40
  pub(crate) fn injected_py_name(name: &str) -> Option<&'static str> {
31
41
  match name {
42
+ // Awk variables
32
43
  SNAIL_AWK_NR => Some(SNAIL_AWK_NR_PYVAR),
33
44
  SNAIL_AWK_FNR => Some(SNAIL_AWK_FNR_PYVAR),
34
45
  SNAIL_AWK_PATH => Some(SNAIL_AWK_PATH_PYVAR),
35
46
  SNAIL_AWK_MATCH => Some(SNAIL_AWK_MATCH_PYVAR),
47
+ SNAIL_AWK_FIELDS => Some(SNAIL_AWK_FIELDS_PYVAR),
48
+ // Map variables
49
+ SNAIL_MAP_SRC => Some(SNAIL_MAP_SRC_PYVAR),
50
+ SNAIL_MAP_FD => Some(SNAIL_MAP_FD_PYVAR),
51
+ SNAIL_MAP_TEXT => Some(SNAIL_MAP_TEXT_PYVAR),
36
52
  _ => None,
37
53
  }
38
54
  }
@@ -3,6 +3,7 @@ mod awk;
3
3
  mod constants;
4
4
  mod expr;
5
5
  mod helpers;
6
+ mod map;
6
7
  mod operators;
7
8
  mod program;
8
9
  mod py_ast;
@@ -10,6 +11,7 @@ mod stmt;
10
11
 
11
12
  // Re-export public API
12
13
  pub use constants::*;
14
+ pub use map::{lower_map_program, lower_map_program_with_auto_print};
13
15
  pub use program::{
14
16
  lower_awk_program, lower_awk_program_with_auto_print, lower_program,
15
17
  lower_program_with_auto_print,
@@ -0,0 +1,235 @@
1
+ use pyo3::prelude::*;
2
+ use pyo3::types::PyList;
3
+ use snail_ast::*;
4
+ use snail_error::LowerError;
5
+
6
+ use crate::constants::*;
7
+ use crate::helpers::{assign_name, name_expr, string_expr};
8
+ use crate::py_ast::{AstBuilder, py_err_to_lower};
9
+ use crate::stmt::lower_block_with_auto_print;
10
+
11
+ pub fn lower_map_program(py: Python<'_>, program: &Program) -> Result<PyObject, LowerError> {
12
+ lower_map_program_with_auto_print(py, program, false)
13
+ }
14
+
15
+ pub fn lower_map_program_with_auto_print(
16
+ py: Python<'_>,
17
+ program: &Program,
18
+ auto_print_last: bool,
19
+ ) -> Result<PyObject, LowerError> {
20
+ let builder = AstBuilder::new(py).map_err(py_err_to_lower)?;
21
+ let span = program.span.clone();
22
+ let mut body = Vec::new();
23
+
24
+ // import sys
25
+ let sys_import = builder
26
+ .call_node(
27
+ "Import",
28
+ vec![
29
+ PyList::new_bound(
30
+ builder.py(),
31
+ vec![
32
+ builder
33
+ .call_node_no_loc(
34
+ "alias",
35
+ vec![
36
+ "sys".to_string().into_py(builder.py()),
37
+ builder.py().None().into_py(builder.py()),
38
+ ],
39
+ )
40
+ .map_err(py_err_to_lower)?,
41
+ ],
42
+ )
43
+ .into_py(builder.py()),
44
+ ],
45
+ &span,
46
+ )
47
+ .map_err(py_err_to_lower)?;
48
+ body.push(sys_import);
49
+
50
+ // __snail_paths = sys.argv[1:] or [line.rstrip('\n') for line in sys.stdin if line.rstrip('\n')]
51
+ let paths_expr = lower_paths_source(&builder, &span)?;
52
+ body.push(assign_name(&builder, "__snail_paths", paths_expr, &span)?);
53
+
54
+ // Generate file loop
55
+ let file_loop = lower_map_file_loop(&builder, program, &span, auto_print_last)?;
56
+ body.push(file_loop);
57
+
58
+ builder.module(body, &span).map_err(py_err_to_lower)
59
+ }
60
+
61
+ fn lower_paths_source(builder: &AstBuilder<'_>, span: &SourceSpan) -> Result<PyObject, LowerError> {
62
+ // sys.argv[1:]
63
+ builder
64
+ .call_node(
65
+ "Subscript",
66
+ vec![
67
+ builder
68
+ .call_node(
69
+ "Attribute",
70
+ vec![
71
+ name_expr(
72
+ builder,
73
+ "sys",
74
+ span,
75
+ builder.load_ctx().map_err(py_err_to_lower)?,
76
+ )?,
77
+ "argv".to_string().into_py(builder.py()),
78
+ builder.load_ctx().map_err(py_err_to_lower)?,
79
+ ],
80
+ span,
81
+ )
82
+ .map_err(py_err_to_lower)?,
83
+ builder
84
+ .call_node(
85
+ "Slice",
86
+ vec![
87
+ builder
88
+ .call_node("Constant", vec![1i64.into_py(builder.py())], span)
89
+ .map_err(py_err_to_lower)?,
90
+ builder.py().None().into_py(builder.py()),
91
+ builder.py().None().into_py(builder.py()),
92
+ ],
93
+ span,
94
+ )
95
+ .map_err(py_err_to_lower)?,
96
+ builder.load_ctx().map_err(py_err_to_lower)?,
97
+ ],
98
+ span,
99
+ )
100
+ .map_err(py_err_to_lower)
101
+ }
102
+
103
+ fn lower_map_file_loop(
104
+ builder: &AstBuilder<'_>,
105
+ program: &Program,
106
+ span: &SourceSpan,
107
+ auto_print_last: bool,
108
+ ) -> Result<PyObject, LowerError> {
109
+ // for __snail_src in __snail_paths:
110
+ // with open(__snail_src, 'r') as __snail_fd:
111
+ // __snail_text = __SnailLazyText(__snail_fd)
112
+ // # user code
113
+
114
+ // Build the with statement body
115
+ let mut with_body = Vec::new();
116
+
117
+ // __snail_text = __SnailLazyText(__snail_fd)
118
+ let lazy_text_call = builder
119
+ .call_node(
120
+ "Call",
121
+ vec![
122
+ name_expr(
123
+ builder,
124
+ SNAIL_LAZY_TEXT_CLASS,
125
+ span,
126
+ builder.load_ctx().map_err(py_err_to_lower)?,
127
+ )?,
128
+ PyList::new_bound(
129
+ builder.py(),
130
+ vec![name_expr(
131
+ builder,
132
+ SNAIL_MAP_FD_PYVAR,
133
+ span,
134
+ builder.load_ctx().map_err(py_err_to_lower)?,
135
+ )?],
136
+ )
137
+ .into_py(builder.py()),
138
+ PyList::empty_bound(builder.py()).into_py(builder.py()),
139
+ ],
140
+ span,
141
+ )
142
+ .map_err(py_err_to_lower)?;
143
+ with_body.push(assign_name(
144
+ builder,
145
+ SNAIL_MAP_TEXT_PYVAR,
146
+ lazy_text_call,
147
+ span,
148
+ )?);
149
+
150
+ // Lower user code
151
+ let user_code =
152
+ lower_block_with_auto_print(builder, &program.stmts, auto_print_last, &program.span)?;
153
+ with_body.extend(user_code);
154
+
155
+ // open(__snail_src, 'r')
156
+ let open_call = builder
157
+ .call_node(
158
+ "Call",
159
+ vec![
160
+ name_expr(
161
+ builder,
162
+ "open",
163
+ span,
164
+ builder.load_ctx().map_err(py_err_to_lower)?,
165
+ )?,
166
+ PyList::new_bound(
167
+ builder.py(),
168
+ vec![
169
+ name_expr(
170
+ builder,
171
+ SNAIL_MAP_SRC_PYVAR,
172
+ span,
173
+ builder.load_ctx().map_err(py_err_to_lower)?,
174
+ )?,
175
+ string_expr(builder, "r", false, StringDelimiter::Double, span)?,
176
+ ],
177
+ )
178
+ .into_py(builder.py()),
179
+ PyList::empty_bound(builder.py()).into_py(builder.py()),
180
+ ],
181
+ span,
182
+ )
183
+ .map_err(py_err_to_lower)?;
184
+
185
+ // with open(...) as __snail_fd:
186
+ let with_item = builder
187
+ .call_node_no_loc(
188
+ "withitem",
189
+ vec![
190
+ open_call,
191
+ name_expr(
192
+ builder,
193
+ SNAIL_MAP_FD_PYVAR,
194
+ span,
195
+ builder.store_ctx().map_err(py_err_to_lower)?,
196
+ )?,
197
+ ],
198
+ )
199
+ .map_err(py_err_to_lower)?;
200
+
201
+ let with_stmt = builder
202
+ .call_node(
203
+ "With",
204
+ vec![
205
+ PyList::new_bound(builder.py(), vec![with_item]).into_py(builder.py()),
206
+ PyList::new_bound(builder.py(), with_body).into_py(builder.py()),
207
+ ],
208
+ span,
209
+ )
210
+ .map_err(py_err_to_lower)?;
211
+
212
+ // for __snail_src in __snail_paths:
213
+ builder
214
+ .call_node(
215
+ "For",
216
+ vec![
217
+ name_expr(
218
+ builder,
219
+ SNAIL_MAP_SRC_PYVAR,
220
+ span,
221
+ builder.store_ctx().map_err(py_err_to_lower)?,
222
+ )?,
223
+ name_expr(
224
+ builder,
225
+ "__snail_paths",
226
+ span,
227
+ builder.load_ctx().map_err(py_err_to_lower)?,
228
+ )?,
229
+ PyList::new_bound(builder.py(), vec![with_stmt]).into_py(builder.py()),
230
+ PyList::empty_bound(builder.py()).into_py(builder.py()),
231
+ ],
232
+ span,
233
+ )
234
+ .map_err(py_err_to_lower)
235
+ }
@@ -1,6 +1,6 @@
1
1
  [package]
2
2
  name = "snail-parser"
3
- version = "0.5.3"
3
+ version = "0.6.0"
4
4
  edition = "2024"
5
5
  readme = "README.md"
6
6