145 lines
5.4 KiB
Lua
145 lines
5.4 KiB
Lua
-- handles core detection logic for workspace in models
|
|
local roblox = require("@lune/roblox")
|
|
local stdio = require("@lune/stdio")
|
|
local types = require("./types")
|
|
local compression = require("./compression")
|
|
|
|
export type Core = {
|
|
scanForWorkspace: (model: types.model, printInstanceNames: boolean) -> boolean,
|
|
isValidModelFile: (fileName: string) -> boolean,
|
|
fileContainsWorkspace: (fileContents: string, opts: types.opts) -> boolean,
|
|
formatResult: (result: boolean, fileName: string) -> string
|
|
}
|
|
|
|
local core = {} :: Core
|
|
|
|
--[[
|
|
Recursively scans a model (table of Instances) for a Workspace instance.
|
|
|
|
### Example usage
|
|
|
|
```lua
|
|
local roblox = require("@lune/roblox")
|
|
local core = require("./core")
|
|
|
|
local model = roblox.deserializeModel(fileContents)
|
|
if core.scanForWorkspace(model) then
|
|
print("This model contains a Workspace instance.")
|
|
else
|
|
print("This model does not contain a Workspace instance.")
|
|
end
|
|
```
|
|
|
|
@param model The deserialized Roblox model to scan.
|
|
@param printInstanceNames If true, prints the full names of all instances in the model during the scan.
|
|
@return True if a Workspace instance is found, false otherwise.
|
|
]]--
|
|
function core.scanForWorkspace(model: types.model, printInstanceNames: boolean): boolean
|
|
assert(typeof(model) == "table", "Expected model to be of type 'table'")
|
|
assert(typeof(printInstanceNames) == "boolean", "Expected printInstanceNames to be of type 'boolean'")
|
|
for _, child in pairs(model) do
|
|
if printInstanceNames then
|
|
print(child:GetFullName())
|
|
end
|
|
if child:IsA("Workspace") or core.scanForWorkspace(child:GetChildren(), printInstanceNames) then
|
|
return true
|
|
end
|
|
end
|
|
return false
|
|
end
|
|
|
|
--[[
|
|
Returns true if the file has a .rbxm or .rbxmx extension, false otherwise.
|
|
|
|
### Example usage
|
|
|
|
```lua
|
|
local fs = require("@lune/fs")
|
|
|
|
local modelFile = fs.readFile("filePath.rbxm")
|
|
if core.isValidModelFile("filePath.rbxm") then
|
|
print("This is a valid Roblox model file.")
|
|
else
|
|
print("This is not a valid Roblox model file.")
|
|
end
|
|
```
|
|
|
|
@param contents The filename of the model file to check.
|
|
@return True if the file has a valid model extension, false otherwise.
|
|
]]--
|
|
function core.isValidModelFile(fileName: string): boolean
|
|
assert(typeof(fileName) == "string", "Expected fileName to be of type 'string'")
|
|
local ext = string.match(fileName, "%.([^%.]+)$")
|
|
return ext == "rbxm" or ext == "rbxmx"
|
|
end
|
|
|
|
|
|
--[[
|
|
Checks if the given file contents contain a Workspace instance.
|
|
|
|
If the file is a valid Roblox model file, attempts to deserialize it and scan for a Workspace instance.
|
|
If deserialization fails, falls back to a simple string search for the Workspace tag.
|
|
|
|
If zlib decompression options are enabled in opts, attempts to decompress the file contents before scanning.
|
|
|
|
### Example usage
|
|
|
|
```lua
|
|
local fs = require("@lune/fs")
|
|
local core = require("./core")
|
|
|
|
local fileContents = fs.readFile("model.rbxm")
|
|
local opts = {
|
|
printInstanceNames = true,
|
|
zlibDecompressFiles = true,
|
|
zlibDecompressFilesRecursive = false
|
|
-- other options...
|
|
}
|
|
if core.fileContainsWorkspace(fileContents, opts) then
|
|
print("The file contains a Workspace instance.")
|
|
else
|
|
print("The file does not contain a Workspace instance.")
|
|
end
|
|
```
|
|
@param fileContents The contents of the file to check.
|
|
@param opts Options table of type `types.opts`
|
|
@return True if a Workspace instance is found, false otherwise.
|
|
]]
|
|
function core.fileContainsWorkspace(fileContents: string, opts: types.opts): boolean
|
|
assert(typeof(fileContents) == "string", "Expected fileContents to be of type 'string'")
|
|
assert(typeof(opts) == "table", "Expected opts to be of type 'table'")
|
|
if opts.zlibDecompressFiles or opts.zlibDecompressFilesRecursive then
|
|
local success = pcall(function() fileContents = compression.zlibDecompress(fileContents, opts) end)
|
|
if not success then
|
|
stdio.write(stdio.color("yellow"))
|
|
stdio.write("Warning: Failed to decompress file with zlib. Proceeding with original contents.\n")
|
|
stdio.write(stdio.color("reset"))
|
|
end
|
|
end
|
|
local success, instances = pcall(function() return roblox.deserializeModel(fileContents) end)
|
|
if not success then
|
|
return string.find(fileContents, "Item class=\"Workspace\"") and true or false
|
|
end
|
|
return core.scanForWorkspace(instances, opts.printInstanceNames)
|
|
end
|
|
|
|
|
|
--[[
|
|
Formats the workspace detection result as a colored string.
|
|
|
|
@param result The boolean result of the workspace detection.
|
|
@param fileName The name of the file that was checked.
|
|
@return A formatted string indicating whether the file contains a Workspace instance, colored green for true and yellow for false.
|
|
]]--
|
|
function core.formatResult(result: boolean, fileName: string): string
|
|
assert(typeof(result) == "boolean", "Expected result to be of type 'boolean'")
|
|
assert(typeof(fileName) == "string", "Expected fileName to be of type 'string'")
|
|
if result then
|
|
return stdio.color("green") .. `File {fileName} contains a Workspace instance.` .. stdio.color("reset")
|
|
else
|
|
return stdio.color("yellow") .. `File {fileName} does not contain a Workspace instance.` .. stdio.color("reset")
|
|
end
|
|
end
|
|
|
|
return core
|