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.
- {snail_lang-0.5.3 → snail_lang-0.6.0}/Cargo.lock +7 -7
- {snail_lang-0.5.3 → snail_lang-0.6.0}/PKG-INFO +14 -4
- {snail_lang-0.5.3 → snail_lang-0.6.0}/README.md +14 -4
- {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-ast/Cargo.toml +1 -1
- {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-ast/src/awk.rs +1 -0
- {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-core/Cargo.toml +1 -1
- {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-core/src/lib.rs +5 -0
- {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-error/Cargo.toml +1 -1
- {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-lower/Cargo.toml +1 -1
- {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-lower/README.md +1 -1
- {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-lower/src/constants.rs +16 -0
- {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-lower/src/lib.rs +2 -0
- snail_lang-0.6.0/crates/snail-lower/src/map.rs +235 -0
- {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-parser/Cargo.toml +1 -1
- {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-parser/src/lib.rs +386 -1
- {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-parser/src/snail.pest +2 -2
- {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-parser/src/string.rs +17 -0
- {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-parser/tests/errors.rs +35 -0
- {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-python/Cargo.toml +1 -1
- {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-python/src/lib.rs +6 -2
- {snail_lang-0.5.3 → snail_lang-0.6.0}/pyproject.toml +1 -1
- {snail_lang-0.5.3 → snail_lang-0.6.0}/python/snail/cli.py +76 -12
- {snail_lang-0.5.3 → snail_lang-0.6.0}/python/snail/runtime/__init__.py +11 -0
- snail_lang-0.6.0/python/snail/runtime/lazy_text.py +50 -0
- {snail_lang-0.5.3 → snail_lang-0.6.0}/Cargo.toml +0 -0
- {snail_lang-0.5.3 → snail_lang-0.6.0}/LICENSE +0 -0
- {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-ast/README.md +0 -0
- {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-ast/src/ast.rs +0 -0
- {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-ast/src/lib.rs +0 -0
- {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-core/README.md +0 -0
- {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-error/README.md +0 -0
- {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-error/src/lib.rs +0 -0
- {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-lower/src/awk.rs +0 -0
- {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-lower/src/expr.rs +0 -0
- {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-lower/src/helpers.rs +0 -0
- {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-lower/src/operators.rs +0 -0
- {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-lower/src/program.rs +0 -0
- {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-lower/src/py_ast.rs +0 -0
- {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-lower/src/stmt.rs +0 -0
- {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-parser/README.md +0 -0
- {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-parser/src/awk.rs +0 -0
- {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-parser/src/expr.rs +0 -0
- {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-parser/src/literal.rs +0 -0
- {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-parser/src/stmt.rs +0 -0
- {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-parser/src/util.rs +0 -0
- {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-parser/tests/common.rs +0 -0
- {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-parser/tests/parser.rs +0 -0
- {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-parser/tests/statements.rs +0 -0
- {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-parser/tests/syntax_expressions.rs +0 -0
- {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-parser/tests/syntax_strings.rs +0 -0
- {snail_lang-0.5.3 → snail_lang-0.6.0}/crates/snail-python/build.rs +0 -0
- {snail_lang-0.5.3 → snail_lang-0.6.0}/python/snail/__init__.py +0 -0
- {snail_lang-0.5.3 → snail_lang-0.6.0}/python/snail/runtime/compact_try.py +0 -0
- {snail_lang-0.5.3 → snail_lang-0.6.0}/python/snail/runtime/regex.py +0 -0
- {snail_lang-0.5.3 → snail_lang-0.6.0}/python/snail/runtime/structured_accessor.py +0 -0
- {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.
|
|
488
|
+
version = "0.6.0"
|
|
489
489
|
|
|
490
490
|
[[package]]
|
|
491
491
|
name = "snail-core"
|
|
492
|
-
version = "0.
|
|
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.
|
|
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.
|
|
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.
|
|
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.
|
|
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.
|
|
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.
|
|
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
|
|
65
|
+
**Built-in variables:**
|
|
66
66
|
|
|
67
|
-
|
|
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
|
|
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
|
|
54
|
-
|
|
55
|
-
|
|
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
|
|
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
|
|
@@ -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
|
|
|
@@ -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
|
+
}
|