123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562 |
- --Minetest
- --Copyright (C) 2013 sapier
- --
- --This program is free software; you can redistribute it and/or modify
- --it under the terms of the GNU Lesser General Public License as published by
- --the Free Software Foundation; either version 2.1 of the License, or
- --(at your option) any later version.
- --
- --This program is distributed in the hope that it will be useful,
- --but WITHOUT ANY WARRANTY; without even the implied warranty of
- --MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- --GNU Lesser General Public License for more details.
- --
- --You should have received a copy of the GNU Lesser General Public License along
- --with this program; if not, write to the Free Software Foundation, Inc.,
- --51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
- --------------------------------------------------------------------------------
- function get_mods(path,retval,modpack)
- local mods = core.get_dirlist(path,true)
- for i=1,#mods,1 do
- local toadd = {}
- local modpackfile = nil
- toadd.name = mods[i]
- toadd.path = path .. DIR_DELIM .. mods[i] .. DIR_DELIM
- if modpack ~= nil and
- modpack ~= "" then
- toadd.modpack = modpack
- else
- local filename = path .. DIR_DELIM .. mods[i] .. DIR_DELIM .. "modpack.txt"
- local error = nil
- modpackfile,error = io.open(filename,"r")
- end
- if modpackfile ~= nil then
- modpackfile:close()
- toadd.is_modpack = true
- table.insert(retval,toadd)
- get_mods(path .. DIR_DELIM .. mods[i],retval,mods[i])
- else
- table.insert(retval,toadd)
- end
- end
- end
- --modmanager implementation
- modmgr = {}
- --------------------------------------------------------------------------------
- function modmgr.extract(modfile)
- if modfile.type == "zip" then
- local tempfolder = os.tempfolder()
- if tempfolder ~= nil and
- tempfolder ~= "" then
- core.create_dir(tempfolder)
- if core.extract_zip(modfile.name,tempfolder) then
- return tempfolder
- end
- end
- end
- return nil
- end
- -------------------------------------------------------------------------------
- function modmgr.getbasefolder(temppath)
- if temppath == nil then
- return {
- type = "invalid",
- path = ""
- }
- end
- local testfile = io.open(temppath .. DIR_DELIM .. "init.lua","r")
- if testfile ~= nil then
- testfile:close()
- return {
- type="mod",
- path=temppath
- }
- end
- testfile = io.open(temppath .. DIR_DELIM .. "modpack.txt","r")
- if testfile ~= nil then
- testfile:close()
- return {
- type="modpack",
- path=temppath
- }
- end
- local subdirs = core.get_dirlist(temppath,true)
- --only single mod or modpack allowed
- if #subdirs ~= 1 then
- return {
- type = "invalid",
- path = ""
- }
- end
- testfile =
- io.open(temppath .. DIR_DELIM .. subdirs[1] ..DIR_DELIM .."init.lua","r")
- if testfile ~= nil then
- testfile:close()
- return {
- type="mod",
- path= temppath .. DIR_DELIM .. subdirs[1]
- }
- end
- testfile =
- io.open(temppath .. DIR_DELIM .. subdirs[1] ..DIR_DELIM .."modpack.txt","r")
- if testfile ~= nil then
- testfile:close()
- return {
- type="modpack",
- path=temppath .. DIR_DELIM .. subdirs[1]
- }
- end
- return {
- type = "invalid",
- path = ""
- }
- end
- --------------------------------------------------------------------------------
- function modmgr.isValidModname(modpath)
- if modpath:find("-") ~= nil then
- return false
- end
- return true
- end
- --------------------------------------------------------------------------------
- function modmgr.parse_register_line(line)
- local pos1 = line:find("\"")
- local pos2 = nil
- if pos1 ~= nil then
- pos2 = line:find("\"",pos1+1)
- end
- if pos1 ~= nil and pos2 ~= nil then
- local item = line:sub(pos1+1,pos2-1)
- if item ~= nil and
- item ~= "" then
- local pos3 = item:find(":")
- if pos3 ~= nil then
- local retval = item:sub(1,pos3-1)
- if retval ~= nil and
- retval ~= "" then
- return retval
- end
- end
- end
- end
- return nil
- end
- --------------------------------------------------------------------------------
- function modmgr.parse_dofile_line(modpath,line)
- local pos1 = line:find("\"")
- local pos2 = nil
- if pos1 ~= nil then
- pos2 = line:find("\"",pos1+1)
- end
- if pos1 ~= nil and pos2 ~= nil then
- local filename = line:sub(pos1+1,pos2-1)
- if filename ~= nil and
- filename ~= "" and
- filename:find(".lua") then
- return modmgr.identify_modname(modpath,filename)
- end
- end
- return nil
- end
- --------------------------------------------------------------------------------
- function modmgr.identify_modname(modpath,filename)
- local testfile = io.open(modpath .. DIR_DELIM .. filename,"r")
- if testfile ~= nil then
- local line = testfile:read()
- while line~= nil do
- local modname = nil
- if line:find("minetest.register_tool") then
- modname = modmgr.parse_register_line(line)
- end
- if line:find("minetest.register_craftitem") then
- modname = modmgr.parse_register_line(line)
- end
- if line:find("minetest.register_node") then
- modname = modmgr.parse_register_line(line)
- end
- if line:find("dofile") then
- modname = modmgr.parse_dofile_line(modpath,line)
- end
- if modname ~= nil then
- testfile:close()
- return modname
- end
- line = testfile:read()
- end
- testfile:close()
- end
- return nil
- end
- --------------------------------------------------------------------------------
- function modmgr.render_modlist(render_list)
- local retval = ""
- if render_list == nil then
- if modmgr.global_mods == nil then
- modmgr.refresh_globals()
- end
- render_list = modmgr.global_mods
- end
- local list = render_list:get_list()
- local last_modpack = nil
- for i,v in ipairs(list) do
- if retval ~= "" then
- retval = retval ..","
- end
- local color = ""
- if v.is_modpack then
- local rawlist = render_list:get_raw_list()
- local all_enabled = true
- for j=1,#rawlist,1 do
- if rawlist[j].modpack == list[i].name and
- rawlist[j].enabled ~= true then
- all_enabled = false
- break
- end
- end
- if all_enabled == false then
- color = mt_color_grey
- else
- color = mt_color_dark_green
- end
- end
- if v.typ == "game_mod" then
- color = mt_color_blue
- else
- if v.enabled then
- color = mt_color_green
- end
- end
- retval = retval .. color
- if v.modpack ~= nil then
- retval = retval .. " "
- end
- retval = retval .. v.name
- end
- return retval
- end
- --------------------------------------------------------------------------------
- function modmgr.get_dependencies(modfolder)
- local toadd = ""
- if modfolder ~= nil then
- local filename = modfolder ..
- DIR_DELIM .. "depends.txt"
- local dependencyfile = io.open(filename,"r")
- if dependencyfile then
- local dependency = dependencyfile:read("*l")
- while dependency do
- if toadd ~= "" then
- toadd = toadd .. ","
- end
- toadd = toadd .. dependency
- dependency = dependencyfile:read()
- end
- dependencyfile:close()
- end
- end
- return toadd
- end
- --------------------------------------------------------------------------------
- function modmgr.get_worldconfig(worldpath)
- local filename = worldpath ..
- DIR_DELIM .. "world.mt"
- local worldfile = Settings(filename)
- local worldconfig = {}
- worldconfig.global_mods = {}
- worldconfig.game_mods = {}
- for key,value in pairs(worldfile:to_table()) do
- if key == "gameid" then
- worldconfig.id = value
- else
- worldconfig.global_mods[key] = core.is_yes(value)
- end
- end
- --read gamemods
- local gamespec = gamemgr.find_by_gameid(worldconfig.id)
- gamemgr.get_game_mods(gamespec, worldconfig.game_mods)
- return worldconfig
- end
- --------------------------------------------------------------------------------
- function modmgr.installmod(modfilename,basename)
- local modfile = modmgr.identify_filetype(modfilename)
- local modpath = modmgr.extract(modfile)
- if modpath == nil then
- gamedata.errormessage = fgettext("Install Mod: file: \"$1\"", modfile.name) ..
- fgettext("\nInstall Mod: unsupported filetype \"$1\" or broken archive", modfile.type)
- return
- end
- local basefolder = modmgr.getbasefolder(modpath)
- if basefolder.type == "modpack" then
- local clean_path = nil
- if basename ~= nil then
- clean_path = "mp_" .. basename
- end
- if clean_path == nil then
- clean_path = get_last_folder(cleanup_path(basefolder.path))
- end
- if clean_path ~= nil then
- local targetpath = core.get_modpath() .. DIR_DELIM .. clean_path
- if not core.copy_dir(basefolder.path,targetpath) then
- gamedata.errormessage = fgettext("Failed to install $1 to $2", basename, targetpath)
- end
- else
- gamedata.errormessage = fgettext("Install Mod: unable to find suitable foldername for modpack $1", modfilename)
- end
- end
- if basefolder.type == "mod" then
- local targetfolder = basename
- if targetfolder == nil then
- targetfolder = modmgr.identify_modname(basefolder.path,"init.lua")
- end
- --if heuristic failed try to use current foldername
- if targetfolder == nil then
- targetfolder = get_last_folder(basefolder.path)
- end
- if targetfolder ~= nil and modmgr.isValidModname(targetfolder) then
- local targetpath = core.get_modpath() .. DIR_DELIM .. targetfolder
- core.copy_dir(basefolder.path,targetpath)
- else
- gamedata.errormessage = fgettext("Install Mod: unable to find real modname for: $1", modfilename)
- end
- end
- core.delete_dir(modpath)
- modmgr.refresh_globals()
- end
- --------------------------------------------------------------------------------
- function modmgr.preparemodlist(data)
- local retval = {}
- local global_mods = {}
- local game_mods = {}
- --read global mods
- local modpath = core.get_modpath()
- if modpath ~= nil and
- modpath ~= "" then
- get_mods(modpath,global_mods)
- end
- for i=1,#global_mods,1 do
- global_mods[i].typ = "global_mod"
- table.insert(retval,global_mods[i])
- end
- --read game mods
- local gamespec = gamemgr.find_by_gameid(data.gameid)
- gamemgr.get_game_mods(gamespec, game_mods)
- for i=1,#game_mods,1 do
- game_mods[i].typ = "game_mod"
- table.insert(retval,game_mods[i])
- end
- if data.worldpath == nil then
- return retval
- end
- --read world mod configuration
- local filename = data.worldpath ..
- DIR_DELIM .. "world.mt"
- local worldfile = Settings(filename)
- for key,value in pairs(worldfile:to_table()) do
- if key:sub(1, 9) == "load_mod_" then
- key = key:sub(10)
- local element = nil
- for i=1,#retval,1 do
- if retval[i].name == key and
- not retval[i].is_modpack then
- element = retval[i]
- break
- end
- end
- if element ~= nil then
- element.enabled = core.is_yes(value)
- else
- core.log("info", "Mod: " .. key .. " " .. dump(value) .. " but not found")
- end
- end
- end
- return retval
- end
- --------------------------------------------------------------------------------
- function modmgr.comparemod(elem1,elem2)
- if elem1 == nil or elem2 == nil then
- return false
- end
- if elem1.name ~= elem2.name then
- return false
- end
- if elem1.is_modpack ~= elem2.is_modpack then
- return false
- end
- if elem1.typ ~= elem2.typ then
- return false
- end
- if elem1.modpack ~= elem2.modpack then
- return false
- end
- if elem1.path ~= elem2.path then
- return false
- end
- return true
- end
- --------------------------------------------------------------------------------
- function modmgr.mod_exists(basename)
- if modmgr.global_mods == nil then
- modmgr.refresh_globals()
- end
- if modmgr.global_mods:raw_index_by_uid(basename) > 0 then
- return true
- end
- return false
- end
- --------------------------------------------------------------------------------
- function modmgr.get_global_mod(idx)
- if modmgr.global_mods == nil then
- return nil
- end
- if idx == nil or idx < 1 or
- idx > modmgr.global_mods:size() then
- return nil
- end
- return modmgr.global_mods:get_list()[idx]
- end
- --------------------------------------------------------------------------------
- function modmgr.refresh_globals()
- modmgr.global_mods = filterlist.create(
- modmgr.preparemodlist, --refresh
- modmgr.comparemod, --compare
- function(element,uid) --uid match
- if element.name == uid then
- return true
- end
- end,
- nil, --filter
- {}
- )
- modmgr.global_mods:add_sort_mechanism("alphabetic", sort_mod_list)
- modmgr.global_mods:set_sortmode("alphabetic")
- end
- --------------------------------------------------------------------------------
- function modmgr.identify_filetype(name)
- if name:sub(-3):lower() == "zip" then
- return {
- name = name,
- type = "zip"
- }
- end
- if name:sub(-6):lower() == "tar.gz" or
- name:sub(-3):lower() == "tgz"then
- return {
- name = name,
- type = "tgz"
- }
- end
- if name:sub(-6):lower() == "tar.bz2" then
- return {
- name = name,
- type = "tbz"
- }
- end
- if name:sub(-2):lower() == "7z" then
- return {
- name = name,
- type = "7z"
- }
- end
- return {
- name = name,
- type = "ukn"
- }
- end
|