import json import threading import time import shutil import logging from datetime import datetime import requests from app.classes.controllers.servers_controller import ServersController from app.classes.models.server_permissions import PermissionsServers from app.classes.shared.websocket_manager import WebSocketManager logger = logging.getLogger(__name__) PAPERJARS = ["paper", "folia"] class ServerJars: def __init__(self, helper): self.helper = helper self.base_url = "https://serverjars.com" self.paper_base = "https://api.papermc.io" @staticmethod def get_paper_jars(): return PAPERJARS def _get_api_result(self, call_url: str): full_url = f"{self.base_url}{call_url}" try: response = requests.get(full_url, timeout=2) response.raise_for_status() api_data = json.loads(response.content) except Exception as e: logger.error(f"Unable to load {full_url} api due to error: {e}") return {} api_result = api_data.get("status") api_response = api_data.get("response", {}) if api_result != "success": logger.error(f"Api returned a failed status: {api_result}") return {} return api_response def get_paper_versions(self, project): try: response = requests.get( f"{self.paper_base}/v2/projects/{project}/", timeout=2 ) response.raise_for_status() api_data = json.loads(response.content) except Exception as e: logger.error( f"Unable to load https://api.papermc.io/v2/projects/{project}/" f"api due to error: {e}" ) return {} versions = api_data.get("versions", []) versions.reverse() return versions def get_paper_build(self, project, version): try: response = requests.get( f"{self.paper_base}/v2/projects/{project}/versions/{version}/builds/", timeout=2, ) response.raise_for_status() api_data = json.loads(response.content) except Exception as e: logger.error( f"Unable to load https://api.papermc.io/v2/projects/{project}/" f"api due to error: {e}" ) return {} build = api_data.get("builds", [])[-1] return build def _read_cache(self): cache_file = self.helper.serverjar_cache cache = {} try: with open(cache_file, "r", encoding="utf-8") as f: cache = json.load(f) except Exception as e: logger.error(f"Unable to read serverjars.com cache file: {e}") return cache def get_serverjar_data(self): data = self._read_cache() return data.get("types") def _check_api_alive(self): logger.info("Checking serverjars.com API status") check_url = f"{self.base_url}/api/fetchTypes" try: response = requests.get(check_url, timeout=2) if response.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 {} logger.error("unable to contact serverjars.com api") return False def manual_refresh_cache(self): cache_file = self.helper.serverjar_cache # debug override # cache_old = True # if the API is down... we bomb out if not self._check_api_alive(): return False logger.info("Manual Refresh requested.") now = datetime.now() data = { "last_refreshed": now.strftime("%m/%d/%Y, %H:%M:%S"), "types": {}, } jar_types = self._get_server_type_list() data["types"].update(jar_types) for s in data["types"]: data["types"].update({s: dict.fromkeys(data["types"].get(s), {})}) for j in data["types"].get(s): versions = self._get_jar_details(j, s) data["types"][s].update({j: versions}) for item in PAPERJARS: data["types"]["servers"][item] = self.get_paper_versions(item) # save our cache try: with open(cache_file, "w", encoding="utf-8") as f: 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}") def refresh_cache(self): cache_file = self.helper.serverjar_cache cache_old = self.helper.is_file_older_than_x_days(cache_file) # 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"), "types": {}, } jar_types = self._get_server_type_list() data["types"].update(jar_types) for s in data["types"]: data["types"].update({s: dict.fromkeys(data["types"].get(s), {})}) for j in data["types"].get(s): versions = self._get_jar_details(j, s) data["types"][s].update({j: versions}) for item in PAPERJARS: data["types"]["servers"][item] = self.get_paper_versions(item) # save our cache try: with open(cache_file, "w", encoding="utf-8") as f: 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}") def _get_jar_details(self, server_type, jar_type="servers"): url = f"/api/fetchAll/{jar_type}/{server_type}" response = self._get_api_result(url) temp = [] for v in response: temp.append(v.get("version")) time.sleep(0.5) return temp def _get_server_type_list(self): url = "/api/fetchTypes/" response = self._get_api_result(url) if "bedrock" in response.keys(): # remove pocketmine from options del response["bedrock"] return response def download_jar(self, jar, 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=(jar, server, version, path, server_id), ) update_thread.start() def a_download_jar(self, jar, server, version, path, server_id): # delaying download for server register to finish time.sleep(3) if server not in PAPERJARS: fetch_url = f"{self.base_url}/api/fetchJar/{jar}/{server}/{version}" else: build = self.get_paper_build(server, version).get("build", None) if not build: return fetch_url = ( f"{self.paper_base}/v2/projects" f"/{server}/versions/{version}/builds/{build}/downloads/" f"{server}-{version}-{build}.jar" ) server_users = PermissionsServers.get_server_user_list(server_id) # We need to make sure the server is registered before # we submit a db update for it's stats. while True: try: ServersController.set_import(server_id) for user in server_users: WebSocketManager().broadcast_user(user, "send_start_reload", {}) break except Exception as ex: logger.debug(f"server not registered yet. Delaying download - {ex}") # open a file stream with requests.get(fetch_url, timeout=2, stream=True) as r: success = False try: with open(path, "wb") as output: shutil.copyfileobj(r.raw, output) # If this is the newer forge version we will run the installer if server == "forge": ServersController.finish_import(server_id, True) else: ServersController.finish_import(server_id) success = True except Exception as e: logger.error(f"Unable to save jar to {path} due to error:{e}") ServersController.finish_import(server_id) server_users = PermissionsServers.get_server_user_list(server_id) for user in server_users: WebSocketManager().broadcast_user( user, "notification", "Executable download finished" ) time.sleep(3) WebSocketManager().broadcast_user(user, "send_start_reload", {}) return success