/var/datum/controller/subsystem/statistics/SSfeedback /datum/controller/subsystem/statistics name = "Statistics & Inactivity" wait = 1 MINUTE flags = SS_NO_TICK_CHECK | SS_BACKGROUND priority = SS_PRIORITY_STATISTICS 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_raider = list() var/list/msg_cargo = list() var/list/msg_service = list() var/list/datum/statistic/simple_statistics = list() var/list/datum/feedback_variable/feedback = list() /datum/controller/subsystem/statistics/New() NEW_SS_GLOBAL(SSfeedback) /datum/controller/subsystem/statistics/Initialize(timeofday) if (!config.kick_inactive && !(config.sql_enabled && config.sql_stats)) can_fire = FALSE 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, /proc/cmp_name_asc, TRUE) /datum/controller/subsystem/statistics/fire() // Handle AFK. if (config.kick_inactive) var/inactivity_threshold = config.kick_inactive MINUTES for (var/client/C in clients) if (!istype(C.mob, /mob/dead)) if (C.is_afk(inactivity_threshold)) log_access("AFK: [key_name(C)]") C << span("warning", "You have been inactive for more than [config.kick_inactive] minute\s and have been disconnected.") qdel(C) // Handle population polling. if (config.sql_enabled && config.sql_stats) var/admincount = admins.len var/playercount = 0 for(var/mob/M in player_list) if(M.client) playercount += 1 establish_db_connection(dbcon) if(!dbcon.IsConnected()) 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 = 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") /datum/controller/subsystem/statistics/Recover() src.messages = SSfeedback.messages src.messages_admin = SSfeedback.messages_admin src.msg_common = SSfeedback.msg_common src.msg_science = SSfeedback.msg_science src.msg_command = SSfeedback.msg_command src.msg_medical = SSfeedback.msg_medical src.msg_engineering = SSfeedback.msg_engineering src.msg_security = SSfeedback.msg_security src.msg_deathsquad = SSfeedback.msg_deathsquad src.msg_syndicate = SSfeedback.msg_syndicate src.msg_cargo = SSfeedback.msg_cargo src.msg_service = SSfeedback.msg_service src.feedback = SSfeedback.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/message_server/MS in SSmachinery.all_machines) 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","PDA-[pda_msg_amt]") 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 += "

Round Statistics

