LazyModules (LM) is an opinionated framework for roblox games. It is based on streamlining issues we (Rusty and Vijet) have encountered making moderate-to-large, complex games in roblox. If you use it the way we have used it, it is transformational in what it feels like to write and debug a roblox game.
The minimal release is recommended; the libraries and practices for the base and full releases are still being ironed out. In fact the full release is just bonkers and ultimately an experiment with how much of the "new game" process we can pre-solve in principle.
A template file that uses all stages supported in the full release with their types.
LM does Module-globbing, i.e. it requires all modules (specified by directory whitelist/blacklists in Config.luau). However, you are never intended to require LM in your modules, so your game itself still uses require
trees to form is structure. The globbing process uses a require wrapper that can be used to automate error reporting if production has some startup issue not present in dev environments.
Similar to Knit, LM supports an "execution model" which we refer to as startup stages, but they are available everywhere. Through these stages, LM is passed into your modules as long as you define e.g. a mod.__init(G)
function. The Game
object (abbreviated as G
) can be used to solve file-dependency issues with G:Get("<ModuleName>")
. However you should structure your project as if you do not have ability and only use it when you deem it absolutely necessary.
Some startup stages are used to remove the need for boilerplate when doing common things like networking. The __build_signals(G, B)
stage provides B
which can be used to create remote events with run-context-agnostic usage.
local MyTransmitter
function mod.__build_signals(G, B)
MyTransmitter = B:NewTransmitter("MyTransmitter")
:ServerConnection(function(plr, ...)
end)
:ClientConnection(function(_, ...)
end)
end
A certain startup stage being executed has undefined module-order, so definitions like the example above are provided through callbacks when the stage ends:
local MyTransmitter
function mod2.__build_signals(G, B)
B:GetTransmitter("MyTransmitter", function(v), MyTransmitter = v end)
end
But wait, how do I know when it's safe to use MyTransmitter
? This is a problem that tends to crop up in complex/large codebases, not just in the example above, in fact it's exactly the reason we want execution stages to begin with:
local MyTransmitter
function mod.__init(G)
MyTransmitter:Transmit("data") -- error: MyTransmitter is nil
end
function mod.__run(G)
MyTransmitter:Transmit("data") -- OK: __run is after __build_signals which is after __init
end
function mod2.__build_signals(G, B)
B:GetTransmitter("MyTransmitter", function(v), MyTransmitter = v end)
end
Instead of firing signals or putting callbacks everywhere, we use startup stages to synchronize a known "when" for functionality. Advanced use cases can even implement custom startup stages that build the synchronization directly into LM.
One of the startup stages available is __tests. This allows in-file (or out of file) testing of the game as it is when it runs. In principal, your tests can move the player, use abilities, test datastores, etc.
function mod.__tests(G, T)
T:Test("Do this thing", function()
local a = 1
T:ForContext("in this context",
T.Equal, a, 1,
T.LessThan, a, 2
)
end)
end
LM has "tiers" of releases which correspond to different levels of invasiveness or bloat.
- Minimal
- The barebones of just startup stages, including a tests stage
- Base
Adds DebugMenu(on hold from Pumpkin removal)- Add basic debug visualizer library
- Full
- Adds a Player class accessible from
Game[plr]
after DataStore for associated player is fetched - Adds
PlayerDataModules
associated with the Player class - Guarantees the player object, game state, and datastore data exists by the time
__run
is running on the client
- Adds a Player class accessible from
- Dedicated Docs
- Improve types
- Add types to dependencies that are missing them
- Investigate the quality of all packaged dependencies
- BIG: Optimize networking library (buffer packing, single remote)
- Re-examine the usage of Pumpkin in place of lighter weight frameworks
- Integrate our replacement for ContextActionSerice and Controllers (requires the default PlayerModule to be overriden)