1
0
mirror of https://github.com/ComfyFactory/ComfyFactorio.git synced 2025-01-24 03:47:58 +02:00
ComfyFactorio/utils/datastore/session_data.lua
2022-08-18 21:09:13 +02:00

425 lines
12 KiB
Lua

local Global = require 'utils.global'
local Core = require 'utils.core'
local Color = require 'utils.color_presets'
local Game = require 'utils.game'
local Token = require 'utils.token'
local Task = require 'utils.task'
local Server = require 'utils.server'
local Event = require 'utils.event'
local table = require 'utils.table'
local set_timeout_in_ticks = Task.set_timeout_in_ticks
local session_data_set = 'sessions'
local session = {}
local online_track = {}
local trusted = {}
local settings = {
-- local trusted_value = 2592000 -- 12h
trusted_value = 5184000, -- 24h
required_only_time_to_save_time = 36000, -- nearest prime to 10 minutes in ticks
nth_tick = 18000 -- nearest prime to 5 minutes in ticks
}
local set_data = Server.set_data
local try_get_data = Server.try_get_data
local try_get_data_and_print = Server.try_get_data_and_print
local concat = table.concat
Global.register(
{
session = session,
online_track = online_track,
trusted = trusted,
settings = settings
},
function(tbl)
session = tbl.session
online_track = tbl.online_track
trusted = tbl.trusted
settings = tbl.settings
end
)
local Public = {
events = {
on_player_removed = Event.generate_event_name('on_player_removed')
}
}
local try_download_data_token =
Token.register(
function(data)
local player_index = data.key
local value = data.value
if value then
session[player_index] = value
if value > settings.trusted_value then
trusted[player_index] = true
end
else
local player = game.get_player(player_index)
session[player_index] = 0
trusted[player_index] = false
-- we don't want to clutter the database with players less than 10 minutes played.
if player.online_time >= settings.required_only_time_to_save_time then
set_data(session_data_set, player_index, session[player_index])
end
end
end
)
local try_upload_data_token =
Token.register(
function(data)
local player_index = data.key
if not player_index then
return
end
local value = data.value
local player = game.get_player(player_index)
if value then
-- we don't want to clutter the database with players less than 10 minutes played.
if player.online_time <= settings.required_only_time_to_save_time then
return
end
local old_time_ingame = value
if not online_track[player_index] then
online_track[player_index] = 0
end
if online_track[player_index] > player.online_time then
-- instance has been reset but scenario owner did not clear the player.
-- so we clear it here and return.
online_track[player_index] = 0
return
end
local new_time = old_time_ingame + player.online_time - online_track[player_index]
if new_time <= 0 then
new_time = old_time_ingame + player.online_time
online_track[player_index] = 0
print('[ERROR] ' .. player_index .. ' had new time set as negative value: ' .. new_time)
return
end
if new_time > settings.trusted_value then
trusted[player_index] = true
end
set_data(session_data_set, player_index, new_time)
session[player_index] = new_time
online_track[player_index] = player.online_time
else
if player.online_time >= settings.required_only_time_to_save_time then
if not session[player_index] then
session[player_index] = 0
end
set_data(session_data_set, player_index, session[player_index])
end
end
end
)
local get_total_playtime_token =
Token.register(
function(data)
if not data then
return
end
if not data.to_print then
return
end
local player_index = data.key
local value = data.value
local to_print = data.to_print
local player = game.get_player(to_print)
if player and player.valid then
if player_index then
if value then
player.play_sound {path = 'utility/scenario_message', volume_modifier = 1}
player.print('[color=blue]' .. player_index .. '[/color] has a total playtime of: ' .. Core.get_formatted_playtime(value))
else
player.play_sound {path = 'utility/cannot_build', volume_modifier = 1}
player.print('[color=red]' .. player_index .. '[/color] was not found.')
end
end
end
end
)
local nth_tick_token =
Token.register(
function(data)
local index = data.index
local player = game.get_player(index)
if player and player.valid then
Public.try_ul_data(player.name)
end
end
)
--- Uploads each connected players play time to the dataset
local function upload_data()
local players = game.connected_players
local count = 0
for i = 1, #players do
count = count + 10
local player = players[i]
set_timeout_in_ticks(count, nth_tick_token, {index = player.index})
end
end
--- Prints out game.tick to real hour/minute
---@param ticks int
---@param h int
---@param m int
function Public.format_time(ticks, h, m)
local seconds = ticks / 60
local minutes = math.floor((seconds) / 60)
local hours = math.floor((minutes) / 60)
local min = math.floor(minutes - 60 * hours)
if h and m then
return string.format('%dh:%02dm', hours, minutes, min)
elseif h then
return string.format('%dh', hours)
elseif m then
return string.format('%02dm', minutes, min)
end
end
--- Tries to get data from the webpanel and prints it out to the player
---@param player LuaPlayer
---@param target_player string
function Public.get_and_print_to_player(player, target_player)
if not (player and player.valid) then
return
end
local p = player.print
if not target_player then
p('[ERROR] No player was provided.', Color.fail)
return
end
if not player.admin then
p("[ERROR] You're not admin.", Color.fail)
return
end
local secs = Server.get_current_time()
if secs == nil then
return
else
try_get_data_and_print(session_data_set, target_player, player.name, get_total_playtime_token)
end
end
--- Tries to get data from the webpanel and updates the local table with values.
---@param player_index string
function Public.try_dl_data(player_index)
player_index = tostring(player_index)
local secs = Server.get_current_time()
if secs == nil then
session[player_index] = game.players[player_index].online_time
return
else
try_get_data(session_data_set, player_index, try_download_data_token)
end
end
--- Tries to get data from the webpanel and updates the local table with values.
---@param player_index string
function Public.try_ul_data(player_index)
player_index = tostring(player_index)
local secs = Server.get_current_time()
if secs == nil then
return
else
try_get_data(session_data_set, player_index, try_upload_data_token)
end
end
--- Checks if a player exists within the table
---@param player_name string
---@return boolean
function Public.exists(player_name)
return session[player_name] ~= nil
end
--- Prints a list of all players in the player_session table.
function Public.print_sessions()
local result = {}
for k, _ in pairs(session) do
result[#result + 1] = k
end
result = concat(result, ', ')
Game.player_print(result)
end
--- Returns the table of session
---@return table
function Public.get_session_table()
return session
end
--- Returns the table of online_track
---@return table
function Public.get_tracker_table()
return online_track
end
--- Returns the table of trusted
---@return table
function Public.get_trusted_table()
return trusted
end
--- Returns the table of trusted
---@param player LuaPlayer
---@return table|boolean
function Public.get_trusted_player(player)
return trusted and player and player.valid and trusted[player.name] or false
end
--- Set a player as trusted
---@param player LuaPlayer
function Public.set_trusted_player(player)
if trusted and player and player.valid then
trusted[player.name] = true
end
end
--- Set a player as untrusted
---@param player LuaPlayer
function Public.set_untrusted_player(player)
if trusted and player and player.valid then
trusted[player.name] = nil
end
end
--- Returns the table of session
---@param player LuaPlayer
---@return table|boolean
function Public.get_session_player(player)
return session and player and player.valid and session[player.name] or false
end
--- Returns the table of settings
---@return table
function Public.get_settings_table()
return settings
end
--- Clears a given player from the session tables.
---@param player LuaPlayer
function Public.clear_player(player)
if player and player.valid then
local name = player.name
local connected = player.connected
if not connected then
if session[name] then
session[name] = nil
end
if online_track[name] then
online_track[name] = nil
end
if trusted[name] then
trusted[name] = nil
end
end
end
end
--- Resets a given player from the online_track table.
---@param player LuaPlayer
function Public.reset_online_track(player)
local name = player.name
if online_track[name] then
online_track[name] = 0
end
end
--- It's vital that we reset the online_track so we
--- don't calculate the values wrong.
Event.add(
Public.events.on_player_removed,
function()
for name, _ in pairs(online_track) do
local player = game.get_player(name)
Public.clear_player(player)
end
end
)
Event.add(
defines.events.on_player_joined_game,
function(event)
local player = game.get_player(event.player_index)
if not player or not player.valid then
return
end
Public.try_dl_data(player.name)
end
)
Event.add(
defines.events.on_player_left_game,
function(event)
local player = game.get_player(event.player_index)
if not player or not player.valid then
return
end
Public.try_ul_data(player.name)
end
)
Event.on_nth_tick(settings.nth_tick, upload_data)
Server.on_data_set_changed(
session_data_set,
function(data)
local player = game.get_player(data.key)
if player and player.valid then
session[data.key] = data.value
if data.value > settings.trusted_value then
trusted[data.key] = true
else
if trusted[data.key] then
trusted[data.key] = false
end
end
end
end
)
commands.add_command(
'playtime',
'Fetches a player total playtime or nil.',
function(cmd)
local player = game.player
if not (player and player.valid) then
return
end
local p = player.print
local param = cmd.parameter
if not param then
p('[ERROR] No player was provided.', Color.fail)
return
end
Public.get_and_print_to_player(player, param)
end
)
Public.upload_data = upload_data
return Public