forked from rojo-rbx/rojo
210 lines
6.4 KiB
Rust
210 lines
6.4 KiB
Rust
use std::{borrow::Cow, str};
|
|
|
|
use maplit::hashmap;
|
|
use rbx_dom_weak::{RbxId, RbxTree, RbxValue};
|
|
|
|
use crate::{
|
|
imfs::new::{FsResultExt, Imfs, ImfsEntry, ImfsFetcher},
|
|
snapshot::InstanceSnapshot,
|
|
};
|
|
|
|
use super::{
|
|
dir::SnapshotDir,
|
|
middleware::{SnapshotFileResult, SnapshotInstanceResult, SnapshotMiddleware},
|
|
};
|
|
|
|
pub struct SnapshotLua;
|
|
|
|
impl SnapshotMiddleware for SnapshotLua {
|
|
fn from_imfs<F: ImfsFetcher>(
|
|
imfs: &mut Imfs<F>,
|
|
entry: &ImfsEntry,
|
|
) -> SnapshotInstanceResult<'static> {
|
|
let file_name = entry.path().file_name().unwrap().to_string_lossy();
|
|
|
|
// These paths alter their parent instance, so we don't need to turn
|
|
// them into a script instance here.
|
|
match &*file_name {
|
|
"init.lua" | "init.server.lua" | "init.client.lua" => return Ok(None),
|
|
_ => {}
|
|
}
|
|
|
|
if entry.is_file() {
|
|
snapshot_lua_file(imfs, entry)
|
|
} else {
|
|
if let Some(snapshot) = snapshot_init(imfs, entry, "init.lua")? {
|
|
// An `init.lua` file turns its parent into a ModuleScript
|
|
Ok(Some(snapshot))
|
|
} else if let Some(snapshot) = snapshot_init(imfs, entry, "init.server.lua")? {
|
|
// An `init.server.lua` file turns its parent into a Script
|
|
Ok(Some(snapshot))
|
|
} else if let Some(snapshot) = snapshot_init(imfs, entry, "init.client.lua")? {
|
|
// An `init.client.lua` file turns its parent into a LocalScript
|
|
Ok(Some(snapshot))
|
|
} else {
|
|
Ok(None)
|
|
}
|
|
}
|
|
}
|
|
|
|
fn from_instance(tree: &RbxTree, id: RbxId) -> SnapshotFileResult {
|
|
let instance = tree.get_instance(id).unwrap();
|
|
|
|
match instance.class_name.as_str() {
|
|
"ModuleScript" | "LocalScript" | "Script" => {
|
|
unimplemented!("Snapshotting Script instances")
|
|
}
|
|
_ => None,
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Core routine for turning Lua files into snapshots.
|
|
fn snapshot_lua_file<F: ImfsFetcher>(
|
|
imfs: &mut Imfs<F>,
|
|
entry: &ImfsEntry,
|
|
) -> SnapshotInstanceResult<'static> {
|
|
let file_name = entry.path().file_name().unwrap().to_string_lossy();
|
|
|
|
let (class_name, instance_name) = if let Some(name) = match_trailing(&file_name, ".server.lua")
|
|
{
|
|
("Script", name)
|
|
} else if let Some(name) = match_trailing(&file_name, ".client.lua") {
|
|
("LocalScript", name)
|
|
} else if let Some(name) = match_trailing(&file_name, ".lua") {
|
|
("ModuleScript", name)
|
|
} else {
|
|
return Ok(None);
|
|
};
|
|
|
|
let contents = entry.contents(imfs)?;
|
|
let contents_str = str::from_utf8(contents)
|
|
.expect("File content was not valid UTF-8")
|
|
.to_string();
|
|
|
|
let properties = hashmap! {
|
|
"Source".to_owned() => RbxValue::String {
|
|
value: contents_str,
|
|
},
|
|
};
|
|
|
|
Ok(Some(InstanceSnapshot {
|
|
snapshot_id: None,
|
|
metadata: Default::default(), // TODO
|
|
name: Cow::Owned(instance_name.to_owned()),
|
|
class_name: Cow::Borrowed(class_name),
|
|
properties,
|
|
children: Vec::new(),
|
|
}))
|
|
}
|
|
|
|
/// Attempts to snapshot an 'init' Lua script contained inside of a folder with
|
|
/// the given name.
|
|
///
|
|
/// Scripts named `init.lua`, `init.server.lua`, or `init.client.lua` usurp
|
|
/// their parents, which acts similarly to `__init__.py` from the Python world.
|
|
fn snapshot_init<F: ImfsFetcher>(
|
|
imfs: &mut Imfs<F>,
|
|
folder_entry: &ImfsEntry,
|
|
init_name: &str,
|
|
) -> SnapshotInstanceResult<'static> {
|
|
let init_path = folder_entry.path().join(init_name);
|
|
|
|
if let Some(init_entry) = imfs.get(init_path).with_not_found()? {
|
|
if let Some(mut dir_snapshot) = SnapshotDir::from_imfs(imfs, folder_entry)? {
|
|
if let Some(mut init_snapshot) = snapshot_lua_file(imfs, &init_entry)? {
|
|
init_snapshot.name = dir_snapshot.name;
|
|
init_snapshot.children = dir_snapshot.children;
|
|
|
|
return Ok(Some(init_snapshot));
|
|
}
|
|
}
|
|
}
|
|
|
|
Ok(None)
|
|
}
|
|
|
|
fn match_trailing<'a>(input: &'a str, trailer: &str) -> Option<&'a str> {
|
|
if input.ends_with(trailer) {
|
|
let end = input.len().saturating_sub(trailer.len());
|
|
Some(&input[..end])
|
|
} else {
|
|
None
|
|
}
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod test {
|
|
use super::*;
|
|
|
|
use maplit::hashmap;
|
|
|
|
use crate::imfs::new::{ImfsSnapshot, NoopFetcher};
|
|
|
|
#[test]
|
|
fn module_from_imfs() {
|
|
let mut imfs = Imfs::new(NoopFetcher);
|
|
let file = ImfsSnapshot::file("Hello there!");
|
|
|
|
imfs.load_from_snapshot("/foo.lua", file);
|
|
|
|
let entry = imfs.get("/foo.lua").unwrap();
|
|
let instance_snapshot = SnapshotLua::from_imfs(&mut imfs, &entry).unwrap().unwrap();
|
|
|
|
assert_eq!(instance_snapshot.name, "foo");
|
|
assert_eq!(instance_snapshot.class_name, "ModuleScript");
|
|
assert_eq!(
|
|
instance_snapshot.properties,
|
|
hashmap! {
|
|
"Source".to_owned() => RbxValue::String {
|
|
value: "Hello there!".to_owned(),
|
|
},
|
|
}
|
|
);
|
|
}
|
|
|
|
#[test]
|
|
fn server_from_imfs() {
|
|
let mut imfs = Imfs::new(NoopFetcher);
|
|
let file = ImfsSnapshot::file("Hello there!");
|
|
|
|
imfs.load_from_snapshot("/foo.server.lua", file);
|
|
|
|
let entry = imfs.get("/foo.server.lua").unwrap();
|
|
let instance_snapshot = SnapshotLua::from_imfs(&mut imfs, &entry).unwrap().unwrap();
|
|
|
|
assert_eq!(instance_snapshot.name, "foo");
|
|
assert_eq!(instance_snapshot.class_name, "Script");
|
|
assert_eq!(
|
|
instance_snapshot.properties,
|
|
hashmap! {
|
|
"Source".to_owned() => RbxValue::String {
|
|
value: "Hello there!".to_owned(),
|
|
},
|
|
}
|
|
);
|
|
}
|
|
|
|
#[test]
|
|
fn client_from_imfs() {
|
|
let mut imfs = Imfs::new(NoopFetcher);
|
|
let file = ImfsSnapshot::file("Hello there!");
|
|
|
|
imfs.load_from_snapshot("/foo.client.lua", file);
|
|
|
|
let entry = imfs.get("/foo.client.lua").unwrap();
|
|
let instance_snapshot = SnapshotLua::from_imfs(&mut imfs, &entry).unwrap().unwrap();
|
|
|
|
assert_eq!(instance_snapshot.name, "foo");
|
|
assert_eq!(instance_snapshot.class_name, "LocalScript");
|
|
assert_eq!(
|
|
instance_snapshot.properties,
|
|
hashmap! {
|
|
"Source".to_owned() => RbxValue::String {
|
|
value: "Hello there!".to_owned(),
|
|
},
|
|
}
|
|
);
|
|
}
|
|
}
|