" for (var/statistic in simple_statistics) var/datum/statistic/S = simple_statistics[statistic] if (S.broadcast_at_roundend && S.has_value()) dat += span("notice", "[S]: [S.get_roundend_lines()]") to_chat(world, dat.Join("\n")) // Called on world reboot. /datum/controller/subsystem/statistics/Shutdown() if(!feedback) return if (!config.sql_enabled || !config.sql_stats) return round_end_data_gathering() //round_end time logging and some other data processing establish_db_connection(dbcon) if(!dbcon.IsConnected()) return for(var/datum/feedback_variable/FV in feedback) var/sql = "INSERT INTO ss13_feedback VALUES (null, Now(), \"[game_id]\", \"[FV.get_variable()]\", [FV.get_value()], \"[FV.get_details()]\")" var/DBQuery/query_insert = dbcon.NewQuery(sql) query_insert.Execute() // Sanitize inputs to avoid SQL injection attacks /proc/sql_sanitize_text(var/text) text = replacetext(text, "'", "''") text = replacetext(text, ";", "") text = replacetext(text, "&", "") return text /proc/feedback_set(var/variable,var/value) if(!SSfeedback) return variable = sql_sanitize_text(variable) var/datum/feedback_variable/FV = SSfeedback.find_feedback_datum(variable) if(!FV) return FV.set_value(value) /proc/feedback_inc(var/variable,var/value) if(!SSfeedback) return variable = sql_sanitize_text(variable) var/datum/feedback_variable/FV = SSfeedback.find_feedback_datum(variable) if(!FV) return FV.inc(value) /proc/feedback_dec(var/variable,var/value) if(!SSfeedback) return variable = sql_sanitize_text(variable) var/datum/feedback_variable/FV = SSfeedback.find_feedback_datum(variable) if(!FV) return FV.dec(value) /proc/feedback_set_details(var/variable,var/details) if(!SSfeedback) return variable = sql_sanitize_text(variable) details = sql_sanitize_text(details) var/datum/feedback_variable/FV = SSfeedback.find_feedback_datum(variable) if(!FV) return FV.set_details(details) /proc/feedback_add_details(var/variable,var/details) if(!SSfeedback) return variable = sql_sanitize_text(variable) details = sql_sanitize_text(details) var/datum/feedback_variable/FV = SSfeedback.find_feedback_datum(variable) if(!FV) return FV.add_details(details) /proc/sql_report_death(var/mob/living/carbon/human/H) if(!config.sql_enabled || !config.sql_stats) return if(!H) return if(!H.key || !H.mind) return var/area/placeofdeath = get_area(H) var/podname = placeofdeath ? placeofdeath.name : "Unknown area" var/sqlname = sanitizeSQL(H.real_name) var/sqlkey = sanitizeSQL(H.key) var/sqlpod = sanitizeSQL(podname) var/sqlspecial = sanitizeSQL(H.mind.special_role) var/sqljob = sanitizeSQL(H.mind.assigned_role) var/laname var/lakey if(H.lastattacker) laname = sanitizeSQL(H.lastattacker:real_name) lakey = sanitizeSQL(H.lastattacker:key) var/sqltime = time2text(world.realtime, "YYYY-MM-DD hh:mm:ss") var/coord = "[H.x], [H.y], [H.z]" //world << "INSERT INTO death (name, byondkey, job, special, pod, tod, laname, lakey, gender, bruteloss, fireloss, brainloss, oxyloss) VALUES ('[sqlname]', '[sqlkey]', '[sqljob]', '[sqlspecial]', '[sqlpod]', '[sqltime]', '[laname]', '[lakey]', '[H.gender]', [H.bruteloss], [H.getFireLoss()], [H.brainloss], [H.getOxyLoss()])" establish_db_connection(dbcon) if(!dbcon.IsConnected()) log_game("SQL ERROR during death reporting. Failed to connect.") else var/DBQuery/query = dbcon.NewQuery("INSERT INTO ss13_death (name, byondkey, job, special, pod, tod, laname, lakey, gender, bruteloss, fireloss, brainloss, oxyloss, coord) VALUES ('[sqlname]', '[sqlkey]', '[sqljob]', '[sqlspecial]', '[sqlpod]', '[sqltime]', '[laname]', '[lakey]', '[H.gender]', [H.getBruteLoss()], [H.getFireLoss()], [H.brainloss], [H.getOxyLoss()], '[coord]')") if(!query.Execute()) var/err = query.ErrorMsg() log_game("SQL ERROR during death reporting. Error : \[[err]\]\n") /proc/sql_report_cyborg_death(var/mob/living/silicon/robot/H) if(!config.sql_enabled || !config.sql_stats) return if(!H) return if(!H.key || !H.mind) return var/area/placeofdeath = get_area(H) var/podname = placeofdeath ? placeofdeath.name : "Unknown area" var/sqlname = sanitizeSQL(H.real_name) var/sqlkey = sanitizeSQL(H.key) var/sqlpod = sanitizeSQL(podname) var/sqlspecial = sanitizeSQL(H.mind.special_role) var/sqljob = sanitizeSQL(H.mind.assigned_role) var/laname var/lakey if(H.lastattacker) laname = sanitizeSQL(H.lastattacker:real_name) lakey = sanitizeSQL(H.lastattacker:key) var/sqltime = time2text(world.realtime, "YYYY-MM-DD hh:mm:ss") var/coord = "[H.x], [H.y], [H.z]" //world << "INSERT INTO death (name, byondkey, job, special, pod, tod, laname, lakey, gender, bruteloss, fireloss, brainloss, oxyloss) VALUES ('[sqlname]', '[sqlkey]', '[sqljob]', '[sqlspecial]', '[sqlpod]', '[sqltime]', '[laname]', '[lakey]', '[H.gender]', [H.bruteloss], [H.getFireLoss()], [H.brainloss], [H.getOxyLoss()])" establish_db_connection(dbcon) if(!dbcon.IsConnected()) log_game("SQL ERROR during death reporting. Failed to connect.") else var/DBQuery/query = dbcon.NewQuery("INSERT INTO ss13_death (name, byondkey, job, special, pod, tod, laname, lakey, gender, bruteloss, fireloss, brainloss, oxyloss, coord) VALUES ('[sqlname]', '[sqlkey]', '[sqljob]', '[sqlspecial]', '[sqlpod]', '[sqltime]', '[laname]', '[lakey]', '[H.gender]', [H.getBruteLoss()], [H.getFireLoss()], [H.brainloss], [H.getOxyLoss()], '[coord]')") if(!query.Execute()) var/err = query.ErrorMsg() log_game("SQL ERROR during death reporting. Error : \[[err]\]\n") /datum/controller/subsystem/statistics/proc/IncrementSimpleStat(stat) . = TRUE var/datum/statistic/numeric/S = simple_statistics[stat] if (!S) return FALSE S.increment_value() /datum/controller/subsystem/statistics/proc/IncrementGroupedStat(stat, key) . = TRUE var/datum/statistic/grouped/S = simple_statistics[stat] if (!S) return FALSE S.increment_value(key)