crafty-4/app/classes/minecraft/serverjars.py

219 lines
7.2 KiB
Python
Raw Normal View History

2020-08-23 22:43:28 +00:00
import json
import threading
2020-08-23 22:43:28 +00:00
import time
import shutil
2020-08-23 22:43:28 +00:00
import logging
from datetime import datetime
import requests
2020-08-23 22:43:28 +00:00
2022-03-04 00:36:36 +00:00
from app.classes.controllers.servers_controller import Servers_Controller
from app.classes.models.server_permissions import Permissions_Servers
2020-08-23 22:43:28 +00:00
logger = logging.getLogger(__name__)
2022-04-11 10:08:36 +00:00
class ServerJars:
def __init__(self, helper):
self.helper = helper
self.base_url = "https://serverjars.com"
2020-08-23 22:43:28 +00:00
def _get_api_result(self, call_url: str):
full_url = f"{self.base_url}{call_url}"
2020-08-23 22:43:28 +00:00
try:
r = requests.get(full_url, timeout=2)
2020-08-23 22:43:28 +00:00
if r.status_code not in [200, 201]:
return {}
except Exception as e:
logger.error(f"Unable to connect to serverjar.com api due to error: {e}")
2020-08-23 22:43:28 +00:00
return {}
try:
api_data = json.loads(r.content)
except Exception as e:
logger.error(f"Unable to parse serverjar.com api result due to error: {e}")
2020-08-23 22:43:28 +00:00
return {}
api_result = api_data.get("status")
api_response = api_data.get("response", {})
2020-08-23 22:43:28 +00:00
if api_result != "success":
logger.error(f"Api returned a failed status: {api_result}")
2020-08-23 22:43:28 +00:00
return {}
return api_response
def _read_cache(self):
cache_file = self.helper.serverjar_cache
2020-08-23 22:43:28 +00:00
cache = {}
try:
with open(cache_file, "r", encoding="utf-8") as f:
2020-08-23 22:43:28 +00:00
cache = json.load(f)
except Exception as e:
logger.error(f"Unable to read serverjars.com cache file: {e}")
2020-08-23 22:43:28 +00:00
return cache
def get_serverjar_data(self):
data = self._read_cache()
return data.get("servers")
2020-08-23 22:43:28 +00:00
def get_serverjar_data_sorted(self):
data = self.get_serverjar_data()
def str_to_int(x, counter=0):
try:
return ord(x[0]) + str_to_int(x[1:], counter + 1) + len(x)
except IndexError:
return 0
def to_int(x):
try:
return int(x)
except ValueError:
temp = x.split("-")
return to_int(temp[0]) + str_to_int(temp[1]) / 100000
sort_key_fn = lambda x: [to_int(y) for y in x.split(".")]
for key in data.keys():
data[key] = sorted(data[key], key=sort_key_fn)
return data
def _check_api_alive(self):
2020-08-23 22:43:28 +00:00
logger.info("Checking serverjars.com API status")
check_url = f"{self.base_url}/api/fetchTypes"
try:
r = requests.get(check_url, timeout=2)
2020-08-23 22:43:28 +00:00
if r.status_code in [200, 201]:
logger.info("Serverjars.com API is alive")
return True
except Exception as e:
logger.error(f"Unable to connect to serverjar.com api due to error: {e}")
return {}
2020-08-23 22:43:28 +00:00
logger.error("unable to contact serverjars.com api")
2020-08-23 22:43:28 +00:00
return False
def refresh_cache(self):
cache_file = self.helper.serverjar_cache
cache_old = self.helper.is_file_older_than_x_days(cache_file)
2020-08-23 22:43:28 +00:00
# debug override
# cache_old = True
# if the API is down... we bomb out
if not self._check_api_alive():
return False
logger.info("Checking Cache file age")
# if file is older than 1 day
if cache_old:
logger.info("Cache file is over 1 day old, refreshing")
now = datetime.now()
data = {"last_refreshed": now.strftime("%m/%d/%Y, %H:%M:%S"), "servers": {}}
2020-08-23 22:43:28 +00:00
jar_types = self._get_server_type_list()
# for each jar type
for j in jar_types:
# for each server
for s in jar_types.get(j):
# jar versions for this server
versions = self._get_jar_details(s)
# add these versions (a list) to the dict with
# a key of the server type
data["servers"].update({s: versions})
2020-08-23 22:43:28 +00:00
# save our cache
try:
with open(cache_file, "w", encoding="utf-8") as f:
2020-08-23 22:43:28 +00:00
f.write(json.dumps(data, indent=4))
logger.info("Cache file refreshed")
except Exception as e:
logger.error(f"Unable to update serverjars.com cache file: {e}")
2020-08-23 22:43:28 +00:00
def _get_jar_details(self, jar_type="servers"):
url = f"/api/fetchAll/{jar_type}"
2020-08-23 22:43:28 +00:00
response = self._get_api_result(url)
temp = []
for v in response:
temp.append(v.get("version"))
time.sleep(0.5)
2020-08-23 22:43:28 +00:00
return temp
def _get_server_type_list(self):
url = "/api/fetchTypes/"
2020-08-23 22:43:28 +00:00
response = self._get_api_result(url)
return response
2022-03-04 00:36:36 +00:00
def download_jar(self, server, version, path, server_id):
update_thread = threading.Thread(
name=f"server_download-{server_id}-{server}-{version}",
target=self.a_download_jar,
daemon=True,
args=(server, version, path, server_id),
)
update_thread.start()
2022-03-04 00:36:36 +00:00
def a_download_jar(self, server, version, path, server_id):
# delaying download for server register to finish
2022-03-04 00:36:36 +00:00
time.sleep(3)
fetch_url = f"{self.base_url}/api/fetchJar/{server}/{version}"
server_users = Permissions_Servers.get_server_user_list(server_id)
2022-03-04 00:36:36 +00:00
# We need to make sure the server is registered before
# we submit a db update for it's stats.
2022-03-04 00:36:36 +00:00
while True:
try:
Servers_Controller.set_download(server_id)
for user in server_users:
2022-04-11 10:08:36 +00:00
self.helper.websocket_helper.broadcast_user(
user, "send_start_reload", {}
)
2022-03-04 00:36:36 +00:00
break
except Exception as ex:
logger.debug(f"server not registered yet. Delaying download - {ex}")
2020-08-23 22:43:28 +00:00
# open a file stream
with requests.get(fetch_url, timeout=2, stream=True) as r:
2020-08-23 22:43:28 +00:00
try:
with open(path, "wb") as output:
shutil.copyfileobj(r.raw, output)
2022-03-04 00:36:36 +00:00
Servers_Controller.finish_download(server_id)
for user in server_users:
self.helper.websocket_helper.broadcast_user(
user, "notification", "Executable download finished"
)
2022-03-04 00:36:36 +00:00
time.sleep(3)
2022-04-11 10:08:36 +00:00
self.helper.websocket_helper.broadcast_user(
user, "send_start_reload", {}
)
2022-03-04 00:36:36 +00:00
return True
2020-08-23 22:43:28 +00:00
except Exception as e:
logger.error(f"Unable to save jar to {path} due to error:{e}")
2022-03-04 00:36:36 +00:00
Servers_Controller.finish_download(server_id)
server_users = Permissions_Servers.get_server_user_list(server_id)
2022-03-04 00:36:36 +00:00
for user in server_users:
self.helper.websocket_helper.broadcast_user(
user, "notification", "Executable download finished"
)
2022-03-04 00:36:36 +00:00
time.sleep(3)
2022-04-11 10:08:36 +00:00
self.helper.websocket_helper.broadcast_user(
user, "send_start_reload", {}
)
2020-08-23 22:43:28 +00:00
2022-03-04 00:36:36 +00:00
return False