教程模块
This commit is contained in:
@@ -1,4 +1,5 @@
|
||||
from django.contrib.auth import authenticate, login, logout
|
||||
from django.contrib.auth.decorators import login_required
|
||||
from ninja import Router
|
||||
from ninja.errors import HttpError
|
||||
from .schemas import UserRegistrationSchema, UserLoginSchema
|
||||
@@ -7,40 +8,36 @@ from .models import RoleChoices, User
|
||||
router = Router()
|
||||
|
||||
|
||||
@router.get("/")
|
||||
def account(request):
|
||||
return {"message": "success"}
|
||||
|
||||
|
||||
@router.post("/register")
|
||||
def user_register(request, payload: UserRegistrationSchema):
|
||||
if User.objects.filter(username=payload.username).exists():
|
||||
raise HttpError(400, "Username already exists")
|
||||
raise HttpError(400, "用户已存在")
|
||||
User.objects.create_user(
|
||||
username=payload.username,
|
||||
email=payload.email,
|
||||
password=payload.password,
|
||||
)
|
||||
return {"message": "User created successfully"}
|
||||
return {"message": "创建成功"}
|
||||
|
||||
|
||||
@router.post("/login")
|
||||
def user_login(request, payload: UserLoginSchema):
|
||||
user = authenticate(username=payload.username, password=payload.password)
|
||||
if user is not None:
|
||||
if user:
|
||||
login(request, user)
|
||||
return {"username": user.username, "role": user.role}
|
||||
else:
|
||||
raise HttpError(401, "Invalid credentials")
|
||||
raise HttpError(401, "账号密码错误")
|
||||
|
||||
|
||||
@router.post("/logout")
|
||||
@login_required
|
||||
def user_logout(request):
|
||||
logout(request)
|
||||
|
||||
|
||||
@router.get("/profile")
|
||||
def current_user_profile(request):
|
||||
def my_profile(request):
|
||||
# 暂时这样写
|
||||
if request.user.is_authenticated:
|
||||
return {"username": request.user.get_username(), "role": request.user.role}
|
||||
|
||||
28
account/decorators.py
Normal file
28
account/decorators.py
Normal file
@@ -0,0 +1,28 @@
|
||||
from ninja.errors import HttpError
|
||||
from ninja import NinjaAPI
|
||||
from functools import wraps
|
||||
from .models import User, RoleChoices
|
||||
|
||||
api = NinjaAPI()
|
||||
|
||||
|
||||
def _require(roles):
|
||||
def decorator(func):
|
||||
@wraps(func)
|
||||
def wrapper(request, *args, **kwargs):
|
||||
if not request.user.is_authenticated:
|
||||
raise HttpError(401, "用户未登录")
|
||||
try:
|
||||
if request.user.role not in roles:
|
||||
raise HttpError(403, "你没有权限")
|
||||
except User.DoesNotExist:
|
||||
raise HttpError(404, "用户不存在")
|
||||
return func(request, *args, **kwargs)
|
||||
|
||||
return wrapper
|
||||
|
||||
return decorator
|
||||
|
||||
admin_required = _require([RoleChoices.ADMIN, RoleChoices.SUPER])
|
||||
|
||||
super_required = _require([RoleChoices.SUPER])
|
||||
@@ -42,7 +42,9 @@ INSTALLED_APPS = [
|
||||
"django.contrib.staticfiles",
|
||||
"ninja",
|
||||
"corsheaders",
|
||||
"django_extensions",
|
||||
"account",
|
||||
"tutorial",
|
||||
]
|
||||
|
||||
MIDDLEWARE = [
|
||||
|
||||
@@ -22,6 +22,7 @@ from ninja import NinjaAPI
|
||||
api = NinjaAPI()
|
||||
|
||||
api.add_router("account/", "account.api.router")
|
||||
api.add_router("tutorial/", "tutorial.api.router")
|
||||
|
||||
urlpatterns = [
|
||||
path("admin/", admin.site.urls),
|
||||
|
||||
BIN
requirements.txt
BIN
requirements.txt
Binary file not shown.
0
tutorial/__init__.py
Normal file
0
tutorial/__init__.py
Normal file
4
tutorial/admin.py
Normal file
4
tutorial/admin.py
Normal file
@@ -0,0 +1,4 @@
|
||||
from django.contrib import admin
|
||||
from .models import Tutorial
|
||||
|
||||
admin.site.register(Tutorial)
|
||||
41
tutorial/api.py
Normal file
41
tutorial/api.py
Normal file
@@ -0,0 +1,41 @@
|
||||
from ninja import Router
|
||||
from ninja.errors import HttpError
|
||||
|
||||
from account.decorators import super_required
|
||||
|
||||
# from account.decorators import super_required
|
||||
from .schemas import TutorialAll, TutorialIn, TutorialReturn
|
||||
from .models import Tutorial
|
||||
|
||||
router = Router()
|
||||
|
||||
|
||||
@router.get("/", response=TutorialReturn)
|
||||
@super_required
|
||||
def tutorial(request):
|
||||
return {
|
||||
"total": Tutorial.objects.count(),
|
||||
"list": Tutorial.objects.all(),
|
||||
"first": Tutorial.objects.first(),
|
||||
}
|
||||
|
||||
|
||||
@router.get("/{display}", response=TutorialAll)
|
||||
def get(request, display: str):
|
||||
return Tutorial.objects.get(display=display)
|
||||
|
||||
|
||||
@router.post("/")
|
||||
@super_required
|
||||
def create(request, payload: TutorialIn):
|
||||
if Tutorial.objects.filter(display=payload.display):
|
||||
raise HttpError(400, "有序号相同的教程存在")
|
||||
Tutorial.objects.create(**payload.dict())
|
||||
return {"message": "创建成功"}
|
||||
|
||||
|
||||
@router.delete("/{display}")
|
||||
@super_required
|
||||
def remove(request, display: str):
|
||||
Tutorial.objects.filter(display=display).delete()
|
||||
return {"message": "删除成功"}
|
||||
6
tutorial/apps.py
Normal file
6
tutorial/apps.py
Normal file
@@ -0,0 +1,6 @@
|
||||
from django.apps import AppConfig
|
||||
|
||||
|
||||
class TutorialConfig(AppConfig):
|
||||
default_auto_field = 'django.db.models.BigAutoField'
|
||||
name = 'tutorial'
|
||||
31
tutorial/migrations/0001_initial.py
Normal file
31
tutorial/migrations/0001_initial.py
Normal file
@@ -0,0 +1,31 @@
|
||||
# Generated by Django 5.1.6 on 2025-03-03 11:38
|
||||
|
||||
import django_extensions.db.fields
|
||||
from django.db import migrations, models
|
||||
|
||||
|
||||
class Migration(migrations.Migration):
|
||||
|
||||
initial = True
|
||||
|
||||
dependencies = [
|
||||
]
|
||||
|
||||
operations = [
|
||||
migrations.CreateModel(
|
||||
name='Tutorial',
|
||||
fields=[
|
||||
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||
('created', django_extensions.db.fields.CreationDateTimeField(auto_now_add=True, verbose_name='created')),
|
||||
('modified', django_extensions.db.fields.ModificationDateTimeField(auto_now=True, verbose_name='modified')),
|
||||
('display', models.CharField(max_length=10, unique=True)),
|
||||
('title', models.CharField(max_length=100)),
|
||||
('content', models.TextField()),
|
||||
('is_public', models.BooleanField(default=False)),
|
||||
],
|
||||
options={
|
||||
'get_latest_by': 'modified',
|
||||
'abstract': False,
|
||||
},
|
||||
),
|
||||
]
|
||||
0
tutorial/migrations/__init__.py
Normal file
0
tutorial/migrations/__init__.py
Normal file
12
tutorial/models.py
Normal file
12
tutorial/models.py
Normal file
@@ -0,0 +1,12 @@
|
||||
from django.db import models
|
||||
from django_extensions.db.models import TimeStampedModel
|
||||
|
||||
|
||||
class Tutorial(TimeStampedModel):
|
||||
display = models.CharField(max_length=10, unique=True)
|
||||
title = models.CharField(max_length=100)
|
||||
content = models.TextField()
|
||||
is_public = models.BooleanField(default=False)
|
||||
|
||||
def __str__(self):
|
||||
return self.title
|
||||
28
tutorial/schemas.py
Normal file
28
tutorial/schemas.py
Normal file
@@ -0,0 +1,28 @@
|
||||
from ninja import Schema, ModelSchema
|
||||
from typing import List, Optional
|
||||
from .models import Tutorial
|
||||
|
||||
|
||||
class TutorialSlim(Schema):
|
||||
display: str
|
||||
title: str
|
||||
is_public: bool
|
||||
|
||||
|
||||
class TutorialAll(ModelSchema):
|
||||
class Meta:
|
||||
model = Tutorial
|
||||
fields = "__all__"
|
||||
|
||||
|
||||
class TutorialReturn(Schema):
|
||||
total: int
|
||||
list: List[TutorialSlim]
|
||||
first: Optional[TutorialAll]
|
||||
|
||||
|
||||
class TutorialIn(Schema):
|
||||
display: str
|
||||
title: str
|
||||
content: str
|
||||
is_public: bool = False
|
||||
Reference in New Issue
Block a user