2023-01-24 20:27:38 +00:00
|
|
|
use std::{
|
|
|
|
env::{self, current_dir},
|
2023-02-17 18:20:17 +00:00
|
|
|
fs,
|
2023-01-24 20:27:38 +00:00
|
|
|
path::PathBuf,
|
|
|
|
};
|
2023-01-24 07:05:54 +00:00
|
|
|
|
|
|
|
use mlua::prelude::*;
|
|
|
|
|
2023-02-10 11:14:28 +00:00
|
|
|
use crate::utils::table::TableBuilder;
|
|
|
|
|
2023-02-11 11:39:39 +00:00
|
|
|
pub fn create(lua: &'static Lua) -> LuaResult<LuaTable> {
|
2023-02-17 14:03:13 +00:00
|
|
|
// Preserve original require behavior if we have a special env var set,
|
|
|
|
// returning an empty table since there are no globals to overwrite
|
2023-01-24 07:05:54 +00:00
|
|
|
if env::var_os("LUAU_PWD_REQUIRE").is_some() {
|
2023-02-13 14:28:18 +00:00
|
|
|
return TableBuilder::new(lua)?.build_readonly();
|
2023-01-24 07:05:54 +00:00
|
|
|
}
|
2023-02-17 18:20:17 +00:00
|
|
|
// Store the current pwd, and make the functions for path conversions & loading a file
|
|
|
|
let mut require_pwd = current_dir()?.to_string_lossy().to_string();
|
|
|
|
if !require_pwd.ends_with('/') {
|
|
|
|
require_pwd = format!("{require_pwd}/")
|
|
|
|
}
|
2023-02-17 14:03:13 +00:00
|
|
|
let require_info: LuaFunction = lua.named_registry_value("dbg.info")?;
|
|
|
|
let require_error: LuaFunction = lua.named_registry_value("error")?;
|
|
|
|
let require_get_abs_rel_paths = lua
|
|
|
|
.create_function(
|
|
|
|
|_, (require_pwd, require_source, require_path): (String, String, String)| {
|
2023-02-17 18:20:17 +00:00
|
|
|
let path_relative_to_pwd = PathBuf::from(
|
2023-02-17 14:03:13 +00:00
|
|
|
&require_source
|
|
|
|
.trim_start_matches("[string \"")
|
|
|
|
.trim_end_matches("\"]"),
|
|
|
|
)
|
|
|
|
.parent()
|
|
|
|
.unwrap()
|
2023-02-17 18:20:17 +00:00
|
|
|
.join(&require_path);
|
2023-02-17 14:03:13 +00:00
|
|
|
// Try to normalize and resolve relative path segments such as './' and '../'
|
2023-02-17 18:20:17 +00:00
|
|
|
let file_path = match (
|
|
|
|
path_relative_to_pwd.with_extension("luau").canonicalize(),
|
|
|
|
path_relative_to_pwd.with_extension("lua").canonicalize(),
|
|
|
|
) {
|
|
|
|
(Ok(luau), _) => luau,
|
|
|
|
(_, Ok(lua)) => lua,
|
|
|
|
_ => {
|
|
|
|
return Err(LuaError::RuntimeError(format!(
|
|
|
|
"File does not exist at path '{require_path}'"
|
|
|
|
)))
|
|
|
|
}
|
|
|
|
};
|
|
|
|
let absolute = file_path.to_string_lossy().to_string();
|
2023-02-17 14:03:13 +00:00
|
|
|
let relative = absolute.trim_start_matches(&require_pwd).to_string();
|
|
|
|
Ok((absolute, relative))
|
|
|
|
},
|
|
|
|
)?
|
|
|
|
.bind(require_pwd)?;
|
2023-02-17 18:20:17 +00:00
|
|
|
// Note that file loading must be blocking to guarantee the require cache works, if it
|
|
|
|
// were async then one lua script may require a module during the file reading process
|
|
|
|
let require_get_loaded_file = lua.create_function(
|
|
|
|
|lua: &Lua, (path_absolute, path_relative): (String, String)| {
|
|
|
|
// Use a name without extensions for loading the chunk, the
|
|
|
|
// above code assumes the require path is without extensions
|
|
|
|
let path_relative_no_extension = path_relative
|
|
|
|
.trim_end_matches(".lua")
|
|
|
|
.trim_end_matches(".luau");
|
|
|
|
// Try to read the wanted file, note that we use bytes instead of reading
|
|
|
|
// to a string since lua scripts are not necessarily valid utf-8 strings
|
|
|
|
match fs::read(path_absolute) {
|
|
|
|
Ok(contents) => lua
|
|
|
|
.load(&contents)
|
|
|
|
.set_name(path_relative_no_extension)?
|
|
|
|
.eval::<LuaValue>(),
|
|
|
|
Err(e) => Err(LuaError::external(e)),
|
|
|
|
}
|
|
|
|
},
|
|
|
|
)?;
|
2023-01-24 20:27:38 +00:00
|
|
|
/*
|
2023-02-17 14:03:13 +00:00
|
|
|
We need to get the source file where require was
|
|
|
|
called to be able to do path-relative requires,
|
|
|
|
so we make a small wrapper to do that here, this
|
|
|
|
will then call our actual async require function
|
2023-01-24 20:27:38 +00:00
|
|
|
|
2023-02-17 14:03:13 +00:00
|
|
|
This must be done in lua because due to how our
|
|
|
|
scheduler works mlua can not preserve debug info
|
2023-01-24 20:27:38 +00:00
|
|
|
*/
|
2023-02-17 14:03:13 +00:00
|
|
|
let require_env = TableBuilder::new(lua)?
|
|
|
|
.with_value("loaded", lua.create_table()?)?
|
|
|
|
.with_value("cache", lua.create_table()?)?
|
|
|
|
.with_value("info", require_info)?
|
|
|
|
.with_value("error", require_error)?
|
|
|
|
.with_value("paths", require_get_abs_rel_paths)?
|
2023-02-17 18:20:17 +00:00
|
|
|
.with_value("load", require_get_loaded_file)?
|
2023-02-17 14:03:13 +00:00
|
|
|
.build_readonly()?;
|
|
|
|
let require_fn_lua = lua
|
|
|
|
.load(
|
|
|
|
r#"
|
2023-02-17 18:20:17 +00:00
|
|
|
local source = info(1, "s")
|
|
|
|
if source == '[string "require"]' then
|
|
|
|
source = info(2, "s")
|
|
|
|
end
|
2023-02-17 14:03:13 +00:00
|
|
|
local absolute, relative = paths(source, ...)
|
|
|
|
if loaded[absolute] ~= true then
|
|
|
|
local first, second = load(absolute, relative)
|
|
|
|
if first == nil or second ~= nil then
|
|
|
|
error("Module did not return exactly one value")
|
|
|
|
end
|
|
|
|
loaded[absolute] = true
|
|
|
|
cache[absolute] = first
|
|
|
|
return first
|
|
|
|
else
|
|
|
|
return cache[absolute]
|
|
|
|
end
|
|
|
|
"#,
|
2023-02-13 14:28:18 +00:00
|
|
|
)
|
2023-02-17 14:03:13 +00:00
|
|
|
.set_name("require")?
|
|
|
|
.set_environment(require_env)?
|
|
|
|
.into_function()?;
|
2023-02-10 11:14:28 +00:00
|
|
|
TableBuilder::new(lua)?
|
2023-02-17 14:03:13 +00:00
|
|
|
.with_value("require", require_fn_lua)?
|
2023-02-10 11:14:28 +00:00
|
|
|
.build_readonly()
|
2023-01-24 07:05:54 +00:00
|
|
|
}
|