2022-05-11 17:02:21 +00:00
|
|
|
--
|
|
|
|
-- MODBUS Transaction Controller
|
|
|
|
--
|
|
|
|
|
|
|
|
local util = require("scada-common.util")
|
|
|
|
|
|
|
|
local txnctrl = {}
|
|
|
|
|
2023-02-25 04:36:16 +00:00
|
|
|
local TIMEOUT = 2000 -- 2000ms max wait
|
2022-05-11 17:02:21 +00:00
|
|
|
|
|
|
|
-- create a new transaction controller
|
2023-02-25 04:36:16 +00:00
|
|
|
---@nodiscard
|
2022-05-31 19:36:17 +00:00
|
|
|
function txnctrl.new()
|
2022-05-11 17:02:21 +00:00
|
|
|
local self = {
|
|
|
|
list = {},
|
|
|
|
next_id = 0
|
|
|
|
}
|
|
|
|
|
|
|
|
---@class transaction_controller
|
|
|
|
local public = {}
|
|
|
|
|
|
|
|
local insert = table.insert
|
2022-09-19 02:25:59 +00:00
|
|
|
local remove = table.remove
|
2022-05-11 17:02:21 +00:00
|
|
|
|
|
|
|
-- get the length of the transaction list
|
2023-02-25 04:36:16 +00:00
|
|
|
---@nodiscard
|
2022-05-31 19:36:17 +00:00
|
|
|
function public.length()
|
2022-05-11 17:02:21 +00:00
|
|
|
return #self.list
|
|
|
|
end
|
|
|
|
|
|
|
|
-- check if there are no active transactions
|
2023-02-25 04:36:16 +00:00
|
|
|
---@nodiscard
|
2022-05-31 19:36:17 +00:00
|
|
|
function public.empty()
|
2022-05-11 17:02:21 +00:00
|
|
|
return #self.list == 0
|
|
|
|
end
|
|
|
|
|
|
|
|
-- create a new transaction of the given type
|
2023-02-25 04:36:16 +00:00
|
|
|
---@nodiscard
|
2022-05-11 17:02:21 +00:00
|
|
|
---@param txn_type integer
|
|
|
|
---@return integer txn_id
|
2022-05-31 19:36:17 +00:00
|
|
|
function public.create(txn_type)
|
2022-05-11 17:02:21 +00:00
|
|
|
local txn_id = self.next_id
|
|
|
|
|
|
|
|
insert(self.list, {
|
|
|
|
txn_id = txn_id,
|
|
|
|
txn_type = txn_type,
|
|
|
|
expiry = util.time() + TIMEOUT
|
|
|
|
})
|
|
|
|
|
|
|
|
self.next_id = self.next_id + 1
|
|
|
|
|
|
|
|
return txn_id
|
|
|
|
end
|
|
|
|
|
|
|
|
-- mark a transaction as resolved to get its transaction type
|
2023-02-25 04:36:16 +00:00
|
|
|
---@nodiscard
|
2022-05-11 17:02:21 +00:00
|
|
|
---@param txn_id integer
|
|
|
|
---@return integer txn_type
|
2022-05-31 19:36:17 +00:00
|
|
|
function public.resolve(txn_id)
|
2022-05-11 17:02:21 +00:00
|
|
|
local txn_type = nil
|
|
|
|
|
|
|
|
for i = 1, public.length() do
|
|
|
|
if self.list[i].txn_id == txn_id then
|
2022-09-19 02:25:59 +00:00
|
|
|
local entry = remove(self.list, i)
|
|
|
|
txn_type = entry.txn_type
|
|
|
|
break
|
2022-05-11 17:02:21 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
return txn_type
|
|
|
|
end
|
|
|
|
|
2022-05-17 21:16:04 +00:00
|
|
|
-- renew a transaction by re-inserting it with its ID and type
|
|
|
|
---@param txn_id integer
|
|
|
|
---@param txn_type integer
|
2022-05-31 19:36:17 +00:00
|
|
|
function public.renew(txn_id, txn_type)
|
2022-05-17 21:16:04 +00:00
|
|
|
insert(self.list, {
|
|
|
|
txn_id = txn_id,
|
|
|
|
txn_type = txn_type,
|
|
|
|
expiry = util.time() + TIMEOUT
|
|
|
|
})
|
|
|
|
end
|
|
|
|
|
2022-05-11 17:02:21 +00:00
|
|
|
-- close timed-out transactions
|
2022-05-31 19:36:17 +00:00
|
|
|
function public.cleanup()
|
2022-05-11 17:02:21 +00:00
|
|
|
local now = util.time()
|
2022-05-16 21:11:46 +00:00
|
|
|
util.filter_table(self.list, function (txn) return txn.expiry > now end)
|
2022-05-11 17:02:21 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
-- clear the transaction list
|
2022-05-31 19:36:17 +00:00
|
|
|
function public.clear()
|
2022-05-11 17:02:21 +00:00
|
|
|
self.list = {}
|
|
|
|
end
|
|
|
|
|
|
|
|
return public
|
|
|
|
end
|
|
|
|
|
|
|
|
return txnctrl
|