2021-09-09 00:01:10 +02:00
|
|
|
import logging
|
|
|
|
import datetime
|
2022-05-08 00:07:55 +03:00
|
|
|
import typing as t
|
2022-04-11 00:23:55 -05:00
|
|
|
from peewee import (
|
|
|
|
ForeignKeyField,
|
|
|
|
CharField,
|
|
|
|
AutoField,
|
|
|
|
DateTimeField,
|
|
|
|
BooleanField,
|
|
|
|
IntegerField,
|
|
|
|
FloatField,
|
|
|
|
)
|
2022-05-08 00:07:55 +03:00
|
|
|
from playhouse.shortcuts import model_to_dict
|
2021-09-09 00:01:10 +02:00
|
|
|
|
2022-04-14 03:10:25 +01:00
|
|
|
from app.classes.shared.main_models import DatabaseShortcuts
|
2022-04-11 00:23:55 -05:00
|
|
|
from app.classes.models.base_model import BaseModel
|
2021-09-09 00:01:10 +02:00
|
|
|
|
2022-03-08 04:40:44 +00:00
|
|
|
logger = logging.getLogger(__name__)
|
2021-09-09 00:01:10 +02:00
|
|
|
|
2022-03-23 06:06:13 +00:00
|
|
|
# **********************************************************************************
|
2021-09-09 00:01:10 +02:00
|
|
|
# Servers Class
|
2022-03-23 06:06:13 +00:00
|
|
|
# **********************************************************************************
|
2022-04-11 00:23:55 -05:00
|
|
|
class Servers(BaseModel):
|
2021-09-09 00:01:10 +02:00
|
|
|
server_id = AutoField()
|
|
|
|
created = DateTimeField(default=datetime.datetime.now)
|
|
|
|
server_uuid = CharField(default="", index=True)
|
|
|
|
server_name = CharField(default="Server", index=True)
|
|
|
|
path = CharField(default="")
|
|
|
|
backup_path = CharField(default="")
|
|
|
|
executable = CharField(default="")
|
|
|
|
log_path = CharField(default="")
|
|
|
|
execution_command = CharField(default="")
|
|
|
|
auto_start = BooleanField(default=0)
|
|
|
|
auto_start_delay = IntegerField(default=10)
|
|
|
|
crash_detection = BooleanField(default=0)
|
|
|
|
stop_command = CharField(default="stop")
|
|
|
|
executable_update_url = CharField(default="")
|
|
|
|
server_ip = CharField(default="127.0.0.1")
|
|
|
|
server_port = IntegerField(default=25565)
|
|
|
|
logs_delete_after = IntegerField(default=0)
|
2022-01-31 20:50:25 -05:00
|
|
|
type = CharField(default="minecraft-java")
|
2021-09-09 00:01:10 +02:00
|
|
|
|
|
|
|
class Meta:
|
|
|
|
table_name = "servers"
|
|
|
|
|
|
|
|
|
2022-03-23 06:06:13 +00:00
|
|
|
# **********************************************************************************
|
2021-09-09 00:01:10 +02:00
|
|
|
# Servers Stats Class
|
2022-03-23 06:06:13 +00:00
|
|
|
# **********************************************************************************
|
2022-04-14 03:10:25 +01:00
|
|
|
class ServerStats(BaseModel):
|
2021-09-09 00:01:10 +02:00
|
|
|
stats_id = AutoField()
|
|
|
|
created = DateTimeField(default=datetime.datetime.now)
|
2022-03-23 02:50:12 +00:00
|
|
|
server_id = ForeignKeyField(Servers, backref="server", index=True)
|
2021-09-09 00:01:10 +02:00
|
|
|
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)
|
2021-09-21 21:13:17 +02:00
|
|
|
waiting_start = BooleanField(default=False)
|
2022-01-09 13:14:59 -05:00
|
|
|
first_run = BooleanField(default=True)
|
2022-01-27 20:43:23 -05:00
|
|
|
crashed = BooleanField(default=False)
|
2022-03-04 00:36:36 +00:00
|
|
|
downloading = BooleanField(default=False)
|
2021-09-09 00:01:10 +02:00
|
|
|
|
|
|
|
class Meta:
|
|
|
|
table_name = "server_stats"
|
|
|
|
|
|
|
|
|
2022-03-23 06:06:13 +00:00
|
|
|
# **********************************************************************************
|
2021-09-09 00:01:10 +02:00
|
|
|
# Servers Class
|
2022-03-23 06:06:13 +00:00
|
|
|
# **********************************************************************************
|
2022-04-14 03:10:25 +01:00
|
|
|
class HelperServers:
|
2022-04-11 00:23:55 -05:00
|
|
|
def __init__(self, database):
|
|
|
|
self.database = database
|
2021-11-21 11:52:29 +01:00
|
|
|
|
2022-03-23 06:06:13 +00:00
|
|
|
# **********************************************************************************
|
2021-09-09 00:01:10 +02:00
|
|
|
# Generic Servers Methods
|
2022-03-23 06:06:13 +00:00
|
|
|
# **********************************************************************************
|
2021-09-09 00:01:10 +02:00
|
|
|
@staticmethod
|
2022-01-26 01:45:30 +00:00
|
|
|
def create_server(
|
|
|
|
name: str,
|
|
|
|
server_uuid: str,
|
|
|
|
server_dir: str,
|
|
|
|
backup_path: str,
|
|
|
|
server_command: str,
|
|
|
|
server_file: str,
|
|
|
|
server_log_file: str,
|
|
|
|
server_stop: str,
|
2022-02-10 18:20:36 -05:00
|
|
|
server_type: str,
|
2022-05-05 03:32:09 +03:00
|
|
|
server_port: int = 25565,
|
2022-05-05 14:02:23 +03:00
|
|
|
server_host: str = "127.0.0.1",
|
2022-05-05 03:32:09 +03:00
|
|
|
) -> int:
|
|
|
|
"""Create a server in the database
|
|
|
|
|
|
|
|
Args:
|
|
|
|
name: The name of the server
|
|
|
|
server_uuid: This is the UUID of the server
|
|
|
|
server_dir: The directory where the server is located
|
|
|
|
backup_path: The path to the backup folder
|
|
|
|
server_command: The command to start the server
|
|
|
|
server_file: The name of the server file
|
|
|
|
server_log_file: The path to the server log file
|
|
|
|
server_stop: This is the command to stop the server
|
|
|
|
server_type: This is the type of server you're creating.
|
2022-05-05 14:02:23 +03:00
|
|
|
server_port: The port the server will be monitored on, defaults to 25565
|
|
|
|
server_host: The host the server will be monitored on, defaults to 127.0.0.1
|
2022-05-05 03:32:09 +03:00
|
|
|
|
|
|
|
Returns:
|
|
|
|
int: The new server's id
|
|
|
|
|
|
|
|
Raises:
|
|
|
|
PeeweeException: If the server already exists
|
|
|
|
"""
|
2022-03-23 02:50:12 +00:00
|
|
|
return Servers.insert(
|
|
|
|
{
|
|
|
|
Servers.server_name: name,
|
|
|
|
Servers.server_uuid: server_uuid,
|
|
|
|
Servers.path: server_dir,
|
|
|
|
Servers.executable: server_file,
|
|
|
|
Servers.execution_command: server_command,
|
|
|
|
Servers.auto_start: False,
|
|
|
|
Servers.auto_start_delay: 10,
|
|
|
|
Servers.crash_detection: False,
|
|
|
|
Servers.log_path: server_log_file,
|
|
|
|
Servers.server_port: server_port,
|
2022-05-05 14:02:23 +03:00
|
|
|
Servers.server_ip: server_host,
|
2022-03-23 02:50:12 +00:00
|
|
|
Servers.stop_command: server_stop,
|
|
|
|
Servers.backup_path: backup_path,
|
|
|
|
Servers.type: server_type,
|
|
|
|
}
|
|
|
|
).execute()
|
2021-11-21 11:52:29 +01:00
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def get_server_obj(server_id):
|
|
|
|
return Servers.get_by_id(server_id)
|
|
|
|
|
2022-02-10 18:20:36 -05:00
|
|
|
@staticmethod
|
|
|
|
def get_server_type_by_id(server_id):
|
|
|
|
server_type = Servers.select().where(Servers.server_id == server_id).get()
|
|
|
|
return server_type.type
|
|
|
|
|
2021-11-21 11:52:29 +01:00
|
|
|
@staticmethod
|
|
|
|
def update_server(server_obj):
|
|
|
|
return server_obj.save()
|
|
|
|
|
2022-04-11 00:23:55 -05:00
|
|
|
def remove_server(self, server_id):
|
|
|
|
with self.database.atomic():
|
2022-04-10 22:00:44 +00:00
|
|
|
Servers.delete().where(Servers.server_id == server_id).execute()
|
2021-09-09 00:01:10 +02:00
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def get_server_data_by_id(server_id):
|
|
|
|
query = Servers.select().where(Servers.server_id == server_id).limit(1)
|
|
|
|
try:
|
2022-04-14 03:10:25 +01:00
|
|
|
return DatabaseShortcuts.return_rows(query)[0]
|
2021-09-09 00:01:10 +02:00
|
|
|
except IndexError:
|
|
|
|
return {}
|
|
|
|
|
2022-05-08 00:07:55 +03:00
|
|
|
@staticmethod
|
|
|
|
def get_server_columns(
|
|
|
|
server_id: t.Union[str, int], column_names: t.List[str]
|
|
|
|
) -> t.List[t.Any]:
|
|
|
|
columns = [getattr(Servers, column) for column in column_names]
|
|
|
|
return model_to_dict(
|
|
|
|
Servers.select(*columns).where(Servers.server_id == server_id).get(),
|
|
|
|
only=columns,
|
|
|
|
)
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def get_server_column(server_id: t.Union[str, int], column_name: str) -> t.Any:
|
|
|
|
column = getattr(Servers, column_name)
|
|
|
|
return model_to_dict(
|
|
|
|
Servers.select(column).where(Servers.server_id == server_id).get(),
|
|
|
|
only=[column],
|
|
|
|
)[column_name]
|
|
|
|
|
2022-03-23 06:06:13 +00:00
|
|
|
# **********************************************************************************
|
2021-09-09 00:01:10 +02:00
|
|
|
# Servers Methods
|
2022-03-23 06:06:13 +00:00
|
|
|
# **********************************************************************************
|
2021-09-09 00:01:10 +02:00
|
|
|
@staticmethod
|
|
|
|
def get_all_defined_servers():
|
|
|
|
query = Servers.select()
|
2022-04-14 03:10:25 +01:00
|
|
|
return DatabaseShortcuts.return_rows(query)
|
2021-09-09 00:01:10 +02:00
|
|
|
|
2022-05-10 02:08:49 +03:00
|
|
|
@staticmethod
|
|
|
|
def get_all_server_ids() -> t.List[int]:
|
|
|
|
return [server.server_id for server in Servers.select(Servers.server_id)]
|
|
|
|
|
2021-09-09 00:01:10 +02:00
|
|
|
@staticmethod
|
|
|
|
def get_all_servers_stats():
|
2022-04-14 03:10:25 +01:00
|
|
|
servers = HelperServers.get_all_defined_servers()
|
2021-09-09 00:01:10 +02:00
|
|
|
server_data = []
|
2022-03-01 01:02:30 +00:00
|
|
|
try:
|
|
|
|
for s in servers:
|
2022-03-23 02:50:12 +00:00
|
|
|
latest = (
|
2022-04-14 03:10:25 +01:00
|
|
|
ServerStats.select()
|
|
|
|
.where(ServerStats.server_id == s.get("server_id"))
|
|
|
|
.order_by(ServerStats.created.desc())
|
2022-03-23 02:50:12 +00:00
|
|
|
.limit(1)
|
|
|
|
)
|
|
|
|
server_data.append(
|
|
|
|
{
|
|
|
|
"server_data": s,
|
2022-04-14 03:10:25 +01:00
|
|
|
"stats": DatabaseShortcuts.return_rows(latest)[0],
|
2022-03-23 02:50:12 +00:00
|
|
|
"user_command_permission": True,
|
|
|
|
}
|
|
|
|
)
|
2022-03-01 01:02:30 +00:00
|
|
|
except IndexError as ex:
|
2022-03-23 02:50:12 +00:00
|
|
|
logger.error(
|
|
|
|
f"Stats collection failed with error: {ex}. Was a server just created?"
|
|
|
|
)
|
2021-09-09 00:01:10 +02:00
|
|
|
return server_data
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def get_server_friendly_name(server_id):
|
2022-04-14 03:10:25 +01:00
|
|
|
server_data = HelperServers.get_server_data_by_id(server_id)
|
2022-03-23 06:06:13 +00:00
|
|
|
friendly_name = (
|
|
|
|
f"{server_data.get('server_name', None)} "
|
|
|
|
f"with ID: {server_data.get('server_id', 0)}"
|
|
|
|
)
|
2021-09-09 00:01:10 +02:00
|
|
|
return friendly_name
|
|
|
|
|
2022-03-23 06:06:13 +00:00
|
|
|
# **********************************************************************************
|
2021-09-09 00:01:10 +02:00
|
|
|
# Servers_Stats Methods
|
2022-03-23 06:06:13 +00:00
|
|
|
# **********************************************************************************
|
2021-09-09 00:01:10 +02:00
|
|
|
@staticmethod
|
|
|
|
def get_latest_server_stats(server_id):
|
2022-03-23 02:50:12 +00:00
|
|
|
return (
|
2022-04-14 03:10:25 +01:00
|
|
|
ServerStats.select()
|
|
|
|
.where(ServerStats.server_id == server_id)
|
|
|
|
.order_by(ServerStats.created.desc())
|
2022-03-23 02:50:12 +00:00
|
|
|
.limit(1)
|
|
|
|
)
|
2021-09-09 00:01:10 +02:00
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def get_server_stats_by_id(server_id):
|
2022-03-23 02:50:12 +00:00
|
|
|
stats = (
|
2022-04-14 03:10:25 +01:00
|
|
|
ServerStats.select()
|
|
|
|
.where(ServerStats.server_id == server_id)
|
|
|
|
.order_by(ServerStats.created.desc())
|
2022-03-23 02:50:12 +00:00
|
|
|
.limit(1)
|
|
|
|
)
|
2022-04-14 03:10:25 +01:00
|
|
|
return DatabaseShortcuts.return_rows(stats)[0]
|
2021-09-09 00:01:10 +02:00
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def server_id_exists(server_id):
|
2022-04-14 03:10:25 +01:00
|
|
|
if not HelperServers.get_server_data_by_id(server_id):
|
2021-09-09 00:01:10 +02:00
|
|
|
return False
|
|
|
|
return True
|
|
|
|
|
2022-01-27 20:43:23 -05:00
|
|
|
@staticmethod
|
|
|
|
def sever_crashed(server_id):
|
2022-04-14 03:10:25 +01:00
|
|
|
ServerStats.update(crashed=True).where(
|
|
|
|
ServerStats.server_id == server_id
|
2022-04-11 00:23:55 -05:00
|
|
|
).execute()
|
2022-01-27 20:43:23 -05:00
|
|
|
|
2022-03-04 00:36:36 +00:00
|
|
|
@staticmethod
|
|
|
|
def set_download(server_id):
|
2022-04-14 03:10:25 +01:00
|
|
|
ServerStats.update(downloading=True).where(
|
|
|
|
ServerStats.server_id == server_id
|
2022-04-11 00:23:55 -05:00
|
|
|
).execute()
|
2022-03-04 00:36:36 +00:00
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def finish_download(server_id):
|
2022-04-14 03:10:25 +01:00
|
|
|
ServerStats.update(downloading=False).where(
|
|
|
|
ServerStats.server_id == server_id
|
2022-04-11 00:23:55 -05:00
|
|
|
).execute()
|
2022-03-04 00:36:36 +00:00
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def get_download_status(server_id):
|
2022-03-23 02:50:12 +00:00
|
|
|
download_status = (
|
2022-04-14 03:10:25 +01:00
|
|
|
ServerStats.select().where(ServerStats.server_id == server_id).get()
|
2022-03-23 02:50:12 +00:00
|
|
|
)
|
2022-03-04 00:36:36 +00:00
|
|
|
return download_status.downloading
|
|
|
|
|
2022-01-27 20:43:23 -05:00
|
|
|
@staticmethod
|
|
|
|
def server_crash_reset(server_id):
|
2022-04-14 03:10:25 +01:00
|
|
|
ServerStats.update(crashed=False).where(
|
|
|
|
ServerStats.server_id == server_id
|
2022-04-11 00:23:55 -05:00
|
|
|
).execute()
|
2022-01-27 20:43:23 -05:00
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def is_crashed(server_id):
|
2022-04-14 03:10:25 +01:00
|
|
|
svr = ServerStats.select().where(ServerStats.server_id == server_id).get()
|
2022-04-02 02:58:31 +01:00
|
|
|
if svr.crashed is True:
|
2022-01-27 20:43:23 -05:00
|
|
|
return True
|
|
|
|
else:
|
|
|
|
return False
|
|
|
|
|
2021-09-09 00:01:10 +02:00
|
|
|
@staticmethod
|
|
|
|
def set_update(server_id, value):
|
|
|
|
try:
|
2022-03-23 02:50:12 +00:00
|
|
|
# Checks if server even exists
|
2022-04-14 03:10:25 +01:00
|
|
|
ServerStats.select().where(ServerStats.server_id == server_id)
|
2021-09-09 00:01:10 +02:00
|
|
|
except Exception as ex:
|
2022-01-26 01:45:30 +00:00
|
|
|
logger.error(f"Database entry not found! {ex}")
|
2022-04-14 03:10:25 +01:00
|
|
|
ServerStats.update(updating=value).where(
|
|
|
|
ServerStats.server_id == server_id
|
2022-04-11 00:23:55 -05:00
|
|
|
).execute()
|
2021-09-09 00:01:10 +02:00
|
|
|
|
2022-01-10 14:58:38 -05:00
|
|
|
@staticmethod
|
|
|
|
def get_update_status(server_id):
|
2022-03-23 02:50:12 +00:00
|
|
|
update_status = (
|
2022-04-14 03:10:25 +01:00
|
|
|
ServerStats.select().where(ServerStats.server_id == server_id).get()
|
2022-03-23 02:50:12 +00:00
|
|
|
)
|
2022-01-27 20:43:23 -05:00
|
|
|
return update_status.updating
|
2022-01-10 14:58:38 -05:00
|
|
|
|
2022-01-09 13:14:59 -05:00
|
|
|
@staticmethod
|
|
|
|
def set_first_run(server_id):
|
2022-03-23 02:50:12 +00:00
|
|
|
# Sets first run to false
|
2022-01-09 13:14:59 -05:00
|
|
|
try:
|
2022-03-23 02:50:12 +00:00
|
|
|
# Checks if server even exists
|
2022-04-14 03:10:25 +01:00
|
|
|
ServerStats.select().where(ServerStats.server_id == server_id)
|
2022-01-09 13:14:59 -05:00
|
|
|
except Exception as ex:
|
2022-01-26 01:45:30 +00:00
|
|
|
logger.error(f"Database entry not found! {ex}")
|
|
|
|
return
|
2022-04-14 03:10:25 +01:00
|
|
|
ServerStats.update(first_run=False).where(
|
|
|
|
ServerStats.server_id == server_id
|
2022-04-11 00:23:55 -05:00
|
|
|
).execute()
|
2022-01-09 13:14:59 -05:00
|
|
|
|
2022-01-10 14:58:38 -05:00
|
|
|
@staticmethod
|
|
|
|
def get_first_run(server_id):
|
2022-04-14 03:10:25 +01:00
|
|
|
first_run = ServerStats.select().where(ServerStats.server_id == server_id).get()
|
2022-01-10 14:58:38 -05:00
|
|
|
return first_run.first_run
|
|
|
|
|
2021-09-09 00:01:10 +02:00
|
|
|
@staticmethod
|
2022-04-14 03:10:25 +01:00
|
|
|
def get_ttl_without_player(server_id):
|
2022-03-23 02:50:12 +00:00
|
|
|
last_stat = (
|
2022-04-14 03:10:25 +01:00
|
|
|
ServerStats.select()
|
|
|
|
.where(ServerStats.server_id == server_id)
|
|
|
|
.order_by(ServerStats.created.desc())
|
2022-03-23 02:50:12 +00:00
|
|
|
.first()
|
|
|
|
)
|
|
|
|
last_stat_with_player = (
|
2022-04-14 03:10:25 +01:00
|
|
|
ServerStats.select()
|
|
|
|
.where(ServerStats.server_id == server_id)
|
|
|
|
.where(ServerStats.online > 0)
|
|
|
|
.order_by(ServerStats.created.desc())
|
2022-03-23 02:50:12 +00:00
|
|
|
.first()
|
|
|
|
)
|
2021-09-09 00:01:10 +02:00
|
|
|
return last_stat.created - last_stat_with_player.created
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def can_stop_no_players(server_id, time_limit):
|
|
|
|
can = False
|
2022-04-14 03:10:25 +01:00
|
|
|
ttl_no_players = HelperServers.get_ttl_without_player(server_id)
|
2021-09-09 00:01:10 +02:00
|
|
|
if (time_limit == -1) or (ttl_no_players > time_limit):
|
|
|
|
can = True
|
|
|
|
return can
|
2021-11-21 11:52:29 +01:00
|
|
|
|
2021-09-21 21:13:17 +02:00
|
|
|
@staticmethod
|
|
|
|
def set_waiting_start(server_id, value):
|
|
|
|
try:
|
2022-01-26 01:45:30 +00:00
|
|
|
# Checks if server even exists
|
2022-04-14 03:10:25 +01:00
|
|
|
ServerStats.select().where(ServerStats.server_id == server_id)
|
2021-09-21 21:13:17 +02:00
|
|
|
except Exception as ex:
|
2022-01-26 01:45:30 +00:00
|
|
|
logger.error(f"Database entry not found! {ex}")
|
2022-04-14 03:10:25 +01:00
|
|
|
ServerStats.update(waiting_start=value).where(
|
|
|
|
ServerStats.server_id == server_id
|
2022-04-11 00:23:55 -05:00
|
|
|
).execute()
|
2021-09-21 21:13:17 +02:00
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def get_waiting_start(server_id):
|
2022-03-23 02:50:12 +00:00
|
|
|
waiting_start = (
|
2022-04-14 03:10:25 +01:00
|
|
|
ServerStats.select().where(ServerStats.server_id == server_id).get()
|
2022-03-23 02:50:12 +00:00
|
|
|
)
|
2021-09-21 21:13:17 +02:00
|
|
|
return waiting_start.waiting_start
|