From a9257a6bf521e455c2be42362e7fcb565a128587 Mon Sep 17 00:00:00 2001 From: Brett Williams Date: Sat, 3 Jun 2023 16:59:01 -0500 Subject: [PATCH] Improve performance of job status updates through caching hashes --- lib/server/job_server.py | 25 ++++++++++++++++--------- lib/server/server_proxy.py | 30 ++++++++++++++++++++++++------ lib/server/static/js/job_table.js | 2 +- lib/utilities/server_helper.py | 5 ++++- 4 files changed, 45 insertions(+), 17 deletions(-) diff --git a/lib/server/job_server.py b/lib/server/job_server.py index 4c9d944..6f7238a 100755 --- a/lib/server/job_server.py +++ b/lib/server/job_server.py @@ -55,6 +55,22 @@ def index(): render_clients=render_clients(), preset_list=presets) +@server.get('/api/jobs') +def jobs_json(): + try: + hash_token = request.args.get('token', None) + all_jobs = [x.json() for x in RenderQueue.all_jobs()] + job_cache_token = str(json.dumps(all_jobs).__hash__()) + + if hash_token and hash_token == job_cache_token: + return [], 204 # no need to update + else: + return {'jobs': all_jobs, 'token': job_cache_token} + except Exception as e: + logger.exception(f"Exception fetching all_jobs_cached: {e}") + return [], 500 + + @server.route('/ui/job//full_details') def job_detail(job_id): found_job = RenderQueue.job_with_id(job_id) @@ -131,15 +147,6 @@ def get_job_file(job_id, filename): abort(404) -@server.get('/api/jobs') -def jobs_json(): - try: - return [x.json() for x in RenderQueue.all_jobs()] - except Exception as e: - logger.exception(f"Exception fetching all_jobs: {e}") - return [], 500 - - @server.get('/api/jobs/') def filtered_jobs_json(status_val): state = string_to_status(status_val) diff --git a/lib/server/server_proxy.py b/lib/server/server_proxy.py index 667176a..ff29098 100644 --- a/lib/server/server_proxy.py +++ b/lib/server/server_proxy.py @@ -1,3 +1,4 @@ +import logging import os import json import requests @@ -10,13 +11,25 @@ status_colors = {RenderStatus.ERROR: "red", RenderStatus.CANCELLED: 'orange1', R categories = [RenderStatus.RUNNING, RenderStatus.ERROR, RenderStatus.NOT_STARTED, RenderStatus.SCHEDULED, RenderStatus.COMPLETED, RenderStatus.CANCELLED, RenderStatus.UNDEFINED] +logger = logging.getLogger() + class RenderServerProxy: def __init__(self, hostname=None, server_port="8080"): - self.hostname = hostname + self._hostname = hostname self.port = server_port self.fetched_status_data = None + self.__jobs_cache_token = None + + @property + def hostname(self): + return self._hostname + + @hostname.setter + def hostname(self, value): + self._hostname = value + self.__jobs_cache_token = None def connect(self): status = self.request_data('status') @@ -25,24 +38,29 @@ class RenderServerProxy: def request_data(self, payload, timeout=5): try: req = self.request(payload, timeout) - if req.ok: + if req.ok and req.status_code == 200: return req.json() + except requests.ConnectionError as e: + logger.error(f"Connection error: {e}") except Exception as e: - pass + logger.exception(f"Uncaught exception: {e}") return None def request(self, payload, timeout=5): return requests.get(f'http://{self.hostname}:{self.port}/api/{payload}', timeout=timeout) def get_jobs(self, timeout=5): - all_jobs = self.request_data('jobs', timeout=timeout) - if all_jobs is not None: + url = f'jobs?token={self.__jobs_cache_token}' if self.__jobs_cache_token else 'jobs' + status_result = self.request_data(url, timeout=timeout) + all_jobs = None + if status_result is not None: sorted_jobs = [] for status_category in categories: - found_jobs = [x for x in all_jobs if x['status'] == status_category.value] + found_jobs = [x for x in status_result['jobs'] if x['status'] == status_category.value] if found_jobs: sorted_jobs.extend(found_jobs) all_jobs = sorted_jobs + self.__jobs_cache_token = status_result['token'] return all_jobs def get_data(self, timeout=5): diff --git a/lib/server/static/js/job_table.js b/lib/server/static/js/job_table.js index a2b93d9..12b3ca4 100644 --- a/lib/server/static/js/job_table.js +++ b/lib/server/static/js/job_table.js @@ -58,7 +58,7 @@ columns: [ autoWidth: true, server: { url: '/api/jobs', - then: results => results, + then: results => results['jobs'], }, sort: true, }).render(document.getElementById('table')); \ No newline at end of file diff --git a/lib/utilities/server_helper.py b/lib/utilities/server_helper.py index 6db4e99..42f048e 100644 --- a/lib/utilities/server_helper.py +++ b/lib/utilities/server_helper.py @@ -32,8 +32,11 @@ def generate_thumbnail_for_job(job, thumb_video_path, thumb_image_path, max_widt generate_thumbnail(source_path=source, dest_path=thumb_video_path, max_width=max_width) except Exception as e: logger.error(f"Error generating thumbnail for {source}: {e}") - finally: + + try: os.remove(in_progress_path) + except FileNotFoundError: + pass # Determine best source file to use for thumbs if job.status == RenderStatus.COMPLETED: # use finished file for thumb