|
|
import logging
|
|
|
import re
|
|
|
import os
|
|
|
from abc import abstractmethod
|
|
|
from django.conf import settings
|
|
|
from django.core.exceptions import ValidationError
|
|
|
from django.db import models
|
|
|
from django.urls import reverse
|
|
|
from django.utils.timezone import now
|
|
|
from django.utils.translation import gettext_lazy as _
|
|
|
from mdeditor.fields import MDTextField
|
|
|
from uuslug import slugify
|
|
|
|
|
|
from djangoblog.utils import cache_decorator, cache
|
|
|
from djangoblog.utils import get_current_site
|
|
|
|
|
|
# 确保导入了 post_save 信号和 receiver 装饰器
|
|
|
from django.db.models.signals import post_save
|
|
|
from django.dispatch import receiver
|
|
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
|
def user_avatar_path(instance, filename):
|
|
|
"""
|
|
|
定义用户头像的上传路径
|
|
|
"""
|
|
|
# 文件将被上传到 MEDIA_ROOT/user_<id>/avatar/<filename>
|
|
|
return os.path.join(f'user_{instance.user.id}', 'avatar', filename)
|
|
|
|
|
|
class LinkShowType(models.TextChoices):
|
|
|
I = ('i', _('index'))
|
|
|
L = ('l', _('list'))
|
|
|
P = ('p', _('post'))
|
|
|
A = ('a', _('all'))
|
|
|
S = ('s', _('slide'))
|
|
|
|
|
|
class BaseModel(models.Model):
|
|
|
id = models.AutoField(primary_key=True)
|
|
|
creation_time = models.DateTimeField(_('creation time'), default=now)
|
|
|
last_modify_time = models.DateTimeField(_('modify time'), default=now)
|
|
|
|
|
|
def save(self, *args, **kwargs):
|
|
|
is_update_views = isinstance(
|
|
|
self,
|
|
|
Article) and 'update_fields' in kwargs and kwargs['update_fields'] == ['views']
|
|
|
if is_update_views:
|
|
|
Article.objects.filter(pk=self.pk).update(views=self.views)
|
|
|
else:
|
|
|
if 'slug' in self.__dict__:
|
|
|
slug = getattr(
|
|
|
self, 'title') if 'title' in self.__dict__ else getattr(
|
|
|
self, 'name')
|
|
|
setattr(self, 'slug', slugify(slug))
|
|
|
super().save(*args, **kwargs)
|
|
|
|
|
|
def get_full_url(self):
|
|
|
site = get_current_site().domain
|
|
|
url = "https://{site}{path}".format(site=site,
|
|
|
path=self.get_absolute_url())
|
|
|
return url
|
|
|
|
|
|
class Meta:
|
|
|
abstract = True
|
|
|
|
|
|
@abstractmethod
|
|
|
def get_absolute_url(self):
|
|
|
pass
|
|
|
|
|
|
class Article(BaseModel):
|
|
|
"""文章"""
|
|
|
STATUS_CHOICES = (
|
|
|
('d', _('Draft')),
|
|
|
('p', _('Published')),
|
|
|
)
|
|
|
COMMENT_STATUS = (
|
|
|
('o', _('Open')),
|
|
|
('c', _('Close')),
|
|
|
)
|
|
|
TYPE = (
|
|
|
('a', _('Article')),
|
|
|
('p', _('Page')),
|
|
|
)
|
|
|
title = models.CharField(_('title'), max_length=200, unique=True)
|
|
|
body = MDTextField(_('body'))
|
|
|
pub_time = models.DateTimeField(
|
|
|
_('publish time'), blank=False, null=False, default=now)
|
|
|
status = models.CharField(
|
|
|
_('status'),
|
|
|
max_length=1,
|
|
|
choices=STATUS_CHOICES,
|
|
|
default='p')
|
|
|
comment_status = models.CharField(
|
|
|
_('comment status'),
|
|
|
max_length=1,
|
|
|
choices=COMMENT_STATUS,
|
|
|
default='o')
|
|
|
type = models.CharField(_('type'), max_length=1, choices=TYPE, default='a')
|
|
|
views = models.PositiveIntegerField(_('views'), default=0)
|
|
|
author = models.ForeignKey(
|
|
|
settings.AUTH_USER_MODEL,
|
|
|
verbose_name=_('author'),
|
|
|
blank=False,
|
|
|
null=False,
|
|
|
on_delete=models.CASCADE)
|
|
|
article_order = models.IntegerField(
|
|
|
_('order'), blank=False, null=False, default=0)
|
|
|
show_toc = models.BooleanField(_('show toc'), blank=False, null=False, default=False)
|
|
|
category = models.ForeignKey(
|
|
|
'Category',
|
|
|
verbose_name=_('category'),
|
|
|
on_delete=models.CASCADE,
|
|
|
blank=False,
|
|
|
null=False)
|
|
|
tags = models.ManyToManyField('Tag', verbose_name=_('tag'), blank=True)
|
|
|
|
|
|
def body_to_string(self):
|
|
|
return self.body
|
|
|
|
|
|
def __str__(self):
|
|
|
return self.title
|
|
|
|
|
|
class Meta:
|
|
|
ordering = ['-article_order', '-pub_time']
|
|
|
verbose_name = _('article')
|
|
|
verbose_name_plural = verbose_name
|
|
|
get_latest_by = 'id'
|
|
|
|
|
|
def get_absolute_url(self):
|
|
|
return reverse('blog:detailbyid', kwargs={
|
|
|
'article_id': self.id,
|
|
|
'year': self.creation_time.year,
|
|
|
'month': self.creation_time.month,
|
|
|
'day': self.creation_time.day
|
|
|
})
|
|
|
|
|
|
@cache_decorator(60 * 60 * 10)
|
|
|
def get_category_tree(self):
|
|
|
tree = self.category.get_category_tree()
|
|
|
names = list(map(lambda c: (c.name, c.get_absolute_url()), tree))
|
|
|
return names
|
|
|
|
|
|
def viewed(self):
|
|
|
self.views += 1
|
|
|
self.save(update_fields=['views'])
|
|
|
|
|
|
def comment_list(self):
|
|
|
cache_key = 'article_comments_{id}'.format(id=self.id)
|
|
|
value = cache.get(cache_key)
|
|
|
if value:
|
|
|
logger.info('get article comments:{id}'.format(id=self.id))
|
|
|
return value
|
|
|
else:
|
|
|
comments = self.comment_set.filter(is_enable=True).order_by('-id')
|
|
|
cache.set(cache_key, comments, 60 * 100)
|
|
|
logger.info('set article comments:{id}'.format(id=self.id))
|
|
|
return comments
|
|
|
|
|
|
def get_admin_url(self):
|
|
|
info = (self._meta.app_label, self._meta.model_name)
|
|
|
return reverse('admin:%s_%s_change' % info, args=(self.pk,))
|
|
|
|
|
|
@cache_decorator(expiration=60 * 100)
|
|
|
def next_article(self):
|
|
|
# 下一篇
|
|
|
return Article.objects.filter(
|
|
|
id__gt=self.id, status='p').order_by('id').first()
|
|
|
|
|
|
@cache_decorator(expiration=60 * 100)
|
|
|
def prev_article(self):
|
|
|
# 前一篇
|
|
|
return Article.objects.filter(id__lt=self.id, status='p').first()
|
|
|
|
|
|
def get_first_image_url(self):
|
|
|
"""
|
|
|
Get the first image url from article.body.
|
|
|
:return:
|
|
|
"""
|
|
|
match = re.search(r'!\[.*?\]\((.+?)\)', self.body)
|
|
|
if match:
|
|
|
return match.group(1)
|
|
|
return ""
|
|
|
|
|
|
@cache_decorator(60 * 60 * 24) # 缓存24小时
|
|
|
def get_related_articles(self, limit=5):
|
|
|
"""
|
|
|
根据当前文章的标签和分类获取相关推荐文章
|
|
|
:param limit: 推荐文章数量上限
|
|
|
:return: 相关文章列表
|
|
|
"""
|
|
|
# 1. 获取当前文章的所有标签ID
|
|
|
tag_ids = self.tags.values_list('id', flat=True)
|
|
|
|
|
|
# 2. 查询有相同标签的已发布文章(排除当前文章),并去重
|
|
|
related_by_tag = Article.objects.filter(
|
|
|
tags__id__in=tag_ids,
|
|
|
status='p'
|
|
|
).exclude(id=self.id).distinct()
|
|
|
|
|
|
# 3. 如果标签匹配的文章不足,补充同分类的已发布文章
|
|
|
if related_by_tag.count() < limit:
|
|
|
# 计算还需要补充的文章数量
|
|
|
need = limit - related_by_tag.count()
|
|
|
# 查询同分类的文章(排除当前文章和已通过标签匹配的文章)
|
|
|
related_by_category = Article.objects.filter(
|
|
|
category=self.category,
|
|
|
status='p'
|
|
|
).exclude(
|
|
|
id__in=list(related_by_tag.values_list('id', flat=True)) + [self.id]
|
|
|
).order_by('-pub_time')[:need]
|
|
|
|
|
|
# 合并结果(标签匹配的文章在前,分类匹配的在后)
|
|
|
related_articles = list(related_by_tag) + list(related_by_category)
|
|
|
else:
|
|
|
# 标签匹配的文章足够,直接取前 limit 篇
|
|
|
related_articles = list(related_by_tag)[:limit]
|
|
|
|
|
|
return related_articles
|
|
|
|
|
|
# 新增:获取文章的收藏数
|
|
|
def get_favorite_count(self):
|
|
|
"""获取文章的收藏数"""
|
|
|
return self.favorites.count()
|
|
|
|
|
|
|
|
|
class Category(BaseModel):
|
|
|
"""文章分类"""
|
|
|
name = models.CharField(_('category name'), max_length=30, unique=True)
|
|
|
parent_category = models.ForeignKey(
|
|
|
'self',
|
|
|
verbose_name=_('parent category'),
|
|
|
blank=True,
|
|
|
null=True,
|
|
|
on_delete=models.CASCADE)
|
|
|
slug = models.SlugField(default='no-slug', max_length=60, blank=True)
|
|
|
index = models.IntegerField(default=0, verbose_name=_('index'))
|
|
|
|
|
|
class Meta:
|
|
|
ordering = ['-index']
|
|
|
verbose_name = _('category')
|
|
|
verbose_name_plural = verbose_name
|
|
|
|
|
|
def get_absolute_url(self):
|
|
|
return reverse(
|
|
|
'blog:category_detail', kwargs={
|
|
|
'category_name': self.slug})
|
|
|
|
|
|
def __str__(self):
|
|
|
return self.name
|
|
|
|
|
|
@cache_decorator(60 * 60 * 10)
|
|
|
def get_category_tree(self):
|
|
|
"""
|
|
|
递归获得分类目录的父级
|
|
|
:return:
|
|
|
"""
|
|
|
categorys = []
|
|
|
|
|
|
def parse(category):
|
|
|
categorys.append(category)
|
|
|
if category.parent_category:
|
|
|
parse(category.parent_category)
|
|
|
|
|
|
parse(self)
|
|
|
return categorys
|
|
|
|
|
|
@cache_decorator(60 * 60 * 10)
|
|
|
def get_sub_categorys(self):
|
|
|
"""
|
|
|
获得当前分类目录所有子集
|
|
|
:return:
|
|
|
"""
|
|
|
categorys = []
|
|
|
all_categorys = Category.objects.all()
|
|
|
|
|
|
def parse(category):
|
|
|
if category not in categorys:
|
|
|
categorys.append(category)
|
|
|
childs = all_categorys.filter(parent_category=category)
|
|
|
for child in childs:
|
|
|
if category not in categorys:
|
|
|
categorys.append(child)
|
|
|
parse(child)
|
|
|
|
|
|
parse(self)
|
|
|
return categorys
|
|
|
|
|
|
class Tag(BaseModel):
|
|
|
"""文章标签"""
|
|
|
name = models.CharField(_('tag name'), max_length=30, unique=True)
|
|
|
slug = models.SlugField(default='no-slug', max_length=60, blank=True)
|
|
|
|
|
|
def __str__(self):
|
|
|
return self.name
|
|
|
|
|
|
def get_absolute_url(self):
|
|
|
return reverse('blog:tag_detail', kwargs={'tag_name': self.slug})
|
|
|
|
|
|
@cache_decorator(60 * 60 * 10)
|
|
|
def get_article_count(self):
|
|
|
return Article.objects.filter(tags__name=self.name).distinct().count()
|
|
|
|
|
|
class Meta:
|
|
|
ordering = ['name']
|
|
|
verbose_name = _('tag')
|
|
|
verbose_name_plural = verbose_name
|
|
|
|
|
|
class Links(models.Model):
|
|
|
"""友情链接"""
|
|
|
|
|
|
name = models.CharField(_('link name'), max_length=30, unique=True)
|
|
|
link = models.URLField(_('link'))
|
|
|
sequence = models.IntegerField(_('order'), unique=True)
|
|
|
is_enable = models.BooleanField(
|
|
|
_('is show'), default=True, blank=False, null=False)
|
|
|
show_type = models.CharField(
|
|
|
_('show type'),
|
|
|
max_length=1,
|
|
|
choices=LinkShowType.choices,
|
|
|
default=LinkShowType.I)
|
|
|
creation_time = models.DateTimeField(_('creation time'), default=now)
|
|
|
last_mod_time = models.DateTimeField(_('modify time'), default=now)
|
|
|
|
|
|
class Meta:
|
|
|
ordering = ['sequence']
|
|
|
verbose_name = _('link')
|
|
|
verbose_name_plural = verbose_name
|
|
|
|
|
|
def __str__(self):
|
|
|
return self.name
|
|
|
|
|
|
class SideBar(models.Model):
|
|
|
"""侧边栏,可以展示一些html内容"""
|
|
|
name = models.CharField(_('title'), max_length=100)
|
|
|
content = models.TextField(_('content'))
|
|
|
sequence = models.IntegerField(_('order'), unique=True)
|
|
|
is_enable = models.BooleanField(_('is enable'), default=True)
|
|
|
creation_time = models.DateTimeField(_('creation time'), default=now)
|
|
|
last_mod_time = models.DateTimeField(_('modify time'), default=now)
|
|
|
|
|
|
class Meta:
|
|
|
ordering = ['sequence']
|
|
|
verbose_name = _('sidebar')
|
|
|
verbose_name_plural = verbose_name
|
|
|
|
|
|
def __str__(self):
|
|
|
return self.name
|
|
|
|
|
|
class BlogSettings(models.Model):
|
|
|
"""blog的配置"""
|
|
|
site_name = models.CharField(
|
|
|
_('site name'),
|
|
|
max_length=200,
|
|
|
null=False,
|
|
|
blank=False,
|
|
|
default='')
|
|
|
site_description = models.TextField(
|
|
|
_('site description'),
|
|
|
max_length=1000,
|
|
|
null=False,
|
|
|
blank=False,
|
|
|
default='')
|
|
|
site_seo_description = models.TextField(
|
|
|
_('site seo description'), max_length=1000, null=False, blank=False, default='')
|
|
|
site_keywords = models.TextField(
|
|
|
_('site keywords'),
|
|
|
max_length=1000,
|
|
|
null=False,
|
|
|
blank=False,
|
|
|
default='')
|
|
|
article_sub_length = models.IntegerField(_('article sub length'), default=300)
|
|
|
sidebar_article_count = models.IntegerField(_('sidebar article count'), default=10)
|
|
|
sidebar_comment_count = models.IntegerField(_('sidebar comment count'), default=5)
|
|
|
article_comment_count = models.IntegerField(_('article comment count'), default=5)
|
|
|
show_google_adsense = models.BooleanField(_('show adsense'), default=False)
|
|
|
google_adsense_codes = models.TextField(
|
|
|
_('adsense code'), max_length=2000, null=True, blank=True, default='')
|
|
|
open_site_comment = models.BooleanField(_('open site comment'), default=True)
|
|
|
global_header = models.TextField("公共头部", null=True, blank=True, default='')
|
|
|
global_footer = models.TextField("公共尾部", null=True, blank=True, default='')
|
|
|
beian_code = models.CharField(
|
|
|
'备案号',
|
|
|
max_length=2000,
|
|
|
null=True,
|
|
|
blank=True,
|
|
|
default='')
|
|
|
analytics_code = models.TextField(
|
|
|
"网站统计代码",
|
|
|
max_length=1000,
|
|
|
null=False,
|
|
|
blank=False,
|
|
|
default='')
|
|
|
show_gongan_code = models.BooleanField(
|
|
|
'是否显示公安备案号', default=False, null=False)
|
|
|
gongan_beiancode = models.TextField(
|
|
|
'公安备案号',
|
|
|
max_length=2000,
|
|
|
null=True,
|
|
|
blank=True,
|
|
|
default='')
|
|
|
comment_need_review = models.BooleanField(
|
|
|
'评论是否需要审核', default=False, null=False)
|
|
|
|
|
|
class Meta:
|
|
|
verbose_name = _('Website configuration')
|
|
|
verbose_name_plural = verbose_name
|
|
|
|
|
|
def __str__(self):
|
|
|
return self.site_name
|
|
|
|
|
|
def clean(self):
|
|
|
if BlogSettings.objects.exclude(id=self.id).count():
|
|
|
raise ValidationError(_('There can only be one configuration'))
|
|
|
|
|
|
def save(self, *args, **kwargs):
|
|
|
super().save(*args, **kwargs)
|
|
|
from djangoblog.utils import cache
|
|
|
cache.clear()
|
|
|
|
|
|
class Favorite(models.Model):
|
|
|
"""
|
|
|
文章收藏模型
|
|
|
"""
|
|
|
article = models.ForeignKey(Article, on_delete=models.CASCADE, related_name='favorites')
|
|
|
# 使用 settings.AUTH_USER_MODEL
|
|
|
user = models.ForeignKey(settings.AUTH_USER_MODEL, on_delete=models.CASCADE, related_name='favorite_articles')
|
|
|
created_at = models.DateTimeField(auto_now_add=True)
|
|
|
|
|
|
class Meta:
|
|
|
unique_together = ('article', 'user') # 一个用户只能收藏同一篇文章一次
|
|
|
verbose_name = "收藏"
|
|
|
verbose_name_plural = "收藏"
|
|
|
|
|
|
def __str__(self):
|
|
|
return f'{self.user.username} favorites {self.article.title}'
|
|
|
|
|
|
class UserProfile(models.Model):
|
|
|
"""
|
|
|
用户资料扩展模型
|
|
|
用于存储用户头像、个人简介等额外信息
|
|
|
"""
|
|
|
user = models.OneToOneField(settings.AUTH_USER_MODEL, on_delete=models.CASCADE, related_name='profile', verbose_name=_('User'))
|
|
|
|
|
|
# 基本信息
|
|
|
avatar = models.ImageField(
|
|
|
upload_to=user_avatar_path,
|
|
|
blank=True,
|
|
|
null=True,
|
|
|
verbose_name=_('Avatar'),
|
|
|
help_text=_('Upload your avatar image (recommended size: 100x100px)')
|
|
|
)
|
|
|
bio = models.TextField(
|
|
|
blank=True,
|
|
|
null=True,
|
|
|
verbose_name=_('Biography'),
|
|
|
help_text=_('Tell us a little about yourself')
|
|
|
)
|
|
|
|
|
|
# 可选的社交链接
|
|
|
website = models.URLField(blank=True, null=True, verbose_name=_('Website'))
|
|
|
github = models.URLField(blank=True, null=True, verbose_name=_('GitHub'))
|
|
|
twitter = models.URLField(blank=True, null=True, verbose_name=_('Twitter'))
|
|
|
weibo = models.URLField(blank=True, null=True, verbose_name=_('Weibo'))
|
|
|
|
|
|
# 时间戳
|
|
|
created_at = models.DateTimeField(auto_now_add=True, verbose_name=_('Created At'))
|
|
|
updated_at = models.DateTimeField(auto_now=True, verbose_name=_('Updated At'))
|
|
|
|
|
|
class Meta:
|
|
|
verbose_name = _('User Profile')
|
|
|
verbose_name_plural = _('User Profiles')
|
|
|
ordering = ['-created_at']
|
|
|
|
|
|
def __str__(self):
|
|
|
return f"{self.user.username}'s Profile"
|
|
|
|
|
|
def get_absolute_url(self):
|
|
|
"""
|
|
|
定义用户资料页的绝对URL
|
|
|
"""
|
|
|
return reverse('blog:user_profile', kwargs={'username': self.user.username})
|
|
|
|
|
|
# 信号:当一个新的自定义用户(settings.AUTH_USER_MODEL)被创建时,自动创建一个关联的UserProfile
|
|
|
@receiver(post_save, sender=settings.AUTH_USER_MODEL)
|
|
|
def create_user_profile(sender, instance, created, **kwargs):
|
|
|
if created:
|
|
|
UserProfile.objects.create(user=instance)
|
|
|
|
|
|
@receiver(post_save, sender=settings.AUTH_USER_MODEL)
|
|
|
def save_user_profile(sender, instance, **kwargs):
|
|
|
# 使用 get_or_create 防止在某些情况下(如手动创建用户时)profile不存在
|
|
|
UserProfile.objects.get_or_create(user=instance)
|
|
|
instance.profile.save()
|