mirror of
https://github.com/yyhuni/xingrin.git
synced 2026-01-31 19:53:11 +08:00
133 lines
4.3 KiB
Python
133 lines
4.3 KiB
Python
"""通知系统数据模型"""
|
||
|
||
import logging
|
||
from datetime import timedelta
|
||
|
||
from django.db import models
|
||
from django.utils import timezone
|
||
|
||
from .types import NotificationCategory, NotificationLevel
|
||
|
||
logger = logging.getLogger(__name__)
|
||
|
||
|
||
class NotificationSettings(models.Model):
|
||
"""
|
||
通知设置(单例模型)
|
||
存储 Discord webhook 配置和各分类的通知开关
|
||
"""
|
||
|
||
# Discord 配置
|
||
discord_enabled = models.BooleanField(default=False, help_text='是否启用 Discord 通知')
|
||
discord_webhook_url = models.URLField(blank=True, default='', help_text='Discord Webhook URL')
|
||
|
||
# 企业微信配置
|
||
wecom_enabled = models.BooleanField(default=False, help_text='是否启用企业微信通知')
|
||
wecom_webhook_url = models.URLField(blank=True, default='', help_text='企业微信机器人 Webhook URL')
|
||
|
||
# 分类开关(使用 JSONField 存储)
|
||
categories = models.JSONField(
|
||
default=dict,
|
||
help_text='各分类通知开关,如 {"scan": true, "vulnerability": true, "asset": true, "system": false}'
|
||
)
|
||
|
||
# 时间信息
|
||
created_at = models.DateTimeField(auto_now_add=True)
|
||
updated_at = models.DateTimeField(auto_now=True)
|
||
|
||
class Meta:
|
||
db_table = 'notification_settings'
|
||
verbose_name = '通知设置'
|
||
verbose_name_plural = '通知设置'
|
||
|
||
def save(self, *args, **kwargs):
|
||
self.pk = 1 # 单例模式
|
||
super().save(*args, **kwargs)
|
||
|
||
@classmethod
|
||
def get_instance(cls) -> 'NotificationSettings':
|
||
"""获取或创建单例实例"""
|
||
obj, _ = cls.objects.get_or_create(
|
||
pk=1,
|
||
defaults={
|
||
'discord_enabled': False,
|
||
'discord_webhook_url': '',
|
||
'categories': {
|
||
'scan': True,
|
||
'vulnerability': True,
|
||
'asset': True,
|
||
'system': False,
|
||
}
|
||
}
|
||
)
|
||
return obj
|
||
|
||
def is_category_enabled(self, category: str) -> bool:
|
||
"""检查指定分类是否启用通知"""
|
||
return self.categories.get(category, False)
|
||
|
||
|
||
class Notification(models.Model):
|
||
"""通知模型"""
|
||
|
||
id = models.AutoField(primary_key=True)
|
||
|
||
category = models.CharField(
|
||
max_length=20,
|
||
choices=NotificationCategory.choices,
|
||
default=NotificationCategory.SYSTEM,
|
||
db_index=True,
|
||
help_text='通知分类'
|
||
)
|
||
|
||
level = models.CharField(
|
||
max_length=20,
|
||
choices=NotificationLevel.choices,
|
||
default=NotificationLevel.LOW,
|
||
db_index=True,
|
||
help_text='通知级别'
|
||
)
|
||
|
||
title = models.CharField(max_length=200, help_text='通知标题')
|
||
message = models.CharField(max_length=2000, help_text='通知内容')
|
||
|
||
created_at = models.DateTimeField(auto_now_add=True, help_text='创建时间')
|
||
|
||
is_read = models.BooleanField(default=False, help_text='是否已读')
|
||
read_at = models.DateTimeField(null=True, blank=True, help_text='阅读时间')
|
||
|
||
class Meta:
|
||
db_table = 'notification'
|
||
verbose_name = '通知'
|
||
verbose_name_plural = '通知'
|
||
ordering = ['-created_at']
|
||
indexes = [
|
||
models.Index(fields=['-created_at']),
|
||
models.Index(fields=['category', '-created_at']),
|
||
models.Index(fields=['level', '-created_at']),
|
||
models.Index(fields=['is_read', '-created_at']),
|
||
]
|
||
|
||
def __str__(self):
|
||
return f"{self.get_level_display()} - {self.title}"
|
||
|
||
@classmethod
|
||
def cleanup_old_notifications(cls) -> int:
|
||
"""清理超过15天的旧通知"""
|
||
cutoff_date = timezone.now() - timedelta(days=15)
|
||
deleted_count, _ = cls.objects.filter(created_at__lt=cutoff_date).delete()
|
||
return deleted_count or 0
|
||
|
||
def save(self, *args, **kwargs):
|
||
"""重写save方法,在创建新通知时自动清理旧通知"""
|
||
is_new = self.pk is None
|
||
super().save(*args, **kwargs)
|
||
|
||
if is_new:
|
||
try:
|
||
deleted_count = self.__class__.cleanup_old_notifications()
|
||
if deleted_count > 0:
|
||
logger.info("自动清理了 %d 条超过15天的旧通知", deleted_count)
|
||
except Exception:
|
||
logger.warning("通知自动清理失败", exc_info=True)
|