DRF 跨域 Vue 登录!

DRF 跨域

 

下载

# 版本, 我 Django3.x Python3.7
# django-cors-headers 4.0.0
pip install django-cors-headers

配置

# 打开 settings.py
# 增加
INSTALLED_APPS = [
...
   'corsheaders',
]


# 添加中间件
MIDDLEWARE = [
   'corsheaders.middleware.CorsMiddleware',
...
]


# 增加
CORS_ORIGIN_ALLOW_ALL = True  # 所有ip都可以访问后端接口
# CORS_ORIGIN_WHITELIST = ["http://127.0.0.1:8080",["http://192.168.10.1:8080"]  # 指定能够访问后端接口的ip或域名列表


# 允许访问的请求方法
CORS_ALLOW_METHODS = (
   'DELETE',
   'GET',
   'OPTIONS',
   'PATCH',
   'POST',
   'PUT',
   'VIEW',
)

# 允许的headers
CORS_ALLOW_HEADERS = (
   'accept',
   'accept-encoding',
   'authorization',
   'content-type',
   'dnt',
   'origin',
   'user-agent',
   'x-csrftoken',
   'x-requested-with',
)

 

后端原生写法

前端 Vue

我使用的是 gitee.com 上开源封装好的 vue
# 下载地址: git clone https://gitee.com/nmgwap/vueproject.git

# 配置
# /vueproject/config/index.js
proxyTable: {
           '/api': {
               target: 'http://127.0.0.1:8000', // 你请求的第三方接口
               changeOrigin: true, // 在本地会创建一个虚拟服务端,然后发送请求的数据,并同时接收请求的数据,这样服务端和服务端进行数据的交互就不会有跨域问题
               pathRewrite: { // 路径重写,
                   '^/api': '/api' // 替换target中的请求地址,也就是说以后你在请求http://api.jisuapi.com/XXXXX这个地址的时候直接写成/api即可。
              }
          }
      },

 

 

 

修改 login.vue, 它里面有 token 还有验证码认证, 我暂时后端还没有做,就需要先注释掉, 否则登录会失败

<template>
 <div class="login-wrap">
   <el-form label-position="left" :model="ruleForm" :rules="rules" ref="ruleForm" label-width="0px" class="demo-ruleForm login-container">
     <h3 class="title">用户登录</h3>
     <el-form-item prop="username">
       <el-input type="text" v-model="ruleForm.username" auto-complete="off" placeholder="账号"></el-input>
     </el-form-item>
     <el-form-item prop="password">
       <el-input type="password" v-model="ruleForm.password" auto-complete="off" placeholder="密码"></el-input>
     </el-form-item>
<!--     <el-row>-->
<!--       <el-col :span="12">-->
<!--         <el-form-item prop="code">-->
<!--           <el-input type="text" v-model="ruleForm.code" auto-complete="off" placeholder="图形验证码" @keyup.enter.native="submitForm('ruleForm')"></el-input>-->
<!--         </el-form-item>-->
<!--       </el-col>-->
<!--       <el-col :span="12" class="code-box">-->
<!--         <img :src="ruleForm.codeimg" alt="" class="codeimg" @click="getcode()">-->
<!--       </el-col>-->
<!--     </el-row>-->
     <el-checkbox class="remember" v-model="rememberpwd">记住密码</el-checkbox>
     <el-form-item style="width:100%;">
       <el-button type="primary" style="width:100%;" @click="submitForm('ruleForm')" :loading="logining">登录</el-button>
     </el-form-item>
   </el-form>
 </div>
</template>
<script type="text/ecmascript-6">
import { login } from '../api/userMG'
import { setCookie, getCookie, delCookie } from '../utils/util'
import md5 from 'js-md5'
export default {
name: 'login',
data() {
  return {
    //定义loading默认为false
    logining: false,
    // 记住密码
    rememberpwd: false,
    ruleForm: {
      //username和password默认为空
      username: '',
      password: '',
      code: '',
      randomStr: '',
      codeimg: ''
    },
    //rules前端验证
    rules: {
      username: [{ required: true, message: '请输入账号', trigger: 'blur' }],
      password: [{ required: true, message: '请输入密码', trigger: 'blur' }],
      // code: [{ required: true, message: '请输入验证码', trigger: 'blur' }]
    }
  }
},
// 创建完毕状态(里面是操作)
created() {
  this.$message({
    message: '账号密码及验证码不为空即可',
    type: 'success'
  })
  // 获取图形验证码
  // this.getcode()
  // 获取存在本地的用户名密码
  this.getuserpwd()

},
// 里面的函数只有调用才会执行
methods: {
  // 获取用户名密码
  getuserpwd() {
    if (getCookie('user') != '' && getCookie('pwd') != '') {
      this.ruleForm.username = getCookie('user')
      this.ruleForm.password = getCookie('pwd')
      this.rememberpwd = true
    }
  },
  //获取info列表
  submitForm(formName) {
    this.$refs[formName].validate(valid => {
      if (valid) {
        this.logining = true
        // 测试通道,不为空直接登录
        // setTimeout(() => {
        //   this.logining = false
        //   this.$store.commit('login', 'true')
        //   this.$router.push({ path: '/goods/Goods' })
        // }, 1000)
        // 注释
        login(this.ruleForm).then(res => {
          // alert(res.status)
          if (res.status === 201) {
            if (this.rememberpwd) {
              //保存帐号到cookie,有效期7天
              setCookie('user', this.ruleForm.username, 7)
              //保存密码到cookie,有效期7天
              setCookie('pwd', this.ruleForm.password, 7)
            } else {
              delCookie('user')
              delCookie('pwd')
            }
            //如果请求成功就让他2秒跳转路由
            setTimeout(() => {
              this.logining = false
              // 缓存token
              // localStorage.setItem('logintoken', res.data.token)
              // 缓存用户个人信息
              localStorage.setItem('userdata', JSON.stringify(res.data))
              this.$store.commit('login', 'true')
              this.$router.push({ path: '/goods/Goods' })
            }, 1000)
          } else {
            this.$message.error(res.msg)
            this.logining = false
            return false
          }
        })
      } else {
        // 获取图形验证码
        this.getcode()
        this.$message.error('请输入用户名密码!')
        this.logining = false
        return false
      }
    })
  },
}
}
</script>

