ФУНКЦИИ - МЕЖПРОЕКТНАЯ КОММУНИКАЦИЯ: retention cleanup файлового хранилища

This commit is contained in:
DCCONSTRUCTIONS 2026-04-27 13:47:46 +03:00
parent 3e328531ec
commit a7606f2e9a
3 changed files with 81 additions and 2 deletions

View File

@ -112,6 +112,7 @@ def restore_related_objects(app_label, model_name, instance_pk, using=None):
@shared_task
def hard_delete():
from plane.bgtasks.file_asset_task import delete_expired_file_asset
from plane.db.models import (
Workspace,
Project,
@ -134,6 +135,9 @@ def hard_delete():
)
days = settings.HARD_DELETE_AFTER_DAYS
delete_expired_file_asset()
# check delete workspace
_ = Workspace.all_objects.filter(deleted_at__lt=timezone.now() - timezone.timedelta(days=days)).delete()
@ -185,6 +189,9 @@ def hard_delete():
# Iterate through all models
for model in all_models:
if model._meta.label_lower in {"db.fileasset", "db.storedblob"}:
continue
# Check if the model has a 'deleted_at' field
if hasattr(model, "deleted_at"):
# Get all instances where 'deleted_at' is greater than 30 days ago

View File

@ -7,6 +7,7 @@ import os
from datetime import timedelta
# Django imports
from django.conf import settings
from django.utils import timezone
from django.db.models import Q
@ -14,10 +15,44 @@ from django.db.models import Q
from celery import shared_task
# Module imports
from plane.db.models import FileAsset
from plane.db.models import FileAsset, StoredBlob
from plane.settings.storage import S3Storage
from plane.utils.file_dedup import release_file_asset_blob
def _asset_object_key(asset):
return str(asset.asset.name or asset.asset)
def _delete_legacy_asset_object(asset, storage):
object_key = _asset_object_key(asset)
if not object_key:
return False
has_other_reference = FileAsset.all_objects.filter(asset=object_key).exclude(pk=asset.pk).exists()
if has_other_reference:
return False
return storage.delete_files(object_names=[object_key])
def _hard_delete_file_asset(asset, storage=None):
blob_id = asset.blob_id
if blob_id:
release_file_asset_blob(asset)
StoredBlob.all_objects.filter(
pk=blob_id,
ref_count=0,
status=StoredBlob.Status.ORPHANED,
).delete()
else:
storage = storage or S3Storage()
_delete_legacy_asset_object(asset, storage)
asset.delete(soft=False)
@shared_task
def delete_unuploaded_file_asset():
"""This task deletes unuploaded file assets older than a certain number of days."""
@ -27,4 +62,16 @@ def delete_unuploaded_file_asset():
)
for asset in stale_assets.iterator():
release_file_asset_blob(asset, delete_untracked_object=True)
asset.delete()
asset.delete(soft=False)
@shared_task
def delete_expired_file_asset():
"""Hard delete soft-deleted file assets after the restore retention window."""
days = int(os.environ.get("FILE_ASSET_HARD_DELETE_AFTER_DAYS", settings.HARD_DELETE_AFTER_DAYS))
cutoff = timezone.now() - timedelta(days=days)
storage = S3Storage()
expired_assets = FileAsset.all_objects.filter(deleted_at__lt=cutoff)
for asset in expired_assets.iterator():
_hard_delete_file_asset(asset, storage=storage)

View File

@ -5,8 +5,10 @@
from unittest.mock import patch
import pytest
from django.utils import timezone
from plane.db.models import FileAsset, StoredBlob
from plane.bgtasks.file_asset_task import delete_expired_file_asset
from plane.utils.file_dedup import attach_existing_blob_to_file_asset, finalize_uploaded_file_asset, release_file_asset_blob
@ -150,3 +152,26 @@ def test_attach_existing_blob_reuses_canonical_object_without_copy(workspace, pr
assert target_asset.asset.name == "workspace/a.txt"
assert blob.ref_count == 2
assert fake_storage.deleted == []
@pytest.mark.django_db
def test_delete_expired_file_asset_releases_blob_and_hard_deletes_rows(workspace, project, fake_storage):
first_asset = create_asset(workspace, project, "workspace/a.txt")
duplicate_asset = create_asset(workspace, project, "workspace/b.txt")
with patch("plane.utils.file_dedup.S3Storage", return_value=fake_storage):
finalize_uploaded_file_asset(first_asset)
finalize_uploaded_file_asset(duplicate_asset)
deleted_at = timezone.now() - timezone.timedelta(days=90)
FileAsset.all_objects.filter(pk__in=[first_asset.pk, duplicate_asset.pk]).update(deleted_at=deleted_at)
with (
patch("plane.utils.file_dedup.S3Storage", return_value=fake_storage),
patch("plane.bgtasks.file_asset_task.S3Storage", return_value=fake_storage),
):
delete_expired_file_asset()
assert FileAsset.all_objects.filter(pk__in=[first_asset.pk, duplicate_asset.pk]).count() == 0
assert StoredBlob.all_objects.count() == 0
assert fake_storage.deleted == ["workspace/b.txt", "workspace/a.txt"]