2020-08-16 22:47:53 -04:00
import os
2020-08-12 21:33:36 -04:00
import sys
import logging
import datetime
from app . classes . shared . helpers import helper
from app . classes . shared . console import console
2020-08-16 22:47:53 -04:00
from app . classes . minecraft . server_props import ServerProps
2021-03-01 02:54:20 +02:00
from app . classes . web . websocket_helper import websocket_helper
2020-08-12 21:33:36 -04:00
logger = logging . getLogger ( __name__ )
2021-03-21 23:02:18 -05:00
peewee_logger = logging . getLogger ( ' peewee ' )
peewee_logger . setLevel ( logging . INFO )
2020-08-12 21:33:36 -04:00
try :
from peewee import *
from playhouse . shortcuts import model_to_dict
2020-08-16 22:47:53 -04:00
import yaml
2020-08-12 21:33:36 -04:00
except ModuleNotFoundError as e :
2021-04-17 23:34:13 +03:00
logger . critical ( " Import Error: Unable to load {} module " . format ( e . name ) , exc_info = True )
console . critical ( " Import Error: Unable to load {} module " . format ( e . name ) )
2020-08-12 21:33:36 -04:00
sys . exit ( 1 )
2021-03-21 23:02:18 -05:00
schema_version = ( 0 , 1 , 0 ) # major, minor, patch semver
2020-08-12 21:33:36 -04:00
database = SqliteDatabase ( helper . db_path , pragmas = {
' journal_mode ' : ' wal ' ,
' cache_size ' : - 1024 * 10 } )
class BaseModel ( Model ) :
class Meta :
database = database
2021-03-21 23:02:18 -05:00
class SchemaVersion ( BaseModel ) :
# DO NOT EVER CHANGE THE SCHEMA OF THIS TABLE
# (unless we have a REALLY good reason to)
# There will only ever be one row, and it allows the database loader to detect
# what it needs to do on major version upgrades so you don't have to wipe the DB
# every time you upgrade
schema_major = IntegerField ( )
schema_minor = IntegerField ( )
schema_patch = IntegerField ( )
class Meta :
table_name = ' schema_version '
primary_key = CompositeKey ( ' schema_major ' , ' schema_minor ' , ' schema_patch ' )
2020-08-12 21:33:36 -04:00
class Users ( BaseModel ) :
user_id = AutoField ( )
created = DateTimeField ( default = datetime . datetime . now )
last_login = DateTimeField ( default = datetime . datetime . now )
2020-12-27 11:00:26 -05:00
last_update = DateTimeField ( default = datetime . datetime . now )
2020-08-12 21:33:36 -04:00
last_ip = CharField ( default = " " )
2020-12-24 17:55:15 -05:00
username = CharField ( default = " " , unique = True , index = True )
2020-08-12 21:33:36 -04:00
password = CharField ( default = " " )
enabled = BooleanField ( default = True )
2020-12-24 17:55:15 -05:00
superuser = BooleanField ( default = False )
api_token = CharField ( default = " " , unique = True , index = True ) # we may need to revisit this
2020-08-12 21:33:36 -04:00
class Meta :
table_name = " users "
2020-09-22 15:00:05 -04:00
2020-12-24 17:55:15 -05:00
class Roles ( BaseModel ) :
role_id = AutoField ( )
created = DateTimeField ( default = datetime . datetime . now )
2020-12-27 11:00:26 -05:00
last_update = DateTimeField ( default = datetime . datetime . now )
2020-12-24 17:55:15 -05:00
role_name = CharField ( default = " " , unique = True , index = True )
class Meta :
table_name = " roles "
class User_Roles ( BaseModel ) :
user_id = ForeignKeyField ( Users , backref = ' user_role ' )
role_id = ForeignKeyField ( Roles , backref = ' user_role ' )
class Meta :
table_name = ' user_roles '
primary_key = CompositeKey ( ' user_id ' , ' role_id ' )
2020-08-31 16:16:45 -04:00
class Audit_Log ( BaseModel ) :
audit_id = AutoField ( )
created = DateTimeField ( default = datetime . datetime . now )
user_name = CharField ( default = " " )
2020-12-24 17:55:15 -05:00
user_id = IntegerField ( default = 0 , index = True )
2020-08-31 16:16:45 -04:00
source_ip = CharField ( default = ' 127.0.0.1 ' )
2020-12-24 17:55:15 -05:00
server_id = IntegerField ( default = None , index = True ) # When auditing global events, use server ID 0
2020-08-31 16:16:45 -04:00
log_msg = TextField ( default = ' ' )
2020-08-12 21:33:36 -04:00
class Host_Stats ( BaseModel ) :
2020-12-24 17:55:15 -05:00
time = DateTimeField ( default = datetime . datetime . now , index = True )
2020-08-12 21:33:36 -04:00
boot_time = CharField ( default = " " )
cpu_usage = FloatField ( default = 0 )
cpu_cores = IntegerField ( default = 0 )
cpu_cur_freq = FloatField ( default = 0 )
cpu_max_freq = FloatField ( default = 0 )
mem_percent = FloatField ( default = 0 )
mem_usage = CharField ( default = " " )
mem_total = CharField ( default = " " )
2020-08-18 21:04:43 -04:00
disk_json = TextField ( default = " " )
2020-08-12 21:33:36 -04:00
class Meta :
table_name = " host_stats "
2020-08-16 22:47:53 -04:00
class Servers ( BaseModel ) :
server_id = AutoField ( )
created = DateTimeField ( default = datetime . datetime . now )
2020-12-24 17:55:15 -05:00
server_uuid = CharField ( default = " " , index = True )
server_name = CharField ( default = " Server " , index = True )
2020-08-16 22:47:53 -04:00
path = CharField ( default = " " )
2021-03-21 23:02:18 -05:00
backup_path = CharField ( default = " " )
2020-08-16 22:47:53 -04:00
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 " )
2021-07-24 21:09:39 -04:00
executable_update_url = CharField ( default = " " )
2020-09-22 12:11:16 -04:00
server_ip = CharField ( default = " 127.0.0.1 " )
2020-08-23 22:17:36 -04:00
server_port = IntegerField ( default = 25565 )
2021-01-20 17:20:42 +02:00
logs_delete_after = IntegerField ( default = 0 )
2020-08-23 22:17:36 -04:00
2020-08-16 22:47:53 -04:00
class Meta :
table_name = " servers "
2020-12-24 17:55:15 -05:00
class Role_Servers ( BaseModel ) :
2020-12-25 22:00:28 -05:00
role_id = ForeignKeyField ( Roles , backref = ' role_server ' )
2020-12-24 17:55:15 -05:00
server_id = ForeignKeyField ( Servers , backref = ' role_server ' )
class Meta :
table_name = ' role_servers '
primary_key = CompositeKey ( ' role_id ' , ' server_id ' )
2020-08-24 13:08:17 -04:00
class Server_Stats ( BaseModel ) :
stats_id = AutoField ( )
created = DateTimeField ( default = datetime . datetime . now )
2020-12-24 17:55:15 -05:00
server_id = ForeignKeyField ( Servers , backref = ' server ' , index = True )
2020-08-24 13:08:17 -04:00
started = CharField ( default = " " )
running = BooleanField ( default = False )
cpu = FloatField ( default = 0 )
mem = FloatField ( default = 0 )
2020-08-25 09:29:11 -04:00
mem_percent = FloatField ( default = 0 )
2020-08-24 13:08:17 -04:00
world_name = CharField ( default = " " )
world_size = CharField ( default = " " )
server_port = IntegerField ( default = 25565 )
int_ping_results = CharField ( default = " " )
2020-08-24 19:11:17 -04:00
online = IntegerField ( default = 0 )
max = IntegerField ( default = 0 )
players = CharField ( default = " " )
desc = CharField ( default = " Unable to Connect " )
version = CharField ( default = " " )
2021-07-25 17:04:10 -04:00
updating = BooleanField ( default = False )
2020-08-24 19:11:17 -04:00
2020-08-24 13:08:17 -04:00
class Meta :
table_name = " server_stats "
2020-08-24 19:11:17 -04:00
class Commands ( BaseModel ) :
command_id = AutoField ( )
created = DateTimeField ( default = datetime . datetime . now )
2020-12-24 17:55:15 -05:00
server_id = ForeignKeyField ( Servers , backref = ' server ' , index = True )
user = ForeignKeyField ( Users , backref = ' user ' , index = True )
2020-08-24 19:11:17 -04:00
source_ip = CharField ( default = ' 127.0.0.1 ' )
command = CharField ( default = ' ' )
2020-08-31 13:46:25 -04:00
executed = BooleanField ( default = False )
2020-08-24 19:11:17 -04:00
class Meta :
table_name = " commands "
2020-08-12 21:33:36 -04:00
class Webhooks ( BaseModel ) :
id = AutoField ( )
2020-12-24 17:55:15 -05:00
name = CharField ( max_length = 64 , unique = True , index = True )
2020-08-12 21:33:36 -04:00
method = CharField ( default = " POST " )
url = CharField ( unique = True )
event = CharField ( default = " " )
send_data = BooleanField ( default = True )
class Meta :
table_name = " webhooks "
2021-03-21 23:02:18 -05:00
class Schedules ( BaseModel ) :
schedule_id = IntegerField ( unique = True , primary_key = True )
server_id = ForeignKeyField ( Servers , backref = ' schedule_server ' )
enabled = BooleanField ( )
action = CharField ( )
interval = IntegerField ( )
interval_type = CharField ( )
start_time = CharField ( null = True )
command = CharField ( null = True )
comment = CharField ( )
class Meta :
table_name = ' schedules '
2020-08-12 21:33:36 -04:00
class Backups ( BaseModel ) :
2021-03-21 23:02:18 -05:00
directories = CharField ( null = True )
2020-08-12 21:33:36 -04:00
max_backups = IntegerField ( )
2021-03-21 23:02:18 -05:00
server_id = ForeignKeyField ( Servers , backref = ' backups_server ' )
schedule_id = ForeignKeyField ( Schedules , backref = ' backups_schedule ' )
2020-08-12 21:33:36 -04:00
class Meta :
table_name = ' backups '
2020-08-16 22:47:53 -04:00
2020-08-12 21:33:36 -04:00
class db_builder :
@staticmethod
def create_tables ( ) :
with database :
database . create_tables ( [
Backups ,
Users ,
2020-12-24 17:55:15 -05:00
Roles ,
User_Roles ,
2020-08-12 21:33:36 -04:00
Host_Stats ,
2020-08-16 22:47:53 -04:00
Webhooks ,
2020-08-18 21:04:43 -04:00
Servers ,
2020-12-25 22:00:28 -05:00
Role_Servers ,
2020-08-24 19:11:17 -04:00
Server_Stats ,
2020-08-31 16:16:45 -04:00
Commands ,
2021-03-21 23:02:18 -05:00
Audit_Log ,
SchemaVersion ,
Schedules
2020-08-12 21:33:36 -04:00
] )
2020-08-13 10:38:36 -04:00
@staticmethod
def default_settings ( ) :
2020-09-22 15:00:05 -04:00
logger . info ( " Fresh Install Detected - Creating Default Settings " )
console . info ( " Fresh Install Detected - Creating Default Settings " )
2021-03-21 23:02:18 -05:00
SchemaVersion . insert ( {
SchemaVersion . schema_major : schema_version [ 0 ] ,
SchemaVersion . schema_minor : schema_version [ 1 ] ,
SchemaVersion . schema_patch : schema_version [ 2 ]
} ) . execute ( )
2020-08-18 21:50:24 -04:00
default_data = helper . find_default_password ( )
2020-09-06 12:37:56 +08:00
username = default_data . get ( " username " , ' admin ' )
2020-08-18 21:50:24 -04:00
password = default_data . get ( " password " , ' crafty ' )
2020-12-24 17:55:15 -05:00
#api_token = helper.random_string_generator(32)
#
#Users.insert({
# Users.username: username.lower(),
# Users.password: helper.encode_pass(password),
# Users.api_token: api_token,
# Users.enabled: True,
# Users.superuser: True
#}).execute()
db_shortcuts . add_user ( username , password = password , superuser = True )
#console.info("API token is {}".format(api_token))
2020-09-06 12:58:17 +08:00
2020-08-13 10:38:36 -04:00
@staticmethod
def is_fresh_install ( ) :
2020-09-22 15:00:05 -04:00
try :
user = Users . get_by_id ( 1 )
2020-08-13 10:38:36 -04:00
return False
2020-09-22 15:00:05 -04:00
except :
return True
pass
2020-08-13 10:38:36 -04:00
2021-03-21 23:02:18 -05:00
@staticmethod
def check_schema_version ( ) :
svs = SchemaVersion . select ( ) . execute ( )
if len ( svs ) != 1 :
raise exceptions . SchemaError ( " Multiple or no schema versions detected - potentially a failed upgrade? " )
sv = svs [ 0 ]
svt = ( sv . schema_major , sv . schema_minor , sv . schema_patch )
logger . debug ( " Schema: found {} , expected {} " . format ( svt , schema_version ) )
console . debug ( " Schema: found {} , expected {} " . format ( svt , schema_version ) )
if sv . schema_major > schema_version [ 0 ] :
raise exceptions . SchemaError ( " Major version mismatch - possible code reversion " )
elif sv . schema_major < schema_version [ 0 ] :
db_shortcuts . upgrade_schema ( )
if sv . schema_minor > schema_version [ 1 ] :
logger . warning ( " Schema minor mismatch detected: found {} , expected {} . Proceed with caution " . format ( svt , schema_version ) )
console . warning ( " Schema minor mismatch detected: found {} , expected {} . Proceed with caution " . format ( svt , schema_version ) )
elif sv . schema_minor < schema_version [ 1 ] :
db_shortcuts . upgrade_schema ( )
if sv . schema_patch > schema_version [ 2 ] :
logger . info ( " Schema patch mismatch detected: found {} , expected {} . Proceed with caution " . format ( svt , schema_version ) )
console . info ( " Schema patch mismatch detected: found {} , expected {} . Proceed with caution " . format ( svt , schema_version ) )
elif sv . schema_patch < schema_version [ 2 ] :
db_shortcuts . upgrade_schema ( )
logger . info ( " Schema validation successful! {} " . format ( schema_version ) )
2020-08-16 22:47:53 -04:00
class db_shortcuts :
2021-03-21 23:02:18 -05:00
@staticmethod
def upgrade_schema ( ) :
raise NotImplemented ( " I don ' t know who you are or how you reached this code, but this should NOT have happened. Please report it to the developer with due haste. " )
2020-12-24 17:55:15 -05:00
@staticmethod
def return_rows ( query ) :
2020-08-16 22:47:53 -04:00
rows = [ ]
2020-08-23 18:43:28 -04:00
try :
if query . count ( ) > 0 :
for s in query :
rows . append ( model_to_dict ( s ) )
except Exception as e :
logger . warning ( " Database Error: {} " . format ( e ) )
pass
2020-08-16 22:47:53 -04:00
return rows
2021-04-04 12:48:02 -05:00
@staticmethod
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 , server_port = 25565 ) :
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 ,
Servers . stop_command : server_stop ,
Servers . backup_path : backup_path
} ) . execute ( )
@staticmethod
def remove_server ( server_id ) :
with database . atomic ( ) :
Role_Servers . delete ( ) . where ( Role_Servers . server_id == server_id ) . execute ( )
Servers . delete ( ) . where ( Servers . server_id == server_id ) . execute ( )
2020-12-24 17:55:15 -05:00
@staticmethod
def get_server_data_by_id ( server_id ) :
2021-03-21 23:02:18 -05:00
query = Servers . select ( ) . where ( Servers . server_id == server_id ) . limit ( 1 )
2020-09-01 13:29:39 -04:00
try :
2021-03-21 23:02:18 -05:00
return db_helper . return_rows ( query ) [ 0 ]
except IndexError :
2020-09-01 13:29:39 -04:00
return { }
2020-12-24 17:55:15 -05:00
@staticmethod
def get_all_defined_servers ( ) :
2020-08-16 22:47:53 -04:00
query = Servers . select ( )
2020-12-24 17:55:15 -05:00
return db_helper . return_rows ( query )
2021-03-09 23:17:50 +01:00
@staticmethod
2021-04-03 13:18:43 -05:00
def get_authorized_servers ( user_id ) :
user_servers = User_Servers . select ( ) . where ( User_Servers . user_id == user_id )
2021-03-09 23:17:50 +01:00
server_data = [ ]
2021-04-03 13:18:43 -05:00
for u in user_servers :
2021-03-09 23:17:50 +01:00
server_data . append ( db_helper . get_server_data_by_id ( u . server_id ) )
return server_data
2021-03-13 23:12:42 +01:00
@staticmethod
2021-04-03 13:18:43 -05:00
def get_authorized_servers_from_roles ( user_id ) :
user_roles = User_Roles . select ( ) . where ( User_Roles . user_id == user_id )
2021-03-13 23:12:42 +01:00
roles_list = [ ]
2021-04-03 13:18:43 -05:00
role_server = [ ]
2021-03-13 23:12:42 +01:00
server_data = [ ]
2021-04-03 13:18:43 -05:00
for u in user_roles :
2021-03-13 23:12:42 +01:00
roles_list . append ( db_helper . get_role ( u . role_id ) )
for r in roles_list :
role_test = Role_Servers . select ( ) . where ( Role_Servers . role_id == r . get ( ' role_id ' ) )
for t in role_test :
2021-04-03 13:18:43 -05:00
role_server . append ( t )
2021-03-13 23:12:42 +01:00
2021-04-03 13:18:43 -05:00
for s in role_server :
2021-03-13 23:12:42 +01:00
server_data . append ( db_helper . get_server_data_by_id ( s . server_id ) )
return server_data
2020-08-16 22:47:53 -04:00
2020-12-24 17:55:15 -05:00
@staticmethod
def get_all_servers_stats ( ) :
servers = db_helper . get_all_defined_servers ( )
2020-08-27 18:30:56 -04:00
server_data = [ ]
for s in servers :
latest = Server_Stats . select ( ) . where ( Server_Stats . server_id == s . get ( ' server_id ' ) ) . order_by ( Server_Stats . created . desc ( ) ) . limit ( 1 )
2021-03-21 23:02:18 -05:00
server_data . append ( { ' server_data ' : s , " stats " : db_helper . return_rows ( latest ) [ 0 ] } )
2020-08-27 18:30:56 -04:00
return server_data
2020-08-24 13:08:17 -04:00
2021-03-09 23:17:50 +01:00
@staticmethod
2021-04-03 13:18:43 -05:00
def get_authorized_servers_stats ( user_id ) :
user_servers = User_Servers . select ( ) . where ( User_Servers . user_id == user_id )
authorized_servers = [ ]
2021-03-09 23:17:50 +01:00
server_data = [ ]
2021-04-03 13:18:43 -05:00
for u in user_servers :
authorized_servers . append ( db_helper . get_server_data_by_id ( u . server_id ) )
2021-03-09 23:17:50 +01:00
2021-04-03 13:18:43 -05:00
for s in authorized_servers :
2021-03-09 23:17:50 +01:00
latest = Server_Stats . select ( ) . where ( Server_Stats . server_id == s . get ( ' server_id ' ) ) . order_by ( Server_Stats . created . desc ( ) ) . limit ( 1 )
server_data . append ( { ' server_data ' : s , " stats " : db_helper . return_rows ( latest ) } )
return server_data
2021-03-13 23:12:42 +01:00
@staticmethod
2021-04-03 13:18:43 -05:00
def get_authorized_servers_stats_from_roles ( user_id ) :
user_roles = User_Roles . select ( ) . where ( User_Roles . user_id == user_id )
2021-03-13 23:12:42 +01:00
roles_list = [ ]
2021-04-03 13:18:43 -05:00
role_server = [ ]
authorized_servers = [ ]
2021-03-13 23:12:42 +01:00
server_data = [ ]
2021-04-03 13:18:43 -05:00
for u in user_roles :
2021-03-13 23:12:42 +01:00
roles_list . append ( db_helper . get_role ( u . role_id ) )
for r in roles_list :
role_test = Role_Servers . select ( ) . where ( Role_Servers . role_id == r . get ( ' role_id ' ) )
for t in role_test :
2021-04-03 13:18:43 -05:00
role_server . append ( t )
2021-03-13 23:12:42 +01:00
2021-04-03 13:18:43 -05:00
for s in role_server :
authorized_servers . append ( db_helper . get_server_data_by_id ( s . server_id ) )
2021-03-13 23:12:42 +01:00
2021-04-03 13:18:43 -05:00
for s in authorized_servers :
2021-03-13 23:12:42 +01:00
latest = Server_Stats . select ( ) . where ( Server_Stats . server_id == s . get ( ' server_id ' ) ) . order_by ( Server_Stats . created . desc ( ) ) . limit ( 1 )
2021-04-03 12:36:01 -05:00
server_data . append ( { ' server_data ' : s , " stats " : db_helper . return_rows ( latest ) [ 0 ] } )
2021-03-13 23:12:42 +01:00
return server_data
2020-12-24 17:55:15 -05:00
@staticmethod
def get_server_stats_by_id ( server_id ) :
2020-09-01 13:29:39 -04:00
stats = Server_Stats . select ( ) . where ( Server_Stats . server_id == server_id ) . order_by ( Server_Stats . created . desc ( ) ) . limit ( 1 )
2021-03-21 23:02:18 -05:00
return db_helper . return_rows ( stats ) [ 0 ]
2020-09-01 13:29:39 -04:00
2020-12-24 17:55:15 -05:00
@staticmethod
def server_id_exists ( server_id ) :
if not db_helper . get_server_data_by_id ( server_id ) :
2020-09-01 13:29:39 -04:00
return False
return True
2021-03-09 23:17:50 +01:00
@staticmethod
2021-04-03 13:18:43 -05:00
def server_id_authorized ( serverId , user_id ) :
2021-03-09 23:17:50 +01:00
userServer = User_Servers . select ( ) . where ( User_Servers . server_id == serverId )
2021-04-03 13:18:43 -05:00
authorized = userServer . select ( ) . where ( User_Servers . user_id == user_id )
2021-03-09 23:17:50 +01:00
#authorized = db_helper.return_rows(authorized)
if authorized . count ( ) == 0 :
return False
return True
2020-09-01 13:29:39 -04:00
2021-03-13 23:12:42 +01:00
@staticmethod
2021-04-03 13:18:43 -05:00
def server_id_authorized_from_roles ( serverId , user_id ) :
2021-03-13 23:12:42 +01:00
cpt_authorized = 0
roles_list = [ ]
2021-04-03 13:18:43 -05:00
role_server = [ ]
2021-03-13 23:12:42 +01:00
authorized = [ ]
2021-04-03 13:18:43 -05:00
user_roles = User_Roles . select ( ) . where ( User_Roles . user_id == user_id )
2021-03-13 23:12:42 +01:00
2021-04-03 13:18:43 -05:00
for u in user_roles :
2021-03-13 23:12:42 +01:00
roles_list . append ( db_helper . get_role ( u . role_id ) )
for r in roles_list :
role_test = Role_Servers . select ( ) . where ( Role_Servers . role_id == r . get ( ' role_id ' ) )
for s in role_test :
if s . server_id . server_id == serverId :
cpt_authorized + = 1
if cpt_authorized == 0 :
return False
return True
2020-08-31 13:46:25 -04:00
@staticmethod
def get_latest_hosts_stats ( ) :
2020-08-18 21:04:43 -04:00
query = Host_Stats . select ( ) . order_by ( Host_Stats . id . desc ( ) ) . get ( )
return model_to_dict ( query )
2020-12-24 17:55:15 -05:00
@staticmethod
def new_api_token ( ) :
while True :
token = helper . random_string_generator ( 32 )
test = list ( Users . select ( Users . user_id ) . where ( Users . api_token == token ) )
if len ( test ) == 0 :
return token
@staticmethod
def get_all_users ( ) :
2020-12-06 15:42:49 -05:00
query = Users . select ( )
return query
2020-12-24 17:55:15 -05:00
@staticmethod
def get_all_roles ( ) :
query = Roles . select ( )
return query
@staticmethod
2021-04-03 13:18:43 -05:00
def get_user_id_by_name ( username ) :
2021-03-21 23:02:18 -05:00
if username == " SYSTEM " :
return 0
2020-12-24 17:55:15 -05:00
try :
return ( Users . get ( Users . username == username ) ) . user_id
except DoesNotExist :
return None
2021-04-17 18:19:19 +03:00
@staticmethod
def get_user_by_api_token ( token : str ) :
query = Users . select ( ) . where ( Users . api_token == token )
if query . exists ( ) :
user = model_to_dict ( Users . get ( Users . api_token == token ) )
# I know it should apply it without setting it but I'm just making sure
user = db_shortcuts . add_user_roles ( user )
return user
else :
return { }
@staticmethod
def add_user_roles ( user ) :
if type ( user ) == dict :
user_id = user [ ' user_id ' ]
else :
user_id = user . user_id
# I just copied this code from get_user, it had those TODOs & comments made by mac - Lukas
roles_query = User_Roles . select ( ) . join ( Roles , JOIN . INNER ) . where ( User_Roles . user_id == user_id )
# TODO: this query needs to be narrower
roles = set ( )
for r in roles_query :
roles . add ( r . role_id . role_id )
#servers_query = User_Servers.select().join(Servers, JOIN.INNER).where(User_Servers.user_id == user_id)
## TODO: this query needs to be narrower
servers = set ( )
#for s in servers_query:
# servers.add(s.server_id.server_id)
user [ ' roles ' ] = roles
#user['servers'] = servers
#logger.debug("user: ({}) {}".format(user_id, user))
return user
2020-12-24 17:55:15 -05:00
@staticmethod
2020-12-25 22:00:28 -05:00
def get_user ( user_id ) :
2021-03-21 23:02:18 -05:00
if user_id == 0 :
return {
user_id : 0 ,
created : None ,
last_login : None ,
last_update : None ,
last_ip : " 127.27.23.89 " ,
username : " SYSTEM " ,
password : None ,
enabled : True ,
superuser : False ,
api_token : None ,
roles : [ ] ,
servers : [ ]
}
2020-12-25 22:00:28 -05:00
user = model_to_dict ( Users . get ( Users . user_id == user_id ) )
if user :
2021-04-17 18:19:19 +03:00
# I know it should apply it without setting it but I'm just making sure
user = db_shortcuts . add_user_roles ( user )
2020-12-24 17:55:15 -05:00
return user
else :
2021-04-03 12:36:01 -05:00
#logger.debug("user: ({}) {}".format(user_id, {}))
2020-12-24 17:55:15 -05:00
return { }
@staticmethod
def update_user ( user_id , user_data = { } ) :
base_data = db_helper . get_user ( user_id )
up_data = { }
2020-12-25 22:00:28 -05:00
added_roles = set ( )
removed_roles = set ( )
added_servers = set ( )
removed_servers = set ( )
2020-12-24 17:55:15 -05:00
for key in user_data :
if key == " user_id " :
continue
elif key == " roles " :
2020-12-25 22:00:28 -05:00
added_roles = user_data [ ' roles ' ] . difference ( base_data [ ' roles ' ] )
removed_roles = base_data [ ' roles ' ] . difference ( user_data [ ' roles ' ] )
2021-04-03 12:36:01 -05:00
#elif key == "servers":
# added_servers = user_data['servers'].difference(base_data['servers'])
# removed_servers = base_data['servers'].difference(user_data['servers'])
2020-12-24 17:55:15 -05:00
elif key == " regen_api " :
2020-12-25 22:00:28 -05:00
if user_data [ ' regen_api ' ] :
up_data [ ' api_token ' ] = db_shortcuts . new_api_token ( )
2020-12-24 17:55:15 -05:00
elif key == " password " :
2020-12-25 22:00:28 -05:00
if user_data [ ' password ' ] is not None and user_data [ ' password ' ] != " " :
up_data [ ' password ' ] = helper . encode_pass ( user_data [ ' password ' ] )
2020-12-24 17:55:15 -05:00
elif base_data [ key ] != user_data [ key ] :
up_data [ key ] = user_data [ key ]
2020-12-27 11:00:26 -05:00
up_data [ ' last_update ' ] = helper . get_time_as_string ( )
2020-12-25 22:00:28 -05:00
logger . debug ( " user: {} +role: {} -role: {} +server: {} -server {} " . format ( user_data , added_roles , removed_roles , added_servers , removed_servers ) )
with database . atomic ( ) :
for role in added_roles :
User_Roles . get_or_create ( user_id = user_id , role_id = role )
# TODO: This is horribly inefficient and we should be using bulk queries but im going for functionality at this point
User_Roles . delete ( ) . where ( User_Roles . user_id == user_id ) . where ( User_Roles . role_id . in_ ( removed_roles ) ) . execute ( )
2021-03-21 23:02:18 -05:00
#for server in added_servers:
# User_Servers.get_or_create(user_id=user_id, server_id=server)
# # TODO: This is horribly inefficient and we should be using bulk queries but im going for functionality at this point
#User_Servers.delete().where(User_Servers.user_id == user_id).where(User_Servers.server_id.in_(removed_servers)).execute()
2020-12-25 22:00:28 -05:00
if up_data :
Users . update ( up_data ) . where ( Users . user_id == user_id ) . execute ( )
2020-12-24 17:55:15 -05:00
@staticmethod
def add_user ( username , password = None , api_token = None , enabled = True , superuser = False ) :
if password is not None :
pw_enc = helper . encode_pass ( password )
else :
pw_enc = None
if api_token is None :
api_token = db_shortcuts . new_api_token ( )
else :
if type ( api_token ) is not str and len ( api_token ) != 32 :
raise ValueError ( " API token must be a 32 character string " )
user_id = Users . insert ( {
Users . username : username . lower ( ) ,
Users . password : pw_enc ,
Users . api_token : api_token ,
Users . enabled : enabled ,
2020-12-27 11:00:26 -05:00
Users . superuser : superuser ,
Users . created : helper . get_time_as_string ( )
2020-12-24 17:55:15 -05:00
} ) . execute ( )
return user_id
@staticmethod
def remove_user ( user_id ) :
2021-04-03 12:36:01 -05:00
with database . atomic ( ) :
2021-04-03 13:18:43 -05:00
User_Roles . delete ( ) . where ( User_Roles . user_id == user_id ) . execute ( )
2021-04-03 12:36:01 -05:00
user = Users . get ( Users . user_id == user_id )
return user . delete_instance ( )
2020-12-24 17:55:15 -05:00
@staticmethod
def user_id_exists ( user_id ) :
if not db_shortcuts . get_user ( user_id ) :
return False
return True
@staticmethod
def get_roleid_by_name ( role_name ) :
2020-12-27 11:00:26 -05:00
try :
return ( Roles . get ( Roles . role_name == role_name ) ) . role_id
except DoesNotExist :
return None
2020-12-24 17:55:15 -05:00
@staticmethod
def get_role ( role_id ) :
2020-12-27 11:00:26 -05:00
role = model_to_dict ( Roles . get ( Roles . role_id == role_id ) )
if role :
servers_query = Role_Servers . select ( ) . join ( Servers , JOIN . INNER ) . where ( Role_Servers . role_id == role_id )
# TODO: this query needs to be narrower
servers = set ( )
for s in servers_query :
servers . add ( s . server_id . server_id )
role [ ' servers ' ] = servers
2021-04-03 12:36:01 -05:00
#logger.debug("role: ({}) {}".format(role_id, role))
2020-12-27 11:00:26 -05:00
return role
else :
2021-04-03 12:36:01 -05:00
#logger.debug("role: ({}) {}".format(role_id, {}))
2020-12-27 11:00:26 -05:00
return { }
@staticmethod
def update_role ( role_id , role_data = { } ) :
base_data = db_helper . get_role ( role_id )
up_data = { }
added_servers = set ( )
removed_servers = set ( )
for key in role_data :
if key == " role_id " :
continue
elif key == " servers " :
added_servers = role_data [ ' servers ' ] . difference ( base_data [ ' servers ' ] )
removed_servers = base_data [ ' servers ' ] . difference ( role_data [ ' servers ' ] )
elif base_data [ key ] != role_data [ key ] :
up_data [ key ] = role_data [ key ]
up_data [ ' last_update ' ] = helper . get_time_as_string ( )
logger . debug ( " role: {} +server: {} -server {} " . format ( role_data , added_servers , removed_servers ) )
with database . atomic ( ) :
for server in added_servers :
Role_Servers . get_or_create ( role_id = role_id , server_id = server )
# TODO: This is horribly inefficient and we should be using bulk queries but im going for functionality at this point
Role_Servers . delete ( ) . where ( Role_Servers . role_id == role_id ) . where ( Role_Servers . server_id . in_ ( removed_servers ) ) . execute ( )
if up_data :
Roles . update ( up_data ) . where ( Roles . role_id == role_id ) . execute ( )
@staticmethod
def add_role ( role_name ) :
role_id = Roles . insert ( {
Roles . role_name : role_name . lower ( ) ,
Roles . created : helper . get_time_as_string ( )
} ) . execute ( )
return role_id
@staticmethod
def remove_role ( role_id ) :
2021-04-03 13:18:43 -05:00
with database . atomic ( ) :
Role_Servers . delete ( ) . where ( Role_Servers . role_id == role_id ) . execute ( )
User_Roles . delete ( ) . where ( User_Roles . role_id == role_id ) . execute ( )
role = Roles . get ( Roles . role_id == role_id )
return role . delete_instance ( )
2020-12-27 11:00:26 -05:00
@staticmethod
def role_id_exists ( role_id ) :
if not db_shortcuts . get_role ( role_id ) :
return False
return True
2020-12-24 17:55:15 -05:00
@staticmethod
def get_unactioned_commands ( ) :
2020-08-31 16:16:45 -04:00
query = Commands . select ( ) . where ( Commands . executed == 0 )
2020-12-24 17:55:15 -05:00
return db_helper . return_rows ( query )
2020-08-16 22:47:53 -04:00
2021-03-05 11:35:05 +02:00
@staticmethod
def get_server_friendly_name ( server_id ) :
server_data = db_helper . get_server_data_by_id ( server_id )
2021-03-01 02:54:20 +02:00
friendly_name = " {} with ID: {} " . format ( server_data . get ( ' server_name ' , None ) , server_data . get ( ' server_id ' , 0 ) )
2020-08-31 16:16:45 -04:00
return friendly_name
2021-03-05 11:35:05 +02:00
@staticmethod
def send_command ( user_id , server_id , remote_ip , command ) :
2020-08-31 16:16:45 -04:00
2021-03-05 11:35:05 +02:00
server_name = db_helper . get_server_friendly_name ( server_id )
2020-08-31 16:16:45 -04:00
2021-03-05 11:35:05 +02:00
# Example: Admin issued command start_server for server Survival
db_helper . add_to_audit_log ( user_id , " issued command {} for server {} " . format ( command , server_name ) ,
2020-08-31 16:16:45 -04:00
server_id , remote_ip )
Commands . insert ( {
Commands . server_id : server_id ,
Commands . user : user_id ,
Commands . source_ip : remote_ip ,
Commands . command : command
} ) . execute ( )
2020-12-24 17:55:15 -05:00
@staticmethod
def get_actity_log ( ) :
2020-12-13 13:40:55 -05:00
q = Audit_Log . select ( )
2020-12-24 17:55:15 -05:00
return db_helper . return_db_rows ( q )
2020-12-13 13:40:55 -05:00
2020-12-24 17:55:15 -05:00
@staticmethod
def return_db_rows ( model ) :
2020-12-13 13:40:55 -05:00
data = [ model_to_dict ( row ) for row in model ]
return data
2020-08-31 16:16:45 -04:00
@staticmethod
def mark_command_complete ( command_id = None ) :
if command_id is not None :
logger . debug ( " Marking Command {} completed " . format ( command_id ) )
Commands . update ( {
Commands . executed : True
} ) . where ( Commands . command_id == command_id ) . execute ( )
2021-04-03 12:36:01 -05:00
def add_to_audit_log ( self , user_id , log_msg , server_id = None , source_ip = None ) :
2020-08-31 16:16:45 -04:00
logger . debug ( " Adding to audit log User: {} - Message: {} " . format ( user_id , log_msg ) )
2021-04-03 12:36:01 -05:00
user_data = self . get_user ( user_id )
2020-08-31 16:16:45 -04:00
2021-04-03 12:36:01 -05:00
audit_msg = " {} {} " . format ( str ( user_data [ ' username ' ] ) . capitalize ( ) , log_msg )
2020-08-31 16:16:45 -04:00
2021-03-01 02:54:20 +02:00
websocket_helper . broadcast ( ' notification ' , audit_msg )
2020-08-31 16:16:45 -04:00
Audit_Log . insert ( {
2021-04-03 12:36:01 -05:00
Audit_Log . user_name : user_data [ ' username ' ] ,
2020-08-31 16:16:45 -04:00
Audit_Log . user_id : user_id ,
Audit_Log . server_id : server_id ,
Audit_Log . log_msg : audit_msg ,
Audit_Log . source_ip : source_ip
} ) . execute ( )
2021-03-01 02:54:20 +02:00
@staticmethod
def add_to_audit_log_raw ( user_name , user_id , server_id , log_msg , source_ip ) :
Audit_Log . insert ( {
Audit_Log . user_name : user_name ,
Audit_Log . user_id : user_id ,
Audit_Log . server_id : server_id ,
Audit_Log . log_msg : log_msg ,
Audit_Log . source_ip : source_ip
} ) . execute ( )
2020-08-31 16:16:45 -04:00
2021-03-21 23:02:18 -05:00
@staticmethod
def create_scheduled_task ( server_id , action , interval , interval_type , start_time , command , comment = None , enabled = True ) :
sch_id = Schedules . insert ( {
Schedules . server_id : server_id ,
Schedules . action : action ,
Schedules . enabled : enabled ,
Schedules . interval : interval ,
Schedules . interval_type : interval_type ,
Schedules . start_time : start_time ,
Schedules . command : command ,
Schedules . comment : comment
} ) . execute ( )
return sch_id
@staticmethod
def delete_scheduled_task ( schedule_id ) :
sch = Schedules . get ( Schedules . schedule_id == schedule_id )
return Schedules . delete_instance ( sch )
@staticmethod
def update_scheduled_task ( schedule_id , updates ) :
Schedules . update ( updates ) . where ( Schedules . schedule_id == schedule_id ) . execute ( )
@staticmethod
def get_scheduled_task ( schedule_id ) :
return model_to_dict ( Schedules . get ( Schedules . schedule_id == schedule_id ) ) . execute ( )
@staticmethod
def get_schedules_by_server ( server_id ) :
return Schedules . select ( ) . where ( Schedules . server_id == server_id ) . execute ( )
@staticmethod
def get_schedules_all ( ) :
return Schedules . select ( ) . execute ( )
2020-08-31 16:16:45 -04:00
2021-03-21 23:02:18 -05:00
@staticmethod
def get_schedules_enabled ( ) :
return Schedules . select ( ) . where ( Schedules . enabled == True ) . execute ( )
2020-12-13 13:40:55 -05:00
2021-03-21 23:02:18 -05:00
@staticmethod
def get_backup_config ( server_id ) :
try :
row = Backups . select ( ) . where ( Backups . server_id == server_id ) . join ( Schedules ) . join ( Servers ) [ 0 ]
conf = {
" backup_path " : row . server_id . backup_path ,
" directories " : row . directories ,
" max_backups " : row . max_backups ,
" auto_enabled " : row . schedule_id . enabled ,
" server_id " : row . server_id . server_id
}
except IndexError :
conf = {
" backup_path " : None ,
" directories " : None ,
" max_backups " : 0 ,
" auto_enabled " : True ,
" server_id " : server_id
}
return conf
2021-07-25 17:04:10 -04:00
@staticmethod
def set_update ( server_id , value ) :
try :
row = Server_Stats . select ( ) . where ( Server_Stats . server_id == server_id )
except Exception as ex :
logger . error ( " Database entry not found. " . format ( ex ) )
with database . atomic ( ) :
Server_Stats . update ( updating = value ) . where ( Server_Stats . server_id == server_id ) . execute ( )
2021-03-21 23:02:18 -05:00
@staticmethod
def set_backup_config ( server_id : int , backup_path : str = None , max_backups : int = None , auto_enabled : bool = True ) :
logger . debug ( " Updating server {} backup config with {} " . format ( server_id , locals ( ) ) )
try :
row = Backups . select ( ) . where ( Backups . server_id == server_id ) . join ( Schedules ) . join ( Servers ) [ 0 ]
new_row = False
conf = { }
schd = { }
except IndexError :
conf = {
" directories " : None ,
" max_backups " : 0 ,
" server_id " : server_id
}
schd = {
" enabled " : True ,
" action " : " backup_server " ,
" interval_type " : " days " ,
" interval " : 1 ,
" start_time " : " 00:00 " ,
" server_id " : server_id ,
" comment " : " Default backup job "
}
new_row = True
if max_backups is not None :
conf [ ' max_backups ' ] = max_backups
schd [ ' enabled ' ] = bool ( auto_enabled )
if not new_row :
with database . atomic ( ) :
if backup_path is not None :
u1 = Servers . update ( backup_path = backup_path ) . where ( Servers . server_id == server_id ) . execute ( )
else :
u1 = 0
u2 = Backups . update ( conf ) . where ( Backups . server_id == server_id ) . execute ( )
u3 = Schedules . update ( schd ) . where ( Schedules . schedule_id == row . schedule_id ) . execute ( )
logger . debug ( " Updating existing backup record. {} + {} + {} rows affected " . format ( u1 , u2 , u3 ) )
else :
with database . atomic ( ) :
conf [ " server_id " ] = server_id
if backup_path is not None :
u = Servers . update ( backup_path = backup_path ) . where ( Servers . server_id == server_id )
s = Schedules . create ( * * schd )
conf [ ' schedule_id ' ] = s . schedule_id
b = Backups . create ( * * conf )
logger . debug ( " Creating new backup record. " )
2020-12-13 13:40:55 -05:00
2020-08-12 21:33:36 -04:00
installer = db_builder ( )
2021-03-21 23:02:18 -05:00
db_helper = db_shortcuts ( )