minecraft-exporter/minecraft_exporter.py

395 lines
22 KiB
Python
Raw Normal View History

2019-10-26 16:20:05 +02:00
import json
import os
import re
import time
2019-10-26 16:20:05 +02:00
from os import listdir
from os.path import isfile, join
import nbt
import requests
import schedule
from mcrcon import MCRcon, MCRconException
from prometheus_client import Metric, REGISTRY, start_http_server
2019-10-26 16:20:05 +02:00
class MinecraftCollector(object):
def __init__(self):
self.stats_directory = "/world/stats"
self.player_directory = "/world/playerdata"
self.advancements_directory = "/world/advancements"
self.better_questing = "/world/betterquesting"
self.player_map = dict()
self.quests_enabled = False
self.rcon = None
self.rcon_connected = False
if all(x in os.environ for x in ['RCON_HOST', 'RCON_PASSWORD']):
self.rcon = MCRcon(os.environ['RCON_HOST'], os.environ['RCON_PASSWORD'], port=int(os.environ['RCON_PORT']))
print("RCON is enabled for " + os.environ['RCON_HOST'])
if os.path.isdir(self.better_questing):
self.quests_enabled = True
schedule.every().day.at("01:00").do(self.flush_playernamecache)
2019-10-26 16:20:05 +02:00
def get_players(self):
return [f[:-5] for f in listdir(self.stats_directory) if isfile(join(self.stats_directory, f))]
2019-10-26 16:20:05 +02:00
def flush_playernamecache(self):
print("flushing playername cache")
self.player_map = dict()
def uuid_to_player(self, uuid):
if uuid in self.player_map:
return self.player_map[uuid]
2019-10-26 16:20:05 +02:00
else:
try:
result = requests.get('https://sessionserver.mojang.com/session/minecraft/profile/' + uuid)
self.player_map[uuid] = result.json()['name']
return (result.json()['name'])
except:
2021-03-16 13:56:48 +01:00
return
def rcon_connect(self):
try:
self.rcon.connect()
self.rcon_connected = True
print("Successfully connected to RCON")
return True
except Exception as e:
print("Failed to connect to RCON")
print(e)
return False
2022-01-23 20:01:04 +01:00
def rcon_disconnect(self):
self.rcon.disconnect()
self.rcon_connected = False
def rcon_command(self, command):
try:
response = self.rcon.command(command)
2022-01-23 20:01:04 +01:00
except MCRconException as e:
response = None
if e == "Connection timeout error":
print("Lost RCON Connection")
self.rcon_disconnect()
else:
print("RCON command failed")
except (ConnectionResetError, ConnectionAbortedError, BrokenPipeError):
print("Lost RCON Connection")
2022-01-23 20:01:04 +01:00
self.rcon_disconnect()
response = None
return response
2020-01-11 17:32:53 +01:00
2019-10-26 16:20:05 +02:00
def get_server_stats(self):
if self.rcon is None or (not self.rcon_connected and not self.rcon_connect()):
2019-10-26 16:20:05 +02:00
return []
metrics = []
2023-10-23 06:40:44 +02:00
dim_tps = Metric('mc_dim_tps', 'TPS of a dimension', "counter")
dim_ticktime = Metric('mc_dim_ticktime', "Time a Tick took in a Dimension", "counter")
overall_tps = Metric('mc_overall_tps', 'overall TPS', "counter")
overall_ticktime = Metric('mc_overall_ticktime', "overall Ticktime", "counter")
player_online = Metric('mc_player_online', "is 1 if player is online", "counter")
entities = Metric('mc_entities', "type and count of active entites", "counter")
tps_1m = Metric('mc_paper_tps_1m', '1 Minute TPS', "counter")
tps_5m = Metric('mc_paper_tps_5m', '5 Minute TPS', "counter")
tps_15m = Metric('mc_paper_tps_15m', '15 Minute TPS', "counter")
metrics.extend(
[dim_tps, dim_ticktime, overall_tps, overall_ticktime, player_online, entities, tps_1m, tps_5m, tps_15m])
if 'PAPER_SERVER' in os.environ and os.environ['PAPER_SERVER'] == "True":
resp = str(self.rcon_command("tps")).strip().replace("§a", "")
tpsregex = re.compile("TPS from last 1m, 5m, 15m: (\d*\.\d*), (\d*\.\d*), (\d*\.\d*)")
for m1, m5, m15 in tpsregex.findall(resp):
2023-10-23 06:40:44 +02:00
tps_1m.add_sample('mc_paper_tps_1m', value=m1, labels={'tps': '1m'})
tps_5m.add_sample('mc_paper_tps_5m', value=m5, labels={'tps': '5m'})
tps_15m.add_sample('mc_paper_tps_15m', value=m15, labels={'tps': '15m'})
if 'FORGE_SERVER' in os.environ and os.environ['FORGE_SERVER'] == "True":
# dimensions
resp = self.rcon_command("forge tps")
dimtpsregex = re.compile("Dim\s*(-*\d*)\s\((.*?)\)\s:\sMean tick time:\s(.*?) ms\. Mean TPS: (\d*\.\d*)")
for dimid, dimname, meanticktime, meantps in dimtpsregex.findall(resp):
2023-10-23 06:40:44 +02:00
dim_tps.add_sample('mc_dim_tps', value=meantps, labels={'dimension_id': dimid, 'dimension_name': dimname})
dim_ticktime.add_sample('mc_dim_ticktime', value=meanticktime,
labels={'dimension_id': dimid, 'dimension_name': dimname})
overallregex = re.compile("Overall\s?: Mean tick time: (.*) ms. Mean TPS: (.*)")
2023-10-23 06:40:44 +02:00
overall_tps.add_sample('mc_overall_tps', value=overallregex.findall(resp)[0][1], labels={})
overall_ticktime.add_sample('mc_overall_ticktime', value=overallregex.findall(resp)[0][0], labels={})
# entites
resp = self.rcon_command("forge entity list")
entityregex = re.compile("(\d+): (.*?:.*?)\s")
for entitycount, entityname in entityregex.findall(resp):
2023-10-23 06:40:44 +02:00
entities.add_sample('mc_entities', value=entitycount, labels={'entity': entityname})
2019-11-05 22:08:19 +01:00
2020-03-27 21:56:05 +01:00
# dynmap
if 'DYNMAP_ENABLED' in os.environ and os.environ['DYNMAP_ENABLED'] == "True":
2023-10-23 06:40:44 +02:00
dynmap_tile_render_statistics = Metric('mc_dynmap_tile_render_statistics',
'Tile Render Statistics reported by Dynmap', "counter")
2023-10-23 06:40:44 +02:00
dynmap_chunk_loading_statistics_count = Metric('mc_dynmap_chunk_loading_statistics_count',
'Chunk Loading Statistics reported by Dynmap', "counter")
2023-10-23 06:40:44 +02:00
dynmap_chunk_loading_statistics_duration = Metric('mc_dynmap_chunk_loading_statistics_duration',
'Chunk Loading Statistics reported by Dynmap', "counter")
metrics.extend([dynmap_tile_render_statistics, dynmap_chunk_loading_statistics_count,
dynmap_chunk_loading_statistics_duration])
2020-03-27 21:56:05 +01:00
resp = self.rcon_command("dynmap stats")
2020-03-27 21:56:05 +01:00
dynmaptilerenderregex = re.compile(" (.*?): processed=(\d*), rendered=(\d*), updated=(\d*)")
for dim, processed, rendered, updated in dynmaptilerenderregex.findall(resp):
2023-10-23 06:40:44 +02:00
dynmap_tile_render_statistics.add_sample('mc_dynmap_tile_render_statistics', value=processed,
labels={'type': 'processed', 'file': dim})
2023-10-23 06:40:44 +02:00
dynmap_tile_render_statistics.add_sample('mc_dynmap_tile_render_statistics', value=rendered,
labels={'type': 'rendered', 'file': dim})
2023-10-23 06:40:44 +02:00
dynmap_tile_render_statistics.add_sample('mc_dynmap_tile_render_statistics', value=updated,
labels={'type': 'updated', 'file': dim})
2020-03-27 21:56:05 +01:00
dynmapchunkloadingregex = re.compile("Chunks processed: (.*?): count=(\d*), (\d*.\d*)")
for state, count, duration_per_chunk in dynmapchunkloadingregex.findall(resp):
2023-10-23 06:40:44 +02:00
dynmap_chunk_loading_statistics_count.add_sample('mc_dynmap_chunk_loading_statistics', value=count,
labels={'type': state})
2023-10-23 06:40:44 +02:00
dynmap_chunk_loading_statistics_duration.add_sample('mc_dynmap_chunk_loading_duration',
value=duration_per_chunk, labels={'type': state})
2020-03-27 21:56:05 +01:00
2019-11-05 22:08:19 +01:00
# player
resp = self.rcon_command("list")
2023-10-23 06:40:44 +02:00
#playerregex = re.compile("players online:(.*)")
#if playerregex.findall(resp):
# for player in playerregex.findall(resp)[0].split(","):
# if not player.isspace():
# player_online.add_sample('mc_player_online', value=1, labels={'player': player.lstrip()})
if resp.startswith('There are 0 '):
player_online.add_sample('mc_player_online', value=0, labels={})
else:
players = resp.split(':')[1].replace(' ','').split(',')
for player in players:
player_online.add_sample('mc_player_online', value=1, labels={'player': player})
2020-04-19 17:25:15 +02:00
return metrics
2019-10-26 16:20:05 +02:00
def get_player_quests_finished(self, uuid):
with open(self.better_questing + "/QuestProgress.json") as json_file:
2020-01-11 17:32:53 +01:00
data = json.load(json_file)
json_file.close()
counter = 0
2020-01-11 17:41:22 +01:00
for _, value in data['questProgress:9'].items():
2020-01-11 17:57:04 +01:00
for _, u in value['tasks:9']['0:10']['completeUsers:9'].items():
if u == uuid:
counter += 1
2020-01-11 17:32:53 +01:00
return counter
2019-10-26 16:20:05 +02:00
def get_player_stats(self, uuid):
with open(self.stats_directory + "/" + uuid + ".json") as json_file:
2019-10-26 16:20:05 +02:00
data = json.load(json_file)
json_file.close()
nbtfile = nbt.nbt.NBTFile(self.player_directory + "/" + uuid + ".dat", 'rb')
data["stat.XpTotal"] = nbtfile.get("XpTotal").value
data["stat.XpLevel"] = nbtfile.get("XpLevel").value
data["stat.Score"] = nbtfile.get("Score").value
data["stat.Health"] = nbtfile.get("Health").value
data["stat.foodLevel"] = nbtfile.get("foodLevel").value
with open(self.advancements_directory + "/" + uuid + ".json") as json_file:
2019-10-26 16:20:05 +02:00
count = 0
advancements = json.load(json_file)
for key, value in advancements.items():
2020-03-07 12:09:24 +01:00
if key in ("DataVersion"):
continue
2020-03-07 12:09:24 +01:00
if value["done"] == True:
2019-10-26 16:20:05 +02:00
count += 1
data["stat.advancements"] = count
if self.quests_enabled:
2020-01-11 18:09:14 +01:00
data["stat.questsFinished"] = self.get_player_quests_finished(uuid)
2019-10-26 16:20:05 +02:00
return data
def update_metrics_for_player(self, uuid):
2019-10-26 16:20:05 +02:00
name = self.uuid_to_player(uuid)
if not name:
return
data = self.get_player_stats(uuid)
2023-10-23 06:40:44 +02:00
blocks_mined = Metric('mc_blocks_mined', 'Blocks a Player mined', "counter")
blocks_picked_up = Metric('mc_blocks_picked_up', 'Blocks a Player picked up', "counter")
player_deaths = Metric('mc_player_deaths', 'How often a Player died', "counter")
player_jumps = Metric('mc_player_jumps', 'How often a Player has jumped', "counter")
cm_traveled = Metric('mc_cm_traveled', 'How many cm a Player traveled, whatever that means', "counter")
player_xp_total = Metric('mc_player_xp_total', "How much total XP a player has", "counter")
player_current_level = Metric('mc_player_current_level', "How much current XP a player has", "counter")
player_food_level = Metric('mc_player_food_level', "How much food the player currently has", "counter")
player_health = Metric('mc_player_health', "How much Health the player currently has", "counter")
player_score = Metric('mc_player_score', "The Score of the player", "counter")
entities_killed = Metric('mc_entities_killed', "Entities killed by player", "counter")
damage_taken = Metric('mc_damage_taken', "Damage Taken by Player", "counter")
damage_dealt = Metric('mc_damage_dealt', "Damage dealt by Player", "counter")
blocks_crafted = Metric('mc_blocks_crafted', "Items a Player crafted", "counter")
player_playtime = Metric('mc_player_playtime', "Time in Minutes a Player was online", "counter")
player_advancements = Metric('mc_player_advancements', "Number of completed advances of a player", "counter")
player_slept = Metric('mc_player_slept', "Times a Player slept in a bed", "counter")
player_quests_finished = Metric('mc_player_quests_finished', 'Number of quests a Player has finished', 'counter')
player_used_crafting_table = Metric('mc_player_used_crafting_table', "Times a Player used a Crafting Table",
"counter")
mc_custom = Metric('mc_custom', "Custom Minecraft stat", "counter")
for key, value in data.items(): # pre 1.15
2020-03-07 12:09:24 +01:00
if key in ("stats", "DataVersion"):
continue
stat = key.split(".")[1] # entityKilledBy
2019-10-26 16:20:05 +02:00
if stat == "mineBlock":
blocks_mined.add_sample("blocks_mined", value=value, labels={'player': name, 'block': '.'.join(
(key.split(".")[2], key.split(".")[3]))})
2019-10-26 16:20:05 +02:00
elif stat == "pickup":
blocks_picked_up.add_sample("blocks_picked_up", value=value, labels={'player': name, 'block': '.'.join(
(key.split(".")[2], key.split(".")[3]))})
2019-10-26 16:20:05 +02:00
elif stat == "entityKilledBy":
if len(key.split(".")) == 4:
2023-10-23 06:40:44 +02:00
player_deaths.add_sample('mc_player_deaths', value=value, labels={'player': name, 'cause': '.'.join(
(key.split(".")[2], key.split(".")[3]))})
2019-10-26 16:20:05 +02:00
else:
2023-10-23 06:40:44 +02:00
player_deaths.add_sample('mc_player_deaths', value=value,
labels={'player': name, 'cause': key.split(".")[2]})
2019-10-26 16:20:05 +02:00
elif stat == "jump":
player_jumps.add_sample("player_jumps", value=value, labels={'player': name})
2019-10-26 16:20:05 +02:00
elif stat == "walkOneCm":
cm_traveled.add_sample("cm_traveled", value=value, labels={'player': name, 'method': "walking"})
2019-10-26 16:20:05 +02:00
elif stat == "swimOneCm":
cm_traveled.add_sample("cm_traveled", value=value, labels={'player': name, 'method': "swimming"})
2019-10-26 16:20:05 +02:00
elif stat == "sprintOneCm":
cm_traveled.add_sample("cm_traveled", value=value, labels={'player': name, 'method': "sprinting"})
2019-10-26 16:20:05 +02:00
elif stat == "diveOneCm":
cm_traveled.add_sample("cm_traveled", value=value, labels={'player': name, 'method': "diving"})
2019-10-26 16:20:05 +02:00
elif stat == "fallOneCm":
cm_traveled.add_sample("cm_traveled", value=value, labels={'player': name, 'method': "falling"})
2019-10-26 16:20:05 +02:00
elif stat == "flyOneCm":
cm_traveled.add_sample("cm_traveled", value=value, labels={'player': name, 'method': "flying"})
2019-10-26 16:20:05 +02:00
elif stat == "boatOneCm":
cm_traveled.add_sample("cm_traveled", value=value, labels={'player': name, 'method': "boat"})
2019-10-26 16:20:05 +02:00
elif stat == "horseOneCm":
cm_traveled.add_sample("cm_traveled", value=value, labels={'player': name, 'method': "horse"})
2019-10-26 16:20:05 +02:00
elif stat == "climbOneCm":
cm_traveled.add_sample("cm_traveled", value=value, labels={'player': name, 'method': "climbing"})
2019-10-26 16:20:05 +02:00
elif stat == "XpTotal":
2023-10-23 06:40:44 +02:00
player_xp_total.add_sample('mc_player_xp_total', value=value, labels={'player': name})
2019-10-26 16:20:05 +02:00
elif stat == "XpLevel":
2023-10-23 06:40:44 +02:00
player_current_level.add_sample('mc_player_current_level', value=value, labels={'player': name})
2019-10-26 16:20:05 +02:00
elif stat == "foodLevel":
2023-10-23 06:40:44 +02:00
player_food_level.add_sample('mc_player_food_level', value=value, labels={'player': name})
2019-10-26 16:20:05 +02:00
elif stat == "Health":
2023-10-23 06:40:44 +02:00
player_health.add_sample('mc_player_health', value=value, labels={'player': name})
2019-10-26 16:20:05 +02:00
elif stat == "Score":
2023-10-23 06:40:44 +02:00
player_score.add_sample('mc_player_score', value=value, labels={'player': name})
2019-10-26 16:20:05 +02:00
elif stat == "killEntity":
2023-10-23 06:40:44 +02:00
entities_killed.add_sample('mc_entities_killed', value=value,
labels={'player': name, "entity": key.split(".")[2]})
2019-10-26 16:20:05 +02:00
elif stat == "damageDealt":
2023-10-23 06:40:44 +02:00
damage_dealt.add_sample('mc_damage_dealt', value=value, labels={'player': name})
2019-10-26 16:20:05 +02:00
elif stat == "damageTaken":
2023-10-23 06:40:44 +02:00
damage_dealt.add_sample('mc_damage_taken', value=value, labels={'player': name})
2019-10-26 16:20:05 +02:00
elif stat == "craftItem":
2023-10-23 06:40:44 +02:00
blocks_crafted.add_sample('mc_blocks_crafted', value=value, labels={'player': name, 'block': '.'.join(
(key.split(".")[2], key.split(".")[3]))})
2019-10-26 16:20:05 +02:00
elif stat == "playOneMinute":
2023-10-23 06:40:44 +02:00
player_playtime.add_sample('mc_player_playtime', value=value, labels={'player': name})
2019-10-26 16:20:05 +02:00
elif stat == "advancements":
2023-10-23 06:40:44 +02:00
player_advancements.add_sample('mc_player_advancements', value=value, labels={'player': name})
2019-10-26 16:20:05 +02:00
elif stat == "sleepInBed":
2023-10-23 06:40:44 +02:00
player_slept.add_sample('mc_player_slept', value=value, labels={'player': name})
2019-10-26 16:20:05 +02:00
elif stat == "craftingTableInteraction":
2023-10-23 06:40:44 +02:00
player_used_crafting_table.add_sample('mc_player_used_crafting_table', value=value,
labels={'player': name})
2020-01-11 17:32:53 +01:00
elif stat == "questsFinished":
2023-10-23 06:40:44 +02:00
player_quests_finished.add_sample('mc_player_quests_finished', value=value, labels={'player': name})
2020-05-09 20:07:07 +02:00
if "stats" in data: # Minecraft > 1.15
if "minecraft:crafted" in data["stats"]:
for block, value in data["stats"]["minecraft:crafted"].items():
2023-10-23 06:40:44 +02:00
blocks_crafted.add_sample('mc_blocks_crafted', value=value, labels={'player': name, 'block': block})
if "minecraft:mined" in data["stats"]:
for block, value in data["stats"]["minecraft:mined"].items():
blocks_mined.add_sample("blocks_mined", value=value, labels={'player': name, 'block': block})
if "minecraft:picked_up" in data["stats"]:
for block, value in data["stats"]["minecraft:picked_up"].items():
blocks_picked_up.add_sample("blocks_picked_up", value=value,
labels={'player': name, 'block': block})
if "minecraft:killed" in data["stats"]:
for entity, value in data["stats"]["minecraft:killed"].items():
2023-10-23 06:40:44 +02:00
entities_killed.add_sample('mc_entities_killed', value=value,
labels={'player': name, "entity": entity})
if "minecraft:killed_by" in data["stats"]:
for entity, value in data["stats"]["minecraft:killed_by"].items():
2023-10-23 06:40:44 +02:00
player_deaths.add_sample('mc_player_deaths', value=value, labels={'player': name, 'cause': entity})
2020-05-09 20:07:07 +02:00
for stat, value in data["stats"]["minecraft:custom"].items():
if stat == "minecraft:jump":
player_jumps.add_sample("player_jumps", value=value, labels={'player': name})
2020-05-09 20:07:07 +02:00
elif stat == "minecraft:deaths":
2023-10-23 06:40:44 +02:00
player_deaths.add_sample('mc_player_deaths', value=value, labels={'player': name})
2020-05-09 20:07:07 +02:00
elif stat == "minecraft:damage_taken":
2023-10-23 06:40:44 +02:00
damage_taken.add_sample('mc_damage_taken', value=value, labels={'player': name})
2020-05-09 20:07:07 +02:00
elif stat == "minecraft:damage_dealt":
2023-10-23 06:40:44 +02:00
damage_dealt.add_sample('mc_damage_dealt',value=value,labels={'player':name})
2022-02-06 01:57:23 +01:00
elif stat == "minecraft:play_time":
2023-10-23 06:40:44 +02:00
player_playtime.add_sample('mc_player_playtime',value=value,labels={'player':name})
2022-02-21 10:02:05 +01:00
elif stat == "minecraft:play_one_minute": # pre 1.17
2023-10-23 06:40:44 +02:00
player_playtime.add_sample('mc_player_playtime',value=value,labels={'player':name})
2020-05-09 20:07:07 +02:00
elif stat == "minecraft:walk_one_cm":
cm_traveled.add_sample("cm_traveled", value=value, labels={'player': name, 'method': "walking"})
2020-05-09 20:07:07 +02:00
elif stat == "minecraft:walk_on_water_one_cm":
cm_traveled.add_sample("cm_traveled", value=value, labels={'player': name, 'method': "swimming"})
2020-05-09 20:07:07 +02:00
elif stat == "minecraft:sprint_one_cm":
cm_traveled.add_sample("cm_traveled", value=value, labels={'player': name, 'method': "sprinting"})
2020-05-09 20:07:07 +02:00
elif stat == "minecraft:walk_under_water_one_cm":
cm_traveled.add_sample("cm_traveled", value=value, labels={'player': name, 'method': "diving"})
2020-05-09 20:07:07 +02:00
elif stat == "minecraft:fall_one_cm":
cm_traveled.add_sample("cm_traveled", value=value, labels={'player': name, 'method': "falling"})
2020-05-09 20:07:07 +02:00
elif stat == "minecraft:fly_one_cm":
cm_traveled.add_sample("cm_traveled", value=value, labels={'player': name, 'method': "flying"})
2020-05-09 20:07:07 +02:00
elif stat == "minecraft:boat_one_cm":
cm_traveled.add_sample("cm_traveled", value=value, labels={'player': name, 'method': "boat"})
2020-05-09 20:07:07 +02:00
elif stat == "minecraft:horse_one_cm":
cm_traveled.add_sample("cm_traveled", value=value, labels={'player': name, 'method': "horse"})
2020-05-09 20:07:07 +02:00
elif stat == "minecraft:climb_one_cm":
cm_traveled.add_sample("cm_traveled", value=value, labels={'player': name, 'method': "climbing"})
2020-05-09 20:07:07 +02:00
elif stat == "minecraft:sleep_in_bed":
2023-10-23 06:40:44 +02:00
player_slept.add_sample('mc_player_slept', value=value, labels={'player': name})
2020-05-09 20:07:07 +02:00
elif stat == "minecraft:interact_with_crafting_table":
2023-10-23 06:40:44 +02:00
player_used_crafting_table.add_sample('mc_player_used_crafting_table', value=value,
labels={'player': name})
2020-05-09 20:07:07 +02:00
else:
mc_custom.add_sample('mc_custom', value=value, labels={'stat': stat})
return [blocks_mined, blocks_picked_up, player_deaths, player_jumps, cm_traveled, player_xp_total,
player_current_level, player_food_level, player_health, player_score, entities_killed, damage_taken,
damage_dealt, blocks_crafted, player_playtime, player_advancements, player_slept,
player_used_crafting_table, player_quests_finished, mc_custom]
2019-10-26 16:20:05 +02:00
def collect(self):
for player in self.get_players():
metrics = self.update_metrics_for_player(player)
if not metrics:
continue
for metric in metrics:
2019-10-26 16:20:05 +02:00
yield metric
for metric in self.get_server_stats():
yield metric
2019-10-26 16:20:05 +02:00
2022-02-23 22:55:20 +01:00
if __name__ == '__main__':
try:
HTTP_PORT = int(os.environ.get('HTTP_PORT'))
except:
HTTP_PORT = 8000
start_http_server(HTTP_PORT)
2019-10-26 16:20:05 +02:00
REGISTRY.register(MinecraftCollector())
print(f'Exporter started on Port {HTTP_PORT}')
2019-10-26 16:20:05 +02:00
while True:
2022-01-23 20:01:04 +01:00
try:
time.sleep(1)
schedule.run_pending()
except MCRconException:
# RCON timeout
collector.rcon_disconnect()