summaryrefslogtreecommitdiffstats
path: root/acf/transaction/backend.lua
blob: 4393101738d445afb6160e8c32dfdc700e214816 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
--[[
Copyright (c) 2012-2013 Kaarle Ritvanen
See LICENSE file for license details
--]]

local M = {}

local err = require('acf.error')

-- TODO each transaction backend (i.e. persistence manager or
-- transaction proper) should be implemented as a thread or have its
-- internal state stored in shared storage (with appropriate locking)


local generation = 0
function M.gen_number()
   generation = generation + 1
   return generation
end


M.TransactionBackend = require('acf.object').class()

function M.TransactionBackend:init() self.mod_time = {} end

function M.TransactionBackend:get_if_older(path, timestamp)
   local value, ts = self:get(path)
   if ts > timestamp then err.raise('conflict', path) end
   return value, ts
end

function M.TransactionBackend:set(path, value)
   self:set_multiple{{path, value}}
end

function M.TransactionBackend:set_multiple(mods)
   -- TODO delegate to PM backends?
   local timestamp = M.gen_number()
   local effective = {}

   local function tostr(s) return s ~= nil and tostring(s) or nil end

   for _, mod in ipairs(mods) do
      local path, value = unpack(mod)

      if type(value) == 'table' or type(
	 self:get(path)
      ) == 'table' or self:get(path) ~= value then

	 table.insert(effective, mod)
	 self.mod_time[path] = timestamp
      end
   end

   self:_set_multiple(effective)
end

-- TODO should be atomic, mutex with set_multiple
function M.TransactionBackend:comp_and_setm(accessed, mods)
   local errors = err.ErrorDict()
   for path, timestamp in pairs(accessed) do
      errors:collect(self.get_if_older, self, path, timestamp)
   end
   errors:raise()

   self:set_multiple(mods)
end


return M