123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536 |
- # -*- encoding: utf-8 -*-
- """
- @File : UnicomComboController.py
- @Time : 2022/6/23 9:18
- @Author : stephen
- @Email : zhangdongming@asj6.wecom.work
- @Software: PyCharm
- """
- import datetime
- import json
- import logging
- import time
- from django.db import transaction
- from django.http import HttpResponse, JsonResponse
- from django.views.generic.base import View
- from Model.models import UnicomDeviceInfo, UnicomCombo, Pay_Type, Order_Model, Store_Meal, AiStoreMeal, \
- UnicomComboOrderInfo, UnicomComboExperienceHistory
- from Object.ResponseObject import ResponseObject
- from Object.TokenObject import TokenObject
- from Object.UnicomObject import UnicomObjeect
- from Object.utils import LocalDateTimeUtil
- from Object.utils.PayUtil import PayService
- from Service.CommonService import CommonService
- class UnicomComboView(View):
- def get(self, request, *args, **kwargs):
- request.encoding = 'utf-8'
- operation = kwargs.get('operation')
- return self.validation(request.GET, request, operation)
- def post(self, request, *args, **kwargs):
- request.encoding = 'utf-8'
- operation = kwargs.get('operation')
- return self.validation(request.POST, request, operation)
- def validation(self, request_dict, request, operation):
- if operation == 'buy-notify':
- return self.package_callback_notify(request_dict, request)
- elif operation == 'test-notify':
- order_id = request_dict.get('orderId', None)
- activate_type = request_dict.get('activateType', 0)
- iccid = request_dict.get('iccid', None)
- combo_id = request_dict.get('comboId', None)
- self.create_combo_order_info(order_id, int(activate_type), iccid, int(combo_id))
- return HttpResponse('SUCCESS')
- elif operation == 'device-queue-monitoring':
- return self.device_queue_monitoring_push(request_dict, request)
- elif operation == 'device-status-change':
- return self.device_status_change_push(request_dict, request)
- elif operation == 'device-bind':
- response = ResponseObject('cn')
- return self.device_add(request_dict, response)
- elif operation == 'device-status':
- response = ResponseObject('cn')
- return self.update_device_status(request_dict, response)
- else:
- token = TokenObject(request.META.get('HTTP_AUTHORIZATION'))
- lang = request_dict.get('lang', token.lang)
- response = ResponseObject(lang)
- if token.code != 0:
- return response.json(token.code)
- user_id = token.userID
- if operation == 'combo-save':
- return self.save_unicom_combo(request_dict, response)
- elif operation == 'combo-pay':
- return self.buy_unicom_combo(user_id, request_dict, request, response)
- elif operation == 'combo-list':
- return self.query_package_list(response)
- elif operation == 'get-device-info':
- return self.get_device_info(request_dict, response)
- @classmethod
- def get_device_info(cls, request_dict, response):
- """
- 获取设备信息
- @param request_dict:
- @param response:
- @return:
- """
- serial_no = request_dict.get('serialNumber', None)
- if not serial_no:
- return response.json(444)
- unicom_device_info_qs = UnicomDeviceInfo.objects.filter(serial_no=serial_no).values()
- if not unicom_device_info_qs.exists():
- return response.json(173)
- return response.json(0, dict(unicom_device_info_qs.first()))
- @classmethod
- def device_add(cls, request_dict, response):
- """
- 设备绑定iccid
- @param request_dict:
- @param response:
- @return:
- """
- iccid = request_dict.get('iccid', None)
- serial_no = request_dict.get('serialNo', None)
- if not all([iccid, serial_no]):
- return response.json(444)
- n_time = int(time.time())
- try:
- with transaction.atomic():
- # 待完善代码 根据uid与用户id验证系统设备
- unicom_device_qs = UnicomDeviceInfo.objects.filter(iccid=iccid)
- if unicom_device_qs.exists():
- return response.json(174)
- unicom_device_serial_qs = UnicomDeviceInfo.objects.filter(serial_no=serial_no)
- if unicom_device_serial_qs.exists():
- return response.json(174)
- unicom_obj = UnicomObjeect()
- result = unicom_obj.verify_device(iccid=iccid)
- if result.status_code == 200 and result.text:
- res_dict = json.loads(result.text)
- if res_dict['success']:
- if res_dict['data']['status'] == 0:
- return response.json(173)
- params = {'iccid': iccid, 'serial_no': serial_no, 'updated_time': n_time,
- 'created_time': n_time}
- UnicomDeviceInfo.objects.create(**params)
- return response.json(0)
- else:
- return response.json(173)
- except Exception as e:
- print(e)
- return response.json(177, repr(e))
- @classmethod
- def user_activate_flow(cls, iccid):
- """
- 用户激活初始化流量套餐
- @param iccid:
- @return:
- """
- logger = logging.getLogger('info')
- try:
- now_time = int(time.time())
- unicom_device_info_qs = UnicomDeviceInfo.objects.filter(iccid=iccid)
- if not unicom_device_info_qs.exists():
- return False
- unicom_device_info_qs = unicom_device_info_qs.first()
- if unicom_device_info_qs.status != 1:
- logger.info('用户激活iccid={},业务系统状态为{}'.format(iccid, unicom_device_info_qs.status))
- return False
- # 联通业务逻辑
- unicom_api = UnicomObjeect()
- re_data = {'iccid': iccid}
- result = unicom_api.query_device_status(**re_data)
- res_dict = unicom_api.get_text_dict(result)
- # 状态不等于1(激活)时进行激活 1:激活;2:停用
- if res_dict['data']['status'] != 1:
- re_data = {"iccid": iccid, "status": 1}
- unicom_api.update_device_state(**re_data)
- # 查看是否体验过免费套餐
- experience_history_qs = UnicomComboExperienceHistory.objects.filter(iccid=iccid)
- if not experience_history_qs.exists():
- logger.info('用户首次激活iccid={}'.format(iccid))
- combo_qs = UnicomCombo.objects.filter(combo_type=1, status=0, is_del=False) \
- .values('id', 'expiration_type', 'expiration_days', 'combo_type')
- if combo_qs.exists():
- combo_qs = combo_qs.first()
- # 保存体验记录
- experience_history_vo = {'iccid': iccid, 'experience_type': 0, 'do_time': now_time}
- UnicomComboExperienceHistory.objects.create(**experience_history_vo)
- # 保存套餐激活信息
- cls.create_combo_order_info('', 0, iccid, combo_qs['id'])
- # 修改业务联通卡设备激活信息
- UnicomDeviceInfo.objects.filter(iccid=iccid).update(status=2, updated_time=now_time)
- return True
- except Exception as e:
- print(e)
- return False
- @classmethod
- def save_unicom_combo(cls, request_dict, response):
- """
- 联通套餐保存
- @param request_dict:
- @param response:
- @return:
- """
- combo_id = request_dict.get('id', None)
- combo_name = request_dict.get('comboName', None)
- flow_total = request_dict.get('flowTotal', None)
- expiration_days = request_dict.get('expirationDays', None)
- expiration_type = request_dict.get('expirationType', None)
- price = request_dict.get('price', None)
- remark = request_dict.get('remark', None)
- pay_type = request_dict.get('payType', '').split(',')
- if not all([pay_type, combo_name, flow_total, expiration_days, expiration_type, price]):
- return response.json(444)
- try:
- flow_total = int(flow_total)
- expiration_days = int(expiration_days)
- expiration_type = int(expiration_type)
- with transaction.atomic():
- re_data = {
- 'combo_name': combo_name,
- 'flow_total': flow_total,
- 'expiration_days': expiration_days,
- 'expiration_type': expiration_type,
- 'price': price,
- }
- if remark:
- re_data['remark'] = remark
- if combo_id:
- UnicomCombo.objects.filter(id=combo_id).update(**re_data)
- UnicomCombo.objects.get(id=combo_id).pay_type.set(pay_type)
- return response.json(0)
- UnicomCombo.objects.create(**re_data).pay_type.set(pay_type)
- return response.json(0)
- except Exception as e:
- print(e)
- return response.json(177, repr(e))
- @classmethod
- def query_package_list(cls, response):
- """
- 查询套餐列表
- @return:
- """
- try:
- combo_qs = UnicomCombo.objects.filter(is_show=1, status=0, is_del=False) \
- .order_by('sort').values('id', 'combo_name',
- 'flow_total',
- 'expiration_days',
- 'expiration_type', 'price',
- 'remark')
- if not combo_qs.exists():
- return response.json(0, [])
- combo_list = []
- for item in combo_qs:
- # 获取支付方式列表
- pay_type_qs = Pay_Type.objects.filter(unicomcombo=item['id']).values('id', 'payment')
- combo_list.append({
- 'id': item['id'],
- 'comboName': item['combo_name'],
- 'flowTotal': item['flow_total'],
- 'expirationDays': item['expiration_days'],
- 'expirationType': item['expiration_type'],
- 'price': item['price'],
- 'remark': item['remark'],
- 'payTypes': list(pay_type_qs),
- })
- return response.json(0, combo_list)
- except Exception as e:
- print(e)
- return response.json(177, repr(e))
- @classmethod
- def buy_unicom_combo(cls, user_id, request_dict, request, response):
- """
- 购买联通套餐
- @return:
- """
- try:
- with transaction.atomic():
- iccid = request_dict.get('iccid', None)
- combo_id = request_dict.get('id', None)
- pay_type = request_dict.get('payType', None)
- activate_type = request_dict.get('activateType', 0)
- if not all([iccid, combo_id, pay_type]):
- return response.json(444)
- combo_id = int(combo_id)
- pay_type = int(pay_type)
- now_time = int(time.time())
- unicom_combo_qs = UnicomCombo.objects.filter(id=combo_id, pay_type=pay_type, is_show=1, is_del=False,
- status=0) \
- .values('id', 'combo_name', 'price', 'remark')
- if not unicom_combo_qs.exists():
- return response.json(173)
- unicom_device_qs = UnicomDeviceInfo.objects.filter(iccid=iccid) \
- .values('serial_no')
- if not unicom_device_qs.exists():
- return response.json(173)
- unicom_combo_qs = unicom_combo_qs.first()
- price = unicom_combo_qs['price']
- if not price:
- return response.json(173)
- unicom_device_qs = unicom_device_qs.first()
- device_uid = CommonService.query_uid_with_serial(unicom_device_qs['serial_no'])
- order_id = CommonService.createOrderID()
- rank_id, ai_rank_id = cls.get_cloud_or_ai_combo()
- order_dict = {'rank_id': rank_id, 'ai_rank_id': ai_rank_id, 'orderID': order_id, 'UID': device_uid,
- 'userID_id': user_id, 'desc': unicom_combo_qs['combo_name'], 'payType': pay_type,
- 'payTime': now_time, 'price': price, 'currency': 'CNY', 'addTime': now_time,
- 'updTime': now_time,
- 'unify_combo_id': str(unicom_combo_qs['id']), 'order_type': 2,
- }
- params = 'lang=cn' + '&activateType=' + activate_type
- result = {'result_code': 0, 'reason': 'success', 'error_code': 0}
- if pay_type == 2: # 支付宝
- pay_price = PayService.get_two_float(price, 2)
- notify_url = 'unicom/wap/pay/ali-notify'
- order_dict['pay_url'] = PayService.create_alipay_payment(params, order_id, pay_price,
- unicom_combo_qs['combo_name'],
- notify_url,
- unicom_combo_qs['remark'], response)
- res_data = {'redirectUrl': order_dict['pay_url'], 'orderID': order_id}
- elif pay_type == 3: # 微信支付
- notify_url = 'unicom/wap/pay/wechat-notify'
- ip = CommonService.get_ip_address(request)
- params = 'activateType=' + activate_type
- sign_params = PayService.create_wechat_payment(params, order_id,
- price, ip,
- notify_url,
- unicom_combo_qs['remark'],
- response)
- result['result'] = sign_params
- else:
- return response.json(444, {'param': 'pay_type'})
- Order_Model.objects.create(**order_dict)
- if pay_type == 3:
- return JsonResponse(status=200, data=result)
- return response.json(0, res_data)
- except Exception as e:
- print(e)
- return response.json(500, repr(e))
- @staticmethod
- def get_cloud_or_ai_combo():
- """
- 获取云存或者AI套餐id
- @return:
- """
- rank_id = Store_Meal.objects.all().order_by('sort').values().first()['id']
- ai_rank_id = AiStoreMeal.objects.all().values().first()['id']
- return rank_id, ai_rank_id
- @classmethod
- def query_device_usage_history(cls):
- """
- 查询用量历史
- @return:
- """
- @staticmethod
- def package_callback_notify(request_dict, request):
- """
- 异步套餐订购回调
- @param request_dict:
- @param request:
- @return:
- """
- logger = logging.getLogger('info')
- try:
- logger.info('联通异步套餐订购回调参数{}'.format(request_dict))
- body = request.body.decode("utf-8")
- if body:
- dict_data = json.loads(body)
- sign = dict_data['sign']
- logger.info('设备订购异步回调请求参数{}'.format(dict_data))
- dict_data.pop('sign')
- unicom_obj = UnicomObjeect()
- generate_sign = unicom_obj.createSign(**dict_data)
- logger.info('设备订购请求签名{}'.format(sign))
- logger.info('设备订购生成签名{}'.format(generate_sign))
- r_data = {'success': True, 'msg': '成功'}
- return HttpResponse(json.dumps(r_data, ensure_ascii=False), content_type="application/json,charset=utf-8")
- except Exception as e:
- print(repr(e))
- r_data = {'success': False, 'msg': '失败'}
- return HttpResponse(json.dumps(r_data, ensure_ascii=False), content_type="application/json,charset=utf-8")
- @staticmethod
- def device_queue_monitoring_push(request_dict, request):
- """
- 设备套餐队列用完或者到期推送
- @param request_dict:
- @param request:
- @return:
- """
- logger = logging.getLogger('info')
- try:
- logger.info('设备套餐队列推送{}'.format(request_dict))
- body = request.body.decode("utf-8")
- if body:
- dict_data = json.loads(body)
- sign = dict_data['sign']
- logger.info('设备套餐队列回调请求参数{}'.format(dict_data))
- dict_data.pop('sign')
- unicom_obj = UnicomObjeect()
- generate_sign = unicom_obj.createSign(**dict_data)
- logger.info('设备套餐队列请求签名{}'.format(sign))
- logger.info('设备套餐队列生成签名{}'.format(generate_sign))
- r_data = {'success': True, 'msg': '成功'}
- return HttpResponse(json.dumps(r_data, ensure_ascii=False), content_type="application/json,charset=utf-8")
- except Exception as e:
- print(repr(e))
- r_data = {'success': False, 'msg': '失败'}
- return HttpResponse(json.dumps(r_data, ensure_ascii=False), content_type="application/json,charset=utf-8")
- @staticmethod
- def device_status_change_push(request_dict, request):
- """
- 设备状态变更推送执行场景说明
- @param request_dict:
- @param request:
- @return:
- """
- logger = logging.getLogger('info')
- try:
- logger.info('设备状态变更推送{}'.format(request_dict))
- body = request.body.decode("utf-8")
- if body:
- dict_data = json.loads(body)
- sign = dict_data['sign']
- logger.info('设备状态变更推送请求参数{}'.format(dict_data))
- dict_data.pop('sign')
- unicom_obj = UnicomObjeect()
- generate_sign = unicom_obj.createSign(**dict_data)
- logger.info('设备状态变更推送请求签名{}'.format(sign))
- logger.info('设备状态变更推送生成签名{}'.format(generate_sign))
- r_data = {'success': True, 'msg': '成功'}
- return HttpResponse(json.dumps(r_data, ensure_ascii=False), content_type="application/json,charset=utf-8")
- except Exception as e:
- print(repr(e))
- r_data = {'success': False, 'msg': '失败'}
- return HttpResponse(json.dumps(r_data, ensure_ascii=False), content_type="application/json,charset=utf-8")
- @classmethod
- def create_combo_order_info(cls, order_id, activate_type, iccid, combo_id):
- """
- 创建套餐生效记录
- @param order_id: 订单id
- @param activate_type: 激活类型
- @param iccid: 联通20位iccid
- @param combo_id: 套餐id
- @return: True Or False
- """
- logger = logging.getLogger('info')
- logger.info('创建联通订单套餐信息,订单id{}'.format(order_id))
- try:
- today = datetime.datetime.today()
- year = today.year
- month = today.month
- with transaction.atomic():
- unicom_combo_qs = UnicomCombo.objects.filter(id=int(combo_id)).values()
- if unicom_combo_qs.exists():
- unicom_combo = unicom_combo_qs.first()
- now_time = int(time.time())
- combo_order_qs = UnicomComboOrderInfo.objects.filter(status=1, iccid=iccid)
- status = 0
- if not combo_order_qs.exists():
- status = 1
- combo_order_data = {'iccid': iccid, 'status': status, 'combo_id': int(combo_id),
- 'updated_time': now_time,
- 'created_time': now_time,
- 'year': year, 'month': month}
- if order_id:
- combo_order_data['order_id'] = order_id
- # 有效期类型 1 等于自然月,0天数
- if unicom_combo['expiration_type'] == 1:
- # 激活类型 1=下月激活 否则等于当月激活
- if activate_type == 1:
- combo_order_data['next_month_activate'] = True
- next_start_time, end_time = cls.get_next_month_data_time()
- combo_order_data['activation_time'] = next_start_time
- combo_order_data['expire_time'] = end_time
- combo_order_data['status'] = 0
- else:
- start_time, month_end_time = cls.get_month_start_and_end_time()
- combo_order_data['activation_time'] = now_time
- combo_order_data['expire_time'] = month_end_time
- elif unicom_combo['expiration_type'] == 0:
- days = unicom_combo['expiration_days']
- zero_today, end_time = cls.get_data_time(days)
- combo_order_data['activation_time'] = now_time
- combo_order_data['expire_time'] = end_time
- # 联通业务逻辑
- unicom_api = UnicomObjeect()
- if status == 1:
- re_data = {'iccid': iccid}
- result = unicom_api.query_device_status(**re_data)
- res_dict = unicom_api.get_text_dict(result)
- # 状态不等于1(激活)时进行激活 1:激活;2:停用
- if res_dict and res_dict['data']['status'] != 1:
- re_data = {"iccid": iccid, "status": 1}
- unicom_api.update_device_state(**re_data)
- flow_total_usage = unicom_api.get_flow_usage_total(year, month, iccid)
- combo_order_data['flow_total_usage'] = str(flow_total_usage)
- UnicomComboOrderInfo.objects.create(**combo_order_data)
- logger.info('保存套餐支付信息success')
- return True
- except Exception as e:
- print(e)
- return False
- @staticmethod
- def get_next_month_data_time():
- """
- 获取下个月开始时间与结束时间戳
- @return: next_start_time,end_time
- """
- next_month_start = LocalDateTimeUtil.get_next_month_start()
- next_start_time, next_end_time = LocalDateTimeUtil.get_start_and_end_time(next_month_start, '%Y-%m-%d')
- next_month_end = LocalDateTimeUtil.get_next_month_end()
- start_time, end_time = LocalDateTimeUtil.get_start_and_end_time(next_month_end, '%Y-%m-%d')
- return next_start_time, end_time
- @staticmethod
- def get_data_time(days):
- """
- 获取今天开始时间以及days后日期结束时间戳
- @return: next_start_time,end_time
- """
- zero_today, last_today = LocalDateTimeUtil.get_today_date(True)
- now_time = int(time.time())
- after_time = LocalDateTimeUtil.get_after_days_timestamp(now_time, days)
- time_array = time.localtime(after_time)
- start_time, end_time = LocalDateTimeUtil.get_start_and_end_time(time.strftime("%Y-%m-%d", time_array),
- '%Y-%m-%d')
- return zero_today, end_time
- @staticmethod
- def get_month_start_and_end_time():
- """
- 获取当天开始时间与当月结束时间戳
- @return:
- """
- zero_today, last_today = LocalDateTimeUtil.get_today_date(True)
- month_end = LocalDateTimeUtil.get_cur_month_end()
- start_time, month_end_time = LocalDateTimeUtil.get_start_and_end_time(month_end, '%Y-%m-%d')
- return zero_today, month_end_time
|