#define PING_BUFFER_TIME 25 SUBSYSTEM_DEF(statistics) name = "Statistics & Inactivity" wait = 1 MINUTE flags = SS_BACKGROUND priority = SS_PRIORITY_STATISTICS var/kicked_clients = 0 var/list/messages = list() //Stores messages of non-standard frequencies var/list/messages_admin = list() var/list/msg_common = list() var/list/msg_science = list() var/list/msg_command = list() var/list/msg_medical = list() var/list/msg_engineering = list() var/list/msg_security = list() var/list/msg_deathsquad = list() var/list/msg_syndicate = list() var/list/msg_coalition = list() var/list/msg_raider = list() var/list/msg_burglar = list() var/list/msg_ninja = list() var/list/msg_bluespace = list() var/list/msg_cargo = list() var/list/msg_service = list() var/list/msg_ship = list() var/list/datum/statistic/simple_statistics = list() var/list/datum/feedback_variable/feedback = list() var/status_needs_update = FALSE GENERAL_PROTECT_DATUM(/datum/controller/subsystem/statistics) /datum/controller/subsystem/statistics/Initialize(timeofday) RegisterSignal(SSdcs, COMSIG_GLOB_MOB_DEATH, PROC_REF(something_died)) for (var/type in subtypesof(/datum/statistic) - list(/datum/statistic/numeric, /datum/statistic/grouped)) var/datum/statistic/S = new type if (!S.name) qdel(S) continue simple_statistics[S.key] = S sortTim(simple_statistics, GLOBAL_PROC_REF(cmp_name_asc), TRUE) return SS_INIT_SUCCESS /datum/controller/subsystem/statistics/fire() // Handle AFK and pings for(var/client/C in GLOB.clients) if(GLOB.config.kick_inactive) var/inactivity_threshold = GLOB.config.kick_inactive MINUTES if(!isobserver(C.mob) && !C.holder) if(C.is_afk(inactivity_threshold)) log_access("AFK: [key_name(C)]") to_chat_immediate(C, SPAN_WARNING("You have been inactive for more than [GLOB.config.kick_inactive] minute\s and have been disconnected.")) qdel(C) kicked_clients++ //Ask the client to ping, this is done in TG by the server_maint subsystems, but I'm not gonna port it all just for this if(!(!C || world.time - C.connection_time < PING_BUFFER_TIME || C.inactivity >= (wait-1))) winset(C, null, "command=.update_ping+[num2text(world.time+world.tick_lag*TICK_USAGE_REAL/100, 32)]") // Handle population polling. if (GLOB.config.sql_enabled && GLOB.config.sql_stats) var/admincount = GLOB.staff.len var/playercount = 0 for(var/mob/M in GLOB.player_list) if(M.client) playercount += 1 if(!establish_db_connection(GLOB.dbcon)) log_game("SQL ERROR during population polling. Failed to connect.") else var/sqltime = time2text(world.realtime, "YYYY-MM-DD hh:mm:ss") var/DBQuery/query = GLOB.dbcon.NewQuery("INSERT INTO `ss13_population` (`playercount`, `admincount`, `time`) VALUES ([playercount], [admincount], '[sqltime]')") if(!query.Execute()) var/err = query.ErrorMsg() log_game("SQL ERROR during population polling. Error : \[[err]\]\n") if (status_needs_update) // Update world status. world.update_status() status_needs_update = FALSE /datum/controller/subsystem/statistics/proc/update_status() status_needs_update = TRUE /datum/controller/subsystem/statistics/Recover() src.messages = SSstatistics.messages src.messages_admin = SSstatistics.messages_admin src.msg_common = SSstatistics.msg_common src.msg_science = SSstatistics.msg_science src.msg_command = SSstatistics.msg_command src.msg_medical = SSstatistics.msg_medical src.msg_engineering = SSstatistics.msg_engineering src.msg_security = SSstatistics.msg_security src.msg_deathsquad = SSstatistics.msg_deathsquad src.msg_syndicate = SSstatistics.msg_syndicate src.msg_cargo = SSstatistics.msg_cargo src.msg_service = SSstatistics.msg_service src.feedback = SSstatistics.feedback /datum/controller/subsystem/statistics/proc/find_feedback_datum(variable) for (var/datum/feedback_variable/FV in feedback) if (FV.get_variable() == variable) return FV var/datum/feedback_variable/FV = new(variable) feedback += FV return FV /datum/controller/subsystem/statistics/proc/get_round_feedback() return feedback /datum/controller/subsystem/statistics/proc/round_end_data_gathering() var/pda_msg_amt = 0 var/rc_msg_amt = 0 for(var/obj/machinery/telecomms/message_server/MS in SSmachinery.all_telecomms) if(MS.pda_msgs.len > pda_msg_amt) pda_msg_amt = MS.pda_msgs.len if(MS.rc_msgs.len > rc_msg_amt) rc_msg_amt = MS.rc_msgs.len feedback_set_details("radio_usage","") feedback_add_details("radio_usage","COM-[msg_common.len]") feedback_add_details("radio_usage","SCI-[msg_science.len]") feedback_add_details("radio_usage","HEA-[msg_command.len]") feedback_add_details("radio_usage","MED-[msg_medical.len]") feedback_add_details("radio_usage","ENG-[msg_engineering.len]") feedback_add_details("radio_usage","SEC-[msg_security.len]") feedback_add_details("radio_usage","DTH-[msg_deathsquad.len]") feedback_add_details("radio_usage","SYN-[msg_syndicate.len]") feedback_add_details("radio_usage","CAR-[msg_cargo.len]") feedback_add_details("radio_usage","SRV-[msg_service.len]") feedback_add_details("radio_usage","OTH-[messages.len]") feedback_add_details("radio_usage","RC-[rc_msg_amt]") for (var/key in simple_statistics) var/datum/statistic/S = simple_statistics[key] if (S.write_to_db && S.key) S.write_to_database() feedback_set_details("round_end","[time2text(world.realtime)]") //This one MUST be the last one that gets set. /datum/controller/subsystem/statistics/proc/print_round_end_message() var/list/dat = list() dat += "