nu_std/
lib.rs

1#![doc = include_str!("../README.md")]
2use log::trace;
3use nu_parser::parse;
4use nu_protocol::{
5    VirtualPathId,
6    engine::{FileStack, StateWorkingSet, VirtualPath},
7    report_parse_error,
8};
9use std::path::PathBuf;
10
11fn create_virt_file(working_set: &mut StateWorkingSet, name: &str, content: &str) -> VirtualPathId {
12    let sanitized_name = PathBuf::from(name).to_string_lossy().to_string();
13    let file_id = working_set.add_file(sanitized_name.clone(), content.as_bytes());
14
15    working_set.add_virtual_path(sanitized_name, VirtualPath::File(file_id))
16}
17
18pub fn load_standard_library(
19    engine_state: &mut nu_protocol::engine::EngineState,
20) -> Result<(), miette::ErrReport> {
21    trace!("load_standard_library");
22
23    let mut working_set = StateWorkingSet::new(engine_state);
24
25    // Contents of the std virtual directory
26    let mut std_virt_paths = vec![];
27
28    // std/mod.nu
29    let std_mod_virt_file_id = create_virt_file(
30        &mut working_set,
31        "std/mod.nu",
32        include_str!("../std/mod.nu"),
33    );
34    std_virt_paths.push(std_mod_virt_file_id);
35
36    // Submodules/subdirectories ... std/<module>/mod.nu
37    let mut std_submodules = vec![
38        // Loaded at startup - Not technically part of std
39        (
40            "mod.nu",
41            "std/prelude",
42            include_str!("../std/prelude/mod.nu"),
43        ),
44        // std submodules
45        ("mod.nu", "std/assert", include_str!("../std/assert/mod.nu")),
46        ("mod.nu", "std/bench", include_str!("../std/bench/mod.nu")),
47        ("mod.nu", "std/dirs", include_str!("../std/dirs/mod.nu")),
48        ("mod.nu", "std/dt", include_str!("../std/dt/mod.nu")),
49        (
50            "mod.nu",
51            "std/formats",
52            include_str!("../std/formats/mod.nu"),
53        ),
54        ("mod.nu", "std/help", include_str!("../std/help/mod.nu")),
55        ("mod.nu", "std/input", include_str!("../std/input/mod.nu")),
56        ("mod.nu", "std/iter", include_str!("../std/iter/mod.nu")),
57        ("mod.nu", "std/log", include_str!("../std/log/mod.nu")),
58        ("mod.nu", "std/math", include_str!("../std/math/mod.nu")),
59        ("mod.nu", "std/util", include_str!("../std/util/mod.nu")),
60        ("mod.nu", "std/xml", include_str!("../std/xml/mod.nu")),
61        ("mod.nu", "std/config", include_str!("../std/config/mod.nu")),
62        (
63            "mod.nu",
64            "std/testing",
65            include_str!("../std/testing/mod.nu"),
66        ),
67    ];
68
69    for (filename, std_subdir_name, content) in std_submodules.drain(..) {
70        let mod_dir = PathBuf::from(std_subdir_name);
71        let name = mod_dir.join(filename);
72        let virt_file_id = create_virt_file(&mut working_set, &name.to_string_lossy(), content);
73
74        // Place file in virtual subdir
75        let mod_dir_filelist = vec![virt_file_id];
76
77        let virt_dir_id = working_set.add_virtual_path(
78            mod_dir.to_string_lossy().to_string(),
79            VirtualPath::Dir(mod_dir_filelist),
80        );
81        // Add the subdir to the list of paths in std
82        std_virt_paths.push(virt_dir_id);
83    }
84
85    // Create std virtual dir with all subdirs and files
86    let std_dir = PathBuf::from("std").to_string_lossy().to_string();
87    let _ = working_set.add_virtual_path(std_dir, VirtualPath::Dir(std_virt_paths));
88
89    // Add std-rfc files
90    let mut std_rfc_virt_paths = vec![];
91
92    // std-rfc/mod.nu
93    let std_rfc_mod_virt_file_id = create_virt_file(
94        &mut working_set,
95        "std-rfc/mod.nu",
96        include_str!("../std-rfc/mod.nu"),
97    );
98    std_rfc_virt_paths.push(std_rfc_mod_virt_file_id);
99
100    // Submodules/subdirectories ... std-rfc/<module>/mod.nu
101    let mut std_rfc_submodules = vec![
102        (
103            "mod.nu",
104            "std-rfc/clip",
105            include_str!("../std-rfc/clip/mod.nu"),
106        ),
107        (
108            "mod.nu",
109            "std-rfc/conversions",
110            include_str!("../std-rfc/conversions/mod.nu"),
111        ),
112        #[cfg(feature = "sqlite")]
113        ("mod.nu", "std-rfc/kv", include_str!("../std-rfc/kv/mod.nu")),
114        (
115            "mod.nu",
116            "std-rfc/path",
117            include_str!("../std-rfc/path/mod.nu"),
118        ),
119        (
120            "mod.nu",
121            "std-rfc/str",
122            include_str!("../std-rfc/str/mod.nu"),
123        ),
124        (
125            "mod.nu",
126            "std-rfc/tables",
127            include_str!("../std-rfc/tables/mod.nu"),
128        ),
129        (
130            "mod.nu",
131            "std-rfc/iter",
132            include_str!("../std-rfc/iter/mod.nu"),
133        ),
134    ];
135
136    for (filename, std_rfc_subdir_name, content) in std_rfc_submodules.drain(..) {
137        let mod_dir = PathBuf::from(std_rfc_subdir_name);
138        let name = mod_dir.join(filename);
139        let virt_file_id = create_virt_file(&mut working_set, &name.to_string_lossy(), content);
140
141        // Place file in virtual subdir
142        let mod_dir_filelist = vec![virt_file_id];
143
144        let virt_dir_id = working_set.add_virtual_path(
145            mod_dir.to_string_lossy().to_string(),
146            VirtualPath::Dir(mod_dir_filelist),
147        );
148        // Add the subdir to the list of paths in std
149        std_rfc_virt_paths.push(virt_dir_id);
150    }
151
152    // Create std virtual dir with all subdirs and files
153    let std_rfc_dir = PathBuf::from("std-rfc").to_string_lossy().to_string();
154    let _ = working_set.add_virtual_path(std_rfc_dir, VirtualPath::Dir(std_rfc_virt_paths));
155
156    // Load prelude
157    let (_, delta) = {
158        let source = r#"
159# Prelude
160use std/prelude *
161"#;
162
163        // Add a placeholder file to the stack of files being evaluated.
164        // The name of this file doesn't matter; it's only there to set the current working directory to NU_STDLIB_VIRTUAL_DIR.
165        let placeholder = PathBuf::from("load std/prelude");
166        working_set.files = FileStack::with_file(placeholder);
167
168        let block = parse(
169            &mut working_set,
170            Some("loading stdlib prelude"),
171            source.as_bytes(),
172            false,
173        );
174
175        // Remove the placeholder file from the stack of files being evaluated.
176        working_set.files.pop();
177
178        if let Some(err) = working_set.parse_errors.first() {
179            report_parse_error(&working_set, err);
180        }
181
182        (block, working_set.render())
183    };
184
185    engine_state.merge_delta(delta)?;
186
187    Ok(())
188}