crafty-4/app/classes/models/server_stats.py

349 lines
12 KiB
Python
Raw Normal View History

import os
import logging
import datetime
2022-05-26 12:50:20 +00:00
from app.classes.models.servers import Servers, HelperServers
from app.classes.shared.helpers import Helpers
2022-05-26 21:19:59 +00:00
from app.classes.shared.main_models import DatabaseShortcuts
from app.classes.shared.migration import MigrationManager
try:
from peewee import (
SqliteDatabase,
Model,
ForeignKeyField,
CharField,
AutoField,
DateTimeField,
BooleanField,
IntegerField,
FloatField,
2022-05-25 11:16:14 +00:00
DoesNotExist,
)
except ModuleNotFoundError as e:
Helpers.auto_installer_fix(e)
logger = logging.getLogger(__name__)
peewee_logger = logging.getLogger("peewee")
peewee_logger.setLevel(logging.INFO)
# **********************************************************************************
# Servers Stats Class
# **********************************************************************************
class ServerStats(Model):
stats_id = AutoField()
created = DateTimeField(default=datetime.datetime.now)
server_id = ForeignKeyField(Servers, backref="server", index=True)
started = CharField(default="")
running = BooleanField(default=False)
cpu = FloatField(default=0)
mem = FloatField(default=0)
mem_percent = FloatField(default=0)
world_name = CharField(default="")
world_size = CharField(default="")
server_port = IntegerField(default=25565)
int_ping_results = CharField(default="")
online = IntegerField(default=0)
max = IntegerField(default=0)
players = CharField(default="")
desc = CharField(default="Unable to Connect")
version = CharField(default="")
updating = BooleanField(default=False)
waiting_start = BooleanField(default=False)
first_run = BooleanField(default=True)
crashed = BooleanField(default=False)
downloading = BooleanField(default=False)
class Meta:
table_name = "server_stats"
# **********************************************************************************
# Servers_Stats Methods
# **********************************************************************************
class HelperServerStats:
2022-05-26 12:50:20 +00:00
server_id: int
database = None
def __init__(self, server_id):
self.server_id = int(server_id)
self.init_database(self.server_id)
2022-05-25 13:54:35 +00:00
def init_database(self, server_id):
try:
server = HelperServers.get_server_data_by_id(server_id)
db_folder = os.path.join(f"{server['path']}", "db_stats")
db_file = os.path.join(
db_folder,
"crafty_server_stats.sqlite",
)
2022-05-26 12:50:20 +00:00
self.database = SqliteDatabase(
db_file, pragmas={"journal_mode": "wal", "cache_size": -1024 * 10}
)
if not os.path.exists(db_file):
try:
os.mkdir(db_folder)
except Exception as ex:
logger.warning(
f"Error try to create the db_stats folder for server : {ex}"
)
helper_stats = Helpers()
helper_stats.migration_dir = os.path.join(
f"{helper_stats.migration_dir}", "stats"
)
helper_stats.db_path = db_file
2022-05-26 12:50:20 +00:00
migration_manager = MigrationManager(self.database, helper_stats)
migration_manager.up() # Automatically runs migrations
except Exception as ex:
logger.warning(
f"Error try to look for the db_stats files for server : {ex}"
)
2022-05-24 22:48:27 +00:00
return None
2022-05-25 13:54:35 +00:00
def select_database(self):
try:
2022-05-25 13:54:35 +00:00
server = HelperServers.get_server_data_by_id(self.server_id)
db_file = os.path.join(
f"{server['path']}",
"db_stats",
"crafty_server_stats.sqlite",
)
2022-05-26 12:50:20 +00:00
self.database = SqliteDatabase(
db_file, pragmas={"journal_mode": "wal", "cache_size": -1024 * 10}
)
except Exception as ex:
logger.warning(
f"Error try to look for the db_stats files for server : {ex}"
)
2022-05-24 22:48:27 +00:00
return None
2022-05-25 13:54:35 +00:00
def get_all_servers_stats(self):
servers = HelperServers.get_all_defined_servers()
server_data = []
try:
2022-05-24 22:48:27 +00:00
for server in servers:
2022-05-30 20:05:36 +00:00
latest = self.get_latest_server_stats()
server_data.append(
{
2022-05-24 22:48:27 +00:00
"server_data": server,
2022-05-30 20:05:36 +00:00
"stats": latest,
"user_command_permission": True,
}
)
except IndexError as ex:
logger.error(
f"Stats collection failed with error: {ex}. Was a server just created?"
)
return server_data
2022-08-21 21:04:23 +00:00
def get_history_stats(self, server_id, max_age):
return (
ServerStats.select()
.where(ServerStats.created > max_age)
.where(ServerStats.server_id == server_id)
.execute(self.database)
)
def insert_server_stats(self, server_stats):
server_id = server_stats.get("id", 0)
if server_id == 0:
logger.warning("Stats saving failed with error: Server unknown (id = 0)")
return
ServerStats.insert(
{
ServerStats.server_id: server_stats.get("id", 0),
ServerStats.started: server_stats.get("started", ""),
ServerStats.running: server_stats.get("running", False),
ServerStats.cpu: server_stats.get("cpu", 0),
ServerStats.mem: server_stats.get("mem", 0),
ServerStats.mem_percent: server_stats.get("mem_percent", 0),
ServerStats.world_name: server_stats.get("world_name", ""),
ServerStats.world_size: server_stats.get("world_size", ""),
ServerStats.server_port: server_stats.get("server_port", 0),
ServerStats.int_ping_results: server_stats.get(
"int_ping_results", False
),
ServerStats.online: server_stats.get("online", False),
ServerStats.max: server_stats.get("max", False),
ServerStats.players: server_stats.get("players", False),
ServerStats.desc: server_stats.get("desc", False),
ServerStats.version: server_stats.get("version", False),
}
2022-05-26 12:50:20 +00:00
).execute(self.database)
2022-05-25 13:54:35 +00:00
def remove_old_stats(self, last_week):
# self.select_database(self.server_id)
2022-05-26 12:50:20 +00:00
ServerStats.delete().where(ServerStats.created < last_week).execute(
self.database
)
2022-05-25 13:54:35 +00:00
def get_latest_server_stats(self):
2022-05-26 12:50:20 +00:00
latest = (
ServerStats.select()
2022-05-25 13:54:35 +00:00
.where(ServerStats.server_id == self.server_id)
.order_by(ServerStats.created.desc())
.limit(1)
2022-05-26 12:50:20 +00:00
.get(self.database)
)
2022-05-26 12:50:20 +00:00
try:
2022-05-26 21:19:59 +00:00
return DatabaseShortcuts.get_data_obj(latest)
2022-05-26 12:50:20 +00:00
except IndexError:
return {}
2022-05-26 12:50:20 +00:00
def get_server_stats(self):
stats = (
ServerStats.select()
2022-05-25 13:54:35 +00:00
.where(ServerStats.server_id == self.server_id)
.order_by(ServerStats.created.desc())
.limit(1)
.first(self.database)
)
2022-05-26 21:19:59 +00:00
return DatabaseShortcuts.get_data_obj(stats)
2022-05-25 13:54:35 +00:00
def server_id_exists(self):
# self.select_database(self.server_id)
if not HelperServers.get_server_data_by_id(self.server_id):
return False
return True
2022-05-25 13:54:35 +00:00
def sever_crashed(self):
# self.select_database(self.server_id)
ServerStats.update(crashed=True).where(
2022-05-25 13:54:35 +00:00
ServerStats.server_id == self.server_id
2022-05-26 12:50:20 +00:00
).execute(self.database)
2022-05-25 13:54:35 +00:00
def set_download(self):
# self.select_database(self.server_id)
ServerStats.update(downloading=True).where(
2022-05-25 13:54:35 +00:00
ServerStats.server_id == self.server_id
2022-05-26 12:50:20 +00:00
).execute(self.database)
2022-05-25 13:54:35 +00:00
def finish_download(self):
# self.select_database(self.server_id)
ServerStats.update(downloading=False).where(
2022-05-25 13:54:35 +00:00
ServerStats.server_id == self.server_id
2022-05-26 12:50:20 +00:00
).execute(self.database)
2022-05-25 13:54:35 +00:00
def get_download_status(self):
# self.select_database(self.server_id)
download_status = (
2022-05-26 12:50:20 +00:00
ServerStats.select()
.where(ServerStats.server_id == self.server_id)
.get(self.database)
)
return download_status.downloading
2022-05-25 13:54:35 +00:00
def server_crash_reset(self):
if self.server_id is None:
return
2022-05-25 13:54:35 +00:00
# self.select_database(self.server_id)
ServerStats.update(crashed=False).where(
2022-05-25 13:54:35 +00:00
ServerStats.server_id == self.server_id
2022-05-26 12:50:20 +00:00
).execute(self.database)
2022-05-25 13:54:35 +00:00
def is_crashed(self):
# self.select_database(self.server_id)
2022-05-26 12:50:20 +00:00
svr: ServerStats = (
ServerStats.select()
.where(ServerStats.server_id == self.server_id)
.get(self.database)
)
2022-05-24 22:48:27 +00:00
return svr.crashed
2022-05-25 13:54:35 +00:00
def set_update(self, value):
if self.server_id is None:
return
2022-05-25 13:54:35 +00:00
# self.select_database(self.server_id)
try:
# Checks if server even exists
2022-05-26 12:50:20 +00:00
ServerStats.select().where(ServerStats.server_id == self.server_id).execute(
self.database
)
2022-05-24 22:48:27 +00:00
except DoesNotExist as ex:
logger.error(f"Database entry not found! {ex}")
2022-05-24 22:48:27 +00:00
return
ServerStats.update(updating=value).where(
2022-05-25 13:54:35 +00:00
ServerStats.server_id == self.server_id
2022-05-26 12:50:20 +00:00
).execute(self.database)
2022-05-25 13:54:35 +00:00
def get_update_status(self):
# self.select_database(self.server_id)
update_status = (
2022-05-26 12:50:20 +00:00
ServerStats.select()
.where(ServerStats.server_id == self.server_id)
.get(self.database)
)
return update_status.updating
2022-05-25 13:54:35 +00:00
def set_first_run(self):
# self.select_database(self.server_id)
# Sets first run to false
try:
# Checks if server even exists
2022-05-26 12:50:20 +00:00
ServerStats.select().where(ServerStats.server_id == self.server_id).execute(
self.database
)
except Exception as ex:
logger.error(f"Database entry not found! {ex}")
return
ServerStats.update(first_run=False).where(
2022-05-25 13:54:35 +00:00
ServerStats.server_id == self.server_id
2022-05-26 12:50:20 +00:00
).execute(self.database)
2022-05-25 13:54:35 +00:00
def get_first_run(self):
# self.select_database(self.server_id)
first_run = (
2022-05-26 12:50:20 +00:00
ServerStats.select()
.where(ServerStats.server_id == self.server_id)
.get(self.database)
2022-05-25 13:54:35 +00:00
)
return first_run.first_run
2022-05-25 13:54:35 +00:00
def get_ttl_without_player(self):
# self.select_database(self.server_id)
last_stat = (
ServerStats.select()
2022-05-25 13:54:35 +00:00
.where(ServerStats.server_id == self.server_id)
.order_by(ServerStats.created.desc())
2022-05-26 12:50:20 +00:00
.first(self.database)
)
last_stat_with_player = (
ServerStats.select()
2022-05-25 13:54:35 +00:00
.where(ServerStats.server_id == self.server_id)
.where(ServerStats.online > 0)
.order_by(ServerStats.created.desc())
2022-05-26 12:50:20 +00:00
.first(self.database)
)
return last_stat.created - last_stat_with_player.created
2022-05-25 13:54:35 +00:00
def can_stop_no_players(self, time_limit):
ttl_no_players = self.get_ttl_without_player()
2022-05-24 22:48:27 +00:00
return (time_limit == -1) or (ttl_no_players > time_limit)
2022-05-25 13:54:35 +00:00
def set_waiting_start(self, value):
# self.select_database(self.server_id)
try:
# Checks if server even exists
ServerStats.select().where(ServerStats.server_id == self.server_id).execute(
self.database
2022-05-24 22:48:27 +00:00
)
except DoesNotExist as ex:
logger.error(f"Database entry not found! {ex}")
2022-05-24 22:48:27 +00:00
return
ServerStats.update(waiting_start=value).where(
2022-05-25 13:54:35 +00:00
ServerStats.server_id == self.server_id
2022-05-26 12:50:20 +00:00
).execute(self.database)
2022-05-25 13:54:35 +00:00
def get_waiting_start(self):
waiting_start = (
2022-05-26 12:50:20 +00:00
ServerStats.select()
.where(ServerStats.server_id == self.server_id)
.get(self.database)
)
return waiting_start.waiting_start