2023-07-15 09:20:43 +03:00
|
|
|
import json
|
2023-08-10 13:17:45 +03:00
|
|
|
import os
|
2023-07-15 09:20:43 +03:00
|
|
|
import os.path
|
2023-07-16 09:25:32 +03:00
|
|
|
import threading
|
2023-07-17 09:29:36 +03:00
|
|
|
import time
|
2023-07-15 09:20:43 +03:00
|
|
|
|
|
|
|
from modules.paths import data_path, script_path
|
|
|
|
|
2023-08-10 13:17:45 +03:00
|
|
|
cache_filename = os.environ.get('SD_WEBUI_CACHE_FILE', os.path.join(data_path, "cache.json"))
|
2023-07-15 09:20:43 +03:00
|
|
|
cache_data = None
|
2023-07-16 09:25:32 +03:00
|
|
|
cache_lock = threading.Lock()
|
2023-07-15 09:20:43 +03:00
|
|
|
|
2023-07-17 09:29:36 +03:00
|
|
|
dump_cache_after = None
|
|
|
|
dump_cache_thread = None
|
|
|
|
|
2023-07-15 09:20:43 +03:00
|
|
|
|
|
|
|
def dump_cache():
|
|
|
|
"""
|
2023-07-17 09:29:36 +03:00
|
|
|
Marks cache for writing to disk. 5 seconds after no one else flags the cache for writing, it is written.
|
2023-07-15 09:20:43 +03:00
|
|
|
"""
|
|
|
|
|
2023-07-17 09:29:36 +03:00
|
|
|
global dump_cache_after
|
|
|
|
global dump_cache_thread
|
|
|
|
|
|
|
|
def thread_func():
|
|
|
|
global dump_cache_after
|
|
|
|
global dump_cache_thread
|
|
|
|
|
|
|
|
while dump_cache_after is not None and time.time() < dump_cache_after:
|
|
|
|
time.sleep(1)
|
|
|
|
|
|
|
|
with cache_lock:
|
|
|
|
with open(cache_filename, "w", encoding="utf8") as file:
|
|
|
|
json.dump(cache_data, file, indent=4)
|
|
|
|
|
|
|
|
dump_cache_after = None
|
|
|
|
dump_cache_thread = None
|
|
|
|
|
2023-07-16 09:25:32 +03:00
|
|
|
with cache_lock:
|
2023-07-17 09:29:36 +03:00
|
|
|
dump_cache_after = time.time() + 5
|
|
|
|
if dump_cache_thread is None:
|
|
|
|
dump_cache_thread = threading.Thread(name='cache-writer', target=thread_func)
|
|
|
|
dump_cache_thread.start()
|
2023-07-15 09:20:43 +03:00
|
|
|
|
|
|
|
|
|
|
|
def cache(subsection):
|
|
|
|
"""
|
|
|
|
Retrieves or initializes a cache for a specific subsection.
|
|
|
|
|
|
|
|
Parameters:
|
|
|
|
subsection (str): The subsection identifier for the cache.
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
dict: The cache data for the specified subsection.
|
|
|
|
"""
|
|
|
|
|
|
|
|
global cache_data
|
|
|
|
|
|
|
|
if cache_data is None:
|
2023-07-16 09:25:32 +03:00
|
|
|
with cache_lock:
|
|
|
|
if cache_data is None:
|
|
|
|
if not os.path.isfile(cache_filename):
|
2023-07-15 09:20:43 +03:00
|
|
|
cache_data = {}
|
2023-07-16 09:25:32 +03:00
|
|
|
else:
|
|
|
|
try:
|
|
|
|
with open(cache_filename, "r", encoding="utf8") as file:
|
|
|
|
cache_data = json.load(file)
|
|
|
|
except Exception:
|
|
|
|
os.replace(cache_filename, os.path.join(script_path, "tmp", "cache.json"))
|
|
|
|
print('[ERROR] issue occurred while trying to read cache.json, move current cache to tmp/cache.json and create new cache')
|
|
|
|
cache_data = {}
|
2023-07-15 09:20:43 +03:00
|
|
|
|
|
|
|
s = cache_data.get(subsection, {})
|
|
|
|
cache_data[subsection] = s
|
|
|
|
|
|
|
|
return s
|
|
|
|
|
|
|
|
|
|
|
|
def cached_data_for_file(subsection, title, filename, func):
|
|
|
|
"""
|
|
|
|
Retrieves or generates data for a specific file, using a caching mechanism.
|
|
|
|
|
|
|
|
Parameters:
|
|
|
|
subsection (str): The subsection of the cache to use.
|
|
|
|
title (str): The title of the data entry in the subsection of the cache.
|
|
|
|
filename (str): The path to the file to be checked for modifications.
|
|
|
|
func (callable): A function that generates the data if it is not available in the cache.
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
dict or None: The cached or generated data, or None if data generation fails.
|
|
|
|
|
|
|
|
The `cached_data_for_file` function implements a caching mechanism for data stored in files.
|
|
|
|
It checks if the data associated with the given `title` is present in the cache and compares the
|
|
|
|
modification time of the file with the cached modification time. If the file has been modified,
|
|
|
|
the cache is considered invalid and the data is regenerated using the provided `func`.
|
|
|
|
Otherwise, the cached data is returned.
|
|
|
|
|
|
|
|
If the data generation fails, None is returned to indicate the failure. Otherwise, the generated
|
|
|
|
or cached data is returned as a dictionary.
|
|
|
|
"""
|
|
|
|
|
|
|
|
existing_cache = cache(subsection)
|
|
|
|
ondisk_mtime = os.path.getmtime(filename)
|
|
|
|
|
|
|
|
entry = existing_cache.get(title)
|
|
|
|
if entry:
|
2023-07-16 09:49:22 +03:00
|
|
|
cached_mtime = entry.get("mtime", 0)
|
2023-07-15 09:20:43 +03:00
|
|
|
if ondisk_mtime > cached_mtime:
|
|
|
|
entry = None
|
|
|
|
|
2023-07-16 13:46:33 +03:00
|
|
|
if not entry or 'value' not in entry:
|
2023-07-16 09:49:22 +03:00
|
|
|
value = func()
|
|
|
|
if value is None:
|
2023-07-15 09:20:43 +03:00
|
|
|
return None
|
|
|
|
|
2023-07-16 09:49:22 +03:00
|
|
|
entry = {'mtime': ondisk_mtime, 'value': value}
|
2023-07-15 09:20:43 +03:00
|
|
|
existing_cache[title] = entry
|
|
|
|
|
|
|
|
dump_cache()
|
|
|
|
|
2023-07-16 09:49:22 +03:00
|
|
|
return entry['value']
|