feat: fully working async install

This commit is contained in:
Yannik Schmidt
2025-04-13 00:18:02 +02:00
parent 8e9e4db3fa
commit 4e9b85ee6d
6 changed files with 106 additions and 32 deletions

View File

@@ -34,6 +34,7 @@ db = None # app data-backend (i.e. LocalFS or FTP)
CONFIG_FILE = "gamevault_config.json"
def close_input_window(input_window):
'''Close the config window and save the settings'''
@@ -188,7 +189,7 @@ def load_main():
app.title("Lan Vault: Overview")
if not infowidget_window:
infowidget_window = infowidget.ProgressBarApp(app)
infowidget_window = infowidget.ProgressBarApp(app, data_backend=db)
# navbar should not expand when window is resized
app.grid_rowconfigure(0, weight=0)

View File

@@ -112,6 +112,12 @@ class HTTP(DataBackend):
print("Local Target is", local_file)
return local_file
def local_delete_cache_file(self, path, cache_dir=None):
'''Delete a local cache file'''
print("WARNING: removing:", self.get_local_target(path))
os.remove(self.get_local_target(path))
def get(self, path, cache_dir=None, return_content=False, wait=False):
print("Getting", path, "cache dir", cache_dir, "return content:", return_content)
@@ -146,6 +152,7 @@ class HTTP(DataBackend):
if return_content or wait:
print("Sync Requested")
# the content is needed for the UI now and not cached, it's needs to be downloaded synchroniously #
# as there cannot be a meaningful UI-draw without it. #
# THIS IS THE OLD WAY
@@ -156,13 +163,30 @@ class HTTP(DataBackend):
# f.write(r.text)
# this is with streaming
chunk_size = 1024 * 1024 * 5 # 5MB
chunk_size = 1024 * 1024 * 50 # 50MB
r = requests.get(self._get_url(), params={"path": path, "as_string": True}, stream=True)
r.raise_for_status()
with open(local_file, "wb") as f:
if path.endswith(".reg") or path.endswith(".txt"):
TYPE = "w"
else:
TYPE = "wb"
with open(local_file, TYPE) as f:
count = 0
for chunk in r.iter_content(chunk_size=chunk_size, decode_unicode=True):
print(f"Doing chunk.. {chunk_size*count}")
if chunk:
try:
f.write(chunk)
except TypeError as e:
print("Cannot write:", chunk, "..to ", path, " because it is the wrong type.", e)
raise e
f.flush()
count += 1
if return_content:
print("Content for", path, ":", r.text)

6
db.py
View File

@@ -4,6 +4,9 @@ from sqlalchemy.orm import sessionmaker, scoped_session
Base = declarative_base()
import logging
logging.basicConfig()
logging.getLogger('sqlalchemy').setLevel(logging.ERROR)
class Download(Base):
__tablename__ = 'files'
@@ -24,7 +27,8 @@ class Download(Base):
class Database:
def __init__(self, db_url="sqlite:///database.db"):
self.engine = create_engine(db_url, echo=True)
self.engine = create_engine(db_url, echo=False)
self.session_factory = sessionmaker(bind=self.engine)
self.Session = scoped_session(self.session_factory) # Thread-safe sessions

View File