<style scoped>
.login-wrap {
 box-sizing: border-box;
 width: 100%;
 height: 100%;
 padding-top: 10%;
 background-image: url();
 /* */
 background-repeat: no-repeat;
 background-position: center right;
 background-size: 100%;
}
.login-container {
 border-radius: 10px;
 margin: 0px auto;
 width: 350px;
 padding: 30px 35px 15px 35px;
 background: #fff;
 border: 1px solid #eaeaea;
 text-align: left;
 box-shadow: 0 0 20px 2px rgba(0, 0, 0, 0.1);
}
.title {
 margin: 0px auto 40px auto;
 text-align: center;
 color: #505458;
}
.remember {
 margin: 0px 0px 35px 0px;
}
.code-box {
 text-align: right;
}
.codeimg {
 height: 40px;
}
</style>

 

 

后端代码

# 参考我这篇博客, 把表创建出来就可以了。 然后把跨域配置好, 引入 drf  我的应用还是  appdemo
https://www.cnblogs.com/liuxiaomo/p/17480458.html

# 在 view.py 中添加如下代码
from django.core.exceptions import ObjectDoesNotExist
from rest_framework.response import Response
from rest_framework.status import HTTP_200_OK, HTTP_404_NOT_FOUND, HTTP_201_CREATED, HTTP_400_BAD_REQUEST, \
   HTTP_204_NO_CONTENT, HTTP_500_INTERNAL_SERVER_ERROR
from rest_framework.views import APIView

from appdemo.models import User
class LoginView(APIView):
   def post(self, request, *args, **kwargs):
       try:
           u_name = request.data['username']
           u = User.objects.get(u_name=u_name)
       except ObjectDoesNotExist:
           return Response({"status": HTTP_404_NOT_FOUND, "msg": "用户名不存在"})
       try:
           u_password = request.data['password']
           if u.u_password == u_password:
               return Response({"status": HTTP_201_CREATED, "msg": "登录失败!", "success": 201})
           else:
               return Response({"status": HTTP_404_NOT_FOUND, "msg": "密码错误"})
       except ObjectDoesNotExist:
           return Response({"status": HTTP_500_INTERNAL_SERVER_ERROR, "msg": "未知错误!"})

 

# 在 DjangoRestDemo/DjangoRestDemo/urls.py
from appdemo.views importLoginView
urlpatterns = [
....
   path('api/login', LoginView.as_view(), name='login')
]

 

# 这样就可以正常访问, 登录的接口了

 

 

使用序列化器

appdemo/serializers.py
# 添加如下内容:

from appdemo.models import User
from rest_framework import serializers
from django.core.exceptions import ObjectDoesNotExist


class LoginSerializer(serializers.ModelSerializer):
   """
  您可以在序列化器中使用source参数来指定前端发送的字段名与后端数据库中的字段名的映射关系。
  例如,如果前端发送的字段名为username,而后端数据库中的字段名为u_name,您可以将序列化器修改如下:

  这样就可以正确获取到前端发送的数据中的username字段,并将其映射到后端数据库中的u_name字段了。
  如果问题仍然存在,请提供更多的代码和错误信息,以便更好地帮助您解决问题。
   """
   username = serializers.CharField(source='u_name')
   password = serializers.CharField(source='u_password')

   class Meta:
       model = User
       fields = ['username', 'password']

   def validate(self, attrs):
       username = attrs.get('u_name')
       password = attrs.get('u_password')

       if not username:
           raise serializers.ValidationError('用户名不能为空')
       if not password:
           raise serializers.ValidationError('密码不能为空')
       try:
           u = User.objects.get(u_name=username)
       except ObjectDoesNotExist:
           raise serializers.ValidationError('用户名不存在')
       if u.u_password == password:
           return '请求成功'
       else:
           raise serializers.ValidationError('密码错误')

 

appdemo/views.py
# 添加如下内容,

from rest_framework.response import Response
from rest_framework.status import HTTP_200_OK, HTTP_404_NOT_FOUND, HTTP_201_CREATED, HTTP_400_BAD_REQUEST, \
HTTP_204_NO_CONTENT, HTTP_500_INTERNAL_SERVER_ERROR
from rest_framework.views import APIView

from appdemo.models import User
from appdemo.serializers import UserSerializer, UserViewSerializer, LoginSerializer

class LoginView(APIView):
def post(self, request, *args, **kwargs):
serializer = LoginSerializer(data=request.data)
if serializer.is_valid():
return Response({"status": HTTP_201_CREATED, "msg": "登录成功!", "success": 201})
return Response({
"msg": serializer.errors['non_field_errors'][0],
})

 

# 路由, 我是把之前的原生的views 代码注释掉, 所以没有新建新的路由
 
posted @ 2023-06-21 16:35  Handsome、Snake  阅读(62)  评论(0编辑  收藏  举报