celery异步发送短信并注册用户信息

1.1在celery_task/mian.py 中添加发送短信函数

# celery项目中的所有导包地址, 都是以CELERY_BASE_DIR为基准设定.
# 执行celery命令时, 也需要进入CELERY_BASE_DIR目录执行.



# -*- coding: utf-8 -*-
import os
from celery import Celery
import sys
CELERY_BASE_DIR = os.path.dirname(os.path.abspath(__file__))


# 定义celery实例, 需要的参数, 1, 实例名, 2, 任务发布位置, 3, 结果保存位置

app = Celery('mycelery',
            broker='redis://127.0.0.1:6379/14', # 任务存放的地方
            backend='redis://127.0.0.1:6379/15') # 结果存放的地方




@app.task
def add(x, y):
    return x + y


from libs.rl_sms import send_message
@app.task(bind=True)
def send_sms_code(self, mobile, datas):
    sys.path.insert(0, os.path.join(CELERY_BASE_DIR, '../syl'))
    # 在方法中导包

    try:
        # 用 res 接收发送结果, 成功是:0, 失败是:-1
        res = send_message(mobile, datas)

    except Exception as e:
        res = '-1'
    if res == '-1':
        # 如果发送结果是 -1  就重试.
        self.retry(countdown=5, max_retries=3, exc=Exception('短信发送失败'))
celery_task/mian.py

 

1.2 在verifications/views.py中添加celery发送短信试图函数

from rest_framework.permissions import AllowAny
from rest_framework.views import APIView
from rest_framework.response import Response
import re
import random
from libs.rl_sms import send_message
import sys
import os


class SmsCodeView(APIView):
    """使用apiview的限流"""
    # 1. 所有人可以访问
    permission_classes = (AllowAny,)

    def post(self, request, send_sms_code=None):
        # 1. 获取参数
        phone = request.data.get('phone')  # 手机号
        image_code = request.data.get('image_code')  # 图片验证码
        image_code_uuid = request.data.get('image_code_uuid')  # 前端生成的uuid

        # 2. 检查参数
        if not all([phone, image_code, image_code_uuid]):
            return Response({"code": 999, "msg": "参数不全"})
        if not re.match(r'^1[3456789]\d{9}$', phone):
            return Response({"code": 999, "msg": "手机号码不正确"})

        # 3. 检查是否发送
        redis_client = get_redis_connection('img_code')
        phone_exists = redis_client.get(phone)
        if phone_exists:
            return Response({"code": 999, "msg": "频繁发送, 请稍后再试"})

        # # 验证图形验证码
        redis_image_code = redis_client.get(image_code_uuid)  # bytes
        if redis_image_code:
        #     # bytes 转成 string
            redis_image_code = redis_image_code.decode()
        #
        # # 比较用户提供的图片内容是否和redis中保存的一致
        if image_code.upper() != redis_image_code:
            return Response({'code': 999, 'msg': '图片验证码不22正确'})

        # 4. 发送
        code = '%06d' % random.randint(0, 999999)  # 随机6位验证码

        from syl.settings import BASE_DIR
        sys.path.insert(0, os.path.join(BASE_DIR, '../celery_task'))
        from celery_task.main import send_sms_code    # 必须这么写, 从main中导包
        send_sms_code(phone, (code, "5"))

        # # 5.1 保存code 到 redis中
        # redis_client.setex(phone, 60 * 5, code)  # phone:code, 5分钟有效期
        # # 5.2 从redis中删除这个图片验证码, 以防再次被使用
        # redis_client.delete(image_code_uuid)

        # 5.3.使用 pipeline 批量操作
        pl = redis_client.pipeline()    # 实例化pipeline对象
        pl.setex(phone, 60 * 5, code)   # 存储phone:code, 5分钟有效期
        pl.delete(image_code_uuid)      # 从redis中删除这个图片验证码, 以防再次被使用
        pl.execute()

        # 6. 返回结果
        return Response({"code": 0, "msg": "短信发送成功"})
verifications/views.py

 

1.3 添加路由

