|
|
|
|
@ -4,134 +4,133 @@ from django.conf import settings
|
|
|
|
|
from django.db import migrations, models
|
|
|
|
|
import django.db.models.deletion
|
|
|
|
|
import django.utils.timezone
|
|
|
|
|
import mdeditor.fields
|
|
|
|
|
|
|
|
|
|
import mdeditor.fields # aq: 导入Markdown编辑器专用字段类型
|
|
|
|
|
|
|
|
|
|
class Migration(migrations.Migration):
|
|
|
|
|
|
|
|
|
|
initial = True
|
|
|
|
|
initial = True # aq: 标记为初始迁移,负责创建博客核心数据表结构
|
|
|
|
|
|
|
|
|
|
dependencies = [
|
|
|
|
|
dependencies = [ # aq: 迁移依赖——关联Django用户模型(支持自定义用户模型)
|
|
|
|
|
migrations.swappable_dependency(settings.AUTH_USER_MODEL),
|
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
operations = [
|
|
|
|
|
migrations.CreateModel(
|
|
|
|
|
operations = [ # aq: 迁移核心操作:创建所有博客相关数据表
|
|
|
|
|
migrations.CreateModel( # aq: 1. 创建网站配置表(BlogSettings)
|
|
|
|
|
name='BlogSettings',
|
|
|
|
|
fields=[
|
|
|
|
|
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
|
|
|
|
('sitename', models.CharField(default='', max_length=200, verbose_name='网站名称')),
|
|
|
|
|
('site_description', models.TextField(default='', max_length=1000, verbose_name='网站描述')),
|
|
|
|
|
('site_seo_description', models.TextField(default='', max_length=1000, verbose_name='网站SEO描述')),
|
|
|
|
|
('site_keywords', models.TextField(default='', max_length=1000, verbose_name='网站关键字')),
|
|
|
|
|
('article_sub_length', models.IntegerField(default=300, verbose_name='文章摘要长度')),
|
|
|
|
|
('sidebar_article_count', models.IntegerField(default=10, verbose_name='侧边栏文章数目')),
|
|
|
|
|
('sidebar_comment_count', models.IntegerField(default=5, verbose_name='侧边栏评论数目')),
|
|
|
|
|
('article_comment_count', models.IntegerField(default=5, verbose_name='文章页面默认显示评论数目')),
|
|
|
|
|
('show_google_adsense', models.BooleanField(default=False, verbose_name='是否显示谷歌广告')),
|
|
|
|
|
('google_adsense_codes', models.TextField(blank=True, default='', max_length=2000, null=True, verbose_name='广告内容')),
|
|
|
|
|
('open_site_comment', models.BooleanField(default=True, verbose_name='是否打开网站评论功能')),
|
|
|
|
|
('beiancode', models.CharField(blank=True, default='', max_length=2000, null=True, verbose_name='备案号')),
|
|
|
|
|
('analyticscode', models.TextField(default='', max_length=1000, verbose_name='网站统计代码')),
|
|
|
|
|
('show_gongan_code', models.BooleanField(default=False, verbose_name='是否显示公安备案号')),
|
|
|
|
|
('gongan_beiancode', models.TextField(blank=True, default='', max_length=2000, null=True, verbose_name='公安备案号')),
|
|
|
|
|
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), # aq: 自增主键(BigInt类型)
|
|
|
|
|
('sitename', models.CharField(default='', max_length=200, verbose_name='网站名称')), # aq: 网站名称
|
|
|
|
|
('site_description', models.TextField(default='', max_length=1000, verbose_name='网站描述')), # aq: 网站描述
|
|
|
|
|
('site_seo_description', models.TextField(default='', max_length=1000, verbose_name='网站SEO描述')), # aq: SEO优化描述
|
|
|
|
|
('site_keywords', models.TextField(default='', max_length=1000, verbose_name='网站关键字')), # aq: 网站搜索关键字
|
|
|
|
|
('article_sub_length', models.IntegerField(default=300, verbose_name='文章摘要长度')), # aq: 文章摘要默认长度
|
|
|
|
|
('sidebar_article_count', models.IntegerField(default=10, verbose_name='侧边栏文章数目')), # aq: 侧边栏显示文章数量
|
|
|
|
|
('sidebar_comment_count', models.IntegerField(default=5, verbose_name='侧边栏评论数目')), # aq: 侧边栏显示评论数量
|
|
|
|
|
('article_comment_count', models.IntegerField(default=5, verbose_name='文章页面默认显示评论数目')), # aq: 文章页默认显示评论数
|
|
|
|
|
('show_google_adsense', models.BooleanField(default=False, verbose_name='是否显示谷歌广告')), # aq: 是否启用谷歌广告
|
|
|
|
|
('google_adsense_codes', models.TextField(blank=True, default='', max_length=2000, null=True, verbose_name='广告内容')), # aq: 谷歌广告代码
|
|
|
|
|
('open_site_comment', models.BooleanField(default=True, verbose_name='是否打开网站评论功能')), # aq: 是否开放全站评论
|
|
|
|
|
('beiancode', models.CharField(blank=True, default='', max_length=2000, null=True, verbose_name='备案号')), # aq: 网站备案号
|
|
|
|
|
('analyticscode', models.TextField(default='', max_length=1000, verbose_name='网站统计代码')), # aq: 网站统计(如百度统计)代码
|
|
|
|
|
('show_gongan_code', models.BooleanField(default=False, verbose_name='是否显示公安备案号')), # aq: 是否显示公安备案号
|
|
|
|
|
('gongan_beiancode', models.TextField(blank=True, default='', max_length=2000, null=True, verbose_name='公安备案号')), # aq: 公安备案号
|
|
|
|
|
],
|
|
|
|
|
options={
|
|
|
|
|
'verbose_name': '网站配置',
|
|
|
|
|
'verbose_name_plural': '网站配置',
|
|
|
|
|
'verbose_name_plural': '网站配置', # aq: 后台显示名称(单数/复数)
|
|
|
|
|
},
|
|
|
|
|
),
|
|
|
|
|
migrations.CreateModel(
|
|
|
|
|
migrations.CreateModel( # aq: 2. 创建友情链接表(Links)
|
|
|
|
|
name='Links',
|
|
|
|
|
fields=[
|
|
|
|
|
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
|
|
|
|
('name', models.CharField(max_length=30, unique=True, verbose_name='链接名称')),
|
|
|
|
|
('link', models.URLField(verbose_name='链接地址')),
|
|
|
|
|
('sequence', models.IntegerField(unique=True, verbose_name='排序')),
|
|
|
|
|
('is_enable', models.BooleanField(default=True, verbose_name='是否显示')),
|
|
|
|
|
('show_type', models.CharField(choices=[('i', '首页'), ('l', '列表页'), ('p', '文章页面'), ('a', '全站'), ('s', '友情链接页面')], default='i', max_length=1, verbose_name='显示类型')),
|
|
|
|
|
('created_time', models.DateTimeField(default=django.utils.timezone.now, verbose_name='创建时间')),
|
|
|
|
|
('last_mod_time', models.DateTimeField(default=django.utils.timezone.now, verbose_name='修改时间')),
|
|
|
|
|
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), # aq: 自增主键
|
|
|
|
|
('name', models.CharField(max_length=30, unique=True, verbose_name='链接名称')), # aq: 友情链接名称(唯一)
|
|
|
|
|
('link', models.URLField(verbose_name='链接地址')), # aq: 链接URL地址
|
|
|
|
|
('sequence', models.IntegerField(unique=True, verbose_name='排序')), # aq: 排序序号(唯一,决定显示顺序)
|
|
|
|
|
('is_enable', models.BooleanField(default=True, verbose_name='是否显示')), # aq: 是否启用该链接
|
|
|
|
|
('show_type', models.CharField(choices=[('i', '首页'), ('l', '列表页'), ('p', '文章页面'), ('a', '全站'), ('s', '友情链接页面')], default='i', max_length=1, verbose_name='显示类型')), # aq: 链接展示位置
|
|
|
|
|
('created_time', models.DateTimeField(default=django.utils.timezone.now, verbose_name='创建时间')), # aq: 创建时间
|
|
|
|
|
('last_mod_time', models.DateTimeField(default=django.utils.timezone.now, verbose_name='修改时间')), # aq: 最后修改时间
|
|
|
|
|
],
|
|
|
|
|
options={
|
|
|
|
|
'verbose_name': '友情链接',
|
|
|
|
|
'verbose_name_plural': '友情链接',
|
|
|
|
|
'ordering': ['sequence'],
|
|
|
|
|
'ordering': ['sequence'], # aq: 按排序序号升序排列
|
|
|
|
|
},
|
|
|
|
|
),
|
|
|
|
|
migrations.CreateModel(
|
|
|
|
|
migrations.CreateModel( # aq: 3. 创建侧边栏表(SideBar)
|
|
|
|
|
name='SideBar',
|
|
|
|
|
fields=[
|
|
|
|
|
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
|
|
|
|
('name', models.CharField(max_length=100, verbose_name='标题')),
|
|
|
|
|
('content', models.TextField(verbose_name='内容')),
|
|
|
|
|
('sequence', models.IntegerField(unique=True, verbose_name='排序')),
|
|
|
|
|
('is_enable', models.BooleanField(default=True, verbose_name='是否启用')),
|
|
|
|
|
('created_time', models.DateTimeField(default=django.utils.timezone.now, verbose_name='创建时间')),
|
|
|
|
|
('last_mod_time', models.DateTimeField(default=django.utils.timezone.now, verbose_name='修改时间')),
|
|
|
|
|
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), # aq: 自增主键
|
|
|
|
|
('name', models.CharField(max_length=100, verbose_name='标题')), # aq: 侧边栏标题
|
|
|
|
|
('content', models.TextField(verbose_name='内容')), # aq: 侧边栏内容(支持HTML)
|
|
|
|
|
('sequence', models.IntegerField(unique=True, verbose_name='排序')), # aq: 排序序号(唯一)
|
|
|
|
|
('is_enable', models.BooleanField(default=True, verbose_name='是否启用')), # aq: 是否启用该侧边栏
|
|
|
|
|
('created_time', models.DateTimeField(default=django.utils.timezone.now, verbose_name='创建时间')), # aq: 创建时间
|
|
|
|
|
('last_mod_time', models.DateTimeField(default=django.utils.timezone.now, verbose_name='修改时间')), # aq: 最后修改时间
|
|
|
|
|
],
|
|
|
|
|
options={
|
|
|
|
|
'verbose_name': '侧边栏',
|
|
|
|
|
'verbose_name_plural': '侧边栏',
|
|
|
|
|
'ordering': ['sequence'],
|
|
|
|
|
'ordering': ['sequence'], # aq: 按排序序号升序排列
|
|
|
|
|
},
|
|
|
|
|
),
|
|
|
|
|
migrations.CreateModel(
|
|
|
|
|
migrations.CreateModel( # aq: 4. 创建标签表(Tag)
|
|
|
|
|
name='Tag',
|
|
|
|
|
fields=[
|
|
|
|
|
('id', models.AutoField(primary_key=True, serialize=False)),
|
|
|
|
|
('created_time', models.DateTimeField(default=django.utils.timezone.now, verbose_name='创建时间')),
|
|
|
|
|
('last_mod_time', models.DateTimeField(default=django.utils.timezone.now, verbose_name='修改时间')),
|
|
|
|
|
('name', models.CharField(max_length=30, unique=True, verbose_name='标签名')),
|
|
|
|
|
('slug', models.SlugField(blank=True, default='no-slug', max_length=60)),
|
|
|
|
|
('id', models.AutoField(primary_key=True, serialize=False)), # aq: 自增主键(Int类型)
|
|
|
|
|
('created_time', models.DateTimeField(default=django.utils.timezone.now, verbose_name='创建时间')), # aq: 创建时间
|
|
|
|
|
('last_mod_time', models.DateTimeField(default=django.utils.timezone.now, verbose_name='修改时间')), # aq: 最后修改时间
|
|
|
|
|
('name', models.CharField(max_length=30, unique=True, verbose_name='标签名')), # aq: 标签名称(唯一)
|
|
|
|
|
('slug', models.SlugField(blank=True, default='no-slug', max_length=60)), # aq: URL友好的slug名称(自动生成)
|
|
|
|
|
],
|
|
|
|
|
options={
|
|
|
|
|
'verbose_name': '标签',
|
|
|
|
|
'verbose_name_plural': '标签',
|
|
|
|
|
'ordering': ['name'],
|
|
|
|
|
'ordering': ['name'], # aq: 按标签名称升序排列
|
|
|
|
|
},
|
|
|
|
|
),
|
|
|
|
|
migrations.CreateModel(
|
|
|
|
|
migrations.CreateModel( # aq: 5. 创建分类表(Category)
|
|
|
|
|
name='Category',
|
|
|
|
|
fields=[
|
|
|
|
|
('id', models.AutoField(primary_key=True, serialize=False)),
|
|
|
|
|
('created_time', models.DateTimeField(default=django.utils.timezone.now, verbose_name='创建时间')),
|
|
|
|
|
('last_mod_time', models.DateTimeField(default=django.utils.timezone.now, verbose_name='修改时间')),
|
|
|
|
|
('name', models.CharField(max_length=30, unique=True, verbose_name='分类名')),
|
|
|
|
|
('slug', models.SlugField(blank=True, default='no-slug', max_length=60)),
|
|
|
|
|
('index', models.IntegerField(default=0, verbose_name='权重排序-越大越靠前')),
|
|
|
|
|
('parent_category', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.CASCADE, to='blog.category', verbose_name='父级分类')),
|
|
|
|
|
('id', models.AutoField(primary_key=True, serialize=False)), # aq: 自增主键
|
|
|
|
|
('created_time', models.DateTimeField(default=django.utils.timezone.now, verbose_name='创建时间')), # aq: 创建时间
|
|
|
|
|
('last_mod_time', models.DateTimeField(default=django.utils.timezone.now, verbose_name='修改时间')), # aq: 最后修改时间
|
|
|
|
|
('name', models.CharField(max_length=30, unique=True, verbose_name='分类名')), # aq: 分类名称(唯一)
|
|
|
|
|
('slug', models.SlugField(blank=True, default='no-slug', max_length=60)), # aq: URL友好的slug名称
|
|
|
|
|
('index', models.IntegerField(default=0, verbose_name='权重排序-越大越靠前')), # aq: 排序权重(数值越大越靠前)
|
|
|
|
|
('parent_category', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.CASCADE, to='blog.category', verbose_name='父级分类')), # aq: 父分类(自关联,支持多级分类)
|
|
|
|
|
],
|
|
|
|
|
options={
|
|
|
|
|
'verbose_name': '分类',
|
|
|
|
|
'verbose_name_plural': '分类',
|
|
|
|
|
'ordering': ['-index'],
|
|
|
|
|
'ordering': ['-index'], # aq: 按权重降序排列
|
|
|
|
|
},
|
|
|
|
|
),
|
|
|
|
|
migrations.CreateModel(
|
|
|
|
|
migrations.CreateModel( # aq: 6. 创建文章表(Article)——核心数据表
|
|
|
|
|
name='Article',
|
|
|
|
|
fields=[
|
|
|
|
|
('id', models.AutoField(primary_key=True, serialize=False)),
|
|
|
|
|
('created_time', models.DateTimeField(default=django.utils.timezone.now, verbose_name='创建时间')),
|
|
|
|
|
('last_mod_time', models.DateTimeField(default=django.utils.timezone.now, verbose_name='修改时间')),
|
|
|
|
|
('title', models.CharField(max_length=200, unique=True, verbose_name='标题')),
|
|
|
|
|
('body', mdeditor.fields.MDTextField(verbose_name='正文')),
|
|
|
|
|
('pub_time', models.DateTimeField(default=django.utils.timezone.now, verbose_name='发布时间')),
|
|
|
|
|
('status', models.CharField(choices=[('d', '草稿'), ('p', '发表')], default='p', max_length=1, verbose_name='文章状态')),
|
|
|
|
|
('comment_status', models.CharField(choices=[('o', '打开'), ('c', '关闭')], default='o', max_length=1, verbose_name='评论状态')),
|
|
|
|
|
('type', models.CharField(choices=[('a', '文章'), ('p', '页面')], default='a', max_length=1, verbose_name='类型')),
|
|
|
|
|
('views', models.PositiveIntegerField(default=0, verbose_name='浏览量')),
|
|
|
|
|
('article_order', models.IntegerField(default=0, verbose_name='排序,数字越大越靠前')),
|
|
|
|
|
('show_toc', models.BooleanField(default=False, verbose_name='是否显示toc目录')),
|
|
|
|
|
('author', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to=settings.AUTH_USER_MODEL, verbose_name='作者')),
|
|
|
|
|
('category', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to='blog.category', verbose_name='分类')),
|
|
|
|
|
('tags', models.ManyToManyField(blank=True, to='blog.tag', verbose_name='标签集合')),
|
|
|
|
|
('id', models.AutoField(primary_key=True, serialize=False)), # aq: 自增主键
|
|
|
|
|
('created_time', models.DateTimeField(default=django.utils.timezone.now, verbose_name='创建时间')), # aq: 创建时间
|
|
|
|
|
('last_mod_time', models.DateTimeField(default=django.utils.timezone.now, verbose_name='修改时间')), # aq: 最后修改时间
|
|
|
|
|
('title', models.CharField(max_length=200, unique=True, verbose_name='标题')), # aq: 文章标题(唯一)
|
|
|
|
|
('body', mdeditor.fields.MDTextField(verbose_name='正文')), # aq: 文章正文(Markdown格式)
|
|
|
|
|
('pub_time', models.DateTimeField(default=django.utils.timezone.now, verbose_name='发布时间')), # aq: 发布时间
|
|
|
|
|
('status', models.CharField(choices=[('d', '草稿'), ('p', '发表')], default='p', max_length=1, verbose_name='文章状态')), # aq: 文章状态(草稿/发表)
|
|
|
|
|
('comment_status', models.CharField(choices=[('o', '打开'), ('c', '关闭')], default='o', max_length=1, verbose_name='评论状态')), # aq: 评论状态(打开/关闭)
|
|
|
|
|
('type', models.CharField(choices=[('a', '文章'), ('p', '页面')], default='a', max_length=1, verbose_name='类型')), # aq: 内容类型(文章/独立页面)
|
|
|
|
|
('views', models.PositiveIntegerField(default=0, verbose_name='浏览量')), # aq: 阅读量(非负整数)
|
|
|
|
|
('article_order', models.IntegerField(default=0, verbose_name='排序,数字越大越靠前')), # aq: 文章排序权重
|
|
|
|
|
('show_toc', models.BooleanField(default=False, verbose_name='是否显示toc目录')), # aq: 是否显示文章目录
|
|
|
|
|
('author', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to=settings.AUTH_USER_MODEL, verbose_name='作者')), # aq: 关联作者(用户模型)
|
|
|
|
|
('category', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to='blog.category', verbose_name='分类')), # aq: 关联分类(多对一)
|
|
|
|
|
('tags', models.ManyToManyField(blank=True, to='blog.tag', verbose_name='标签集合')), # aq: 关联标签(多对多)
|
|
|
|
|
],
|
|
|
|
|
options={
|
|
|
|
|
'verbose_name': '文章',
|
|
|
|
|
'verbose_name_plural': '文章',
|
|
|
|
|
'ordering': ['-article_order', '-pub_time'],
|
|
|
|
|
'get_latest_by': 'id',
|
|
|
|
|
'ordering': ['-article_order', '-pub_time'], # aq: 先按排序权重降序,再按发布时间降序
|
|
|
|
|
'get_latest_by': 'id', # aq: 按ID字段获取最新记录
|
|
|
|
|
},
|
|
|
|
|
),
|
|
|
|
|
]
|
|
|
|
|
]
|