104 lines
3.0 KiB
Lua
104 lines
3.0 KiB
Lua
|
|
function mtul.utils.get_resource(modname, resource, ...)
|
|
if not resource then
|
|
resource = modname
|
|
modname = minetest.get_current_modname()
|
|
end
|
|
return table.concat({minetest.get_modpath(modname), resource, ...}, "/")
|
|
end
|
|
local function trim_spacing(text)
|
|
return text:match"^%s*(.-)%s*$"
|
|
end
|
|
--I will add a file reading lib eventually...
|
|
local read_file = mtul.file.read or function(mod, filename)
|
|
local filepath = mtul.utils.get_resource(mod, filename)
|
|
local file, err = io.open(filename, "r")
|
|
if file == nil then return nil, err end
|
|
local content = file:read"*a"
|
|
file:close()
|
|
end
|
|
|
|
local mod_info
|
|
function mtul.utils.get_mod_info()
|
|
if mod_info then return mod_info end
|
|
mod_info = {}
|
|
-- TODO validate modnames
|
|
local modnames = minetest.get_modnames()
|
|
for _, mod in pairs(modnames) do
|
|
local info
|
|
local mod_conf = Settings(mtul.utils.get_resource(mod, "mod.conf"))
|
|
if mod_conf then
|
|
info = {}
|
|
mod_conf = mod_conf:to_table()
|
|
local function read_depends(field)
|
|
local depends = {}
|
|
for depend in (mod_conf[field] or ""):gmatch"[^,]+" do
|
|
depends[trim_spacing(depend)] = true
|
|
end
|
|
info[field] = depends
|
|
end
|
|
read_depends"depends"
|
|
read_depends"optional_depends"
|
|
else
|
|
info = {
|
|
description = read_file(mod, "description.txt"),
|
|
depends = {},
|
|
optional_depends = {}
|
|
}
|
|
local depends_txt = read_file(mod, "depends.txt")
|
|
if depends_txt then
|
|
local trimmed = {}
|
|
for key, value in pairs(string.split(depends_txt or "", "\n")) do
|
|
trimmed[key] = trim_spacing(value)
|
|
end
|
|
for _, dependency in ipairs(trimmed) do
|
|
local modname, is_optional = dependency:match"(.+)(%??)"
|
|
table.insert(is_optional == "" and info.depends or info.optional_depends, modname)
|
|
end
|
|
end
|
|
end
|
|
if info.name == nil then
|
|
info.name = mod
|
|
end
|
|
mod_info[mod] = info
|
|
end
|
|
return mod_info
|
|
end
|
|
local mod_load_order
|
|
function mtul.utils.get_mod_load_order()
|
|
if mod_load_order then return mod_load_order end
|
|
mod_load_order = {}
|
|
local mod_info = mtul.utils.get_mod_info()
|
|
-- If there are circular soft dependencies, it is possible that a mod is loaded, but not in the right order
|
|
-- TODO somehow maximize the number of soft dependencies fulfilled in case of circular soft dependencies
|
|
local function load(mod)
|
|
if mod.status == "loaded" then
|
|
return true
|
|
end
|
|
if mod.status == "loading" then
|
|
return false
|
|
end
|
|
-- TODO soft/vs hard loading status, reset?
|
|
mod.status = "loading"
|
|
-- Try hard dependencies first. These must be fulfilled.
|
|
for depend in pairs(mod.depends) do
|
|
if not load(mod_info[depend]) then
|
|
return false
|
|
end
|
|
end
|
|
-- Now, try soft dependencies.
|
|
for depend in pairs(mod.optional_depends) do
|
|
-- Mod may not exist
|
|
if mod_info[depend] then
|
|
load(mod_info[depend])
|
|
end
|
|
end
|
|
mod.status = "loaded"
|
|
table.insert(mod_load_order, mod)
|
|
return true
|
|
end
|
|
for _, mod in pairs(mod_info) do
|
|
assert(load(mod))
|
|
end
|
|
return mod_load_order
|
|
end |