Skip to content

Commit

Permalink
feat: add StorageHealthCheck
Browse files Browse the repository at this point in the history
  • Loading branch information
hartungstenio committed Oct 11, 2024
1 parent b4d4e42 commit 7341484
Show file tree
Hide file tree
Showing 2 changed files with 102 additions and 0 deletions.
38 changes: 38 additions & 0 deletions django_healthy/health_checks/storage.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,38 @@
from __future__ import annotations

from io import StringIO

from asgiref.sync import sync_to_async
from django.core.files.storage import Storage, storages
from django.utils.crypto import get_random_string

from .base import HealthCheck, HealthCheckResult


class StorageHealthCheck(HealthCheck):
__slots__: tuple[str, ...] = ("alias", "filename")

def __init__(self, alias: str = "default", filename: str = "healthy_test_file.txt"):
self.alias = alias
self.filename = filename

async def check_health(self) -> HealthCheckResult:
storage: Storage = storages[self.alias]
given: str = get_random_string(100)

try:
filename = await sync_to_async(storage.get_available_name)(self.filename)
await sync_to_async(storage.save)(filename, StringIO(given))

exists = await sync_to_async(storage.exists)(filename)
if not exists:
return HealthCheckResult.degraded(description="File missing", data={"filename": filename})

await sync_to_async(storage.delete)(filename)
exists = await sync_to_async(storage.exists)(filename)
if exists:
return HealthCheckResult.degraded(description="Could not delete file", data={"filename": filename})
except Exception as exc: # noqa: BLE001
return HealthCheckResult.unhealthy(exception=exc)
else:
return HealthCheckResult.healthy()
64 changes: 64 additions & 0 deletions tests/health_checks/test_storage.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,64 @@
from unittest import mock

import pytest
from django.core.files.storage import storages

from django_healthy.health_checks import HealthStatus
from django_healthy.health_checks.storage import StorageHealthCheck


@pytest.mark.asyncio
class TestStorageHealthCheck:
async def test_with_working_storage(self):
health_check = StorageHealthCheck()

got = await health_check.check_health()

assert got.status == HealthStatus.HEALTHY

async def test_without_saving(self):
health_check = StorageHealthCheck()
storage = storages[health_check.alias]

with mock.patch.object(storage, "save"):
got = await health_check.check_health()

assert got.status == HealthStatus.DEGRADED
assert "filename" in got.data

async def test_without_deleting(self):
health_check = StorageHealthCheck()
storage = storages[health_check.alias]

with mock.patch.object(storage, "delete"):
got = await health_check.check_health()

assert got.status == HealthStatus.DEGRADED
assert "filename" in got.data

async def test_with_save_error(self):
health_check = StorageHealthCheck()
storage = storages[health_check.alias]

with mock.patch.object(storage, "save", side_effect=Exception):
got = await health_check.check_health()

assert got.status == HealthStatus.UNHEALTHY

async def test_with_delete_error(self):
health_check = StorageHealthCheck()
storage = storages[health_check.alias]

with mock.patch.object(storage, "delete", side_effect=Exception):
got = await health_check.check_health()

assert got.status == HealthStatus.UNHEALTHY

async def test_with_exists_error(self):
health_check = StorageHealthCheck()
storage = storages[health_check.alias]

with mock.patch.object(storage, "exists", side_effect=Exception):
got = await health_check.check_health()

assert got.status == HealthStatus.UNHEALTHY

0 comments on commit 7341484

Please sign in to comment.