Module persist

This module allows to save and retrieve Lua values in table-like objects, in a non-volatile way.

Once stored in a persisted table, these Lua values can be retrieved even after the Lua process and/or the CPU running it have been rebooted.

Compared to raw files, the persist module offers a higher level of abstraction, allowing to save and retrieve Lua objects directly, without explicitly dealing with serialization, deserialization or other filesystem issues. It can also be ported to environmnets which don't have a filesystem.

This version of the module naively writes data in a file, and keeps whole tables' content in RAM. Its purpose is to avoid using the more efficient QDBM version, whose LGPL license might be problematic to some use cases.

Persistence services are offered through two public APIs:

  • general purpose persisted tables: they behave mostly as regular tables, except that their content survives across reboots. These tables are created with table.new;

  • To easily save and retrieve isolated objects, persist.save and persist.load allow single-line operations with no extra bookkeeping.

Persisted Tables.

Persisted tables behave mostly as regular Lua tables, except that their content survives across reboots. They can hold strings, numbers, booleans, and possibly nested tables thereof, both as their keys and as their values.

Functions can be persisted if and only if they don't capture any upvalue (see examples below).

Beware that tables are stored structurally; what's retrieved are copies of the objects store in them, not the objects themselves:

local persist = require 'persist'
t = persist.table.new('test')
t[1] = { }
assert(t[1] ~= t[1])

However, loops and shared table parts are preserved within a single item (key or value) stored and retrieved from a persisted table:

local y  = { }
local x1 = { y1=y; y2=y } -- y1 and y2 point to the same object
x1.x = x                  -- x1 points to itself through its field x
assert(x1.x == x1)
assert(x1.y1 == x1.y2)
t[2] = x1                 -- save it in a table
local x2 = t[2]           -- retrieve a copy from the table
assert(x2 ~= x1)          -- it's a copy, not the original
assert(x2.x == x2)        -- but it points to itself
assert(x2.y1 == x2.y2)    -- and its shared parts are still shared

Since tables retrieved from persisted tables are actually copies, alterations of these returned tables won't affect the store's content:

x = { foo = 'bar' }
t.x = x
x.foo = 42 -- modifying `x`, not the copy in `t`
assert (t.x.foo == 'bar') -- `t` remains unchanged
t.x = x -- overriding `t.x` with the whole `x` value
assert (t.x.foo == 42) -- now `t` reflects the change

"Simple" function, which don't capture any local variable, can be saved:

function plus_one(x) return x+1 end
t.plus_one = plus_one
assert(t.plus_one(1) == 2)

As for tables, persisted functions don't retain their identity:

assert(t.plus_one ~= t.plus_one)

Finally, functions which capture local variables cannot be saved. Below, the alternative implementation of plus_one captures a local variable i, and won't be properly serialized:

function make_incrementer (i)
    return function(x) return x+i end
end
plus_one = make_incrementer(1)
t.plus_one = plus_one

Persisted Objects.

The usual way to persist data with the persist module is to create a table object, then fill it with values to persist. However, this is neither practical nor efficient when only one or a couple of small values need to be saved.

To avoid a needless proliferation of tiny persisted tables, a pair of persist.save and persist.load functions are provided, to easily store individual objects with a single line of code.

Objects stored through this API have the same limitations as those stored in full-featured persisted tables: no userdata, no threads, no upvalues in functions, no preservation of table and function identities.

POSIX implementation details.

All objects are saved in a SQLITE3 database file called persist/persist.sqlite3.

Type persist

persist.load(name)

Retrieve from flash an object saved with persist.save.

persist.save(name, obj)

Saves an object for later retrieval.

persist.table

The table sub-module of persist

Type table

table.empty(t)

Empties a table and releases associateed resources.

table.new(name)

Creates or loads a new persisted table.

Type persist

Field(s)

persist.load(name)

Retrieve from flash an object saved with persist.save.

Parameter

  • name : the name of the persisted object to load.

Return value

the object stored under that name, or nil if no such object exists.

persist.save(name, obj)

Saves an object for later retrieval.

If the saving operation cannot be performed successfully, an error is thrown. Objects saved with this function can be retrieved with persist.load, by giving back the same name, even after a reboot.

Parameters

  • name : the name of the persisted object to save.

  • obj : the object to persist.

Usage:


persist.save ('xxx', 1357) -- Save it in the store as 'xxx'
[...] -- reboot
x = persist.load 'xxx' -- Retrieve it from the store
assert(x == 1357)

#table persist.table

The table sub-module of persist

Type table

The table sub-module of persist

Field(s)

table.empty(t)

Empties a table and releases associateed resources.

Parameter

  • t : persited table returned by table.new call.

table.new(name)

Creates or loads a new persisted table.

If a table already exists with the provided name, it is loaded; otherwise, a new one is created.

Parameter

  • name : persisted table name.

Return values

  1. the persisted table on success.

  2. nil + error message otherwise.