2022-12-10 18:58:17 +00:00
|
|
|
local log = require("scada-common.log")
|
|
|
|
local rsio = require("scada-common.rsio")
|
|
|
|
local util = require("scada-common.util")
|
|
|
|
|
|
|
|
local rsctl = require("supervisor.session.rsctl")
|
2022-12-18 18:56:04 +00:00
|
|
|
local unit = require("supervisor.session.unit")
|
|
|
|
|
|
|
|
local HEATING_WATER = 20000
|
|
|
|
local HEATING_SODIUM = 200000
|
|
|
|
|
|
|
|
-- 7.14 kJ per blade for 1 mB of fissile fuel
|
|
|
|
local POWER_PER_BLADE = util.joules_to_fe(7140)
|
|
|
|
|
|
|
|
local function m_avg(length, default)
|
|
|
|
local data = {}
|
|
|
|
local index = 1
|
|
|
|
local last_t = 0 ---@type number|nil
|
|
|
|
|
|
|
|
---@class moving_average
|
|
|
|
local public = {}
|
|
|
|
|
|
|
|
-- reset all to a given value
|
|
|
|
---@param x number value
|
|
|
|
function public.reset(x)
|
|
|
|
data = {}
|
|
|
|
for _ = 1, length do table.insert(data, x) end
|
|
|
|
end
|
|
|
|
|
|
|
|
-- record a new value
|
|
|
|
---@param x number new value
|
|
|
|
---@param t number? optional last update time to prevent duplicated entries
|
|
|
|
function public.record(x, t)
|
|
|
|
if type(t) == "number" and last_t == t then
|
|
|
|
return
|
|
|
|
end
|
|
|
|
|
|
|
|
data[index] = x
|
|
|
|
last_t = t
|
|
|
|
|
|
|
|
index = index + 1
|
|
|
|
if index > length then index = 1 end
|
|
|
|
end
|
|
|
|
|
|
|
|
-- compute the moving average
|
|
|
|
---@return number average
|
|
|
|
function public.compute()
|
|
|
|
local sum = 0
|
|
|
|
for i = 1, length do sum = sum + data[i] end
|
|
|
|
return sum
|
|
|
|
end
|
|
|
|
|
|
|
|
public.reset(default)
|
|
|
|
|
|
|
|
return public
|
|
|
|
end
|
|
|
|
|
|
|
|
---@alias PROCESS integer
|
|
|
|
local PROCESS = {
|
|
|
|
INACTIVE = 1,
|
|
|
|
SIMPLE = 2,
|
|
|
|
CHARGE = 3,
|
|
|
|
GEN_RATE = 4,
|
|
|
|
BURN_RATE = 5
|
|
|
|
}
|
2022-12-10 18:58:17 +00:00
|
|
|
|
|
|
|
---@class facility_management
|
|
|
|
local facility = {}
|
|
|
|
|
2022-12-18 18:56:04 +00:00
|
|
|
facility.PROCESS_MODES = PROCESS
|
|
|
|
|
2022-12-10 18:58:17 +00:00
|
|
|
-- create a new facility management object
|
2022-12-18 18:56:04 +00:00
|
|
|
---@param num_reactors integer number of reactor units
|
|
|
|
---@param cooling_conf table cooling configurations of reactor units
|
|
|
|
function facility.new(num_reactors, cooling_conf)
|
2022-12-10 18:58:17 +00:00
|
|
|
local self = {
|
2022-12-18 18:56:04 +00:00
|
|
|
-- components
|
|
|
|
units = {},
|
2022-12-10 18:58:17 +00:00
|
|
|
induction = {},
|
2022-12-18 18:56:04 +00:00
|
|
|
redstone = {},
|
|
|
|
-- process control
|
|
|
|
mode = PROCESS.INACTIVE,
|
|
|
|
charge_target = 0, -- FE
|
|
|
|
charge_rate = 0, -- FE/t
|
|
|
|
charge_limit = 0.99, -- percentage
|
|
|
|
burn_rate_set = 0,
|
|
|
|
unit_limits = {},
|
|
|
|
-- statistics
|
|
|
|
im_stat_init = false,
|
|
|
|
avg_charge = m_avg(10, 0.0),
|
|
|
|
avg_inflow = m_avg(10, 0.0),
|
|
|
|
avg_outflow = m_avg(10, 0.0)
|
2022-12-10 18:58:17 +00:00
|
|
|
}
|
|
|
|
|
2022-12-18 18:56:04 +00:00
|
|
|
-- create units
|
|
|
|
for i = 1, num_reactors do
|
|
|
|
table.insert(self.units, unit.new(i, cooling_conf[i].BOILERS, cooling_conf[i].TURBINES))
|
|
|
|
|
|
|
|
local u_lim = { burn_rate = -1.0, temp = 1100 } ---@class unit_limit
|
|
|
|
table.insert(self.unit_limits, u_lim)
|
|
|
|
end
|
|
|
|
|
2022-12-10 18:58:17 +00:00
|
|
|
-- init redstone RTU I/O controller
|
|
|
|
local rs_rtu_io_ctl = rsctl.new(self.redstone)
|
|
|
|
|
|
|
|
-- unlink disconnected units
|
|
|
|
---@param sessions table
|
|
|
|
local function _unlink_disconnected_units(sessions)
|
|
|
|
util.filter_table(sessions, function (u) return u.is_connected() end)
|
|
|
|
end
|
|
|
|
|
|
|
|
-- PUBLIC FUNCTIONS --
|
|
|
|
|
|
|
|
---@class facility
|
|
|
|
local public = {}
|
|
|
|
|
|
|
|
-- ADD/LINK DEVICES --
|
|
|
|
|
|
|
|
-- link a redstone RTU session
|
|
|
|
---@param rs_unit unit_session
|
|
|
|
function public.add_redstone(rs_unit)
|
|
|
|
table.insert(self.redstone, rs_unit)
|
|
|
|
end
|
|
|
|
|
|
|
|
-- link an imatrix RTU session
|
|
|
|
---@param imatrix unit_session
|
|
|
|
function public.add_imatrix(imatrix)
|
|
|
|
table.insert(self.induction, imatrix)
|
|
|
|
end
|
|
|
|
|
|
|
|
-- purge devices associated with the given RTU session ID
|
|
|
|
---@param session integer RTU session ID
|
|
|
|
function public.purge_rtu_devices(session)
|
|
|
|
util.filter_table(self.redstone, function (s) return s.get_session_id() ~= session end)
|
|
|
|
util.filter_table(self.induction, function (s) return s.get_session_id() ~= session end)
|
|
|
|
end
|
|
|
|
|
|
|
|
-- UPDATE --
|
|
|
|
|
|
|
|
-- update (iterate) the facility management
|
|
|
|
function public.update()
|
|
|
|
-- unlink RTU unit sessions if they are closed
|
|
|
|
_unlink_disconnected_units(self.induction)
|
|
|
|
_unlink_disconnected_units(self.redstone)
|
|
|
|
end
|
|
|
|
|
2022-12-18 18:56:04 +00:00
|
|
|
function public.update_units()
|
|
|
|
for i = 1, #self.units do
|
|
|
|
local u = self.units[i] ---@type reactor_unit
|
|
|
|
u.update()
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2022-12-10 18:58:17 +00:00
|
|
|
-- READ STATES/PROPERTIES --
|
|
|
|
|
|
|
|
-- get build properties of all machines
|
|
|
|
function public.get_build()
|
|
|
|
local build = {}
|
|
|
|
|
|
|
|
build.induction = {}
|
|
|
|
for i = 1, #self.induction do
|
|
|
|
local matrix = self.induction[i] ---@type unit_session
|
|
|
|
build.induction[matrix.get_device_idx()] = { matrix.get_db().formed, matrix.get_db().build }
|
|
|
|
end
|
|
|
|
|
|
|
|
return build
|
|
|
|
end
|
|
|
|
|
|
|
|
-- get RTU statuses
|
|
|
|
function public.get_rtu_statuses()
|
|
|
|
local status = {}
|
|
|
|
|
|
|
|
-- status of induction matricies (including tanks)
|
|
|
|
status.induction = {}
|
|
|
|
for i = 1, #self.induction do
|
|
|
|
local matrix = self.induction[i] ---@type unit_session
|
|
|
|
status.induction[matrix.get_device_idx()] = {
|
|
|
|
matrix.is_faulted(),
|
|
|
|
matrix.get_db().formed,
|
|
|
|
matrix.get_db().state,
|
|
|
|
matrix.get_db().tanks
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
---@todo other RTU statuses
|
|
|
|
|
|
|
|
return status
|
|
|
|
end
|
|
|
|
|
2022-12-18 18:56:04 +00:00
|
|
|
function public.get_units()
|
|
|
|
return self.units
|
|
|
|
end
|
|
|
|
|
2022-12-10 18:58:17 +00:00
|
|
|
return public
|
|
|
|
end
|
|
|
|
|
|
|
|
return facility
|