import os import json import threading import time 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.file_helpers import FileHelpers from app.classes.shared.websocket_manager import WebSocketManager logger = logging.getLogger(__name__) # Temp type var until sjars restores generic fetchTypes0 class BigBucket: def __init__(self, helper): self.helper = helper # remove any trailing slash from config.json # url since we add it on all the calls self.base_url = str( self.helper.get_setting("big_bucket_repo", "https://jars.arcadiatech.org") ).rstrip("/") def _read_cache(self) -> dict: cache_file = self.helper.big_bucket_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 big_bucket cache file: {e}") return cache def get_bucket_data(self): data = self._read_cache() return data.get("types") def _check_bucket_alive(self) -> bool: logger.info("Checking Big Bucket status") check_url = f"{self.base_url}/healthcheck" try: response = requests.get(check_url, timeout=2) response_json = response.json() if ( response.status_code in [200, 201] and response_json.get("status") == "ok" ): logger.info("Big bucket is alive and responding as expected") return True except Exception as e: logger.error(f"Unable to connect to big bucket due to error: {e}") return False logger.error( "Big bucket manifest is not available as expected or unable to contact" ) return False def _get_big_bucket(self) -> dict: logger.debug("Calling for big bucket manifest.") try: response = requests.get(f"{self.base_url}/manifest.json", timeout=5) if response.status_code in [200, 201]: data = response.json() del data["manifest_version"] return data return {} except TimeoutError as e: logger.error(f"Unable to get jars from remote with error {e}") return {} def _refresh_cache(self): """ Contains the shared logic for refreshing the cache. This method is called by both manual_refresh_cache and refresh_cache methods. """ if not self._check_bucket_alive(): logger.error("big bucket API is not available.") return False cache_data = { "last_refreshed": datetime.now().strftime("%m/%d/%Y, %H:%M:%S"), "types": self._get_big_bucket(), } try: with open( self.helper.big_bucket_cache, "w", encoding="utf-8" ) as cache_file: json.dump(cache_data, cache_file, indent=4) logger.info("Cache file successfully refreshed manually.") except Exception as e: logger.error(f"Failed to update cache file manually: {e}") def manual_refresh_cache(self): """ Manually triggers the cache refresh process. """ logger.info("Manual bucket cache refresh initiated.") self._refresh_cache() logger.info("Manual refresh completed.") def refresh_cache(self): """ Automatically trigger cache refresh process based age. This method checks if the cache file is older than a specified number of days before deciding to refresh. """ cache_file_path = self.helper.big_bucket_cache # Determine if the cache is old and needs refreshing cache_old = self.helper.is_file_older_than_x_days(cache_file_path) # debug override # cache_old = True if not self._check_bucket_alive(): logger.error("big bucket API is not available.") return False if not cache_old: logger.info("Cache file is not old enough to require automatic refresh.") return False logger.info("Automatic cache refresh initiated due to old cache.") self._refresh_cache() def get_fetch_url(self, jar, server, version) -> str: """ Constructs the URL for downloading a server JAR file based on the server type. Parameters: jar (str): The category of the JAR file to download. server (str): Server software name (e.g., "paper"). version (str): Server version. Returns: str or None: URL for downloading the JAR file, or None if URL cannot be constructed or an error occurs. """ try: # Read cache file for URL that is in a list of one item return self._read_cache()["types"][jar][server]["versions"][version]["url"][ 0 ] except Exception as e: logger.error(f"An error occurred while constructing fetch URL: {e}") return None 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): """ Downloads a server JAR file and performs post-download actions including notifying users and setting import status. This method waits for the server registration to complete, retrieves the download URL for the specified server JAR file. Upon successful download, it either runs the installer for Forge servers or simply finishes the import process for other types. It notifies server users about the completion of the download. Parameters: - jar (str): The category of the JAR file to download. - server (str): The type of server software (e.g., 'forge', 'paper'). - version (str): The version of the server software. - path (str): The local filesystem path where the JAR file will be saved. - server_id (str): The unique identifier for the server being updated or imported, used for notifying users and setting the import status. Returns: - bool: True if the JAR file was successfully downloaded and saved; False otherwise. The method ensures that the server is properly registered before proceeding with the download and handles exceptions by logging errors and reverting the import status if necessary. """ # delaying download for server register to finish time.sleep(3) fetch_url = self.get_fetch_url(jar, server, version) if not fetch_url: return False server_users = PermissionsServers.get_server_user_list(server_id) # Make sure the server is registered before updating its 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}") # Initiate Download jar_dir = os.path.dirname(path) jar_name = os.path.basename(path) logger.info(fetch_url) success = FileHelpers.ssl_get_file(fetch_url, jar_dir, jar_name) # Post-download actions if success: if server == "forge-installer": # If this is the newer Forge version, run the installer ServersController.finish_import(server_id, True) else: ServersController.finish_import(server_id) # Notify users for user in server_users: WebSocketManager().broadcast_user( user, "notification", "Executable download finished" ) time.sleep(3) # Delay for user notification WebSocketManager().broadcast_user(user, "send_start_reload", {}) else: logger.error(f"Unable to save jar to {path} due to download failure.") ServersController.finish_import(server_id) return success