django 初始化專案 和modelviewset 使用
阿新 • • 發佈:2020-10-11
django初始化專案
1.初始化專案結構└─shiyanlou_project
│ .gitignore
│ README.en.md # 英文
│ README.md # 中文專案簡介
│
├─celery_task # 用來存放celery相關檔案
│init.py
│
├─db # 資料庫相關:1.資料庫初始化sql指令碼; 2.資料庫維護指令碼,清
理,備份指令碼
├─logs # 存放日誌
├─packages # 外部包,原始的,未配置的:1.七牛雲sdk;2.阿里雲簡訊;
├─scrips # 指令碼目錄:1.定時任務指令碼;2.頁面靜態化指令碼;處理專案指令碼
└─uwsgi_conf # uwsgi配置,日誌,pid
│
│
└─syl
│ apps (python包) # Django各種app模組
│ libs (python包) # 七牛雲sdk+配置
│ utils (python包) # 小工具,常用函式
│ static
│ templates
│ syl(專案配置)
│ manager.py
2.建立Django專案
python manage.py startproject syl
3.建立user模型 建立三個包
syl) root@dev:shiyanlou_project/syl
libs apps utils libs
3.1 建立使用者模型user
syl) root@dev:shiyanlou_project/syl/apps
python manage.py startapp user
3.2 在setting.py中註冊user模型(第一步)
NSTALLED_APPS = [ 'user.apps.UserConfig', ]
3.3 syl/urls.py新增主路由(第二步)
import sys sys.path.insert(0,os.path.join(BASE_DIR,'apps'))
3.4 建立 user/urls.py子路由檔案(第三步)
4.重寫Django預設認證使用者模型
4.1 user/models.py建立模型:從寫django user表
class User(AbstractUser): phone = models.CharField('手機號',max_length=20) img = models.ImageField(upload_to='user',null=True) nick_name = models.CharField('暱稱',max_length=20) address = models.CharField('地址',max_length=255) class Meta: db_table = 'tb_user'
4.2 syl/settings.py中註冊自己的使用者模型類
註冊自己的使用者模型類:應用名.模型名字,指定我們重寫的user表進行身份驗證
AUTH_USER_MODEL = 'user.User'
4.3 配置mysql
DATABASES = { 'default': { 'ENGINE': 'django.db.backends.mysql', 'NAME': 'syldb', # 指定資料庫名稱:MyCRM 'USER': 'root', 'PASSWORD': '1', 'HOST': '127.0.0.1', 'PORT': '3306', } }
4.4生成表
python manager.py makemigrations python manager.py migrate
django 配置 viewset使用
在seettings.py 中註冊
NSTALLED_APPS=[ 'django_filters', 'rest_framework', ]
REST_FRAMEWORK = { # 文件報錯: AttributeError: ‘AutoSchema’ object has no attribute ‘get_link’ # 用下面的設定可以解決 'DEFAULT_SCHEMA_CLASS': 'rest_framework.schemas.AutoSchema', # 預設設定是: # 'DEFAULT_SCHEMA_CLASS': 'rest_framework.schemas.openapi.AutoSchema', # 異常處理器 # 'EXCEPTION_HANDLER': 'user.utils.exception_handler', # Base API policies 'DEFAULT_RENDERER_CLASSES': [ 'rest_framework.renderers.JSONRenderer', 'rest_framework.renderers.BrowsableAPIRenderer', ], 'DEFAULT_PARSER_CLASSES': [ 'rest_framework.parsers.JSONParser', 'rest_framework.parsers.FormParser', 'rest_framework.parsers.MultiPartParser' ], # 1.認證器(全域性) 'DEFAULT_AUTHENTICATION_CLASSES': [ 'rest_framework.authentication.SessionAuthentication', # 使用session時的認證器 'rest_framework.authentication.BasicAuthentication' # 提交表單時的認證器 ], #2.許可權配置(全域性): 順序靠上的嚴格 'DEFAULT_PERMISSION_CLASSES': [ # 'rest_framework.permissions.IsAdminUser', # 管理員可以訪問 # 'rest_framework.permissions.IsAuthenticated', # 認證使用者可以訪問 # 'rest_framework.permissions.IsAuthenticatedOrReadOnly', # 認證使用者可以訪問, 否則只能讀取 # 'rest_framework.permissions.AllowAny', # 所有使用者都可以訪問 ], #3.限流(防爬蟲) 'DEFAULT_THROTTLE_CLASSES': [ 'rest_framework.throttling.AnonRateThrottle', 'rest_framework.throttling.UserRateThrottle', ], #3.1限流策略 'DEFAULT_THROTTLE_RATES': { 'user': '100/hour', # 認證使用者每小時100次 'anon': '3/day', # 未認證使用者每天能訪問3次 }, 'DEFAULT_CONTENT_NEGOTIATION_CLASS': 'rest_framework.negotiation.DefaultContentNegotiation', 'DEFAULT_METADATA_CLASS': 'rest_framework.metadata.SimpleMetadata', 'DEFAULT_VERSIONING_CLASS': None, #4.分頁(全域性):全域性分頁器, 例如 省市區的資料自定義分頁器, 不需要分頁 'DEFAULT_PAGINATION_CLASS': 'rest_framework.pagination.PageNumberPagination', # 每頁返回數量 'PAGE_SIZE': 10, # 預設 None #5.過濾器後端 'DEFAULT_FILTER_BACKENDS': [ 'django_filters.rest_framework.DjangoFilterBackend', # 'django_filters.rest_framework.backends.DjangoFilterBackend', 包路徑有變化 ], #5.1過濾排序(全域性):Filtering 過濾排序 'SEARCH_PARAM': 'search', 'ORDERING_PARAM': 'ordering', 'NUM_PROXIES': None, #6.版本控制:Versioning 介面版本控制 'DEFAULT_VERSION': None, 'ALLOWED_VERSIONS': None, 'VERSION_PARAM': 'version', # Authentication 認證 # 未認證使用者使用的使用者型別 'UNAUTHENTICATED_USER': 'django.contrib.auth.models.AnonymousUser', # 未認證使用者使用的Token值 'UNAUTHENTICATED_TOKEN': None, # View configuration 'VIEW_NAME_FUNCTION': 'rest_framework.views.get_view_name', 'VIEW_DESCRIPTION_FUNCTION': 'rest_framework.views.get_view_description', 'NON_FIELD_ERRORS_KEY': 'non_field_errors', # Testing 'TEST_REQUEST_RENDERER_CLASSES': [ 'rest_framework.renderers.MultiPartRenderer', 'rest_framework.renderers.JSONRenderer' ], 'TEST_REQUEST_DEFAULT_FORMAT': 'multipart', # Hyperlink settings 'URL_FORMAT_OVERRIDE': 'format', 'FORMAT_SUFFIX_KWARG': 'format', 'URL_FIELD_NAME': 'url', # Encoding 'UNICODE_JSON': True, 'COMPACT_JSON': True, 'STRICT_JSON': True, 'COERCE_DECIMAL_TO_STRING': True, 'UPLOADED_FILES_USE_URL': True, # Browseable API 'HTML_SELECT_CUTOFF': 1000, 'HTML_SELECT_CUTOFF_TEXT': "More than {count} items...", # Schemas 'SCHEMA_COERCE_PATH_PK': True, 'SCHEMA_COERCE_METHOD_NAMES': { 'retrieve': 'read', 'destroy': 'delete' }, }
建立 user/serlizers.py 寫序列化器
def validata_address(self,data): if data=='測試': raise serializers.ValidationError('請重新填寫地址') # 有錯就丟擲異常 return data # 沒錯就返回結果 def validate_phone(self,data): # 不符合手機格式 # raise serializer.ValidationError('手機號格式不正確') model=self.root.Meta.model num=model.object.filter(phone=data).count() if num>0: raise serializers.ValidationError('已存在手機號') return data # 3 所有屬性驗證 def validate(self,attrs): # attrs:{"username":"zhangsan","phone":"110",} # 所有屬性 驗證器 #self.context 中有request 和views 上下文 #self.context['view'].action 跨域取到動作 # attrs 是需要序列化資料 # raise serializers.ValidationError('xxx錯誤') return attrs # 沒問題返回資料 class Meta: model=User fields='__all__' read_only_fields=('',) extra_kawargs={ 'address':{ 'min_length':5, # 給地址增加 最小長度 限制 'default':'預設測試地址' #增加預設值 } }
在 views 寫入
from django.shortcuts import render from django.http import HttpResponse from django_filters.rest_framework import DjangoFilterBackend from rest_framework import viewsets from rest_framework.authentication import BasicAuthentication,SessionAuthentication from rest_framework.decorators import action from rest_framework.filters import OrderingFilter from rest_framework.permissions import AllowAny,IsAdminUser,IsAuthenticated,IsAuthenticatedOrReadOnly from rest_framework.response import Response from rest_framework.throttling import UserRateThrottle from rest_framework.pagination import PageNumberPagination from rest_framework.views import APIView from rest_framework.permissions import BasePermission,SAFE_METHODS from user.models import User from user.serializers import UserSerlizer,UserUnActiveSerializer def index(request): # 需要認證才能訪問的檢視 return HttpResponse('hello') class PageNum(PageNumberPagination): # 查詢字串中代表每頁返回資料數量的引數名,預設值為:None page_size_query_param = 'page_size' # 查詢字串中代表頁碼的引數名,預設值為:page #page_query_param='page_size' max_page_size = 2 class MyPermissio(BasePermission): def has_permission(self, request, view): print(view.kwargs.get('pk'),request.user.id) '''判斷使用者對使用次許可權類的檢視都有許可權''' # 任何使用者對使用次許可權的檢視都要訪問許可權 print(request) if request.user.is_superuser: # 管理員對使用者模型具有訪問權 return True elif view.kwargs.get('pk')==str(request.user.id): return True return False class UserViewSet(viewsets.ModelViewSet): queryset = User.objects.all() serializer_class = UserSerlizer pagination_class = PageNum # 限流自定義 throttle_classes = [UserRateThrottle] # 指定過濾方法 filter_backends = (DjangoFilterBackend,OrderingFilter) ordering_fields=('data_joined','id') filter_fields=('username','phone','is_active') def get_serializer_class(self): return UserSerializers
urls.py
from django.contrib import admin from django.urls import include,path from user import views from rest_framework.routers import SimpleRouter,DefaultRouter ''' 自動生成路由方法 必須用檢視集 routed=DefaultRouter() 沒有跟路由 /user/ 無法識別 ''' routed=DefaultRouter() routed.register(r'user',views.UserViewSet) urlpatterns = [ path('index', views.index), path('api-auth/',include('rest_framework.urls',namespace='rest_framework')) # 認證地址 ] urlpatterns+=routed.urls