local Token = require 'utils.token' local Task = require 'utils.task' local Global = require 'utils.global' local Event = require 'utils.event' local Print = require('utils.print_override') -- local constants local floor = math.floor local ceil = math.ceil local insert = table.insert local concat = table.concat local serialize = serpent.serialize local remove = table.remove local tostring = tostring local len = string.len local gmatch = string.gmatch local newline = '\n' local raw_print = Print.raw_print local minutes_to_ticks = 60 * 60 local hours_to_ticks = 60 * 60 * 60 local ticks_to_minutes = 1 / minutes_to_ticks local ticks_to_hours = 1 / hours_to_ticks local serialize_options = {sparse = true, compact = true} local Public = {} local server_time = {secs = nil, tick = 0} local server_ups = {ups = 60} local start_data = {server_id = nil, server_name = nil, start_time = nil} local instances = { data = {} } local admins = {} local requests = {} local jailed_data_set = 'jailed' local data_set_handlers = {} local scenario_handlers = {} Global.register( { server_time = server_time, server_ups = server_ups, start_data = start_data, requests = requests, instances = instances, admins = admins }, function(tbl) server_time = tbl.server_time server_ups = tbl.server_ups start_data = tbl.start_data requests = tbl.requests instances = tbl.instances admins = tbl.admins end ) --- Web panel framework. local discord_tag = '[DISCORD]' local discord_raw_tag = '[DISCORD-RAW]' local discord_bold_tag = '[DISCORD-BOLD]' local discord_admin_tag = '[DISCORD-ADMIN]' local discord_banned_tag = '[DISCORD-BANNED]' local discord_banned_embed_tag = '[DISCORD-BANNED-EMBED]' local discord_unbanned_tag = '[DISCORD-UNBANNED]' local discord_unbanned_embed_tag = '[DISCORD-UNBANNED-EMBED]' local discord_jailed_tag = '[DISCORD-JAILED]' local discord_jailed_embed_tag = '[DISCORD-JAILED-EMBED]' local discord_unjailed_tag = '[DISCORD-UNJAILED]' local discord_unjailed_embed_tag = '[DISCORD-UNJAILED-EMBED]' local discord_jailed_named_embed_tag = '[DISCORD-JAILED-NAMED-EMBED]' local discord_unjailed_named_embed_tag = '[DISCORD-UNJAILED-NAMED-EMBED]' local discord_admin_raw_tag = '[DISCORD-ADMIN-RAW]' local discord_embed_parsed_tag = '[DISCORD-EMBED-PARSED]' local discord_embed_tag = '[DISCORD-EMBED]' local discord_embed_raw_tag = '[DISCORD-EMBED-RAW]' local discord_admin_embed_tag = '[DISCORD-ADMIN-EMBED]' local discord_admin_embed_raw_tag = '[DISCORD-ADMIN-EMBED-RAW]' local discord_named_tag = '[DISCORD-NAMED]' local discord_named_raw_tag = '[DISCORD-NAMED-RAW]' local discord_named_bold_tag = '[DISCORD-NAMED-BOLD]' local discord_named_embed_tag = '[DISCORD-NAMED-EMBED]' local discord_named_embed_parsed_tag = '[DISCORD-NAMED-EMBED-PARSED]' local discord_named_embed_raw_tag = '[DISCORD-NAMED-EMBED-RAW]' local start_scenario_tag = '[START-SCENARIO]' local stop_scenario_tag = '[STOP-SCENARIO]' local ping_tag = '[PING]' local data_set_tag = '[DATA-SET]' local ban_get_tag = '[BAN-GET]' local data_get_tag = '[DATA-GET]' local data_get_and_print_tag = '[DATA-GET-AND-PRINT]' local data_get_all_tag = '[DATA-GET-ALL]' local data_tracked_tag = '[DATA-TRACKED]' local scenario_tag = '[SCENARIO-TRACKED]' local ban_sync_tag = '[BAN-SYNC]' local unbanned_sync_tag = '[UNBANNED-SYNC]' local query_players_tag = '[QUERY-PLAYERS]' local player_join_tag = '[PLAYER-JOIN]' local player_leave_tag = '[PLAYER-LEAVE]' local antigrief_tag = '[ANTIGRIEF-LOG]' Public.raw_print = raw_print local function output_data(primary, secondary) local secs = server_time.secs if secs == nil then return false end if start_data and start_data.output then local write = game.write_file write(start_data.output, primary .. (secondary or '') .. newline, true, 0) else raw_print(primary .. (secondary or '')) end end local function assert_non_empty_string_and_no_spaces(str, argument_name) if type(str) ~= 'string' then error(argument_name .. ' must be a string', 3) end if #str == 0 then error(argument_name .. ' must not be an empty string', 3) end if str:match(' ') then error(argument_name .. " must not contain space ' ' character.", 3) end end local function get_online_admins() local online = game.connected_players local i = 0 for _, p in pairs(online) do if p.admin then i = i + 1 end end return i end local function build_embed_data() local d = { time = Public.format_time(game.ticks_played), onlinePlayers = #game.connected_players, totalPlayers = #game.players, onlineAdmins = get_online_admins() } return d end --- The event id for the on_server_started event. -- The event is raised whenever the server goes from the starting state to the running state. -- It provides a good opportunity to request data from the web server. -- Note that if the server is stopped then started again, this event will be raised again. -- @usage -- local Server = require 'utils.server' -- local Event = require 'utils.event' -- -- Event.add(Server.events.on_server_started, -- function() -- Server.try_get_all_data('regulars', callback) -- end) -- Event.add(Server.events.on_changes_detected, -- function() -- Trigger some sort of automated restart whenever the game ends. -- end) Public.events = {on_server_started = Event.generate_event_name('on_server_started'), on_changes_detected = Event.generate_event_name('on_changes_detected')} --- Sends a message to the linked discord channel. The message is sanitized of markdown server side. -- @param message message to send. -- @usage -- local Server = require 'utils.server' -- Server.to_discord('Hello from scenario script!') -- @param locale if the message should be handled as localized. function Public.to_discord(message, locale) if locale then print(message, discord_tag) else output_data(discord_tag .. message) end end --- Sends a message to the linked discord channel. The message is not sanitized of markdown. -- @param message message to send. -- @param locale if the message should be handled as localized. function Public.to_discord_raw(message, locale) if locale then print(message, discord_raw_tag) else output_data(discord_raw_tag .. message) end end --- Sends a message to the linked discord channel. The message is sanitized of markdown server side, then made bold. -- @param message message to send. -- @param locale if the message should be handled as localized. function Public.to_discord_bold(message, locale) if locale then print(message, discord_bold_tag) else output_data(discord_bold_tag .. message) end end --- Sends a message to the named discord channel. The message is sanitized of markdown server side. -- @param message message to send. function Public.to_discord_named(channel_name, message) assert_non_empty_string_and_no_spaces(channel_name, 'channel_name') output_data(concat({discord_named_tag, channel_name, ' ', message})) end --- Sends a message to the named discord channel. The message is not sanitized of markdown. -- @param message message to send. function Public.to_discord_named_raw(channel_name, message) assert_non_empty_string_and_no_spaces(channel_name, 'channel_name') output_data(concat({discord_named_raw_tag, channel_name, ' ', message})) end --- Sends a message to the named discord channel. The message is sanitized of markdown server side, then made bold. -- @param message message to send. function Public.to_discord_named_bold(channel_name, message) assert_non_empty_string_and_no_spaces(channel_name, 'channel_name') output_data(concat({discord_named_bold_tag, channel_name, ' ', message})) end --- Sends an embed message to the named discord channel. The message is sanitized of markdown server side. -- @param message the content of the embed. function Public.to_discord_named_embed(channel_name, message) assert_non_empty_string_and_no_spaces(channel_name, 'channel_name') output_data(concat({discord_named_embed_tag, channel_name, ' ', message})) end --- Sends an embed message that is parsed to the named discord channel. The message is sanitized of markdown server side. -- @param message the content of the embed. function Public.to_discord_named_parsed_embed(channel_name, message) assert_non_empty_string_and_no_spaces(channel_name, 'channel_name') local table_to_json = game.table_to_json if not type(message) == 'table' then return end if not message.title then return end if not message.description then return end message.channelName = channel_name output_data(discord_named_embed_parsed_tag, table_to_json(message)) end --- Sends an embed message to the named discord channel. The message is not sanitized of markdown. -- @param message the content of the embed. function Public.to_discord_named_embed_raw(channel_name, message) assert_non_empty_string_and_no_spaces(channel_name, 'channel_name') output_data(concat({discord_named_embed_raw_tag, channel_name, ' ', message})) end --- Sends a message to the linked admin discord channel. The message is sanitized of markdown server side. -- @param message message to send. -- @param locale if the message should be handled as localized. function Public.to_admin(message, locale) if locale then print(message, discord_admin_tag) else output_data(discord_admin_tag .. message) end end --- Sends a message to the linked banned discord channel. The message is sanitized of markdown server side. -- @param message message to send. -- @param locale if the message should be handled as localized. function Public.to_banned(message, locale) if locale then print(message, discord_banned_tag) else output_data(discord_banned_tag .. message) end end --- Sends a message to the linked banned discord channel. The message is sanitized of markdown server side. -- @param message message to send. -- @param locale if the message should be handled as localized. function Public.to_unbanned(message, locale) if locale then print(message, discord_unbanned_tag) else output_data(discord_unbanned_tag .. message) end end --- Sends a message to the linked connected discord channel. The message is sanitized of markdown server side. -- @param message message to send. -- @param locale if the message should be handled as localized. function Public.to_jailed(message, locale) if locale then print(message, discord_jailed_tag) else output_data(discord_jailed_tag .. message) end end --- Sends a message to the linked connected discord channel. The message is sanitized of markdown server side. -- @param message message to send. -- @param locale if the message should be handled as localized. function Public.to_unjailed(message, locale) if locale then print(message, discord_unjailed_tag) else output_data(discord_unjailed_tag .. message) end end --- Sends a message to the linked admin discord channel. The message is not sanitized of markdown. -- @param message message to send. -- @param locale if the message should be handled as localized. function Public.to_admin_raw(message, locale) if locale then print(message, discord_admin_raw_tag) else output_data(discord_admin_raw_tag .. message) end end --- Sends a embed message to the linked discord channel. The message is sanitized/parsed of markdown server side. -- @param message the content of the embed. function Public.to_discord_embed_parsed(message) local table_to_json = game.table_to_json if not type(message) == 'table' then return end if not message.title then return end if not message.description then return end output_data(discord_embed_parsed_tag .. table_to_json(message)) end --- Sends a embed message to the linked discord channel. The message is sanitized of markdown server side. -- @param message the content of the embed. -- @param locale if the message should be handled as localized. function Public.to_discord_embed(message, locale) if locale then print(message, discord_embed_tag) else output_data(discord_embed_tag .. message) end end --- Sends a embed message to the linked discord channel. The message is not sanitized of markdown. -- @param message the content of the embed. -- @param locale if the message should be handled as localized. function Public.to_discord_embed_raw(message, locale) if locale then print(message, discord_embed_raw_tag) else output_data(discord_embed_raw_tag .. message) end end --- Sends a embed message to the linked admin discord channel. The message is sanitized of markdown server side. -- @param message the content of the embed. -- @param locale if the message should be handled as localized. function Public.to_admin_embed(message, locale) if locale then print(message, discord_admin_embed_tag) else output_data(discord_admin_embed_tag .. message) end end --- Sends a embed message to the linked banned discord channel. The message is sanitized of markdown server side. -- @param message the content of the embed. -- @param locale if the message should be handled as localized. function Public.to_banned_embed(message, locale) local table_to_json = game.table_to_json if not type(message) == 'table' then return end if locale then print(message, discord_banned_embed_tag) else if not message.username then return end if not message.reason then return end if not message.admin then return end output_data(discord_banned_embed_tag .. table_to_json(message)) end end --- Sends a embed message to the linked banned discord channel. The message is sanitized of markdown server side. -- @param message the content of the embed. -- @param locale if the message should be handled as localized. function Public.to_unbanned_embed(message, locale) local table_to_json = game.table_to_json if not type(message) == 'table' then return end if locale then print(message, discord_unbanned_embed_tag) else if not message.username then return end if not message.admin then return end output_data(discord_unbanned_embed_tag .. table_to_json(message)) end end --- Sends a embed message to the linked connected discord channel. The message is sanitized of markdown server side. -- @param message the content of the embed. -- @param locale if the message should be handled as localized. function Public.to_jailed_embed(message, locale) local table_to_json = game.table_to_json if not type(message) == 'table' then return end if locale then print(message, discord_jailed_embed_tag) else if not message.username then return end if not message.reason then return end if not message.admin then return end output_data(discord_jailed_embed_tag .. table_to_json(message)) end end --- Sends a embed message to the jailed discord channel. The message is sanitized of markdown server side. -- @param message the content of the embed. -- @param locale if the message should be handled as localized. function Public.to_jailed_named_embed(message, locale) local table_to_json = game.table_to_json if not type(message) == 'table' then return end if locale then print(message, discord_jailed_named_embed_tag) else if not message.username then return end if not message.reason then return end if not message.admin then return end output_data(discord_jailed_named_embed_tag .. table_to_json(message)) end end --- Sends a embed message to the linked connected discord channel. The message is sanitized of markdown server side. -- @param message the content of the embed. -- @param locale if the message should be handled as localized. function Public.to_unjailed_embed(message, locale) local table_to_json = game.table_to_json if not type(message) == 'table' then return end if locale then print(message, discord_unjailed_embed_tag) else if not message.username then return end if not message.admin then return end output_data(discord_unjailed_embed_tag .. table_to_json(message)) end end --- Sends a embed message to the linked connected discord channel. The message is sanitized of markdown server side. -- @param message the content of the embed. -- @param locale if the message should be handled as localized. function Public.to_unjailed_named_embed(message, locale) local table_to_json = game.table_to_json if not type(message) == 'table' then return end if locale then print(message, discord_unjailed_named_embed_tag) else if not message.username then return end if not message.admin then return end output_data(discord_unjailed_named_embed_tag .. table_to_json(message)) end end --- Sends a embed message to the linked admin discord channel. The message is not sanitized of markdown. -- @param message the content of the embed. -- @param locale if the message should be handled as localized. function Public.to_admin_embed_raw(message, locale) if locale then print(message, discord_admin_embed_raw_tag) else output_data(discord_admin_embed_raw_tag .. message) end end --- Stops and saves the factorio server and starts the named scenario. -- @param scenario_name The name of the scenario as appears in the scenario table on the panel. -- @usage -- local Server = require 'utils.server' -- Server.start_scenario('my_scenario_name') function Public.start_scenario(scenario_name) if type(scenario_name) ~= 'string' then game.print('start_scenario - scenario_name ' .. tostring(scenario_name) .. ' must be a string.') return end local message = start_scenario_tag .. scenario_name output_data(message) end --- Stops and saves the factorio server. -- @usage -- local Server = require 'utils.server' -- Server.stop_scenario() function Public.stop_scenario() local message = stop_scenario_tag output_data(message) end local default_ping_token = Token.register( function(sent_tick) local now = game.tick local diff = now - sent_tick local message = concat({'Pong in ', diff, ' tick(s) ', 'sent tick: ', sent_tick, ' received tick: ', now}) game.print(message) end ) --- Pings the web server. -- @param func_token The function that is called when the web server replies. -- The function is passed the tick that the ping was sent. function Public.ping(func_token) local message = concat({ping_tag, func_token or default_ping_token, ' ', game.tick}) output_data(message) end --- The backend sets instances with data so a player -- can easily connect to another one via in-game. -- @param data
function Public.set_instances(data) if not data then return end if not type(data) == 'table' then return end instances.data = {} for id, tbl in pairs(data) do instances.data[id] = tbl end end --- Gets each available/non-available instance function Public.get_instances() return instances.data end local function double_escape(str) -- Excessive escaping because the data is serialized twice. if not str then return '' end return str:gsub('\\', '\\\\\\\\'):gsub('"', '\\\\\\"'):gsub('\n', '\\\\n') end --- Sets the web server's persistent data storage. If you pass nil for the value removes the data. -- Data set this will by synced in with other server if they choose to. -- There can only be one key for each data_set. -- @param data_set -- @param key -- @param value Any type that is not a function. set to nil to remove the data. -- @usage -- local Server = require 'utils.server' -- Server.set_data('my data set', 'key 1', 123) -- Server.set_data('my data set', 'key 2', 'abc') -- Server.set_data('my data set', 'key 3', {'some', 'data', ['is_set'] = true}) -- -- Server.set_data('my data set', 'key 1', nil) -- this will remove 'key 1' -- Server.set_data('my data set', 'key 2', 'def') -- this will change the value for 'key 2' to 'def' function Public.set_data(data_set, key, value) if type(data_set) ~= 'string' then error('data_set must be a string', 2) end if type(key) ~= 'string' then error('key must be a string', 2) end data_set = double_escape(data_set) key = double_escape(key) local message local vt = type(value) if vt == 'nil' then message = concat({data_set_tag, '{data_set:"', data_set, '",key:"', key, '"}'}) elseif vt == 'string' then value = double_escape(value) message = concat({data_set_tag, '{data_set:"', data_set, '",key:"', key, '",value:"\\"', value, '\\""}'}) elseif vt == 'number' then message = concat({data_set_tag, '{data_set:"', data_set, '",key:"', key, '",value:"', value, '"}'}) elseif vt == 'boolean' then message = concat({data_set_tag, '{data_set:"', data_set, '",key:"', key, '",value:"', tostring(value), '"}'}) elseif vt == 'function' then error('value cannot be a function', 2) else -- table value = serialize(value, serialize_options) -- Less escaping than the string case as serpent provides one level of escaping. -- Need to escape single quotes as serpent uses double quotes for strings. value = value:gsub('\\', '\\\\'):gsub("'", "\\'") message = concat({data_set_tag, '{data_set:"', data_set, '",key:"', key, "\",value:'", value, "'}"}) end output_data(message) end local function validate_arguments(data_set, key, callback_token) if type(data_set) ~= 'string' then error('data_set must be a string', 3) end if type(key) ~= 'string' then error('key must be a string', 3) end if type(callback_token) ~= 'number' then error('callback_token must be a number', 3) end end local function validate_arguments_of_ban(username, callback_token) if type(username) ~= 'string' then error('username must be a string', 3) end if type(callback_token) ~= 'number' then error('callback_token must be a number', 3) end end local function send_try_get_data(data_set, key, callback_token) data_set = double_escape(data_set) key = double_escape(key) local message = concat {data_get_tag, callback_token, ' {', 'data_set:"', data_set, '",key:"', key, '"}'} output_data(message) end local function send_try_get_ban(username, callback_token) username = double_escape(username) local message = concat {ban_get_tag, callback_token, ' {', 'username:"', username, '"}'} output_data(message) end local function send_try_get_data_and_print(data_set, key, to_print, callback_token) data_set = double_escape(data_set) key = double_escape(key) to_print = double_escape(to_print) local message = concat {data_get_and_print_tag, callback_token, ' {', 'data_set:"', data_set, '",key:"', key, '",to_print:"', to_print, '"}'} output_data(message) end local function log_antigrief_data(category, action) category = double_escape(category) action = double_escape(action) local message = concat {antigrief_tag, '{', 'category:"', category, '",action:"', action, '"}'} output_data(message) end local cancelable_callback_token = Token.register( function(data) local data_set = data.data_set local keys = requests[data_set] if not keys then return end local key = data.key local callbacks = keys[key] if not callbacks then return end keys[key] = nil for c, _ in next, callbacks do local func = Token.get(c) func(data) end end ) --- Gets data from the web server's persistent data storage. -- The callback is passed a table {data_set: string, key: string, value: any}. -- If the value is nil, it means there is no stored data for that data_set key pair. -- @param data_set -- @param key -- @param callback_token -- @usage -- local Server = require 'utils.server' -- local Token = require 'utils.token' -- -- local callback = -- Token.register( -- function(data) -- local data_set = data.data_set -- local key = data.key -- local value = data.value -- will be nil if no data -- -- game.print(data_set .. ':' .. key .. ':' .. tostring(value)) -- end -- ) -- -- Server.try_get_data('my data set', 'key 1', callback) function Public.try_get_data(data_set, key, callback_token) validate_arguments(data_set, key, callback_token) send_try_get_data(data_set, key, callback_token) end --- Same as try_get_data returns if a user is banned. function Public.try_get_ban(username, callback_token) validate_arguments_of_ban(username, callback_token) send_try_get_ban(username, callback_token) end --- Same as try_get_data but prints the returned value to the given player who ran the command. function Public.try_get_data_and_print(data_set, key, to_print, callback_token) validate_arguments(data_set, key, callback_token) send_try_get_data_and_print(data_set, key, to_print, callback_token) end local function try_get_data_cancelable(data_set, key, callback_token) local keys = requests[data_set] if not keys then keys = {} requests[data_set] = keys end local callbacks = keys[key] if not callbacks then callbacks = {} keys[key] = callbacks end if callbacks[callback_token] then return end if next(callbacks) then callbacks[callback_token] = true else callbacks[callback_token] = true send_try_get_data(data_set, key, cancelable_callback_token) end end --- Same Server.try_get_data but the request can be cancelled by calling -- Server.cancel_try_get_data(data_set, key, callback_token) -- If the request is cancelled before it is complete the callback will be called with data.cancelled = true. -- It is safe to cancel a non-existent or completed request, in either case the callback will not be called. -- There can only be one request per data_set, key, callback_token combo. If there is already an ongoing request -- an attempt to make a new one will be ignored. -- @param data_set -- @param key -- @param callback_token function Public.try_get_data_cancelable(data_set, key, callback_token) validate_arguments(data_set, key, callback_token) try_get_data_cancelable(data_set, key, callback_token) end local function cancel_try_get_data(data_set, key, callback_token) local keys = requests[data_set] if not keys then return false end local callbacks = keys[key] if not callbacks then return false end if callbacks[callback_token] then callbacks[callback_token] = nil local func = Token.get(callback_token) local data = {data_set = data_set, key = key, cancelled = true} func(data) return true else return false end end --- Cancels the request. Returns false if the request could not be canceled, either because there is no request -- to cancel or it has been completed or canceled already. Otherwise returns true. -- If the request is cancelled before it is complete the callback will be called with data.cancelled = true. -- It is safe to cancel a non-existent or completed request, in either case the callback will not be called. -- @param data_set -- @param key -- @param callback_token function Public.cancel_try_get_data(data_set, key, callback_token) validate_arguments(data_set, key, callback_token) return cancel_try_get_data(data_set, key, callback_token) end local timeout_token = Token.register( function(data) cancel_try_get_data(data.data_set, data.key, data.callback_token) end ) --- Same as Server.try_get_data but the request is cancelled if the timeout expires before the request is complete. -- If the request is cancelled before it is complete the callback will be called with data.cancelled = true. -- There can only be one request per data_set, key, callback_token combo. If there is already an ongoing request -- an attempt to make a new one will be ignored. -- @param data_set -- @param key -- @param callback_token -- @usage -- local Server = require 'utils.server' -- local Token = require 'utils.token' -- -- local callback = -- Token.register( -- function(data) -- local data_set = data.data_set -- local key = data.key -- -- game.print('data_set: ' .. data_set .. ', key: ' .. key) -- -- if data.cancelled then -- game.print('Timed out') -- return -- end -- -- local value = data.value -- will be nil if no data -- -- game.print('value: ' .. tostring(value)) -- end -- ) -- -- Server.try_get_data_timeout('my data set', 'key 1', callback, 60) function Public.try_get_data_timeout(data_set, key, callback_token, timeout_ticks) validate_arguments(data_set, key, callback_token) try_get_data_cancelable(data_set, key, callback_token) Task.set_timeout_in_ticks(timeout_ticks, timeout_token, {data_set = data_set, key = key, callback_token = callback_token}) end --- Gets all the data for the data_set from the web server's persistent data storage. -- The callback is passed a table {data_set: string, entries: {dictionary key -> value}}. -- If there is no data stored for the data_set entries will be nil. -- @param data_set -- @param callback_token -- @usage -- local Server = require 'utils.server' -- local Token = require 'utils.token' -- -- local callback = -- Token.register( -- function(data) -- local data_set = data.data_set -- local entries = data.entries -- will be nil if no data -- local value2 = entries['key 2'] -- local value3 = entries['key 3'] -- end -- ) -- -- Server.try_get_all_data('my data set', callback) function Public.try_get_all_data(data_set, callback_token) if type(data_set) ~= 'string' then error('data_set must be a string', 2) end if type(callback_token) ~= 'number' then error('callback_token must be a number', 2) end data_set = double_escape(data_set) local message = concat {data_get_all_tag, callback_token, ' {', 'data_set:"', data_set, '"}'} output_data(message) end local function raise_admins(data) if not data or not next(data) then return end if admins and next(admins) then for _, admin in pairs(admins) do admins[admin] = nil end end for _, admin in pairs(data) do admins[admin] = true end end local function data_set_changed(data) local handlers = data_set_handlers[data.data_set] if handlers == nil then return end if _DEBUG then for _, handler in ipairs(handlers) do local success, err = pcall(handler, data) if not success then log(err) error(err, 2) end end else for _, handler in ipairs(handlers) do local success, err = pcall(handler, data) if not success then log(err) end end end end local function scenario_changed(data) local handlers = scenario_handlers[data.scenario] if handlers == nil then return end if _DEBUG then for _, handler in ipairs(handlers) do local success, err = pcall(handler, data) if not success then log(err) error(err, 2) end end else for _, handler in ipairs(handlers) do local success, err = pcall(handler, data) if not success then log(err) end end end end --- Register a handler to be called when the data_set changes. -- The handler is passed a table {data_set:string, key:string, value:any} -- If value is nil that means the key was removed. -- The handler may be called even if the value hasn't changed. It's up to the implementer -- to determine if the value has changed, or not care. -- To prevent desyncs the same handlers must be registered for all clients. The easiest way to do this -- is in the control stage, i.e before on_init or on_load would be called. -- @param data_set -- @param handler -- @usage -- local Server = require 'utils.server' -- Server.on_data_set_changed( -- 'my data set', -- function(data) -- local data_set = data.data_set -- local key = data.key -- local value = data.value -- will be nil if data was removed. -- end -- ) function Public.on_data_set_changed(data_set, handler) if _LIFECYCLE == _STAGE.runtime then error('cannot call during runtime', 2) end if type(data_set) ~= 'string' then error('data_set must be a string', 2) end local handlers = data_set_handlers[data_set] if handlers == nil then handlers = {handler} data_set_handlers[data_set] = handlers else handlers[#handlers + 1] = handler end end --- Register a handler to be called when a scenarios changes. -- The handler is passed a table {scenario:string} -- @param scenario -- @param handler -- @usage -- local Server = require 'utils.server' -- Server.on_scenario_changed( -- 'scenario name', -- function(data) -- local scenario = data.scenario -- end -- ) function Public.on_scenario_changed(scenario, handler) if _LIFECYCLE == _STAGE.runtime then error('cannot call during runtime', 2) end if type(scenario) ~= 'string' then error('scenario must be a string', 2) end local handlers = scenario_handlers[scenario] if handlers == nil then handlers = {handler} scenario_handlers[scenario] = handlers else handlers[#handlers + 1] = handler end end --- Called by the web server to notify the client that a data_set has changed. Public.raise_data_set = data_set_changed --- Called by the web server to notify the client that a data_set has changed. Public.raise_admins = raise_admins --- Called by the web server to notify the client that the subscribed scenario has changed. Public.raise_scenario_changed = scenario_changed -- Tracks antigrief and sends them to a specific log channel. Public.log_antigrief_data = log_antigrief_data --- Called by the web server to determine which data_sets are being tracked. function Public.get_tracked_data_sets() local message = {data_tracked_tag, '['} for k, _ in pairs(data_set_handlers) do k = double_escape(k) local message_length = #message message[message_length + 1] = '"' message[message_length + 2] = k message[message_length + 3] = '"' message[message_length + 4] = ',' end if message[#message] == ',' then remove(message) end message[#message + 1] = ']' message = concat(message) output_data(message) end --- Called by the web server to determine which scenarios is being tracked. function Public.get_tracked_scenario() local message = {scenario_tag, ''} for k, _ in pairs(scenario_handlers) do k = double_escape(k) local message_length = #message message[message_length + 1] = k message[message_length + 2] = ',' end if message[#message] == ',' then remove(message) end message = concat(message) output_data(message) end local function escape(str) return str:gsub('\\', '\\\\'):gsub('"', '\\"') end local statistics = { 'item_production_statistics', 'fluid_production_statistics', 'kill_count_statistics', 'entity_build_count_statistics' } function Public.export_stats() local table_to_json = game.table_to_json local stats = { game_tick = game.tick, player_count = #game.connected_players, game_flow_statistics = { pollution_statistics = { input = game.pollution_statistics.input_counts, output = game.pollution_statistics.output_counts } }, rockets_launched = {}, force_flow_statistics = {} } for _, force in pairs(game.forces) do local flow_statistics = {} for _, statName in pairs(statistics) do flow_statistics[statName] = { input = force[statName].input_counts, output = force[statName].output_counts } end stats.rockets_launched[force.name] = force.rockets_launched stats.force_flow_statistics[force.name] = flow_statistics end rcon.print(table_to_json(stats)) end --- Called by the web server to set the server start data. function Public.set_start_data(data) start_data.server_id = data.server_id start_data.server_name = data.server_name local start_time = start_data.start_time if not start_time then -- Only set start time if it has not been set already, so that we keep the first start time. start_data.start_time = data.start_time end end -- This is the current server's id, in the case the save has been loaded on multiple servers. -- @return string function Public.get_server_id() return start_data.server_id or '' end --- Gets the server's name. Empty string if not known. -- This is the current server's name, in the case the save has been loaded on multiple servers. -- @return string function Public.get_server_name() return start_data.server_name or nil end --- Gets the server's name and matches it against a string. -- This is the current server's name, in the case the save has been loaded on multiple servers. -- @param string -- @return string function Public.check_server_name(string) if start_data.server_name then local server_name = start_data.server_name local str = string.match(server_name, string) if str then return true end end return false end --- Gets the server's start time as a unix epoch timestamp. nil if not known. -- @return number? function Public.get_start_time() return start_data.start_time end --- Gets the server's start data. nil if not known. -- @return table? function Public.get_start_data() return start_data end --- Gets the server's admin list. nil if not known. -- @return table? function Public.get_admins_data() return admins end --- If the player exists bans the player. -- Regardless of whether or not the player exists the name is synchronized with other servers -- and stored in the database. -- @param username -- @param reason defaults to empty string. -- @param admin admin's name, defaults to '