ФУНКЦИИ - МЕЖПРОЕКТНАЯ КОММУНИКАЦИЯ: API мониторинга хранилища воркспейса
This commit is contained in:
parent
a7606f2e9a
commit
d9f534efcd
|
|
@ -36,6 +36,7 @@ from plane.app.views import (
|
||||||
UserRecentVisitViewSet,
|
UserRecentVisitViewSet,
|
||||||
WorkspaceHomePreferenceViewSet,
|
WorkspaceHomePreferenceViewSet,
|
||||||
WorkspaceStickyViewSet,
|
WorkspaceStickyViewSet,
|
||||||
|
WorkspaceStorageSummaryEndpoint,
|
||||||
WorkspaceUserPreferenceViewSet,
|
WorkspaceUserPreferenceViewSet,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
@ -257,6 +258,11 @@ urlpatterns = [
|
||||||
WorkspaceStickyViewSet.as_view({"get": "retrieve", "patch": "partial_update", "delete": "destroy"}),
|
WorkspaceStickyViewSet.as_view({"get": "retrieve", "patch": "partial_update", "delete": "destroy"}),
|
||||||
name="workspace-sticky",
|
name="workspace-sticky",
|
||||||
),
|
),
|
||||||
|
path(
|
||||||
|
"workspaces/<str:slug>/storage/summary/",
|
||||||
|
WorkspaceStorageSummaryEndpoint.as_view(),
|
||||||
|
name="workspace-storage-summary",
|
||||||
|
),
|
||||||
# User Preference
|
# User Preference
|
||||||
path(
|
path(
|
||||||
"workspaces/<str:slug>/sidebar-preferences/",
|
"workspaces/<str:slug>/sidebar-preferences/",
|
||||||
|
|
|
||||||
|
|
@ -83,6 +83,7 @@ from .workspace.module import WorkspaceModulesEndpoint
|
||||||
from .workspace.cycle import WorkspaceCyclesEndpoint
|
from .workspace.cycle import WorkspaceCyclesEndpoint
|
||||||
from .workspace.quick_link import QuickLinkViewSet
|
from .workspace.quick_link import QuickLinkViewSet
|
||||||
from .workspace.sticky import WorkspaceStickyViewSet
|
from .workspace.sticky import WorkspaceStickyViewSet
|
||||||
|
from .workspace.storage import WorkspaceStorageSummaryEndpoint
|
||||||
|
|
||||||
from .state.base import StateViewSet, IntakeStateEndpoint
|
from .state.base import StateViewSet, IntakeStateEndpoint
|
||||||
from .view.base import (
|
from .view.base import (
|
||||||
|
|
|
||||||
|
|
@ -0,0 +1,121 @@
|
||||||
|
# Copyright (c) 2023-present Plane Software, Inc. and contributors
|
||||||
|
# SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
# See the LICENSE file for details.
|
||||||
|
|
||||||
|
from datetime import timedelta
|
||||||
|
|
||||||
|
from django.db.models import Sum
|
||||||
|
from django.utils import timezone
|
||||||
|
from rest_framework import status
|
||||||
|
from rest_framework.response import Response
|
||||||
|
|
||||||
|
from plane.app.permissions import ROLE, allow_permission
|
||||||
|
from plane.app.views.base import BaseAPIView
|
||||||
|
from plane.db.models import FileAsset, Project, StoredBlob, Workspace
|
||||||
|
|
||||||
|
|
||||||
|
def _int_size(value):
|
||||||
|
return int(value or 0)
|
||||||
|
|
||||||
|
|
||||||
|
def _sum_asset_size(queryset):
|
||||||
|
return _int_size(queryset.aggregate(total=Sum("size")).get("total"))
|
||||||
|
|
||||||
|
|
||||||
|
def _sum_blob_size(blob_ids):
|
||||||
|
if not blob_ids:
|
||||||
|
return 0
|
||||||
|
|
||||||
|
return _int_size(StoredBlob.objects.filter(id__in=blob_ids).aggregate(total=Sum("size")).get("total"))
|
||||||
|
|
||||||
|
|
||||||
|
def _dedup_savings(logical_size, physical_size):
|
||||||
|
return max(_int_size(logical_size) - _int_size(physical_size), 0)
|
||||||
|
|
||||||
|
|
||||||
|
class WorkspaceStorageSummaryEndpoint(BaseAPIView):
|
||||||
|
@allow_permission(allowed_roles=[ROLE.ADMIN], level="WORKSPACE")
|
||||||
|
def get(self, request, slug):
|
||||||
|
workspace = Workspace.objects.get(slug=slug)
|
||||||
|
stale_cutoff = timezone.now() - timedelta(days=1)
|
||||||
|
|
||||||
|
active_assets = FileAsset.objects.filter(workspace=workspace, is_uploaded=True)
|
||||||
|
active_blob_ids = list(active_assets.exclude(blob__isnull=True).values_list("blob_id", flat=True).distinct())
|
||||||
|
|
||||||
|
workspace_logical_size = _sum_asset_size(active_assets)
|
||||||
|
workspace_physical_size = _sum_blob_size(active_blob_ids)
|
||||||
|
|
||||||
|
failed_uploads = FileAsset.all_objects.filter(
|
||||||
|
workspace=workspace,
|
||||||
|
is_uploaded=False,
|
||||||
|
deleted_at__isnull=True,
|
||||||
|
)
|
||||||
|
stale_unuploaded = failed_uploads.filter(created_at__lt=stale_cutoff)
|
||||||
|
soft_deleted_assets = FileAsset.all_objects.filter(workspace=workspace, deleted_at__isnull=False)
|
||||||
|
orphaned_blobs = StoredBlob.objects.filter(workspace=workspace, status=StoredBlob.Status.ORPHANED)
|
||||||
|
missing_blobs = StoredBlob.objects.filter(workspace=workspace, status=StoredBlob.Status.MISSING)
|
||||||
|
uploaded_without_blob = active_assets.filter(blob__isnull=True)
|
||||||
|
|
||||||
|
project_rows = []
|
||||||
|
projects = Project.objects.filter(workspace=workspace).order_by("name")
|
||||||
|
for project in projects:
|
||||||
|
project_assets = active_assets.filter(project=project)
|
||||||
|
project_blob_ids = list(
|
||||||
|
project_assets.exclude(blob__isnull=True).values_list("blob_id", flat=True).distinct()
|
||||||
|
)
|
||||||
|
project_logical_size = _sum_asset_size(project_assets)
|
||||||
|
project_physical_size = _sum_blob_size(project_blob_ids)
|
||||||
|
project_failed_uploads = failed_uploads.filter(project=project)
|
||||||
|
project_soft_deleted = soft_deleted_assets.filter(project=project)
|
||||||
|
project_uploaded_without_blob = uploaded_without_blob.filter(project=project)
|
||||||
|
|
||||||
|
project_rows.append(
|
||||||
|
{
|
||||||
|
"id": str(project.id),
|
||||||
|
"name": project.name,
|
||||||
|
"identifier": project.identifier,
|
||||||
|
"file_count": project_assets.count(),
|
||||||
|
"blob_count": len(project_blob_ids),
|
||||||
|
"logical_size": project_logical_size,
|
||||||
|
"physical_size": project_physical_size,
|
||||||
|
"dedup_savings": _dedup_savings(project_logical_size, project_physical_size),
|
||||||
|
"failed_upload_count": project_failed_uploads.count(),
|
||||||
|
"failed_upload_size": _sum_asset_size(project_failed_uploads),
|
||||||
|
"soft_deleted_count": project_soft_deleted.count(),
|
||||||
|
"soft_deleted_size": _sum_asset_size(project_soft_deleted),
|
||||||
|
"uploaded_without_blob_count": project_uploaded_without_blob.count(),
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
data = {
|
||||||
|
"workspace": {
|
||||||
|
"id": str(workspace.id),
|
||||||
|
"name": workspace.name,
|
||||||
|
"slug": workspace.slug,
|
||||||
|
"upload_file_size_limit_enabled": workspace.storage_file_size_limit_enabled,
|
||||||
|
"upload_file_size_limit": workspace.storage_file_size_limit,
|
||||||
|
},
|
||||||
|
"summary": {
|
||||||
|
"file_count": active_assets.count(),
|
||||||
|
"blob_count": len(active_blob_ids),
|
||||||
|
"logical_size": workspace_logical_size,
|
||||||
|
"physical_size": workspace_physical_size,
|
||||||
|
"dedup_savings": _dedup_savings(workspace_logical_size, workspace_physical_size),
|
||||||
|
"uploaded_without_blob_count": uploaded_without_blob.count(),
|
||||||
|
},
|
||||||
|
"diagnostics": {
|
||||||
|
"failed_upload_count": failed_uploads.count(),
|
||||||
|
"failed_upload_size": _sum_asset_size(failed_uploads),
|
||||||
|
"stale_unuploaded_count": stale_unuploaded.count(),
|
||||||
|
"stale_unuploaded_size": _sum_asset_size(stale_unuploaded),
|
||||||
|
"soft_deleted_count": soft_deleted_assets.count(),
|
||||||
|
"soft_deleted_size": _sum_asset_size(soft_deleted_assets),
|
||||||
|
"orphaned_blob_count": orphaned_blobs.count(),
|
||||||
|
"orphaned_blob_size": _sum_blob_size(list(orphaned_blobs.values_list("id", flat=True))),
|
||||||
|
"missing_blob_count": missing_blobs.count(),
|
||||||
|
"missing_blob_size": _sum_blob_size(list(missing_blobs.values_list("id", flat=True))),
|
||||||
|
},
|
||||||
|
"projects": project_rows,
|
||||||
|
}
|
||||||
|
|
||||||
|
return Response(data, status=status.HTTP_200_OK)
|
||||||
Loading…
Reference in New Issue