mirror of
https://github.com/MikaylaFischler/cc-mek-scada.git
synced 2024-08-30 18:22:34 +00:00
874 lines
34 KiB
Lua
874 lines
34 KiB
Lua
local comms = require("scada-common.comms")
|
|
local log = require("scada-common.log")
|
|
local util = require("scada-common.util")
|
|
|
|
local iocontrol = require("pocket.iocontrol")
|
|
|
|
local PROTOCOL = comms.PROTOCOL
|
|
local DEVICE_TYPE = comms.DEVICE_TYPE
|
|
local ESTABLISH_ACK = comms.ESTABLISH_ACK
|
|
local MGMT_TYPE = comms.MGMT_TYPE
|
|
local CRDN_TYPE = comms.CRDN_TYPE
|
|
|
|
local LINK_STATE = iocontrol.LINK_STATE
|
|
|
|
local pocket = {}
|
|
|
|
local MQ__RENDER_CMD = {
|
|
UNLOAD_SV_APPS = 1,
|
|
UNLOAD_API_APPS = 2
|
|
}
|
|
|
|
local MQ__RENDER_DATA = {
|
|
LOAD_APP = 1
|
|
}
|
|
|
|
pocket.MQ__RENDER_CMD = MQ__RENDER_CMD
|
|
pocket.MQ__RENDER_DATA = MQ__RENDER_DATA
|
|
|
|
---@type pkt_config
|
|
local config = {}
|
|
|
|
pocket.config = config
|
|
|
|
-- load the pocket configuration
|
|
function pocket.load_config()
|
|
if not settings.load("/pocket.settings") then return false end
|
|
|
|
config.TempScale = settings.get("TempScale")
|
|
|
|
config.SVR_Channel = settings.get("SVR_Channel")
|
|
config.CRD_Channel = settings.get("CRD_Channel")
|
|
config.PKT_Channel = settings.get("PKT_Channel")
|
|
config.ConnTimeout = settings.get("ConnTimeout")
|
|
config.TrustedRange = settings.get("TrustedRange")
|
|
config.AuthKey = settings.get("AuthKey")
|
|
|
|
config.LogMode = settings.get("LogMode")
|
|
config.LogPath = settings.get("LogPath")
|
|
config.LogDebug = settings.get("LogDebug")
|
|
|
|
local cfv = util.new_validator()
|
|
|
|
cfv.assert_type_int(config.TempScale)
|
|
cfv.assert_range(config.TempScale, 1, 4)
|
|
|
|
cfv.assert_channel(config.SVR_Channel)
|
|
cfv.assert_channel(config.CRD_Channel)
|
|
cfv.assert_channel(config.PKT_Channel)
|
|
cfv.assert_type_num(config.ConnTimeout)
|
|
cfv.assert_min(config.ConnTimeout, 2)
|
|
cfv.assert_type_num(config.TrustedRange)
|
|
cfv.assert_min(config.TrustedRange, 0)
|
|
cfv.assert_type_str(config.AuthKey)
|
|
|
|
if type(config.AuthKey) == "string" then
|
|
local len = string.len(config.AuthKey)
|
|
cfv.assert(len == 0 or len >= 8)
|
|
end
|
|
|
|
cfv.assert_type_int(config.LogMode)
|
|
cfv.assert_range(config.LogMode, 0, 1)
|
|
cfv.assert_type_str(config.LogPath)
|
|
cfv.assert_type_bool(config.LogDebug)
|
|
|
|
return cfv.valid()
|
|
end
|
|
|
|
---@enum POCKET_APP_ID
|
|
local APP_ID = {
|
|
ROOT = 1,
|
|
LOADER = 2,
|
|
-- main app pages
|
|
UNITS = 3,
|
|
GUIDE = 4,
|
|
ABOUT = 5,
|
|
-- diag app page
|
|
ALARMS = 6,
|
|
-- other
|
|
DUMMY = 7,
|
|
NUM_APPS = 7
|
|
}
|
|
|
|
pocket.APP_ID = APP_ID
|
|
|
|
---@class nav_tree_page
|
|
---@field _p nav_tree_page|nil page's parent
|
|
---@field _c table page's children
|
|
---@field nav_to function function to navigate to this page
|
|
---@field switcher function|nil function to switch between children
|
|
---@field tasks table tasks to run while viewing this page
|
|
|
|
-- initialize the page navigation system
|
|
---@param smem pkt_shared_memory
|
|
function pocket.init_nav(smem)
|
|
local self = {
|
|
pane = nil, ---@type graphics_element
|
|
sidebar = nil, ---@type graphics_element
|
|
apps = {},
|
|
containers = {},
|
|
help_map = {},
|
|
help_return = nil,
|
|
loader_return = nil,
|
|
cur_app = APP_ID.ROOT
|
|
}
|
|
|
|
self.cur_page = self.root
|
|
|
|
---@class pocket_nav
|
|
local nav = {}
|
|
|
|
-- set the root pane element to switch between apps with
|
|
---@param root_pane graphics_element
|
|
function nav.set_pane(root_pane) self.pane = root_pane end
|
|
|
|
-- link sidebar element
|
|
---@param sidebar graphics_element
|
|
function nav.set_sidebar(sidebar) self.sidebar = sidebar end
|
|
|
|
-- register an app
|
|
---@param app_id POCKET_APP_ID app ID
|
|
---@param container graphics_element element that contains this app (usually a Div)
|
|
---@param pane? graphics_element multipane if this is a simple paned app, then nav_to must be a number
|
|
---@param require_sv? boolean true to specifiy if this app should be unloaded when the supervisor connection is lost
|
|
---@param require_api? boolean true to specifiy if this app should be unloaded when the api connection is lost
|
|
function nav.register_app(app_id, container, pane, require_sv, require_api)
|
|
---@class pocket_app
|
|
local app = {
|
|
loaded = false,
|
|
cur_page = nil, ---@type nav_tree_page
|
|
pane = pane,
|
|
paned_pages = {},
|
|
sidebar_items = {}
|
|
}
|
|
|
|
app.load = function () app.loaded = true end
|
|
app.unload = function () app.loaded = false end
|
|
|
|
-- check which connections this requires (for unload)
|
|
---@return boolean requires_sv, boolean requires_api
|
|
function app.check_requires() return require_sv or false, require_api or false end
|
|
|
|
-- check if any connection is required (for load)
|
|
function app.requires_conn() return require_sv or require_api or false end
|
|
|
|
-- delayed set of the pane if it wasn't ready at the start
|
|
---@param root_pane graphics_element multipane
|
|
function app.set_root_pane(root_pane)
|
|
app.pane = root_pane
|
|
end
|
|
|
|
-- configure the sidebar
|
|
---@param items table
|
|
function app.set_sidebar(items)
|
|
app.sidebar_items = items
|
|
if self.sidebar then self.sidebar.update(items) end
|
|
end
|
|
|
|
-- function to run on initial load into memory
|
|
---@param on_load function callback
|
|
function app.set_load(on_load)
|
|
app.load = function ()
|
|
on_load()
|
|
app.loaded = true
|
|
end
|
|
end
|
|
|
|
-- function to run to close out the app
|
|
---@param on_unload function callback
|
|
function app.set_unload(on_unload)
|
|
app.unload = function ()
|
|
on_unload()
|
|
app.loaded = false
|
|
end
|
|
end
|
|
|
|
-- if a pane was provided, this will switch between numbered pages
|
|
---@param idx integer page index
|
|
function app.switcher(idx)
|
|
if app.paned_pages[idx] then
|
|
app.paned_pages[idx].nav_to()
|
|
end
|
|
end
|
|
|
|
-- create a new page entry in the app's page navigation tree
|
|
---@param parent nav_tree_page|nil a parent page or nil to set this as the root
|
|
---@param nav_to function|integer function to navigate to this page or pane index
|
|
---@return nav_tree_page new_page this new page
|
|
function app.new_page(parent, nav_to)
|
|
---@type nav_tree_page
|
|
local page = { _p = parent, _c = {}, nav_to = function () end, switcher = function () end, tasks = {} }
|
|
|
|
if parent == nil and app.cur_page == nil then
|
|
app.cur_page = page
|
|
end
|
|
|
|
if type(nav_to) == "number" then
|
|
app.paned_pages[nav_to] = page
|
|
|
|
function page.nav_to()
|
|
app.cur_page = page
|
|
if app.pane then app.pane.set_value(nav_to) end
|
|
end
|
|
else
|
|
function page.nav_to()
|
|
app.cur_page = page
|
|
nav_to()
|
|
end
|
|
end
|
|
|
|
-- switch between children
|
|
---@param id integer child ID
|
|
function page.switcher(id) if page._c[id] then page._c[id].nav_to() end end
|
|
|
|
if parent ~= nil then
|
|
table.insert(page._p._c, page)
|
|
end
|
|
|
|
return page
|
|
end
|
|
|
|
-- delete paned pages and clear the current page
|
|
function app.delete_pages()
|
|
app.paned_pages = {}
|
|
app.cur_page = nil
|
|
end
|
|
|
|
-- get the currently active page
|
|
function app.get_current_page() return app.cur_page end
|
|
|
|
-- attempt to navigate up the tree
|
|
---@return boolean success true if successfully navigated up
|
|
function app.nav_up()
|
|
local parent = app.cur_page._p
|
|
if parent then parent.nav_to() end
|
|
return parent ~= nil
|
|
end
|
|
|
|
self.apps[app_id] = app
|
|
self.containers[app_id] = container
|
|
|
|
return app
|
|
end
|
|
|
|
-- open an app
|
|
---@param app_id POCKET_APP_ID
|
|
function nav.open_app(app_id)
|
|
-- reset help return on navigating out of an app
|
|
if app_id == APP_ID.ROOT then self.help_return = nil end
|
|
|
|
local app = self.apps[app_id] ---@type pocket_app
|
|
if app then
|
|
if app.requires_conn() and not smem.pkt_sys.pocket_comms.is_linked() then
|
|
-- bring up the app loader
|
|
self.loader_return = app_id
|
|
app_id = APP_ID.LOADER
|
|
app = self.apps[app_id]
|
|
else self.loader_return = nil end
|
|
|
|
if not app.loaded then smem.q.mq_render.push_data(MQ__RENDER_DATA.LOAD_APP, app_id) end
|
|
|
|
self.cur_app = app_id
|
|
self.pane.set_value(app_id)
|
|
|
|
if #app.sidebar_items > 0 then
|
|
self.sidebar.update(app.sidebar_items)
|
|
end
|
|
else
|
|
log.debug("tried to open unknown app")
|
|
end
|
|
end
|
|
|
|
-- open the app that was blocked on connecting
|
|
function nav.on_loader_connected()
|
|
if self.loader_return then
|
|
nav.open_app(self.loader_return)
|
|
end
|
|
end
|
|
|
|
-- load a given app
|
|
---@param app_id POCKET_APP_ID
|
|
function nav.load_app(app_id)
|
|
self.apps[app_id].load()
|
|
end
|
|
|
|
-- unload api-dependent apps
|
|
function nav.unload_api()
|
|
for id, app in pairs(self.apps) do
|
|
local _, api = app.check_requires()
|
|
if app.loaded and api then
|
|
if id == self.cur_app then nav.open_app(APP_ID.ROOT) end
|
|
app.unload()
|
|
end
|
|
end
|
|
end
|
|
|
|
-- unload supervisor-dependent apps
|
|
function nav.unload_sv()
|
|
for id, app in pairs(self.apps) do
|
|
local sv, _ = app.check_requires()
|
|
if app.loaded and sv then
|
|
if id == self.cur_app then nav.open_app(APP_ID.ROOT) end
|
|
app.unload()
|
|
end
|
|
end
|
|
end
|
|
|
|
-- get a list of the app containers (usually Div elements)
|
|
function nav.get_containers() return self.containers end
|
|
|
|
-- get the currently active page
|
|
---@return nav_tree_page
|
|
function nav.get_current_page()
|
|
return self.apps[self.cur_app].get_current_page()
|
|
end
|
|
|
|
-- attempt to navigate up within the active app, otherwise open home page<br>
|
|
-- except, this will go back to a prior app if leaving the help app after open_help was used
|
|
function nav.nav_up()
|
|
-- return out of help if opened with open_help
|
|
if self.help_return then
|
|
nav.open_app(self.help_return)
|
|
self.help_return = nil
|
|
return
|
|
end
|
|
|
|
local app = self.apps[self.cur_app] ---@type pocket_app
|
|
log.debug("attempting app nav up for app " .. self.cur_app)
|
|
|
|
if not app.nav_up() then
|
|
log.debug("internal app nav up failed, going to home screen")
|
|
nav.open_app(APP_ID.ROOT)
|
|
end
|
|
end
|
|
|
|
-- open the help app, to show the reference for a key
|
|
function nav.open_help(key)
|
|
self.help_return = self.cur_app
|
|
|
|
nav.open_app(APP_ID.GUIDE)
|
|
|
|
local load = self.help_map[key]
|
|
if load then load() end
|
|
end
|
|
|
|
-- link the help map from the guide app
|
|
function nav.link_help(map) self.help_map = map end
|
|
|
|
return nav
|
|
end
|
|
|
|
-- pocket coordinator + supervisor communications
|
|
---@nodiscard
|
|
---@param version string pocket version
|
|
---@param nic nic network interface device
|
|
---@param sv_watchdog watchdog
|
|
---@param api_watchdog watchdog
|
|
---@param nav pocket_nav
|
|
function pocket.comms(version, nic, sv_watchdog, api_watchdog, nav)
|
|
local self = {
|
|
sv = {
|
|
linked = false,
|
|
addr = comms.BROADCAST,
|
|
seq_num = util.time_ms() * 10, -- unique per peer, restarting will not re-use seq nums due to message rate
|
|
r_seq_num = nil, ---@type nil|integer
|
|
last_est_ack = ESTABLISH_ACK.ALLOW
|
|
},
|
|
api = {
|
|
linked = false,
|
|
addr = comms.BROADCAST,
|
|
seq_num = util.time_ms() * 10, -- unique per peer, restarting will not re-use seq nums due to message rate
|
|
r_seq_num = nil, ---@type nil|integer
|
|
last_est_ack = ESTABLISH_ACK.ALLOW
|
|
},
|
|
establish_delay_counter = 0
|
|
}
|
|
|
|
comms.set_trusted_range(config.TrustedRange)
|
|
|
|
-- PRIVATE FUNCTIONS --
|
|
|
|
-- configure network channels
|
|
nic.closeAll()
|
|
nic.open(config.PKT_Channel)
|
|
|
|
-- send a management packet to the supervisor
|
|
---@param msg_type MGMT_TYPE
|
|
---@param msg table
|
|
local function _send_sv(msg_type, msg)
|
|
local s_pkt = comms.scada_packet()
|
|
local pkt = comms.mgmt_packet()
|
|
|
|
pkt.make(msg_type, msg)
|
|
s_pkt.make(self.sv.addr, self.sv.seq_num, PROTOCOL.SCADA_MGMT, pkt.raw_sendable())
|
|
|
|
nic.transmit(config.SVR_Channel, config.PKT_Channel, s_pkt)
|
|
self.sv.seq_num = self.sv.seq_num + 1
|
|
end
|
|
|
|
-- send a management packet to the coordinator
|
|
---@param msg_type MGMT_TYPE
|
|
---@param msg table
|
|
local function _send_crd(msg_type, msg)
|
|
local s_pkt = comms.scada_packet()
|
|
local pkt = comms.mgmt_packet()
|
|
|
|
pkt.make(msg_type, msg)
|
|
s_pkt.make(self.api.addr, self.api.seq_num, PROTOCOL.SCADA_MGMT, pkt.raw_sendable())
|
|
|
|
nic.transmit(config.CRD_Channel, config.PKT_Channel, s_pkt)
|
|
self.api.seq_num = self.api.seq_num + 1
|
|
end
|
|
|
|
-- send an API packet to the coordinator
|
|
---@param msg_type CRDN_TYPE
|
|
---@param msg table
|
|
local function _send_api(msg_type, msg)
|
|
local s_pkt = comms.scada_packet()
|
|
local pkt = comms.crdn_packet()
|
|
|
|
pkt.make(msg_type, msg)
|
|
s_pkt.make(self.api.addr, self.api.seq_num, PROTOCOL.SCADA_CRDN, pkt.raw_sendable())
|
|
|
|
nic.transmit(config.CRD_Channel, config.PKT_Channel, s_pkt)
|
|
self.api.seq_num = self.api.seq_num + 1
|
|
end
|
|
|
|
-- attempt supervisor connection establishment
|
|
local function _send_sv_establish()
|
|
_send_sv(MGMT_TYPE.ESTABLISH, { comms.version, version, DEVICE_TYPE.PKT })
|
|
end
|
|
|
|
-- attempt coordinator API connection establishment
|
|
local function _send_api_establish()
|
|
_send_crd(MGMT_TYPE.ESTABLISH, { comms.version, version, DEVICE_TYPE.PKT, comms.api_version })
|
|
end
|
|
|
|
-- keep alive ack to supervisor
|
|
---@param srv_time integer
|
|
local function _send_sv_keep_alive_ack(srv_time)
|
|
_send_sv(MGMT_TYPE.KEEP_ALIVE, { srv_time, util.time() })
|
|
end
|
|
|
|
-- keep alive ack to coordinator
|
|
---@param srv_time integer
|
|
local function _send_api_keep_alive_ack(srv_time)
|
|
_send_crd(MGMT_TYPE.KEEP_ALIVE, { srv_time, util.time() })
|
|
end
|
|
|
|
-- PUBLIC FUNCTIONS --
|
|
|
|
---@class pocket_comms
|
|
local public = {}
|
|
|
|
-- close connection to the supervisor
|
|
function public.close_sv()
|
|
sv_watchdog.cancel()
|
|
nav.unload_sv()
|
|
self.sv.linked = false
|
|
self.sv.r_seq_num = nil
|
|
self.sv.addr = comms.BROADCAST
|
|
_send_sv(MGMT_TYPE.CLOSE, {})
|
|
end
|
|
|
|
-- close connection to coordinator API server
|
|
function public.close_api()
|
|
api_watchdog.cancel()
|
|
nav.unload_api()
|
|
self.api.linked = false
|
|
self.api.r_seq_num = nil
|
|
self.api.addr = comms.BROADCAST
|
|
_send_crd(MGMT_TYPE.CLOSE, {})
|
|
end
|
|
|
|
-- close the connections to the servers
|
|
function public.close()
|
|
public.close_sv()
|
|
public.close_api()
|
|
end
|
|
|
|
-- attempt to re-link if any of the dependent links aren't active
|
|
function public.link_update()
|
|
if not self.sv.linked then
|
|
iocontrol.report_link_state(util.trinary(self.api.linked, LINK_STATE.API_LINK_ONLY, LINK_STATE.UNLINKED))
|
|
|
|
if self.establish_delay_counter <= 0 then
|
|
_send_sv_establish()
|
|
self.establish_delay_counter = 4
|
|
else
|
|
self.establish_delay_counter = self.establish_delay_counter - 1
|
|
end
|
|
elseif not self.api.linked then
|
|
iocontrol.report_link_state(LINK_STATE.SV_LINK_ONLY)
|
|
|
|
if self.establish_delay_counter <= 0 then
|
|
_send_api_establish()
|
|
self.establish_delay_counter = 4
|
|
else
|
|
self.establish_delay_counter = self.establish_delay_counter - 1
|
|
end
|
|
else
|
|
-- linked, all good!
|
|
iocontrol.report_link_state(LINK_STATE.LINKED, self.sv.addr, self.api.addr)
|
|
end
|
|
end
|
|
|
|
-- supervisor get active alarm tones
|
|
function public.diag__get_alarm_tones()
|
|
if self.sv.linked then _send_sv(MGMT_TYPE.DIAG_TONE_GET, {}) end
|
|
end
|
|
|
|
-- supervisor test alarm tones by tone
|
|
---@param id TONE|0 tone ID, or 0 to stop all
|
|
---@param state boolean tone state
|
|
function public.diag__set_alarm_tone(id, state)
|
|
if self.sv.linked then _send_sv(MGMT_TYPE.DIAG_TONE_SET, { id, state }) end
|
|
end
|
|
|
|
-- supervisor test alarm tones by alarm
|
|
---@param id ALARM|0 alarm ID, 0 to stop all
|
|
---@param state boolean alarm state
|
|
function public.diag__set_alarm(id, state)
|
|
if self.sv.linked then _send_sv(MGMT_TYPE.DIAG_ALARM_SET, { id, state }) end
|
|
end
|
|
|
|
-- coordinator get unit data
|
|
function public.api__get_unit(unit)
|
|
if self.api.linked then _send_api(CRDN_TYPE.API_GET_UNIT, { unit }) end
|
|
end
|
|
|
|
-- parse a packet
|
|
---@param side string
|
|
---@param sender integer
|
|
---@param reply_to integer
|
|
---@param message any
|
|
---@param distance integer
|
|
---@return mgmt_frame|crdn_frame|nil packet
|
|
function public.parse_packet(side, sender, reply_to, message, distance)
|
|
local s_pkt = nic.receive(side, sender, reply_to, message, distance)
|
|
local pkt = nil
|
|
|
|
if s_pkt then
|
|
-- get as SCADA management packet
|
|
if s_pkt.protocol() == PROTOCOL.SCADA_MGMT then
|
|
local mgmt_pkt = comms.mgmt_packet()
|
|
if mgmt_pkt.decode(s_pkt) then
|
|
pkt = mgmt_pkt.get()
|
|
end
|
|
-- get as coordinator packet
|
|
elseif s_pkt.protocol() == PROTOCOL.SCADA_CRDN then
|
|
local crdn_pkt = comms.crdn_packet()
|
|
if crdn_pkt.decode(s_pkt) then
|
|
pkt = crdn_pkt.get()
|
|
end
|
|
else
|
|
log.debug("attempted parse of illegal packet type " .. s_pkt.protocol(), true)
|
|
end
|
|
end
|
|
|
|
return pkt
|
|
end
|
|
|
|
---@param packet mgmt_frame|crdn_frame
|
|
---@param length integer
|
|
---@param max integer?
|
|
---@return boolean
|
|
local function _check_length(packet, length, max)
|
|
local ok = util.trinary(max == nil, packet.length == length, packet.length >= length and packet.length <= (max or 0))
|
|
if not ok then
|
|
local fmt = "[comms] RX_PACKET{r_chan=%d,proto=%d,type=%d}: packet length mismatch -> expect %d != actual %d"
|
|
log.debug(util.sprintf(fmt, packet.scada_frame.remote_channel(), packet.scada_frame.protocol(), packet.type, length, packet.scada_frame.length()))
|
|
end
|
|
return ok
|
|
end
|
|
|
|
---@param packet mgmt_frame|crdn_frame
|
|
local function _fail_type(packet)
|
|
local fmt = "[comms] RX_PACKET{r_chan=%d,proto=%d,type=%d}: unrecognized packet type"
|
|
log.debug(util.sprintf(fmt, packet.scada_frame.remote_channel(), packet.scada_frame.protocol(), packet.type))
|
|
end
|
|
|
|
-- handle a packet
|
|
---@param packet mgmt_frame|crdn_frame|nil
|
|
function public.handle_packet(packet)
|
|
local diag = iocontrol.get_db().diag
|
|
|
|
if packet ~= nil then
|
|
local l_chan = packet.scada_frame.local_channel()
|
|
local r_chan = packet.scada_frame.remote_channel()
|
|
local protocol = packet.scada_frame.protocol()
|
|
local src_addr = packet.scada_frame.src_addr()
|
|
|
|
if l_chan ~= config.PKT_Channel then
|
|
log.debug("received packet on unconfigured channel " .. l_chan, true)
|
|
elseif r_chan == config.CRD_Channel then
|
|
-- check sequence number
|
|
if self.api.r_seq_num == nil then
|
|
self.api.r_seq_num = packet.scada_frame.seq_num() + 1
|
|
elseif self.api.r_seq_num ~= packet.scada_frame.seq_num() then
|
|
log.warning("sequence out-of-order (API): last = " .. self.api.r_seq_num .. ", new = " .. packet.scada_frame.seq_num())
|
|
return
|
|
elseif self.api.linked and (src_addr ~= self.api.addr) then
|
|
log.debug("received packet from unknown computer " .. src_addr .. " while linked (API expected " .. self.api.addr ..
|
|
"); channel in use by another system?")
|
|
return
|
|
else
|
|
self.api.r_seq_num = packet.scada_frame.seq_num() + 1
|
|
end
|
|
|
|
-- feed watchdog on valid sequence number
|
|
api_watchdog.feed()
|
|
|
|
if protocol == PROTOCOL.SCADA_CRDN then
|
|
---@cast packet crdn_frame
|
|
if self.api.linked then
|
|
if packet.type == CRDN_TYPE.API_GET_FAC then
|
|
if _check_length(packet, 11) then
|
|
iocontrol.record_facility_data(packet.data)
|
|
end
|
|
elseif packet.type == CRDN_TYPE.API_GET_UNIT then
|
|
if _check_length(packet, 11) and type(packet.data[1]) == "number" and iocontrol.get_db().units[packet.data[1]] then
|
|
iocontrol.record_unit_data(packet.data)
|
|
end
|
|
else _fail_type(packet) end
|
|
else
|
|
log.debug("discarding coordinator SCADA_CRDN packet before linked")
|
|
end
|
|
elseif protocol == PROTOCOL.SCADA_MGMT then
|
|
---@cast packet mgmt_frame
|
|
if self.api.linked then
|
|
if packet.type == MGMT_TYPE.KEEP_ALIVE then
|
|
-- keep alive request received, echo back
|
|
if _check_length(packet, 1) then
|
|
local timestamp = packet.data[1]
|
|
local trip_time = util.time() - timestamp
|
|
|
|
if trip_time > 750 then
|
|
log.warning("pocket coordinator KEEP_ALIVE trip time > 750ms (" .. trip_time .. "ms)")
|
|
end
|
|
|
|
-- log.debug("pocket coordinator TT = " .. trip_time .. "ms")
|
|
|
|
_send_api_keep_alive_ack(timestamp)
|
|
|
|
iocontrol.report_crd_tt(trip_time)
|
|
end
|
|
elseif packet.type == MGMT_TYPE.CLOSE then
|
|
-- handle session close
|
|
api_watchdog.cancel()
|
|
nav.unload_api()
|
|
self.api.linked = false
|
|
self.api.r_seq_num = nil
|
|
self.api.addr = comms.BROADCAST
|
|
log.info("coordinator server connection closed by remote host")
|
|
else _fail_type(packet) end
|
|
elseif packet.type == MGMT_TYPE.ESTABLISH then
|
|
-- connection with coordinator established
|
|
if _check_length(packet, 1, 2) then
|
|
local est_ack = packet.data[1]
|
|
|
|
if est_ack == ESTABLISH_ACK.ALLOW then
|
|
if packet.length == 2 then
|
|
local fac_config = packet.data[2]
|
|
|
|
if type(fac_config) == "table" and #fac_config == 2 then
|
|
-- get configuration
|
|
local conf = { num_units = fac_config[1], cooling = fac_config[2] }
|
|
|
|
iocontrol.init_fac(conf, config.TempScale)
|
|
|
|
log.info("coordinator connection established")
|
|
self.establish_delay_counter = 0
|
|
self.api.linked = true
|
|
self.api.addr = src_addr
|
|
|
|
if self.sv.linked then
|
|
iocontrol.report_link_state(LINK_STATE.LINKED, self.sv.addr, self.api.addr)
|
|
else
|
|
iocontrol.report_link_state(LINK_STATE.API_LINK_ONLY)
|
|
end
|
|
else
|
|
log.debug("invalid facility configuration table received from coordinator, establish failed")
|
|
end
|
|
else
|
|
log.debug("received coordinator establish allow without facility configuration")
|
|
end
|
|
elseif est_ack == ESTABLISH_ACK.DENY then
|
|
if self.api.last_est_ack ~= est_ack then
|
|
log.info("coordinator connection denied")
|
|
end
|
|
elseif est_ack == ESTABLISH_ACK.COLLISION then
|
|
if self.api.last_est_ack ~= est_ack then
|
|
log.info("coordinator connection denied due to collision")
|
|
end
|
|
elseif est_ack == ESTABLISH_ACK.BAD_VERSION then
|
|
if self.api.last_est_ack ~= est_ack then
|
|
log.info("coordinator comms version mismatch")
|
|
end
|
|
elseif est_ack == ESTABLISH_ACK.BAD_API_VERSION then
|
|
if self.api.last_est_ack ~= est_ack then
|
|
log.info("coordinator api version mismatch")
|
|
end
|
|
else
|
|
log.debug("coordinator SCADA_MGMT establish packet reply unsupported")
|
|
end
|
|
|
|
self.api.last_est_ack = est_ack
|
|
end
|
|
else
|
|
log.debug("discarding coordinator non-link SCADA_MGMT packet before linked")
|
|
end
|
|
else
|
|
log.debug("illegal packet type " .. protocol .. " from coordinator", true)
|
|
end
|
|
elseif r_chan == config.SVR_Channel then
|
|
-- check sequence number
|
|
if self.sv.r_seq_num == nil then
|
|
self.sv.r_seq_num = packet.scada_frame.seq_num() + 1
|
|
elseif self.sv.r_seq_num ~= packet.scada_frame.seq_num() then
|
|
log.warning("sequence out-of-order (SVR): last = " .. self.sv.r_seq_num .. ", new = " .. packet.scada_frame.seq_num())
|
|
return
|
|
elseif self.sv.linked and (src_addr ~= self.sv.addr) then
|
|
log.debug("received packet from unknown computer " .. src_addr .. " while linked (SVR expected " .. self.sv.addr ..
|
|
"); channel in use by another system?")
|
|
return
|
|
else
|
|
self.sv.r_seq_num = packet.scada_frame.seq_num() + 1
|
|
end
|
|
|
|
-- feed watchdog on valid sequence number
|
|
sv_watchdog.feed()
|
|
|
|
-- handle packet
|
|
if protocol == PROTOCOL.SCADA_MGMT then
|
|
---@cast packet mgmt_frame
|
|
if self.sv.linked then
|
|
if packet.type == MGMT_TYPE.KEEP_ALIVE then
|
|
-- keep alive request received, echo back
|
|
if _check_length(packet, 1) then
|
|
local timestamp = packet.data[1]
|
|
local trip_time = util.time() - timestamp
|
|
|
|
if trip_time > 750 then
|
|
log.warning("pocket supervisor KEEP_ALIVE trip time > 750ms (" .. trip_time .. "ms)")
|
|
end
|
|
|
|
-- log.debug("pocket supervisor TT = " .. trip_time .. "ms")
|
|
|
|
_send_sv_keep_alive_ack(timestamp)
|
|
|
|
iocontrol.report_svr_tt(trip_time)
|
|
end
|
|
elseif packet.type == MGMT_TYPE.CLOSE then
|
|
-- handle session close
|
|
sv_watchdog.cancel()
|
|
nav.unload_sv()
|
|
self.sv.linked = false
|
|
self.sv.r_seq_num = nil
|
|
self.sv.addr = comms.BROADCAST
|
|
log.info("supervisor server connection closed by remote host")
|
|
elseif packet.type == MGMT_TYPE.DIAG_TONE_GET then
|
|
if _check_length(packet, 8) then
|
|
for i = 1, #packet.data do
|
|
diag.tone_test.tone_indicators[i].update(packet.data[i] == true)
|
|
end
|
|
end
|
|
elseif packet.type == MGMT_TYPE.DIAG_TONE_SET then
|
|
if packet.length == 1 and packet.data[1] == false then
|
|
diag.tone_test.ready_warn.set_value("testing denied")
|
|
log.debug("supervisor SCADA diag tone set failed")
|
|
elseif packet.length == 2 and type(packet.data[2]) == "table" then
|
|
local ready = packet.data[1]
|
|
local states = packet.data[2]
|
|
|
|
diag.tone_test.ready_warn.set_value(util.trinary(ready, "", "system not ready"))
|
|
|
|
for i = 1, #states do
|
|
if diag.tone_test.tone_buttons[i] ~= nil then
|
|
diag.tone_test.tone_buttons[i].set_value(states[i] == true)
|
|
diag.tone_test.tone_indicators[i].update(states[i] == true)
|
|
end
|
|
end
|
|
else
|
|
log.debug("supervisor SCADA diag tone set packet length/type mismatch")
|
|
end
|
|
elseif packet.type == MGMT_TYPE.DIAG_ALARM_SET then
|
|
if packet.length == 1 and packet.data[1] == false then
|
|
diag.tone_test.ready_warn.set_value("testing denied")
|
|
log.debug("supervisor SCADA diag alarm set failed")
|
|
elseif packet.length == 2 and type(packet.data[2]) == "table" then
|
|
local ready = packet.data[1]
|
|
local states = packet.data[2]
|
|
|
|
diag.tone_test.ready_warn.set_value(util.trinary(ready, "", "system not ready"))
|
|
|
|
for i = 1, #states do
|
|
if diag.tone_test.alarm_buttons[i] ~= nil then
|
|
diag.tone_test.alarm_buttons[i].set_value(states[i] == true)
|
|
end
|
|
end
|
|
else
|
|
log.debug("supervisor SCADA diag alarm set packet length/type mismatch")
|
|
end
|
|
else _fail_type(packet) end
|
|
elseif packet.type == MGMT_TYPE.ESTABLISH then
|
|
-- connection with supervisor established
|
|
if _check_length(packet, 1) then
|
|
local est_ack = packet.data[1]
|
|
|
|
if est_ack == ESTABLISH_ACK.ALLOW then
|
|
log.info("supervisor connection established")
|
|
self.establish_delay_counter = 0
|
|
self.sv.linked = true
|
|
self.sv.addr = src_addr
|
|
|
|
if self.api.linked then
|
|
iocontrol.report_link_state(LINK_STATE.LINKED, self.sv.addr, self.api.addr)
|
|
else
|
|
iocontrol.report_link_state(LINK_STATE.SV_LINK_ONLY)
|
|
end
|
|
elseif est_ack == ESTABLISH_ACK.DENY then
|
|
if self.sv.last_est_ack ~= est_ack then
|
|
log.info("supervisor connection denied")
|
|
end
|
|
elseif est_ack == ESTABLISH_ACK.COLLISION then
|
|
if self.sv.last_est_ack ~= est_ack then
|
|
log.info("supervisor connection denied due to collision")
|
|
end
|
|
elseif est_ack == ESTABLISH_ACK.BAD_VERSION then
|
|
if self.sv.last_est_ack ~= est_ack then
|
|
log.info("supervisor comms version mismatch")
|
|
end
|
|
else
|
|
log.debug("supervisor SCADA_MGMT establish packet reply unsupported")
|
|
end
|
|
|
|
self.sv.last_est_ack = est_ack
|
|
end
|
|
else
|
|
log.debug("discarding supervisor non-link SCADA_MGMT packet before linked")
|
|
end
|
|
else _fail_type(packet) end
|
|
else
|
|
log.debug("received packet from unconfigured channel " .. r_chan, true)
|
|
end
|
|
end
|
|
end
|
|
|
|
-- check if we are still linked with the supervisor
|
|
---@nodiscard
|
|
function public.is_sv_linked() return self.sv.linked end
|
|
|
|
-- check if we are still linked with the coordinator
|
|
---@nodiscard
|
|
function public.is_api_linked() return self.api.linked end
|
|
|
|
-- check if we are still linked with the supervisor and coordinator
|
|
---@nodiscard
|
|
function public.is_linked() return self.sv.linked and self.api.linked end
|
|
|
|
return public
|
|
end
|
|
|
|
return pocket
|