from django.urls import path
from . import views

urlpatterns = [
    # path('image_codes/', views.ImageCodeView.as_view()),
    path('sms_codes/', views.SmsCodeView.as_view()),
]

 

2配置手机号是否符合,并注册

2.1判断手机号是否符合user/urls.py

urlpatterns = [
    path('register/', views.RegisterView.as_view()),# 注册视图

    path('count/', views.RegCountView.as_view()),  # 查询用户名手机号使用量的视图


]
user/urls.py

 

2.2判断手机号是否符合要求和注册user/views.py

import datetime
import random




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 .models import User
from .serializers import UserSerializer, UserUnActiveSerializer, UserInfoSerializer


from rest_framework_jwt.authentication import JSONWebTokenAuthentication


from django_redis import get_redis_connection










# 注册接口
class RegisterView(APIView):
    """
    用户注册, 权限是: 匿名用户可访问
    """
    # 自定义权限类
    permission_classes = (AllowAny,)

    def post(self, request):
        """
        接收用户名,密码,手机号和验证码, 前端校验两遍一致性, 注册成功后返回成功, 然后用户自行登录获取token
        1. 用户名
        2. 密码
        3. 手机号
        4. 验证码
        :param request:
        :return:  {'code':0,'msg':'注册成功'}
        code: "260361"
        password: "123123"
        phone: "13303479527"
        username: "liangxuepeng"
        """
        username = request.data.get('username')
        phone = request.data.get('phone')
        code = request.data.get('code')
        passwrod = request.data.get('password')
        print(username,passwrod,phone,code)
        print("*"*50)

        if all([username, passwrod, phone, code]):
            pass
        else:
            return Response({'code': 999, 'msg': '参数不全'})

        # rand_name = self.randomUsername()
        # 验证手机验证码


        redis_client = get_redis_connection('img_code')
        # print(redis_client)
        # print("*" * 50)
        code_redis = redis_client.get(phone)
        print(code_redis)
        print(phone)
        print("%"*50)
        # print(code_redis)
        # print("*" * 50)
        # print(code)
        # print("*" * 50)
        if code_redis:
            code_redis = code_redis.decode()

        if not code == code_redis:
            return Response({'code': 999, 'msg': '111手机验证码错误'})

        user = User(username=username, phone=phone)
        user.set_password(passwrod)
        user.save()

        return Response({'code': 0, 'msg': '注册成功'})






# 查询用户数量接口
class RegCountView(APIView):
    # 注册时需要验证的用户名和手机号是否使用

    # 自定义权限类
    permission_classes = (AllowAny,)

    def post(self, request):
        # 接收参数:  验证的内容type: username/phone,  data: '用户名' 或者 '手机号',
        datatype = request.data.get('type')
        # print(datatype)
        data = request.data.get('data')
        print(data)
        if not all([data, datatype]):
            return Response({'code': 999, 'msg': '参数不完整'})
        if datatype == 'username':


            count= User.objects.filter(username=data).count()
        if datatype == 'phone':
            count = User.objects.filter(phone=data).count()

        return Response({'code': 0, 'msg': '查询成功', 'data': {'type': datatype, 'count':count}})
user/views.py

 

 

 

2.3接收手机号验证码及

注册荣联 *云通讯 账号

syl/libs/rl_sms.py

from ronglian_sms_sdk import SmsSDK

accId = '8a216da874af5fff0175009949c119d7'
accToken = '3a65d8861d284a748f08f1a0084d257f'
appId = '8a216da874af5fff017500994aa619de'


def send_message(phone,datas):
    sdk = SmsSDK(accId, accToken, appId)
    tid = '1'
    # mobile = '18238693040'
    # datas = ('666666', '5')
    resp = sdk.sendMessage(tid,phone, datas)
    return resp
syl/libs/rl_sms.py

 

 

 

接口及参数

http://192.168.56.100:8888/verify/sms_codes/

 

 

 

posted @ 2020-11-04 20:12  小虾米爱吃鱼  阅读(97)  评论(0编辑  收藏  举报