2023-01-09 05:04:10 +00:00
|
|
|
import os
|
|
|
|
import platform
|
|
|
|
import zipfile
|
2023-03-18 20:24:12 +00:00
|
|
|
import tarfile
|
2023-01-09 05:04:10 +00:00
|
|
|
import subprocess
|
|
|
|
import urllib.request
|
2023-04-14 20:48:26 +00:00
|
|
|
import logging
|
2023-01-09 05:04:10 +00:00
|
|
|
|
|
|
|
from getpass import getpass
|
2023-03-18 20:59:40 +00:00
|
|
|
from app.classes.steamcmd.steamcmd_command import SteamCMDcommand
|
2023-03-18 20:24:12 +00:00
|
|
|
|
2023-04-14 20:48:26 +00:00
|
|
|
logger = logging.getLogger(__name__)
|
2023-01-09 05:04:10 +00:00
|
|
|
|
|
|
|
package_links = {
|
|
|
|
"Windows": {
|
|
|
|
"url": "https://steamcdn-a.akamaihd.net/client/installer/steamcmd.zip",
|
|
|
|
"extension": ".exe",
|
2023-03-18 20:04:32 +00:00
|
|
|
"d_extension": ".zip",
|
2023-01-09 05:04:10 +00:00
|
|
|
},
|
|
|
|
"Linux": {
|
|
|
|
"url": "https://steamcdn-a.akamaihd.net/client/installer/steamcmd_linux.tar.gz",
|
|
|
|
"extension": ".sh",
|
2023-03-18 20:04:32 +00:00
|
|
|
"d_extension": ".tar.gz",
|
|
|
|
},
|
2023-01-09 05:04:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
class SteamCMD:
|
|
|
|
"""
|
|
|
|
Wrapper for SteamCMD
|
|
|
|
Will install from source depending on OS.
|
|
|
|
"""
|
|
|
|
|
|
|
|
_installation_path = ""
|
|
|
|
_uname = "anonymous"
|
|
|
|
_passw = ""
|
|
|
|
|
|
|
|
def __init__(self, installation_path):
|
|
|
|
self._installation_path = installation_path
|
|
|
|
|
|
|
|
if not os.path.isdir(self._installation_path):
|
2023-03-18 20:04:32 +00:00
|
|
|
raise NotADirectoryError(
|
|
|
|
message=f"No valid directory found at {self._installation_path}"
|
|
|
|
"Please make sure that the directory is correct."
|
|
|
|
)
|
2023-01-09 05:04:10 +00:00
|
|
|
|
|
|
|
self._prepare_installation()
|
|
|
|
|
|
|
|
def _prepare_installation(self):
|
|
|
|
"""
|
|
|
|
Sets internal configuration according to parameters and OS
|
|
|
|
"""
|
|
|
|
|
|
|
|
self.platform = platform.system()
|
|
|
|
if self.platform not in ["Windows", "Linux"]:
|
2023-03-18 20:04:32 +00:00
|
|
|
raise NotImplementedError(
|
2023-03-18 20:24:12 +00:00
|
|
|
message=(
|
|
|
|
f"Non supported operating system. "
|
|
|
|
f"Expected Windows, or Linux, got {self.platform}"
|
|
|
|
)
|
2023-03-18 20:04:32 +00:00
|
|
|
)
|
2023-01-09 05:04:10 +00:00
|
|
|
|
|
|
|
self.steamcmd_url = package_links[self.platform]["url"]
|
|
|
|
self.zip = "steamcmd" + package_links[self.platform]["d_extension"]
|
|
|
|
self.exe = os.path.join(
|
|
|
|
self._installation_path,
|
2023-03-18 20:04:32 +00:00
|
|
|
"steamcmd" + package_links[self.platform]["extension"],
|
2023-01-09 05:04:10 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
def _download(self):
|
|
|
|
"""
|
|
|
|
Internal method to download the SteamCMD Binaries from steams' servers.
|
|
|
|
:return: downloaded data for debug purposes
|
|
|
|
"""
|
|
|
|
|
|
|
|
try:
|
|
|
|
if self.steamcmd_url.lower().startswith("http"):
|
|
|
|
req = urllib.request.Request(self.steamcmd_url)
|
|
|
|
else:
|
|
|
|
raise ValueError from None
|
|
|
|
with urllib.request.urlopen(req) as resp:
|
|
|
|
data = resp.read()
|
|
|
|
with open(self.zip, "wb") as f:
|
|
|
|
f.write(data)
|
|
|
|
return data
|
|
|
|
except Exception as e:
|
2023-03-18 20:24:12 +00:00
|
|
|
raise FileNotFoundError(
|
2023-03-18 20:04:32 +00:00
|
|
|
message=f"An unknown exception occurred during downloading. {e}"
|
2023-03-18 20:24:12 +00:00
|
|
|
) from e
|
2023-01-09 05:04:10 +00:00
|
|
|
|
|
|
|
def _extract_steamcmd(self):
|
|
|
|
"""
|
|
|
|
Internal method for extracting downloaded zip file. Works on both
|
|
|
|
windows and linux.
|
|
|
|
"""
|
2023-03-18 20:04:32 +00:00
|
|
|
if self.platform == "Windows":
|
|
|
|
with zipfile.ZipFile(self.zip, "r") as f:
|
2023-01-09 05:04:10 +00:00
|
|
|
f.extractall(self._installation_path)
|
|
|
|
|
2023-03-18 20:04:32 +00:00
|
|
|
elif self.platform == "Linux":
|
|
|
|
with tarfile.open(self.zip, "r:gz") as f:
|
2023-01-09 05:04:10 +00:00
|
|
|
f.extractall(self._installation_path)
|
|
|
|
|
|
|
|
else:
|
|
|
|
# This should never happen, but let's just throw it just in case.
|
2023-03-18 20:04:32 +00:00
|
|
|
raise NotImplementedError(
|
|
|
|
message="The operating system is not supported."
|
|
|
|
f"Expected Linux or Windows, received: {self.platform}"
|
2023-01-09 05:04:10 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
os.remove(self.zip)
|
|
|
|
|
2023-04-14 20:48:26 +00:00
|
|
|
# @staticmethod
|
|
|
|
# def _print_log(*message):
|
|
|
|
# """
|
|
|
|
# Small helper function for printing log entries.
|
|
|
|
# Helps with output of subprocess.check_call not always having newlines
|
|
|
|
# :param *message: Accepts multiple messages, each will be printed on a
|
|
|
|
# new line
|
|
|
|
# """
|
|
|
|
# # TODO: Handle logs better
|
|
|
|
# print("")
|
|
|
|
# print("")
|
|
|
|
# for msg in message:
|
|
|
|
# print(msg)
|
|
|
|
# print("")
|
2023-01-09 05:04:10 +00:00
|
|
|
|
|
|
|
def install(self, force: bool = False):
|
|
|
|
"""
|
|
|
|
Installs steamcmd if it is not already installed to self.install_path.
|
|
|
|
:param force: forces steamcmd install regardless of its presence
|
|
|
|
:return:
|
|
|
|
"""
|
|
|
|
if not os.path.isfile(self.exe) or force:
|
|
|
|
# Steamcmd isn't installed. Go ahead and install it.
|
|
|
|
self._download()
|
|
|
|
self._extract_steamcmd()
|
|
|
|
|
|
|
|
else:
|
2023-03-18 20:04:32 +00:00
|
|
|
raise FileExistsError(
|
|
|
|
message="Steamcmd is already installed. Reinstall is not necessary."
|
|
|
|
"Use force=True to override."
|
2023-01-09 05:04:10 +00:00
|
|
|
)
|
|
|
|
try:
|
|
|
|
subprocess.check_call((self.exe, "+quit"))
|
|
|
|
except subprocess.CalledProcessError as e:
|
|
|
|
if e.returncode == 7:
|
2023-04-14 23:13:55 +00:00
|
|
|
logger.error("SteamCMD has returned error code 7 on fresh installation")
|
2023-01-09 05:04:10 +00:00
|
|
|
return
|
2023-04-14 20:48:26 +00:00
|
|
|
raise SystemError(
|
|
|
|
message=f"Failed to install, check error code {e.returncode}"
|
|
|
|
) from e
|
2023-01-09 05:04:10 +00:00
|
|
|
|
|
|
|
return
|
|
|
|
|
|
|
|
def login(self, uname: str = None, passw: str = None):
|
|
|
|
"""
|
|
|
|
Login function in order to do a persistent login on the steam servers.
|
|
|
|
Prompts users for their credentials and spawns a child process.
|
|
|
|
:param uname: Steam Username
|
|
|
|
:param passw: Steam Password
|
|
|
|
:return: status code of child process
|
|
|
|
"""
|
|
|
|
self._uname = uname if uname else input("Please enter steam username: ")
|
|
|
|
self._passw = passw if passw else getpass("Please enter steam password: ")
|
|
|
|
|
2023-03-18 20:59:40 +00:00
|
|
|
steam_command = SteamCMDcommand()
|
2023-03-18 20:24:12 +00:00
|
|
|
return self.execute(steam_command)
|
2023-01-09 05:04:10 +00:00
|
|
|
|
2023-03-18 20:04:32 +00:00
|
|
|
def app_update(
|
|
|
|
self,
|
|
|
|
app_id: int,
|
|
|
|
install_dir: str = None,
|
|
|
|
validate: bool = None,
|
|
|
|
beta: str = None,
|
|
|
|
betapassword: str = None,
|
|
|
|
):
|
2023-01-09 05:04:10 +00:00
|
|
|
"""
|
|
|
|
Installer function for apps.
|
|
|
|
:param app_id: The Steam ID for the app you want to install
|
|
|
|
:param install_dir: Optional custom installation directory.
|
2023-03-18 20:24:12 +00:00
|
|
|
:param validate: Optional. Turn this on when updating something.
|
2023-01-09 05:04:10 +00:00
|
|
|
:param beta: Optional parameter for running a beta branch.
|
|
|
|
:param betapassword: Optional parameter for entering beta password.
|
|
|
|
:return: Status code of child process.
|
|
|
|
"""
|
2023-03-18 20:59:40 +00:00
|
|
|
steam_command = SteamCMDcommand()
|
2023-01-09 05:04:10 +00:00
|
|
|
if install_dir:
|
2023-03-18 20:24:12 +00:00
|
|
|
steam_command.force_install_dir(install_dir)
|
|
|
|
steam_command.app_update(app_id, validate, beta, betapassword)
|
2023-04-14 20:48:26 +00:00
|
|
|
logger.debug(
|
2023-04-14 23:13:55 +00:00
|
|
|
f"Downloading item {app_id}\n"
|
|
|
|
f"into {install_dir} with validate set to {validate}"
|
2023-03-18 20:04:32 +00:00
|
|
|
)
|
2023-03-18 20:24:12 +00:00
|
|
|
return self.execute(steam_command)
|
2023-01-09 05:04:10 +00:00
|
|
|
|
2023-03-18 20:04:32 +00:00
|
|
|
def workshop_update(
|
|
|
|
self,
|
|
|
|
app_id: int,
|
|
|
|
workshop_id: int,
|
|
|
|
install_dir: str = None,
|
|
|
|
validate: bool = None,
|
|
|
|
n_tries: int = 5,
|
|
|
|
):
|
2023-01-09 05:04:10 +00:00
|
|
|
"""
|
2023-03-18 20:24:12 +00:00
|
|
|
Installer function for workshop content. Retries multiple times on timeout
|
|
|
|
due to valves' timeout on large downloads.
|
2023-01-09 05:04:10 +00:00
|
|
|
:param app_id: The parent application ID
|
|
|
|
:param workshop_id: The ID for workshop content. Can be found in the url.
|
|
|
|
:param install_dir: Optional custom installation directory.
|
2023-03-18 20:24:12 +00:00
|
|
|
:param validate: Optional. Turn this on when updating something.
|
|
|
|
:param n_tries: Counter for how many redownloads it can make before timing out.
|
2023-01-09 05:04:10 +00:00
|
|
|
:return: Status code of child process.
|
|
|
|
"""
|
|
|
|
|
2023-03-18 20:59:40 +00:00
|
|
|
steam_command = SteamCMDcommand()
|
2023-01-09 05:04:10 +00:00
|
|
|
if install_dir:
|
2023-03-18 20:24:12 +00:00
|
|
|
steam_command.force_install_dir(install_dir)
|
|
|
|
steam_command.workshop_download_item(app_id, workshop_id, validate)
|
|
|
|
return self.execute(steam_command, n_tries)
|
2023-01-09 05:04:10 +00:00
|
|
|
|
2023-03-18 20:59:40 +00:00
|
|
|
def execute(self, cmd: SteamCMDcommand, n_tries: int = 1):
|
2023-01-09 05:04:10 +00:00
|
|
|
"""
|
|
|
|
Executes a SteamCMD_command, with added actions occurring sequentially.
|
2023-03-18 20:24:12 +00:00
|
|
|
May retry multiple times on timeout due to valves' timeout on large downloads.
|
2023-01-09 05:04:10 +00:00
|
|
|
:param cmd: Sequence of commands to execute
|
|
|
|
:param n_tries: Number of times the command will be tried.
|
|
|
|
:return: Status code of child process.
|
|
|
|
"""
|
|
|
|
if n_tries == 0:
|
2023-03-18 20:04:32 +00:00
|
|
|
raise TimeoutError(
|
2023-03-18 20:24:12 +00:00
|
|
|
message="""Error executing command, max number of retries exceeded!
|
2023-01-09 05:04:10 +00:00
|
|
|
Consider increasing the n_tries parameter if the download is
|
|
|
|
particularly large"""
|
|
|
|
)
|
|
|
|
|
|
|
|
params = (
|
2023-04-14 23:13:55 +00:00
|
|
|
f'"{self.exe}"',
|
2023-01-09 05:04:10 +00:00
|
|
|
f"+login {self._uname} {self._passw}",
|
|
|
|
cmd.get_cmd(),
|
|
|
|
"+quit",
|
|
|
|
)
|
2023-04-14 20:48:26 +00:00
|
|
|
logger.debug("Parameters used: ".join(params))
|
2023-01-09 05:04:10 +00:00
|
|
|
try:
|
|
|
|
return subprocess.check_call(" ".join(params), shell=True)
|
|
|
|
|
|
|
|
except subprocess.CalledProcessError as e:
|
2023-03-18 20:24:12 +00:00
|
|
|
# SteamCMD has a habit of timing out large downloads,
|
|
|
|
# so retry on timeout for the remainder of n_tries.
|
2023-01-09 05:04:10 +00:00
|
|
|
if e.returncode == 10:
|
2023-04-14 20:48:26 +00:00
|
|
|
logger.warning(
|
2023-03-18 20:04:32 +00:00
|
|
|
f"Download timeout! Tries remaining: {n_tries}. Retrying..."
|
|
|
|
)
|
2023-01-09 05:04:10 +00:00
|
|
|
return self.execute(cmd, n_tries - 1)
|
2023-03-18 20:59:40 +00:00
|
|
|
|
2023-01-09 05:04:10 +00:00
|
|
|
# SteamCMD sometimes crashes when timing out downloads, due to
|
|
|
|
# an assert checking that the download actually finished.
|
|
|
|
# If this happens, retry.
|
2023-03-18 20:59:40 +00:00
|
|
|
if e.returncode == 134:
|
2023-04-14 20:48:26 +00:00
|
|
|
logger.error(
|
2023-03-18 20:04:32 +00:00
|
|
|
f"SteamCMD errored! Tries remaining: {n_tries}. Retrying..."
|
|
|
|
)
|
2023-01-09 05:04:10 +00:00
|
|
|
return self.execute(cmd, n_tries - 1)
|
|
|
|
|
2023-04-14 23:13:55 +00:00
|
|
|
raise SystemError(f"Steamcmd was unable to run. exit code was {e}") from e
|