majuna/app/models/__init__.py

162 lines
5.2 KiB
Python
Raw Permalink Normal View History

2022-06-17 12:42:42 +01:00
import logging
from abc import abstractmethod
from datetime import datetime, timezone
from typing import Any, Dict, List, Optional, Union
2022-04-22 14:01:16 +01:00
from sqlalchemy.orm import Mapped, mapped_column
2022-06-15 11:50:15 +01:00
from app.brm.brn import BRN
2022-04-22 14:01:16 +01:00
from app.extensions import db
from app.models.types import AwareDateTime
2022-04-22 14:01:16 +01:00
2022-05-16 11:44:03 +01:00
class AbstractConfiguration(db.Model): # type: ignore
2022-04-22 14:01:16 +01:00
__abstract__ = True
id: Mapped[int] = mapped_column(db.Integer, primary_key=True)
description: Mapped[str]
added: Mapped[datetime] = mapped_column(AwareDateTime())
updated: Mapped[datetime] = mapped_column(AwareDateTime())
2024-12-06 18:15:47 +00:00
destroyed: Mapped[Optional[datetime]] = mapped_column(
AwareDateTime(), nullable=True
)
2022-04-22 14:01:16 +01:00
@property
@abstractmethod
def brn(self) -> BRN:
raise NotImplementedError()
2022-05-16 11:44:03 +01:00
def destroy(self) -> None:
self.destroyed = datetime.now(tz=timezone.utc)
self.updated = datetime.now(tz=timezone.utc)
2022-04-22 14:01:16 +01:00
@classmethod
2022-05-16 11:44:03 +01:00
def csv_header(cls) -> List[str]:
2024-12-06 18:15:47 +00:00
return ["id", "description", "added", "updated", "destroyed"]
2022-04-22 14:01:16 +01:00
2022-05-16 11:44:03 +01:00
def csv_row(self) -> List[Any]:
2024-12-06 18:15:47 +00:00
return [getattr(self, x) for x in self.csv_header()]
2022-04-22 14:01:16 +01:00
2023-10-29 15:45:10 +00:00
class Deprecation(db.Model): # type: ignore[name-defined,misc]
id: Mapped[int] = mapped_column(db.Integer, primary_key=True)
resource_type: Mapped[str]
resource_id: Mapped[int]
deprecated_at: Mapped[datetime] = mapped_column(AwareDateTime())
meta: Mapped[Optional[Dict[str, Any]]] = mapped_column(db.JSON())
reason: Mapped[str]
2023-10-29 15:45:10 +00:00
@property
def resource(self) -> "AbstractResource":
from app.models.mirrors import Proxy # pylint: disable=R0401
2024-12-06 18:15:47 +00:00
model = {"Proxy": Proxy}[self.resource_type]
2023-10-29 15:45:10 +00:00
return model.query.get(self.resource_id) # type: ignore[no-any-return]
2022-05-16 11:44:03 +01:00
class AbstractResource(db.Model): # type: ignore
2022-04-22 14:01:16 +01:00
__abstract__ = True
id: Mapped[int] = mapped_column(db.Integer, primary_key=True)
added: Mapped[datetime] = mapped_column(AwareDateTime())
updated: Mapped[datetime] = mapped_column(AwareDateTime())
2024-12-06 18:15:47 +00:00
deprecated: Mapped[Optional[datetime]] = mapped_column(
AwareDateTime(), nullable=True
)
2024-11-16 13:17:39 +00:00
deprecation_reason: Mapped[Optional[str]]
2024-12-06 18:15:47 +00:00
destroyed: Mapped[Optional[datetime]] = mapped_column(
AwareDateTime(), nullable=True
)
def __init__(
self,
*,
id: Optional[int] = None,
added: Optional[datetime] = None,
updated: Optional[datetime] = None,
deprecated: Optional[datetime] = None,
deprecation_reason: Optional[str] = None,
destroyed: Optional[datetime] = None,
**kwargs: Any
) -> None:
if added is None:
added = datetime.now(tz=timezone.utc)
if updated is None:
updated = datetime.now(tz=timezone.utc)
2024-12-06 18:15:47 +00:00
super().__init__(
id=id,
added=added,
updated=updated,
deprecated=deprecated,
deprecation_reason=deprecation_reason,
destroyed=destroyed,
**kwargs
)
@property
@abstractmethod
2022-06-15 11:50:15 +01:00
def brn(self) -> BRN:
raise NotImplementedError()
2023-10-29 15:45:10 +00:00
def deprecate(self, *, reason: str, meta: Optional[Dict[str, Any]] = None) -> bool:
"""
Marks the resource as deprecated. In the event that the resource was already
deprecated, no change will be recorded and the function will return False.
:param reason: an opaque string that records the deprecation reason
2023-10-29 15:45:10 +00:00
:param meta: metadata associated with the deprecation reason, such as the circumstances in which censorship was
detected
:return: if the proxy was deprecated
"""
2022-06-18 12:48:53 +01:00
if self.deprecated is None:
logging.info("Deprecating %s (reason=%s)", self.brn, reason)
self.deprecated = datetime.now(tz=timezone.utc)
2022-06-17 12:42:42 +01:00
self.deprecation_reason = reason
self.updated = datetime.now(tz=timezone.utc)
2023-10-29 15:45:10 +00:00
if reason not in [d.reason for d in self.deprecations]:
new_deprecation = Deprecation(
resource_type=type(self).__name__,
resource_id=self.id,
reason=reason,
2024-12-06 18:15:47 +00:00
meta=meta,
2023-10-29 15:45:10 +00:00
)
db.session.add(new_deprecation)
return True
2024-12-06 18:15:47 +00:00
logging.info(
"Not deprecating %s (reason=%s) because it's already deprecated with that reason.",
self.brn,
reason,
)
return False
2022-04-22 14:01:16 +01:00
2023-10-29 15:45:10 +00:00
@property
def deprecations(self) -> List[Deprecation]:
return Deprecation.query.filter_by( # type: ignore[no-any-return]
2024-12-06 18:15:47 +00:00
resource_type="Proxy", resource_id=self.id
2023-10-29 15:45:10 +00:00
).all()
def destroy(self) -> None:
"""
Marks the resource for destruction.
:return: None
"""
2022-04-22 14:01:16 +01:00
if self.deprecated is None:
self.deprecate(reason="destroyed")
self.destroyed = datetime.now(tz=timezone.utc)
self.updated = datetime.now(tz=timezone.utc)
2022-04-22 14:01:16 +01:00
@classmethod
def csv_header(cls) -> List[str]:
2022-04-22 14:01:16 +01:00
return [
2024-12-06 18:15:47 +00:00
"id",
"added",
"updated",
"deprecated",
"deprecation_reason",
"destroyed",
2022-04-22 14:01:16 +01:00
]
def csv_row(self) -> List[Union[datetime, bool, int, str]]:
2024-12-06 18:15:47 +00:00
return [getattr(self, x) for x in self.csv_header()]