From 4edcfe620e5c959230e36d20f6251baef77cdb41 Mon Sep 17 00:00:00 2001 From: LarryMartell Date: Tue, 18 Jul 2023 08:06:33 -0700 Subject: [PATCH 1/5] Delete old thumbnails from the media/cache directory and the database --- docs/management.rst | 11 +++++++ docs/reference/settings.rst | 9 +++++ sorl/thumbnail/conf/defaults.py | 1 + sorl/thumbnail/kvstores/base.py | 30 +++++++++++++++-- .../management/commands/thumbnail.py | 33 ++++++++++++++++++- 5 files changed, 81 insertions(+), 3 deletions(-) diff --git a/docs/management.rst b/docs/management.rst index 26fe4e718..74cb7ab17 100644 --- a/docs/management.rst +++ b/docs/management.rst @@ -29,6 +29,17 @@ useful if your Key Value Store has garbage data not dealt with by cleanup or you're switching Key Value Store backend. +.. _thumbnail-cleanup-delete-timeout: + +thumbnail cleanup_delete_timeout +================================= +``python manage.py thumbnail cleanup_delete_timeout`` + +Deletes thumbnails in the cache, kvstore (database) and storage (filesystem), +if the file's created time is before THUMBNAIL_CLEANUP_DELETE_TIMEOUT seconds ago. +No action will be taken if THUMBNAIL_CLEANUP_DELETE_TIMEOUT is as ``None`` + + .. _thumbnail-clear-delete-referenced: thumbnail clear_delete_referenced diff --git a/docs/reference/settings.rst b/docs/reference/settings.rst index 7d34c2f12..2f18732ae 100644 --- a/docs/reference/settings.rst +++ b/docs/reference/settings.rst @@ -236,6 +236,15 @@ at maximum or ``None`` if your caching backend can handle that as infinite. Only applicable for the Cached DB Key Value Store. +``THUMBNAIL_CLEANUP_DELETE_TIMEOUT`` +=========================== + +- Default: ``3600 * 24 * 365 * 10`` + +Timeout to deletes thumbnails in the cache, kvstore (database) and storage (filesystem), +based on file's created time. If set as ``None`` then no action will be taken. + + ``THUMBNAIL_CACHE`` =================== diff --git a/sorl/thumbnail/conf/defaults.py b/sorl/thumbnail/conf/defaults.py index f1eb6855b..7af1b9ea9 100644 --- a/sorl/thumbnail/conf/defaults.py +++ b/sorl/thumbnail/conf/defaults.py @@ -53,6 +53,7 @@ # Cache timeout for ``cached_db`` store. You should probably keep this at # maximum or ``0`` if your caching backend can handle that as infinite. THUMBNAIL_CACHE_TIMEOUT = 3600 * 24 * 365 * 10 # 10 years +THUMBNAIL_CLEANUP_DELETE_TIMEOUT = 3600 * 24 * 365 * 10 # 10 years # The cache configuration to use for storing thumbnail data THUMBNAIL_CACHE = 'default' diff --git a/sorl/thumbnail/kvstores/base.py b/sorl/thumbnail/kvstores/base.py index eef411d71..f5e80eab4 100644 --- a/sorl/thumbnail/kvstores/base.py +++ b/sorl/thumbnail/kvstores/base.py @@ -93,8 +93,22 @@ def cleanup(self): Cleans up the key value store. In detail: 1. Deletes all key store references for image_files that do not exist and all key references for its thumbnails *and* their image_files. - 2. Deletes or updates all invalid thumbnail keys + 2. Deletes or updates all invalid thumbnail keys. """ + self._cleanup() + + def cleanup_and_delete_if_created_time_before_dt(self, dt): + """ + Cleans up the key value store. In detail: + 1. Deletes all key store references for image_files that: + - do not exist, or + - created time before ``dt`` + and all key references for its thumbnails *and* their image_files. + 2. Deletes or updates all invalid thumbnail keys. + """ + self._cleanup(delete_if_created_time_before_dt=dt) + + def _cleanup(self, delete_if_created_time_before_dt=None): for key in self._find_keys(identity='image'): image_file = self._get(key) @@ -113,8 +127,20 @@ def cleanup(self): thumbnail_keys_set = set(thumbnail_keys) for thumbnail_key in thumbnail_keys: - if not self._get(thumbnail_key): + thumbnail = self._get(thumbnail_key) + if not thumbnail: thumbnail_keys_set.remove(thumbnail_key) + else: + if delete_if_created_time_before_dt is not None: + try: + created_time = thumbnail.storage.get_created_time(thumbnail.name) + except NotImplementedError: + pass + else: + if created_time < delete_if_created_time_before_dt: + thumbnail_keys_set.remove(thumbnail_key) + self.delete(thumbnail, False) + thumbnail.delete() # delete the actual file thumbnail_keys = list(thumbnail_keys_set) diff --git a/sorl/thumbnail/management/commands/thumbnail.py b/sorl/thumbnail/management/commands/thumbnail.py index f7b7468a2..26906dc05 100644 --- a/sorl/thumbnail/management/commands/thumbnail.py +++ b/sorl/thumbnail/management/commands/thumbnail.py @@ -1,10 +1,20 @@ +from datetime import timedelta + from django.core.management.base import BaseCommand +from django.utils import timezone from sorl.thumbnail import default +from sorl.thumbnail.conf import settings from sorl.thumbnail.images import delete_all_thumbnails -VALID_LABELS = ['cleanup', 'clear', 'clear_delete_referenced', 'clear_delete_all'] +VALID_LABELS = [ + 'cleanup', + 'cleanup_delete_timeout', + 'clear', + 'clear_delete_referenced', + 'clear_delete_all', +] class Command(BaseCommand): @@ -31,6 +41,27 @@ def handle(self, *labels, **options): return + if label == 'cleanup_delete_timeout': + if not settings.THUMBNAIL_CLEANUP_DELETE_TIMEOUT: + self.stdout.write( + "THUMBNAIL_CLEANUP_DELETE_TIMEOUT is empty. No action taken", + ending=' ... ' + ) + return + if verbosity >= 1: + self.stdout.write( + "Cleanup thumbnails and delete if created time before THUMBNAIL_CLEANUP_DELETE_TIMEOUT seconds ago", + ending=' ... ' + ) + + thumbnail_cache_timeout_dt = timezone.now() - timedelta(seconds=settings.THUMBNAIL_CLEANUP_DELETE_TIMEOUT) + default.kvstore.cleanup_and_delete_if_created_time_before_dt(thumbnail_cache_timeout_dt) + + if verbosity >= 1: + self.stdout.write('[Done]') + + return + if label == 'clear_delete_referenced': if verbosity >= 1: self.stdout.write( From 84ff14a419b4179cf3b3ec05127d98bef05d648b Mon Sep 17 00:00:00 2001 From: LarryMartell Date: Wed, 19 Jul 2023 12:37:50 -0700 Subject: [PATCH 2/5] dummy commit to trigger tests --- sorl/thumbnail/base.py | 1 + 1 file changed, 1 insertion(+) diff --git a/sorl/thumbnail/base.py b/sorl/thumbnail/base.py index 64155afaa..71e0f2e8e 100644 --- a/sorl/thumbnail/base.py +++ b/sorl/thumbnail/base.py @@ -1,3 +1,4 @@ +# dummy commit to triggger tests import logging import os import re From 4227842c051e6d1d8614b498b240d2e2abb8a529 Mon Sep 17 00:00:00 2001 From: LarryMartell Date: Wed, 19 Jul 2023 12:40:44 -0700 Subject: [PATCH 3/5] dummy commit to trigger tests --- sorl/thumbnail/base.py | 1 - 1 file changed, 1 deletion(-) diff --git a/sorl/thumbnail/base.py b/sorl/thumbnail/base.py index 71e0f2e8e..64155afaa 100644 --- a/sorl/thumbnail/base.py +++ b/sorl/thumbnail/base.py @@ -1,4 +1,3 @@ -# dummy commit to triggger tests import logging import os import re From d11b1108191dd0a18897ebac80a5c4a0f3a29702 Mon Sep 17 00:00:00 2001 From: Claude Paroz Date: Sun, 23 Jul 2023 11:42:49 +0200 Subject: [PATCH 4/5] Upgraded GitHub workflow actions --- .github/workflows/test.yml | 14 ++++++++------ 1 file changed, 8 insertions(+), 6 deletions(-) diff --git a/.github/workflows/test.yml b/.github/workflows/test.yml index c289bfd6f..0b4ee512d 100644 --- a/.github/workflows/test.yml +++ b/.github/workflows/test.yml @@ -14,16 +14,18 @@ jobs: - python-version: '3.8' target: 'qa' steps: - - uses: actions/checkout@v2 + - uses: actions/checkout@v3 - name: Start Redis - uses: supercharge/redis-github-action@1.1.0 + uses: supercharge/redis-github-action@1.5.0 - name: Install system dependencies - run: sudo apt-get install libgraphicsmagick1-dev graphicsmagick libjpeg62 zlib1g-dev + run: | + sudo apt-get update + sudo apt-get install libgraphicsmagick1-dev graphicsmagick libjpeg62 zlib1g-dev - name: Set up Python ${{ matrix.python-version }} - uses: actions/setup-python@v2 + uses: actions/setup-python@v4 with: python-version: ${{ matrix.python-version }} @@ -33,7 +35,7 @@ jobs: echo "::set-output name=dir::$(pip cache dir)" - name: Cache - uses: actions/cache@v2 + uses: actions/cache@v3 with: path: ${{ steps.pip-cache.outputs.dir }} key: @@ -53,6 +55,6 @@ jobs: TARGET: ${{ matrix.target }} - name: Upload coverage - uses: codecov/codecov-action@v1 + uses: codecov/codecov-action@v3 with: name: Python ${{ matrix.python-version }} From 337a51f6109533bab8854a5a628d340b593f57d1 Mon Sep 17 00:00:00 2001 From: LarryMartell Date: Mon, 24 Jul 2023 11:15:26 -0700 Subject: [PATCH 5/5] fixes for tests --- .../management/commands/thumbnail.py | 27 ++++++++++++------- 1 file changed, 17 insertions(+), 10 deletions(-) diff --git a/sorl/thumbnail/management/commands/thumbnail.py b/sorl/thumbnail/management/commands/thumbnail.py index 26906dc05..447a9617a 100644 --- a/sorl/thumbnail/management/commands/thumbnail.py +++ b/sorl/thumbnail/management/commands/thumbnail.py @@ -26,6 +26,7 @@ class Command(BaseCommand): def add_arguments(self, parser): parser.add_argument('args', choices=VALID_LABELS, nargs=1) + # flake8: noqa: C901 def handle(self, *labels, **options): verbosity = int(options.get('verbosity')) label = labels[0] @@ -41,20 +42,26 @@ def handle(self, *labels, **options): return - if label == 'cleanup_delete_timeout': - if not settings.THUMBNAIL_CLEANUP_DELETE_TIMEOUT: - self.stdout.write( - "THUMBNAIL_CLEANUP_DELETE_TIMEOUT is empty. No action taken", - ending=' ... ' - ) - return + elif label == 'cleanup_delete_timeout' and not settings.THUMBNAIL_CLEANUP_DELETE_TIMEOUT: + self.stdout.write( + "THUMBNAIL_CLEANUP_DELETE_TIMEOUT is empty. No action taken", + ending=' ... ' + ) + return + + elif label == 'cleanup_delete_timeout': if verbosity >= 1: self.stdout.write( - "Cleanup thumbnails and delete if created time before THUMBNAIL_CLEANUP_DELETE_TIMEOUT seconds ago", + """ + Cleanup thumbnails and delete if created time before + THUMBNAIL_CLEANUP_DELETE_TIMEOUT seconds ago + """, ending=' ... ' ) - thumbnail_cache_timeout_dt = timezone.now() - timedelta(seconds=settings.THUMBNAIL_CLEANUP_DELETE_TIMEOUT) + thumbnail_cache_timeout_dt = timezone.now() - timedelta( + seconds=settings.THUMBNAIL_CLEANUP_DELETE_TIMEOUT + ) default.kvstore.cleanup_and_delete_if_created_time_before_dt(thumbnail_cache_timeout_dt) if verbosity >= 1: @@ -62,7 +69,7 @@ def handle(self, *labels, **options): return - if label == 'clear_delete_referenced': + elif label == 'clear_delete_referenced': if verbosity >= 1: self.stdout.write( "Delete all thumbnail files referenced in Key Value Store",