summaryrefslogtreecommitdiff
path: root/datastorage
diff options
context:
space:
mode:
Diffstat (limited to 'datastorage')
-rw-r--r--datastorage/README.md22
-rw-r--r--datastorage/depends.txt1
-rw-r--r--datastorage/init.lua98
3 files changed, 121 insertions, 0 deletions
diff --git a/datastorage/README.md b/datastorage/README.md
new file mode 100644
index 0000000..b15b07a
--- /dev/null
+++ b/datastorage/README.md
@@ -0,0 +1,22 @@
+datastorage
+===========
+
+Helper mod to manage players data.
+All the mods can acces a single file (container) and easily have the data saved/loaded for them.
+
+Usage
+-----
+
+ local data = datastorage.get(id, ...)
+
+Returns a reference to a data container. The id is normally a player name.
+Following arguments are keys to recurse into, normally only one, a string
+describing the type of data, is used. If the container doesn't exist it will
+be created, otherwise it will contain all previously stored data. The table
+can store any data. Player's containers will be saved to disk when the player
+leaves, and all references to the player's data should be dropped. All of the
+containers will be saved on server shutdown. To forcibly save a container's
+data use:
+
+ datastorage.save(id)
+
diff --git a/datastorage/depends.txt b/datastorage/depends.txt
new file mode 100644
index 0000000..8b13789
--- /dev/null
+++ b/datastorage/depends.txt
@@ -0,0 +1 @@
+
diff --git a/datastorage/init.lua b/datastorage/init.lua
new file mode 100644
index 0000000..30677fc
--- /dev/null
+++ b/datastorage/init.lua
@@ -0,0 +1,98 @@
+datastorage = {data = {}}
+
+local DIR_DELIM = DIR_DELIM or "/"
+local data_path = minetest.get_worldpath()..DIR_DELIM.."datastorage"..DIR_DELIM
+
+function datastorage.save(id)
+ local data = datastorage.data[id]
+ -- Check if the container is empty
+ if not data or not next(data) then return end
+ for _, sub_data in pairs(data) do
+ if not next(sub_data) then return end
+ end
+
+ local file = io.open(data_path..id, "w")
+ if not file then
+ -- Most likely the data directory doesn't exist, create it
+ -- and try again.
+ if minetest.mkdir then
+ minetest.mkdir(data_path)
+ else
+ -- Using os.execute like this is not very platform
+ -- independent or safe, but most platforms name their
+ -- directory creation utility mkdir, the data path is
+ -- unlikely to contain special characters, and the
+ -- data path is only mutable by the admin.
+ os.execute('mkdir "'..data_path..'"')
+ end
+ file = io.open(data_path..id, "w")
+ if not file then return end
+ end
+
+ local datastr = minetest.serialize(data)
+ if not datastr then return end
+
+ file:write(datastr)
+ file:close()
+ return true
+end
+
+function datastorage.load(id)
+ local file = io.open(data_path..id, "r")
+ if not file then return end
+
+ local data = minetest.deserialize(file:read("*all"))
+ datastorage.data[id] = data
+
+ file:close()
+ return data
+end
+
+-- Compatability
+function datastorage.get_container(player, id)
+ return datastorage.get(player:get_player_name(), id)
+end
+
+-- Retrieves a value from the data storage
+function datastorage.get(id, ...)
+ local last = datastorage.data[id]
+ if last == nil then last = datastorage.load(id) end
+ if last == nil then
+ last = {}
+ datastorage.data[id] = last
+ end
+ local cur = last
+ for _, sub_id in ipairs({...}) do
+ last = cur
+ cur = cur[sub_id]
+ if cur == nil then
+ cur = {}
+ last[sub_id] = cur
+ end
+ end
+ return cur
+end
+
+-- Saves a container and reomves it from memory
+function datastorage.finish(id)
+ datastorage.save(id)
+ datastorage.data[id] = nil
+end
+
+-- Compatability
+function datastorage.save_container(player)
+ return datastorage.save(player:get_player_name())
+end
+
+minetest.register_on_leaveplayer(function(player)
+ local player_name = player:get_player_name()
+ datastorage.save(player_name)
+ datastorage.data[player_name] = nil
+end)
+
+minetest.register_on_shutdown(function()
+ for id in pairs(datastorage.data) do
+ datastorage.save(id)
+ end
+end)
+