2023-01-24 20:27:38 +00:00
|
|
|
use std::{
|
|
|
|
env::{self, current_dir},
|
2023-02-17 14:03:13 +00:00
|
|
|
io,
|
2023-01-24 20:27:38 +00:00
|
|
|
path::PathBuf,
|
|
|
|
};
|
2023-01-24 07:05:54 +00:00
|
|
|
|
|
|
|
use mlua::prelude::*;
|
2023-02-17 14:03:13 +00:00
|
|
|
use tokio::fs;
|
2023-01-24 07:05:54 +00:00
|
|
|
|
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 14:03:13 +00:00
|
|
|
// Store the current pwd, and make helper functions for path conversions
|
|
|
|
let require_pwd = current_dir()?.to_string_lossy().to_string();
|
|
|
|
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)| {
|
|
|
|
let mut path_relative_to_pwd = PathBuf::from(
|
|
|
|
&require_source
|
|
|
|
.trim_start_matches("[string \"")
|
|
|
|
.trim_end_matches("\"]"),
|
|
|
|
)
|
|
|
|
.parent()
|
|
|
|
.unwrap()
|
|
|
|
.join(require_path);
|
|
|
|
// Try to normalize and resolve relative path segments such as './' and '../'
|
|
|
|
if let Ok(canonicalized) =
|
|
|
|
path_relative_to_pwd.with_extension("luau").canonicalize()
|
|
|
|
{
|
|
|
|
path_relative_to_pwd = canonicalized;
|
|
|
|
}
|
|
|
|
if let Ok(canonicalized) = path_relative_to_pwd.with_extension("lua").canonicalize()
|
|
|
|
{
|
|
|
|
path_relative_to_pwd = canonicalized;
|
|
|
|
}
|
|
|
|
let absolute = path_relative_to_pwd.to_string_lossy().to_string();
|
|
|
|
let relative = absolute.trim_start_matches(&require_pwd).to_string();
|
|
|
|
Ok((absolute, relative))
|
|
|
|
},
|
|
|
|
)?
|
|
|
|
.bind(require_pwd)?;
|
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)?
|
|
|
|
.with_async_function("load", load_file)?
|
|
|
|
.build_readonly()?;
|
|
|
|
let require_fn_lua = lua
|
|
|
|
.load(
|
|
|
|
r#"
|
|
|
|
local source = info(2, "s")
|
|
|
|
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
|
|
|
}
|
2023-02-17 14:03:13 +00:00
|
|
|
|
|
|
|
async fn load_file(
|
|
|
|
lua: &Lua,
|
|
|
|
(path_absolute, path_relative): (String, String),
|
|
|
|
) -> LuaResult<LuaValue> {
|
|
|
|
// 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).await {
|
|
|
|
Ok(contents) => lua.load(&contents).set_name(path_relative)?.eval(),
|
|
|
|
Err(e) => match e.kind() {
|
|
|
|
io::ErrorKind::NotFound => Err(LuaError::RuntimeError(format!(
|
|
|
|
"No lua module exists at the path '{path_relative}'"
|
|
|
|
))),
|
|
|
|
_ => Err(LuaError::external(e)),
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|