@@ -18,8 +18,10 @@ import statekeeper
import os
class ProgressBarApp:
def __init__(self, parent):
def __init__(self, parent, data_backend):
self.data_backend = data_backend
self.parent = parent
self.root = tk.Toplevel(parent)
self.root.title("Dynamic Progress Bars")
@@ -32,19 +34,20 @@ class ProgressBarApp:
self.progress_bars = [] # Store tuples of (progressbar, frame, duration, delete_button)
self.running = True
threading.Thread(target=self.add_progress_bars, daemon=True).start()
self.root.after(0, self.start_tracking_progress_bars)
def add_progress_bars(self):
def start_tracking_progress_bars(self):
already_tracked = set()
while self.running:
self.already_tracked = set()
self.check_for_new_progress_bars()
def check_for_new_progress_bars(self):
downloads = set(statekeeper.get_download())
new = downloads - already_tracked
already_tracked |= set(downloads)
new = downloads - self.already_tracked
self.already_tracked |= downloads
for element in new:
frame = tk.Frame(self.frame)
frame.pack(fill=tk.X, pady=2)
@@ -63,13 +66,19 @@ class ProgressBarApp:
print("Starting tracker for", element.path)
threading.Thread(target=self.fill_progress, args=(progress, element.path, frame, delete_button), daemon=True).start()
time.sleep(2) # Wait before adding a new progress bar
# Schedule the next check in 2 seconds
if self.running:
self.root.after(2000, self.check_for_new_progress_bars)
def fill_progress(self, progress, path, frame, delete_button):
fail_count = 0
same_size_count = 0
prev_precent = 0
while True:
print("Checking download progress..")
if not progress.winfo_exists(): # Check if progress bar still exists
return
@@ -84,13 +93,29 @@ class ProgressBarApp:
continue
print("Percent filled:", percent_filled, path)
if not percent_filled or percent_filled >= 99.9:
if percent_filled >= 99.9:
self.root.after(0, progress.config, { "value" : 100 })
print("Finished", path)
break
else:
self.root.after(0, progress.config, { "value" : percent_filled })
time.sleep(0.5)
# check for stuck downloads #
print("same size count", same_size_count)
if prev_precent == percent_filled:
same_size_count += 1
else:
same_size_count = 0
if same_size_count > 5:
self.root.after(0, delete_button.config, {"state": tk.NORMAL, "text": "Failed - Delete file manually!"})
self.progress_bars.append((progress, frame, path, delete_button))
self.update_delete_all_button()
statekeeper.log_end_download(path)
self.data_backend.local_delete_cache_file(path)
break
prev_precent = percent_filled
# handle finished download #
self.root.after(0, delete_button.config, {"state": tk.NORMAL})
self.progress_bars.append((progress, frame, path, delete_button))
@@ -98,7 +123,7 @@ class ProgressBarApp:
def delete_progress(self, frame):
frame.destroy()
self.progress_bars = [(p, f, d, b) for p, f, d, b in self.progress_bars if f != frame]
self.progress_bars = [(p, f, d) for p, f, d in self.progress_bars if f != frame]
self.update_delete_all_button()
def delete_all_finished(self):

View File

@@ -12,6 +12,7 @@ import threading
import sys
import tkinter
import statekeeper
from tkinter import messagebox
class Software:
@@ -82,8 +83,16 @@ class Software:
'''Extract a cached, downloaded zip to the target location'''
software_path = os.path.join(target, self.title)
if os.path.isdir(software_path):
return # TODO better skip
overwrite = messagebox.askyesno(
"Overwrite Existing Directory",
f"The directory '{software_path}' already exists.\nDo you want to overwrite it?"
)
if not overwrite:
print("Skipping install as instructed by user...")
return
os.makedirs(software_path, exist_ok=True)
with zipfile.ZipFile(cache_src, 'r') as zip_ref:
@@ -99,6 +108,7 @@ class Software:
#self.progress_bar_wrapper.set_text(
# text="Extracting: {:.2f}%".format(count/len(total_count)*100))
except zipfile.error as e:
print(e)
pass # TODO ???
#zip_ref.extractall(software_path)
@@ -137,13 +147,17 @@ class Software:
local_file = self.backend.get(remote_file, self.cache_dir, wait=True)
statekeeper.log_end_download(remote_file)
print("Deciding on installer...")
# execute or unpack #
if local_file.endswith(".exe"):
print("Target is an executable.. running as installer.")
if os.name != "nt" and not os.path.isabs(local_file):
# need abs path for wine #
local_file = os.path.join(os.getcwd(), local_file)
localaction.run_exe(local_file)
elif local_file.endswith(".zip"):
print("Target is a zip.. unpacking first.")
self._extract_to_target(local_file, self.backend.install_dir)
# download & install registry files #

View File

@@ -44,9 +44,9 @@ def _download(url, path):
def log_begin_download(path, local_path, url):
session = db.session()
print("Current path", path)
print("Download path", path)
path_exists = session.query(Download).filter(and_(Download.path==path, Download.finished==False)).first()
if path_exists:
if path_exists and False: # TODO FIX THIS
print("DAFUG", path_exists)
print("WTF", path_exists.path)
raise AssertionError("ERROR: {} is already downloading.".format(path))
@@ -59,6 +59,7 @@ def log_begin_download(path, local_path, url):
def log_end_download(path):
print("Downlod end logged", path)
session = db.session()
obj = session.query(Download).filter(Download.path==path).first()
if not obj:
@@ -99,11 +100,16 @@ def get_percent_filled(path):
session = db.session()
obj = session.query(Download).filter(Download.path==path, Download.finished==False).first()
if not obj:
return 100 # means its finished
size = _bytes_to_mb(os.stat(obj.local_path).st_size)
total_size = get_download_size(obj.path)
session.close()
if total_size == 0:
return 0
print("Current filled:", size / total_size * 100)
return size / total_size * 100
def get_download(path=None):