| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539 | 
							- # -*- coding: utf-8 -*-
 
- """
 
- @Author : Rocky
 
- @Time : 2022/6/29 9:31
 
- @File :SmartSceneController.py
 
- """
 
- import json
 
- import time
 
- import requests
 
- from django.core.exceptions import ObjectDoesNotExist
 
- from django.db import transaction
 
- from django.db.models import F, Q, Count
 
- from django.views import View
 
- from Ansjer.config import DETECT_PUSH_DOMAINS
 
- from Ansjer.Config.gatewaySensorConfig import SMART_SCENE_TOPIC, SENSOR_TYPE, EVENT_TYPE, SCENE_EVENT_CREATE, \
 
-     SCENE_EVENT_EDIT, SCENE_EVENT_DELETE, SCENE_STATUS_ON, SCENE_STATUS_OFF, SCENE_EVENT_EDIT_STATUS, \
 
-     VOICE_AUDITION_TOPIC, SMART_SOCKET_TOPIC, DEVICE_TYPE
 
- from Model.models import FamilyRoomDevice, GatewaySubDevice, FamilyRoom, SmartScene, EffectiveTime, Device_Info, \
 
-     SceneLog
 
- from Object.ApschedulerObject import ApschedulerObject
 
- from Object.CeleryBeatObject import CeleryBeatObj
 
- from Object.ResponseObject import ResponseObject
 
- from Service.CommonService import CommonService
 
- class SmartSceneView(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 == 'get-scene':  # 设备获取智能场景数据
 
-             return self.get_scene_data(request_dict, ResponseObject('cn'))
 
-         token_code, user_id, response = CommonService.verify_token_get_user_id(request_dict, request)
 
-         if token_code != 0:
 
-             return response.json(token_code)
 
-         if operation == 'condition-devices':  # 添加条件-查询设备
 
-             return self.condition_devices(request_dict, response)
 
-         elif operation == 'task-devices':  # 添加任务-查询设备
 
-             return self.task_devices(request_dict, user_id, response)
 
-         elif operation == 'create':  # 创建智能场景
 
-             return self.create_smart_scene(request_dict, user_id, response)
 
-         elif operation == 'scene-list':  # 查询智能场景列表
 
-             return self.scene_list(request_dict, user_id, response)
 
-         elif operation == 'smart-button-scene-list':  # 查询智能按钮场景列表
 
-             return self.smart_button_scene_list(request_dict, user_id, response)
 
-         elif operation == 'update-status':  # 更新智能场景状态
 
-             return self.update_status(request_dict, response)
 
-         elif operation == 'detail':  # 查询智能场景详情
 
-             return self.scene_detail(request_dict, response)
 
-         elif operation == 'edit':  # 编辑智能场景
 
-             return self.edit_smart_scene(request_dict, user_id, response)
 
-         elif operation == 'delete':  # 删除智能场景
 
-             return self.delete_smart_scene(request_dict, response)
 
-         elif operation == 'log':  # 查询智能场景日志
 
-             return self.scene_log(request_dict, response)
 
-         elif operation == 'log-date':  # 查询智能场景日志日期
 
-             return self.scene_log_date(request_dict, response)
 
-         elif operation == 'voice-audition':  # 智能场景音频试听
 
-             return self.voice_audition(request_dict, response)
 
-         else:
 
-             return response.json(414)
 
-     @classmethod
 
-     def condition_devices(cls, request_dict, response):
 
-         """
 
-         添加条件-查询设备
 
-         @param request_dict: 请求参数
 
-         @request_dict deviceId: 网关设备id
 
-         @request_dict subDeviceId: 子设备id
 
-         @param response: 响应对象
 
-         @return: response
 
-         """
 
-         device_id = request_dict.get('deviceId', None)
 
-         sub_device_id = request_dict.get('subDeviceId', None)
 
-         if not any([device_id, sub_device_id]):
 
-             return response.json(444, {'error param': 'deviceId or subDeviceId'})
 
-         try:
 
-             if sub_device_id:
 
-                 device_id = GatewaySubDevice.objects.get(id=sub_device_id).device_id
 
-             gateway_sub_device_qs = GatewaySubDevice.objects.filter(device_id=device_id)
 
-             if not gateway_sub_device_qs.exists():
 
-                 return response.json(173)
 
-             res = cls.get_sub_device_room_name(gateway_sub_device_qs)
 
-             return response.json(0, res)
 
-         except Exception as e:
 
-             return response.json(500, 'error_line:{}, error_msg:{}'.format(e.__traceback__.tb_lineno, repr(e)))
 
-     @classmethod
 
-     def task_devices(cls, request_dict, user_id, response):
 
-         """
 
-         添加任务-查询设备
 
-         @param request_dict: 请求参数
 
-         @request_dict deviceId: 网关设备id
 
-         @param user_id: 用户id
 
-         @param response: 响应对象
 
-         @return: response
 
-         """
 
-         sub_device_id = request_dict.get('subDeviceId', None)
 
-         device_id = request_dict.get('deviceId', None)
 
-         if not any([device_id, sub_device_id]):
 
-             return response.json(444, {'error param': 'deviceId or subDeviceId'})
 
-         try:
 
-             if device_id:
 
-                 res = [cls.get_gateway_data(device_id)]
 
-             else:
 
-                 sub_device_qs = GatewaySubDevice.objects.filter(id=sub_device_id).values('device_id', 'device_type')
 
-                 device_id = sub_device_qs[0]['device_id']
 
-                 device_type = sub_device_qs[0]['device_type']
 
-                 if device_type != SENSOR_TYPE['smart_button']:  # 非智能按钮只返回网关
 
-                     res = [cls.get_gateway_data(device_id)]
 
-                 else:  # 智能按钮返回网关,门磁和人体传感器(如果存在)
 
-                     gateway_data = cls.get_gateway_data(device_id)
 
-                     sub_device_qs = GatewaySubDevice.objects.filter(
 
-                         Q(Q(device_id=device_id) & Q(device_type=SENSOR_TYPE['door_magnet'])) |
 
-                         Q(Q(device_id=device_id) & Q(device_type=SENSOR_TYPE['body_sensor']))
 
-                     ).values('id', 'nickname', 'status', 'device_type')
 
-                     if sub_device_qs.exists():
 
-                         res = cls.get_sub_device_room_name(sub_device_qs, gateway_data)
 
-                     else:
 
-                         res = [gateway_data]
 
-             # 添加插座和开关数据
 
-             res = cls.append_plug_and_switch_data(res, user_id)
 
-             return response.json(0, res)
 
-         except Exception as e:
 
-             return response.json(500, 'error_line:{}, error_msg:{}'.format(e.__traceback__.tb_lineno, repr(e)))
 
-     @staticmethod
 
-     def get_sub_device_room_name(sub_device_qs, gateway_data=None):
 
-         """
 
-         获取房间名称
 
-         @param sub_device_qs: 子设备信息
 
-         @param gateway_data: 网关参数
 
-         @return: sub_device_list
 
-         """
 
-         sub_device_list = []
 
-         if gateway_data:
 
-             sub_device_list.append(gateway_data)
 
-         sub_device_qs = sub_device_qs.annotate(gatewaySubId=F('id'),
 
-                                                deviceType=F('device_type'),
 
-                                                deviceNickName=F('nickname')). \
 
-             values('gatewaySubId', 'deviceType', 'deviceNickName', 'status')
 
-         for sub_device in sub_device_qs:
 
-             family_room_device_qs = FamilyRoomDevice.objects.filter(sub_device=sub_device['gatewaySubId']). \
 
-                 values('room_id')
 
-             if not family_room_device_qs.exists():
 
-                 sub_device['roomName'] = ''
 
-             else:
 
-                 room_id = family_room_device_qs[0]['room_id']
 
-                 try:
 
-                     sub_device['roomName'] = FamilyRoom.objects.get(id=room_id).name
 
-                 except ObjectDoesNotExist:
 
-                     sub_device['roomName'] = ''
 
-             sub_device_list.append(sub_device)
 
-         return sub_device_list
 
-     @staticmethod
 
-     def get_gateway_data(device_id):
 
-         """
 
-         获取网关数据
 
-         @param device_id: 网关设备id
 
-         @return: res
 
-         """
 
-         device_info_qs = Device_Info.objects.filter(id=device_id).values('NickName', 'Type', 'serial_number')
 
-         nickname = device_info_qs[0]['NickName']
 
-         device_type = device_info_qs[0]['Type']
 
-         serial_number = device_info_qs[0]['serial_number']
 
-         room_id = FamilyRoomDevice.objects.filter(device_id=device_id).values('room_id')[0]['room_id']
 
-         room_id_qs = FamilyRoom.objects.filter(id=room_id).values('name')
 
-         room_name = room_id_qs.first()['name'] if room_id_qs.exists() else ''
 
-         res = {
 
-             'serialNumber': serial_number,
 
-             'deviceNickName': nickname,
 
-             'deviceType': device_type,
 
-             'roomName': room_name,
 
-             'status': 1,
 
-         }
 
-         return res
 
-     @staticmethod
 
-     def append_plug_and_switch_data(res, user_id):
 
-         """
 
-         添加插座和开关数据
 
-         @param res: 网关等设备数据
 
-         @param user_id: 用户id
 
-         @return: res
 
-         """
 
-         device_info_qs = Device_Info.objects.filter(userID_id=user_id, Type__in=[DEVICE_TYPE['socket']]).\
 
-             values('id', 'NickName', 'Type', 'serial_number')
 
-         if device_info_qs.exists():
 
-             for device_info in device_info_qs:
 
-                 nickname = device_info['NickName']
 
-                 device_type = device_info['Type']
 
-                 serial_number = device_info['serial_number']
 
-                 device_id = device_info['id']
 
-                 room_id = FamilyRoomDevice.objects.filter(device_id=device_id).values('room_id')[0]['room_id']
 
-                 room_id_qs = FamilyRoom.objects.filter(id=room_id).values('name')
 
-                 room_name = room_id_qs.first()['name'] if room_id_qs.exists() else ''
 
-                 data = {
 
-                     'serialNumber': serial_number,
 
-                     'deviceNickName': nickname,
 
-                     'deviceType': device_type,
 
-                     'roomName': room_name,
 
-                     'status': 1,
 
-                 }
 
-                 res.append(data)
 
-         return res
 
-     @classmethod
 
-     def create_smart_scene(cls, request_dict, user_id, response):
 
-         """
 
-         创建智能场景
 
-         @param request_dict: 请求参数
 
-         @param user_id: 用户id
 
-         @request_dict deviceId: 网关设备id
 
-         @request_dict subDeviceId: 子设备id
 
-         @request_dict sceneName: 场景名称
 
-         @request_dict conditions: 条件
 
-         @request_dict tasks: 任务
 
-         @request_dict isAllDay: 是否全天执行
 
-         @request_dict startTime: 开始时间
 
-         @request_dict endTime: 结束时间
 
-         @request_dict repeat: 重复周期
 
-         @param response: 响应对象
 
-         @return: response
 
-         """
 
-         device_id = request_dict.get('deviceId', None)
 
-         sub_device_id = request_dict.get('subDeviceId', None)
 
-         scene_name = request_dict.get('sceneName', None)
 
-         conditions = request_dict.get('conditions', None)
 
-         tasks = request_dict.get('tasks', None)
 
-         is_all_day = request_dict.get('isAllDay', None)
 
-         if not all([scene_name, conditions, tasks]):
 
-             return response.json(444, {'error param': 'scene_name and conditions and tasks'})
 
-         now_time = int(time.time())
 
-         conditions_dict = eval(conditions)
 
-         tasks_list = eval(tasks)
 
-         try:
 
-             # 判断是否已存在该场景名
 
-             smart_scene_qs = SmartScene.objects.filter(user_id=user_id, scene_name=scene_name)
 
-             if smart_scene_qs.exists():
 
-                 return response.json(179)
 
-             tz = CommonService.get_user_tz(user_id)
 
-             smart_scene_dict = {
 
-                 'user_id': user_id,
 
-                 'scene_name': scene_name,
 
-                 'conditions': conditions,
 
-                 'tasks': tasks,
 
-                 'tz': tz,
 
-                 'created_time': now_time,
 
-                 'updated_time': now_time,
 
-             }
 
-             msg = {
 
-                 'scene_event': SCENE_EVENT_CREATE,
 
-                 'scene_status': SCENE_STATUS_ON
 
-             }
 
-             # 处理设置时间
 
-             if is_all_day is not None:
 
-                 is_all_day = int(is_all_day)
 
-                 smart_scene_dict['is_all_day'] = is_all_day
 
-             # 判断条件是否为设置时间
 
-             is_set_time, minutes, repeat = False, 0, 0
 
-             if conditions_dict['type'] == 1:
 
-                 is_set_time = True
 
-                 minutes = conditions_dict['time']['minutes']
 
-                 repeat = conditions_dict['time']['repeat']
 
-             # 条件为设置时间
 
-             if is_set_time:
 
-                 if not device_id:
 
-                     return response.json(444, {'error param': 'deviceId'})
 
-                 smart_scene_dict['device_id'] = device_id
 
-                 device_info_qs = Device_Info.objects.filter(id=device_id).values('serial_number')
 
-                 if not device_info_qs.exists():
 
-                     return response.json(173)
 
-                 serial_number = device_info_qs[0]['serial_number']
 
-                 # 网关数据
 
-                 msg['sensor_type'] = DEVICE_TYPE['gateway']
 
-                 msg['sensor_status'] = 2002
 
-                 msg['sensor_ieee_addr'] = 'FFFFFFFFFFFFFFFF'
 
-             # 条件为传感器设备
 
-             else:
 
-                 if not sub_device_id:
 
-                     return response.json(444, {'error param': 'subDeviceId'})
 
-                 # 查询数据
 
-                 sub_device_qs = GatewaySubDevice.objects.filter(id=sub_device_id).\
 
-                     values('device__serial_number', 'ieee_addr', 'is_tampered')
 
-                 if not sub_device_qs.exists():
 
-                     return response.json(173)
 
-                 if cls.time_conflict(sub_device_id, conditions, is_all_day, request_dict):
 
-                     return response.json(182)
 
-                 device_type = int(conditions_dict['sensor']['device_type'])
 
-                 # 智能按钮不能创建触发条件相同的场景
 
-                 if device_type == SENSOR_TYPE['smart_button']:
 
-                     event_type = conditions_dict['sensor']['eventValues'][0]['event_type']
 
-                     smart_scene_qs = SmartScene.objects.filter(sub_device_id=sub_device_id,
 
-                                                                conditions__contains=event_type)
 
-                     if smart_scene_qs.exists():
 
-                         return response.json(180)
 
-                     # 紧急按钮打开时,创建的场景状态默认为关闭
 
-                     if sub_device_qs[0]['is_tampered'] == 1:
 
-                         smart_scene_dict['is_enable'] = False
 
-                 # 温湿度传感器返回温湿度
 
-                 elif device_type == SENSOR_TYPE['tem_hum_sensor']:
 
-                     event_values = conditions_dict['sensor']['eventValues'][0]
 
-                     if '≥' in event_values['value']:
 
-                         replace_str = '≥ '
 
-                         msg['sensor_symbol'] = 2
 
-                     else:
 
-                         replace_str = '≤ '
 
-                         msg['sensor_symbol'] = 1
 
-                     value = event_values['value'].replace(replace_str, '')
 
-                     msg['sensor_data'] = float(value)
 
-                 smart_scene_dict['sub_device_id'] = sub_device_id
 
-                 serial_number = sub_device_qs[0]['device__serial_number']
 
-                 msg['sensor_type'] = int(conditions_dict['sensor']['device_type'])
 
-                 msg['sensor_ieee_addr'] = sub_device_qs[0]['ieee_addr']
 
-                 msg['sensor_status'] = int(conditions_dict['sensor']['eventValues'][0]['event_type'])
 
-             with transaction.atomic():
 
-                 if is_all_day is None:  # 不设置时间
 
-                     smart_scene_qs = SmartScene.objects.create(**smart_scene_dict)
 
-                     # 设备的time数据,分钟转为秒
 
-                     time_dict = {
 
-                         'start_time': conditions_dict['time']['minutes'] * 60,
 
-                         'repeat': conditions_dict['time']['repeat']
 
-                     }
 
-                 elif is_all_day == 1:  # 全天
 
-                     smart_scene_qs = SmartScene.objects.create(**smart_scene_dict)
 
-                     # 设备的time数据
 
-                     time_dict = {
 
-                         'is_all_day': is_all_day
 
-                     }
 
-                 elif is_all_day == 2:  # 非全天
 
-                     start_time = int(request_dict.get('startTime', None))
 
-                     end_time = int(request_dict.get('endTime', None))
 
-                     repeat = int(request_dict.get('repeat', None))
 
-                     effective_time_qs = EffectiveTime.objects.filter(start_time=start_time, end_time=end_time,
 
-                                                                      repeat=repeat).values('id')
 
-                     if effective_time_qs.exists():
 
-                         effective_time_id = effective_time_qs[0]['id']
 
-                     else:
 
-                         effective_time_id = EffectiveTime.objects.create(start_time=start_time, end_time=end_time,
 
-                                                                          repeat=repeat).id
 
-                     smart_scene_dict['effective_time_id'] = effective_time_id
 
-                     smart_scene_qs = SmartScene.objects.create(**smart_scene_dict)
 
-                     # 设备的time数据,分钟转为秒
 
-                     time_dict = {
 
-                         'is_all_day': is_all_day,
 
-                         'start_time': start_time * 60,
 
-                         'end_time': end_time * 60,
 
-                         'repeat': repeat
 
-                     }
 
-                 else:
 
-                     return response.json(444, {'error param': 'invalid isAllDay'})
 
-                 msg['time'] = time_dict
 
-                 scene_id = smart_scene_qs.id
 
-                 msg['scene_id'] = scene_id
 
-                 # 获取设备任务数据
 
-                 msg['task'], scene_data = cls.get_task_list_and_scene_data(
 
-                     conditions_dict, is_set_time, minutes, repeat, tz, now_time, tasks_list, scene_id)
 
-                 smart_scene_qs.device_data = json.dumps(msg)
 
-                 smart_scene_qs.scene_data = scene_data
 
-                 smart_scene_qs.save()
 
-                 # 设备任务列表不为空,发布MQTT消息通知网关设备
 
-                 if msg['task']:
 
-                     thing_name = serial_number
 
-                     topic_name = SMART_SCENE_TOPIC.format(serial_number)
 
-                     success = CommonService.req_publish_mqtt_msg(thing_name, topic_name, msg)
 
-                     try:
 
-                         assert success
 
-                     except AssertionError:
 
-                         return response.json(10044)
 
-             return response.json(0)
 
-         except Exception as e:
 
-             return response.json(500, 'error_line:{}, error_msg:{}'.format(e.__traceback__.tb_lineno, repr(e)))
 
-     @staticmethod
 
-     def scene_list(request_dict, user_id, response):
 
-         """
 
-         查询智能场景列表
 
-         @param request_dict: 请求参数
 
-         @param user_id: 用户id
 
-         @request_dict deviceId: 网关设备id
 
-         @request_dict subDeviceId: 子设备id
 
-         @param response: 响应对象
 
-         @return: response
 
-         """
 
-         device_id = request_dict.get('deviceId', None)
 
-         sub_device_id = request_dict.get('subDeviceId', None)
 
-         if not any([device_id, sub_device_id]):
 
-             return response.json(444, {'error param': 'deviceId or subDeviceId'})
 
-         try:
 
-             if device_id:
 
-                 sub_device_id = GatewaySubDevice.objects.filter(device_id=device_id).values('id')
 
-                 smart_scene_qs = SmartScene.objects.filter(
 
-                     Q(user_id=user_id) & Q(device_id=device_id) | Q(sub_device_id__in=sub_device_id))
 
-             else:
 
-                 smart_scene_qs = SmartScene.objects.filter(user_id=user_id, sub_device_id=sub_device_id)
 
-             if not smart_scene_qs.exists():
 
-                 return response.json(173)
 
-             smart_scene_qs = smart_scene_qs.values(
 
-                 'id', 'scene_name', 'is_enable', 'device_id', 'sub_device_id', 'tasks')
 
-             smart_scene_list = []
 
-             for item in smart_scene_qs:
 
-                 smart_scene_dict = {
 
-                     'id': item['id'],
 
-                     'scene_name': item['scene_name'],
 
-                     'is_enable': item['is_enable']
 
-                 }
 
-                 if item['device_id']:
 
-                     data = {
 
-                         'userID': user_id
 
-                     }
 
-                     # 如果最后一个任务执行的设备是插座,返回插座数据,否则返回网关数据
 
-                     tasks_list = eval(item['tasks'])
 
-                     last_task = tasks_list[-1]
 
-                     if int(last_task['device_type']) == DEVICE_TYPE['socket']:
 
-                         data['serial_number'] = last_task['serial_number']
 
-                     else:
 
-                         data['id'] = item['device_id']
 
-                     device_qs = Device_Info.objects.filter(**data).values('NickName', 'Type')
 
-                     smart_scene_dict['device_type'] = device_qs[0]['Type'] if device_qs.exists() else ''
 
-                     smart_scene_dict['device_nickname'] = device_qs[0]['NickName'] if device_qs.exists() else ''
 
-                 else:
 
-                     device_qs = GatewaySubDevice.objects.filter(id=item['sub_device_id']).values('device_type',
 
-                                                                                                  'nickname')
 
-                     smart_scene_dict['device_type'] = device_qs[0]['device_type'] if device_qs.exists() else ''
 
-                     smart_scene_dict['device_nickname'] = device_qs[0]['nickname'] if device_qs.exists() else ''
 
-                 smart_scene_list.append(smart_scene_dict)
 
-             return response.json(0, smart_scene_list)
 
-         except Exception as e:
 
-             return response.json(500, 'error_line:{}, error_msg:{}'.format(e.__traceback__.tb_lineno, repr(e)))
 
-     @staticmethod
 
-     def smart_button_scene_list(request_dict, user_id, response):
 
-         """
 
-         查询智能按钮场景列表
 
-         @param request_dict: 请求参数
 
-         @param user_id: 用户id
 
-         @request_dict subDeviceId: 子设备id
 
-         @param response: 响应对象
 
-         @return: response
 
-         """
 
-         sub_device_id = request_dict.get('subDeviceId', None)
 
-         if not sub_device_id:
 
-             return response.json(444, {'error param': 'subDeviceId'})
 
-         try:
 
-             click_scene_qs = SmartScene.objects.filter(user_id=user_id, sub_device_id=sub_device_id,
 
-                                                        conditions__contains=str(
 
-                                                            EVENT_TYPE['smart_button_click'])).values('id', 'scene_name',
 
-                                                                                                      'is_enable')
 
-             double_click_scene_qs = SmartScene.objects.filter(user_id=user_id, sub_device_id=sub_device_id,
 
-                                                               conditions__contains=str(
 
-                                                                   EVENT_TYPE['smart_button_double_click'])). \
 
-                 values('id', 'scene_name', 'is_enable')
 
-             three_click_scene_qs = SmartScene.objects.filter(user_id=user_id, sub_device_id=sub_device_id,
 
-                                                              conditions__contains=str(
 
-                                                                  EVENT_TYPE['smart_button_three_click'])). \
 
-                 values('id', 'scene_name', 'is_enable')
 
-             scene_list = []
 
-             if click_scene_qs.exists():
 
-                 scene_list.append({
 
-                     'trigger_type': 1,
 
-                     'id': click_scene_qs[0]['id'],
 
-                     'scene_name': click_scene_qs[0]['scene_name'],
 
-                     'is_enable': click_scene_qs[0]['is_enable']
 
-                 })
 
-             if double_click_scene_qs.exists():
 
-                 scene_list.append({
 
-                     'trigger_type': 2,
 
-                     'id': double_click_scene_qs[0]['id'],
 
-                     'scene_name': double_click_scene_qs[0]['scene_name'],
 
-                     'is_enable': double_click_scene_qs[0]['is_enable']
 
-                 })
 
-             if three_click_scene_qs.exists():
 
-                 scene_list.append({
 
-                     'trigger_type': 3,
 
-                     'id': three_click_scene_qs[0]['id'],
 
-                     'scene_name': three_click_scene_qs[0]['scene_name'],
 
-                     'is_enable': three_click_scene_qs[0]['is_enable']
 
-                 })
 
-             return response.json(0, scene_list)
 
-         except Exception as e:
 
-             return response.json(500, 'error_line:{}, error_msg:{}'.format(e.__traceback__.tb_lineno, repr(e)))
 
-     @classmethod
 
-     def update_status(cls, request_dict, response):
 
-         """
 
-         更新智能场景状态
 
-         @param request_dict: 请求参数
 
-         @request_dict smartSceneId: 智能场景id
 
-         @request_dict isEnable: 状态,True or False
 
-         @param response: 响应对象
 
-         @return: response
 
-         """
 
-         smart_scene_id = request_dict.get('smartSceneId', None)
 
-         is_enable = request_dict.get('isEnable', None)
 
-         if not all([smart_scene_id, is_enable]):
 
-             return response.json(444, {'error param': 'smartSceneId and status'})
 
-         try:
 
-             smart_scene_id = int(smart_scene_id)
 
-             scene_status = SCENE_STATUS_ON if is_enable == 'True' else SCENE_STATUS_OFF
 
-             # 查询序列号
 
-             smart_scene_qs = SmartScene.objects.filter(id=smart_scene_id).\
 
-                 values('device_id', 'sub_device_id', 'scene_data')
 
-             device_id = smart_scene_qs[0]['device_id']
 
-             if device_id:
 
-                 serial_number = Device_Info.objects.filter(id=device_id).values('serial_number')[0]['serial_number']
 
-             else:
 
-                 sub_device_id = smart_scene_qs[0]['sub_device_id']
 
-                 serial_number = GatewaySubDevice.objects.filter(id=sub_device_id).values('device__serial_number')[0][
 
-                     'device__serial_number']
 
-             smart_scene_data = {
 
-                 'is_enable': is_enable
 
-             }
 
-             no_device_task = False
 
-             # 如果存在定时任务,暂停或恢复任务
 
-             scene_data = smart_scene_qs[0]['scene_data']
 
-             if scene_data:
 
-                 scene_data_dict = eval(scene_data)
 
-                 task_list = scene_data_dict.get('task_list')
 
-                 no_device_task = scene_data_dict.get('no_device_task')
 
-                 if no_device_task is not None and task_list:
 
-                     # 无设备任务,需要scene_id上报场景日志
 
-                     scene_id = smart_scene_id if no_device_task else 0
 
-                     new_scene_data_dict = cls.pause_or_resume_job(scene_data_dict, scene_status, scene_id)
 
-                     if new_scene_data_dict:
 
-                         smart_scene_data['scene_data'] = new_scene_data_dict
 
-             with transaction.atomic():
 
-                 SmartScene.objects.filter(id=smart_scene_id).update(**smart_scene_data)
 
-                 # 存在设备任务,通过mqtt发送设备数据
 
-                 if not no_device_task:
 
-                     topic_name = SMART_SCENE_TOPIC.format(serial_number)
 
-                     msg = {
 
-                         'scene_event': SCENE_EVENT_EDIT_STATUS,
 
-                         'scene_id': smart_scene_id,
 
-                         'scene_status': scene_status
 
-                     }
 
-                     success = CommonService.req_publish_mqtt_msg(serial_number, topic_name, msg)
 
-                     try:
 
-                         assert success
 
-                     except AssertionError:
 
-                         return response.json(10044)
 
-             return response.json(0)
 
-         except Exception as e:
 
-             return response.json(500, 'error_line:{}, error_msg:{}'.format(e.__traceback__.tb_lineno, repr(e)))
 
-     @staticmethod
 
-     def scene_detail(request_dict, response):
 
-         """
 
-         查询智能场景详情
 
-         @param request_dict: 请求参数
 
-         @request_dict smartSceneId: 智能场景id
 
-         @param response: 响应对象
 
-         @return: response
 
-         """
 
-         smart_scene_id = request_dict.get('smartSceneId', None)
 
-         if not smart_scene_id:
 
-             return response.json(444, {'error param': 'smartSceneId'})
 
-         try:
 
-             smart_scene_qs = SmartScene.objects.filter(id=smart_scene_id).values('id', 'scene_name', 'conditions',
 
-                                                                                  'tasks', 'effective_time_id',
 
-                                                                                  'is_all_day')
 
-             if not smart_scene_qs.exists():
 
-                 return response.json(173)
 
-             res = {
 
-                 'scene_name': smart_scene_qs[0]['scene_name'],
 
-                 'condition': eval(smart_scene_qs[0]['conditions']),
 
-                 'task': eval(smart_scene_qs[0]['tasks']),
 
-             }
 
-             # 如果存在关联的时间数据,组织时间数据
 
-             is_all_day = smart_scene_qs[0]['is_all_day']
 
-             effectiveTime = {}
 
-             if is_all_day != 0:
 
-                 effectiveTime['isAllDay'] = is_all_day
 
-             if is_all_day == 2:
 
-                 try:
 
-                     effective_time_qs = EffectiveTime.objects.get(id=smart_scene_qs[0]['effective_time_id'])
 
-                     effectiveTime['startTime'] = effective_time_qs.start_time
 
-                     effectiveTime['endTime'] = effective_time_qs.end_time
 
-                     effectiveTime['repeat'] = effective_time_qs.repeat
 
-                 except ObjectDoesNotExist:
 
-                     return response.json(0, res)
 
-             res['effectiveTime'] = effectiveTime
 
-             return response.json(0, res)
 
-         except Exception as e:
 
-             return response.json(500, 'error_line:{}, error_msg:{}'.format(e.__traceback__.tb_lineno, repr(e)))
 
-     @classmethod
 
-     def edit_smart_scene(cls, request_dict, user_id, response):
 
-         """
 
-         编辑智能场景
 
-         @param request_dict: 请求参数
 
-         @param user_id: 用户id
 
-         @request_dict smartSceneId: 智能场景id
 
-         @param response: 响应对象
 
-         @return: response
 
-         """
 
-         smart_scene_id = int(request_dict.get('smartSceneId', None))
 
-         device_id = request_dict.get('deviceId', None)
 
-         sub_device_id = request_dict.get('subDeviceId', None)
 
-         scene_name = request_dict.get('sceneName', None)
 
-         conditions = request_dict.get('conditions', None)
 
-         tasks = request_dict.get('tasks', None)
 
-         is_all_day = request_dict.get('isAllDay', None)
 
-         conditions_dict = eval(conditions)
 
-         tasks_list = eval(tasks)
 
-         now_time = int(time.time())
 
-         smart_scene_qs = SmartScene.objects.filter(user_id=user_id, scene_name=scene_name).filter(~Q(id=smart_scene_id))
 
-         if smart_scene_qs.exists():
 
-             return response.json(179)
 
-         res = {
 
-             'scene_name': scene_name,
 
-             'conditions': conditions_dict,
 
-             'tasks': tasks_list
 
-         }
 
-         effective_time = {}
 
-         if is_all_day:
 
-             is_all_day = int(is_all_day)
 
-             effective_time['is_all_day'] = is_all_day
 
-         if not all([smart_scene_id, scene_name, conditions, tasks]):
 
-             return response.json(444, {'error param': 'smartSceneId,sceneName,conditions or tasks'})
 
-         try:
 
-             smart_scene_qs = SmartScene.objects.filter(id=smart_scene_id)
 
-             if not smart_scene_qs.exists():
 
-                 return response.json(173)
 
-             tz = smart_scene_qs[0].tz
 
-             scene_data = smart_scene_qs[0].scene_data
 
-             scene_data_dict = eval(scene_data) if scene_data else None
 
-             scene_status = 1 if smart_scene_qs[0].is_enable else 0
 
-             msg = {
 
-                 'scene_id': smart_scene_id,
 
-                 'scene_event': SCENE_EVENT_EDIT,
 
-                 'scene_status': scene_status
 
-             }
 
-             # 判断条件是否为设置时间
 
-             is_set_time, minutes, repeat = False, 0, 0
 
-             if conditions_dict['type'] == 1:
 
-                 is_set_time = True
 
-                 minutes = conditions_dict['time']['minutes']
 
-                 repeat = conditions_dict['time']['repeat']
 
-             # 条件为设置时间
 
-             if is_set_time:
 
-                 if not device_id:
 
-                     return response.json(444, {'error param': 'deviceId'})
 
-                 sub_device_id = 0
 
-                 device_qs = Device_Info.objects.filter(id=device_id).values('serial_number')
 
-                 if not device_qs.exists():
 
-                     return response.json(173)
 
-                 serial_number = device_qs[0]['serial_number']
 
-                 # 网关数据
 
-                 msg['sensor_type'] = DEVICE_TYPE['gateway']
 
-                 msg['sensor_status'] = 2002
 
-                 msg['sensor_ieee_addr'] = 'FFFFFFFFFFFFFFFF'
 
-             # 条件为选择子设备
 
-             else:
 
-                 if not sub_device_id:
 
-                     return response.json(444, {'error param': 'subDeviceId'})
 
-                 if cls.time_conflict(sub_device_id, conditions, is_all_day, request_dict, smart_scene_id):
 
-                     return response.json(182)
 
-                 device_type = int(conditions_dict['sensor']['device_type'])
 
-                 # 智能按钮不能创建触发条件相同的场景
 
-                 if device_type == SENSOR_TYPE['smart_button']:
 
-                     event_type = conditions_dict['sensor']['eventValues'][0]['event_type']
 
-                     smart_scene_temp_qs = SmartScene.objects.filter(Q(sub_device_id=sub_device_id),
 
-                                                                     ~Q(id=smart_scene_id),
 
-                                                                     conditions__contains=event_type)
 
-                     if smart_scene_temp_qs.exists():
 
-                         return response.json(180)
 
-                 # 温湿度传感器返回温湿度
 
-                 elif device_type == SENSOR_TYPE['tem_hum_sensor']:
 
-                     event_values = conditions_dict['sensor']['eventValues'][0]
 
-                     if '≥' in event_values['value']:
 
-                         replace_str = '≥ '
 
-                         msg['sensor_symbol'] = 2
 
-                     else:
 
-                         replace_str = '≤ '
 
-                         msg['sensor_symbol'] = 1
 
-                     value = event_values['value'].replace(replace_str, '')
 
-                     msg['sensor_data'] = float(value)
 
-                 device_id = ''
 
-                 sub_device_qs = GatewaySubDevice.objects.filter(id=sub_device_id).values('ieee_addr',
 
-                                                                                          'device__serial_number')
 
-                 if not sub_device_qs.exists():
 
-                     return response.json(173)
 
-                 serial_number = sub_device_qs[0]['device__serial_number']
 
-                 msg['sensor_type'] = int(conditions_dict['sensor']['device_type'])
 
-                 msg['sensor_ieee_addr'] = sub_device_qs[0]['ieee_addr']
 
-                 msg['sensor_status'] = int(conditions_dict['sensor']['eventValues'][0]['event_type'])
 
-             # 获取设备任务数据
 
-             msg['task'], scene_data = cls.get_task_list_and_scene_data(
 
-                 conditions_dict, is_set_time, minutes, repeat, tz, now_time, tasks_list, smart_scene_id, scene_data_dict
 
-             )
 
-             with transaction.atomic():
 
-                 smart_scene_qs.update(scene_name=scene_name, conditions=conditions, tasks=tasks, scene_data=scene_data,
 
-                                       device_data=json.dumps(msg), updated_time=now_time, device_id=device_id,
 
-                                       sub_device_id=sub_device_id, is_enable=True)
 
-                 if is_all_day is None:  # 不设置时间或全天
 
-                     smart_scene_qs.update(effective_time_id=0, is_all_day=0)
 
-                     time_dict = {
 
-                         'start_time': conditions_dict['time']['minutes'] * 60,
 
-                         'repeat': conditions_dict['time']['repeat']
 
-                     }
 
-                 elif is_all_day == 1:
 
-                     smart_scene_qs.update(effective_time_id=0, is_all_day=is_all_day)
 
-                     time_dict = {
 
-                         'is_all_day': is_all_day
 
-                     }
 
-                 else:
 
-                     start_time = int(request_dict.get('startTime', None))
 
-                     end_time = int(request_dict.get('endTime', None))
 
-                     repeat = int(request_dict.get('repeat', None))
 
-                     effective_time_qs = EffectiveTime.objects.filter(start_time=start_time, end_time=end_time,
 
-                                                                      repeat=repeat).values('id')
 
-                     if effective_time_qs.exists():
 
-                         effective_time_id = effective_time_qs[0]['id']
 
-                     else:
 
-                         effective_time_id = EffectiveTime.objects.create(start_time=start_time, end_time=end_time,
 
-                                                                          repeat=repeat).id
 
-                     smart_scene_qs.update(effective_time_id=effective_time_id, is_all_day=is_all_day)
 
-                     time_dict = {
 
-                         'is_all_day': is_all_day,
 
-                         'start_time': start_time * 60,
 
-                         'end_time': end_time * 60,
 
-                         'repeat': repeat
 
-                     }
 
-                     effective_time = {
 
-                         'isAllDay': is_all_day,
 
-                         'startTime': start_time,
 
-                         'endTime': end_time,
 
-                         'repeat': repeat
 
-                     }
 
-                 msg['time'] = time_dict
 
-             # 设备任务列表不为空,发布MQTT消息通知网关设备
 
-             if msg['task']:
 
-                 thing_name = serial_number
 
-                 topic_name = SMART_SCENE_TOPIC.format(serial_number)
 
-                 success = CommonService.req_publish_mqtt_msg(thing_name, topic_name, msg)
 
-                 try:
 
-                     assert success
 
-                 except AssertionError:
 
-                     return response.json(10044)
 
-             res['effectiveTime'] = effective_time
 
-             return response.json(0, res)
 
-         except Exception as e:
 
-             return response.json(500, 'error_line:{}, error_msg:{}'.format(e.__traceback__.tb_lineno, repr(e)))
 
-     @classmethod
 
-     def delete_smart_scene(cls, request_dict, response):
 
-         """
 
-         删除智能场景
 
-         @param request_dict: 请求参数
 
-         @request_dict smartSceneIds: 智能场景id
 
-         @param response: 响应对象
 
-         @return: response
 
-         """
 
-         smart_scene_ids = request_dict.get('smartSceneIds', None)
 
-         if not smart_scene_ids:
 
-             return response.json(444, {'error param': 'smartSceneIds'})
 
-         try:
 
-             smart_scene_id_list = smart_scene_ids.split(',')
 
-             # 获取序列号
 
-             smart_scene_id = smart_scene_id_list[0]
 
-             smart_scene_qs = SmartScene.objects.filter(id=smart_scene_id).values('device_id', 'sub_device_id')
 
-             device_id = smart_scene_qs[0]['device_id']
 
-             if device_id:
 
-                 serial_number = Device_Info.objects.filter(id=device_id).values('serial_number')[0]['serial_number']
 
-             else:
 
-                 serial_number = GatewaySubDevice.objects.filter(id=smart_scene_qs[0]['sub_device_id']). \
 
-                     values('device__serial_number')[0]['device__serial_number']
 
-             topic_name = SMART_SCENE_TOPIC.format(serial_number)
 
-             with transaction.atomic():
 
-                 smart_scene_qs = SmartScene.objects.filter(id__in=smart_scene_id_list)
 
-                 # 删除定时任务
 
-                 smart_scene_data = smart_scene_qs.values('scene_data')
 
-                 for smart_scene in smart_scene_data:
 
-                     scene_data = smart_scene['scene_data']
 
-                     if scene_data:
 
-                         scene_data_dict = eval(scene_data)
 
-                         cls.del_aps_job(scene_data_dict)
 
-                 smart_scene_qs.delete()
 
-                 for smart_scene_id in smart_scene_id_list:
 
-                     # 通知设备删除场景id
 
-                     msg = {
 
-                         'scene_event': SCENE_EVENT_DELETE,
 
-                         'scene_id': int(smart_scene_id)
 
-                     }
 
-                     success = CommonService.req_publish_mqtt_msg(serial_number, topic_name, msg)
 
-                     try:
 
-                         assert success
 
-                     except AssertionError:
 
-                         return response.json(10044)
 
-                     time.sleep(0.3)
 
-             return response.json(0)
 
-         except Exception as e:
 
-             return response.json(500, 'error_line:{}, error_msg:{}'.format(e.__traceback__.tb_lineno, repr(e)))
 
-     @staticmethod
 
-     def scene_log(request_dict, response):
 
-         """
 
-         查询场景日志
 
-         @param request_dict: 请求参数
 
-         @request_dict deviceId: 网关id
 
-         @request_dict subDeviceId: 子设备id
 
-         @request_dict page: 页数
 
-         @request_dict size: 条数
 
-         @request_dict startTime: 开始时间
 
-         @request_dict endTime: 结束时间
 
-         @param response: 响应对象
 
-         @return: response
 
-         """
 
-         family_id = request_dict.get('familyId', None)
 
-         device_id = request_dict.get('deviceId', None)
 
-         sub_device_id = request_dict.get('subDeviceId', None)
 
-         page = request_dict.get('page', None)
 
-         size = request_dict.get('size', None)
 
-         start_time = request_dict.get('startTime', None)
 
-         end_time = request_dict.get('endTime', None)
 
-         if not all([family_id, page, size]):
 
-             return response.json(444, {'error param': 'familyId or page or size'})
 
-         device_list = []
 
-         sub_device_list = []
 
-         if not device_id and not sub_device_id:
 
-             family_room_device_qs = FamilyRoomDevice.objects.filter(family_id=family_id).values('device', 'sub_device')
 
-             for device in family_room_device_qs:
 
-                 if device['device'] not in device_list:
 
-                     device_list.append(device['device'])
 
-                 if device['sub_device']:
 
-                     sub_device_list.append(device['sub_device'])
 
-         elif sub_device_id:  # 查询子设备
 
-             family_room_device_qs = FamilyRoomDevice.objects.filter(family_id=family_id,
 
-                                                                     sub_device=sub_device_id).values('device_id')
 
-             for device in family_room_device_qs:
 
-                 device_list.append(device['device_id'])
 
-             sub_device_list.append(sub_device_id)
 
-         else:  # 查询网关
 
-             family_room_device_qs = FamilyRoomDevice.objects.filter(Q(family_id=family_id) & Q(device=device_id) &
 
-                                                                     ~Q(sub_device=0)).values(
 
-                 'sub_device')
 
-             device_list.append(device_id)
 
-             for device in family_room_device_qs:
 
-                 sub_device_list.append(device['sub_device'])
 
-         try:
 
-             page, size = int(page), int(size)
 
-             scene_log_qs = SceneLog.objects.filter(Q(device_id__in=device_list) | Q(sub_device_id__in=sub_device_list))
 
-             if start_time and end_time:
 
-                 scene_log_qs = scene_log_qs.filter(created_time__range=(start_time, end_time)).values(
 
-                     'status',
 
-                     'created_time',
 
-                     'device_id',
 
-                     'sub_device_id',
 
-                     'scene_name',
 
-                     'tasks').order_by(
 
-                     '-created_time')[(page - 1) * size:page * size]
 
-             else:
 
-                 scene_log_qs = scene_log_qs.values('status', 'created_time', 'device_id', 'sub_device_id', 'scene_name',
 
-                                                    'tasks').order_by(
 
-                     '-created_time')[(page - 1) * size:page * size]
 
-             if not scene_log_qs.exists():
 
-                 return response.json(0, [])
 
-             for item in scene_log_qs:
 
-                 if not device_id and not sub_device_id:
 
-                     if not item['sub_device_id']:
 
-                         device_qs = Device_Info.objects.filter(id=item['device_id']).values('Type')
 
-                         item['device_type'] = device_qs[0]['Type'] if device_qs.exists() else ''
 
-                     else:
 
-                         device_qs = GatewaySubDevice.objects.filter(id=item['sub_device_id']).values('device_type')
 
-                         item['device_type'] = device_qs[0]['device_type'] if device_qs.exists() else ''
 
-                 elif sub_device_id:
 
-                     device_qs = GatewaySubDevice.objects.filter(id=item['sub_device_id']).values('device_type')
 
-                     item['device_type'] = device_qs[0]['device_type'] if device_qs.exists() else ''
 
-                 else:
 
-                     device_qs = Device_Info.objects.filter(id=item['device_id']).values('Type')
 
-                     item['device_type'] = device_qs[0]['Type'] if device_qs.exists() else ''
 
-                 if item['tasks'] != '':
 
-                     item['tasks'] = eval(item['tasks'])
 
-             scene_log_list = list(scene_log_qs)
 
-             return response.json(0, scene_log_list)
 
-         except Exception as e:
 
-             print('error_line:{}, error_msg:{}'.format(e.__traceback__.tb_lineno, repr(e)))
 
-             return response.json(500, 'error_line:{}, error_msg:{}'.format(e.__traceback__.tb_lineno, repr(e)))
 
-     @staticmethod
 
-     def scene_log_date(request_dict, response):
 
-         """
 
-         查询场景日志日期
 
-         @param request_dict: 请求参数
 
-         @request_dict deviceId: 网关id
 
-         @request_dict subDeviceId: 子设备id
 
-         @param response: 响应对象
 
-         @return: response
 
-         """
 
-         family_id = request_dict.get('familyId', None)
 
-         device_id = request_dict.get('deviceId', None)
 
-         sub_device_id = request_dict.get('subDeviceId', None)
 
-         if not family_id:
 
-             return response.json(444, {'error param': 'familyId'})
 
-         device_list = []
 
-         sub_device_list = []
 
-         if not device_id and not sub_device_id:
 
-             family_room_device_qs = FamilyRoomDevice.objects.filter(family_id=family_id).values('device', 'sub_device')
 
-             for device in family_room_device_qs:
 
-                 if device['device'] not in device_list:
 
-                     device_list.append(device['device'])
 
-                 if device['sub_device']:
 
-                     sub_device_list.append(device['sub_device'])
 
-         elif sub_device_id:
 
-             family_room_device_qs = FamilyRoomDevice.objects.filter(family_id=family_id, sub_device=sub_device_id)
 
-             sub_device_list.append(sub_device_id)
 
-         else:
 
-             family_room_device_qs = FamilyRoomDevice.objects.filter(family_id=family_id, device=device_id).values(
 
-                 'sub_device')
 
-             device_list.append(device_id)
 
-             for device in family_room_device_qs:
 
-                 sub_device_list.append(device['sub_device'])
 
-         if not family_room_device_qs.exists():
 
-             return response.json(173)
 
-         try:
 
-             scene_log_qs = SceneLog.objects.extra(
 
-                 select={'date': "FROM_UNIXTIME(created_time,'%%Y-%%m-%%d')"}).values('date'). \
 
-                                filter(Q(device_id__in=device_list) | Q(sub_device_id__in=sub_device_list)). \
 
-                                annotate(count=Count('created_time')). \
 
-                                order_by('-date')[:31]
 
-             log_date_list = []
 
-             for scene_log in scene_log_qs:
 
-                 log_date_list.append({
 
-                     'timestamp': CommonService.str_to_timestamp(scene_log['date'], '%Y-%m-%d'),
 
-                     'count': scene_log['count'],
 
-                     'format': scene_log['date'],
 
-                 })
 
-             return response.json(0, log_date_list)
 
-         except Exception as e:
 
-             return response.json(500, 'error_line:{}, error_msg:{}'.format(e.__traceback__.tb_lineno, repr(e)))
 
-     @staticmethod
 
-     def get_scene_data(request_dict, response):
 
-         """
 
-         设备获取智能场景数据
 
-         @param request_dict: 请求参数
 
-         @request_dict serial_number: 序列号
 
-         @param response: 响应对象
 
-         @return: response
 
-         """
 
-         serial_number = request_dict.get('serial_number', None)
 
-         if not serial_number:
 
-             return response.json(444, {'error param': 'serial_number'})
 
-         try:
 
-             device_info_qs = Device_Info.objects.filter(serial_number=serial_number).values('id')
 
-             if not device_info_qs.exists():
 
-                 return response.json(173)
 
-             device_id = device_info_qs[0]['id']
 
-             sub_device_id_list = GatewaySubDevice.objects.filter(device_id=device_id).values_list('id', flat=True)
 
-             if sub_device_id_list:
 
-                 smart_scene_qs = SmartScene.objects.filter(
 
-                     Q(device_id=device_id) | Q(sub_device_id__in=sub_device_id_list))
 
-             else:
 
-                 smart_scene_qs = SmartScene.objects.filter(device_id=device_id)
 
-             if not smart_scene_qs.exists():
 
-                 return response.json(173)
 
-             # 下发智能场景数据
 
-             smart_scene_qs = smart_scene_qs.values('device_data')
 
-             topic_name = SMART_SCENE_TOPIC.format(serial_number)
 
-             for smart_scene in smart_scene_qs:
 
-                 msg = eval(smart_scene['device_data'])
 
-                 success = CommonService.req_publish_mqtt_msg(serial_number, topic_name, msg)
 
-                 try:
 
-                     assert success
 
-                 except AssertionError:
 
-                     return response.json(10044)
 
-                 time.sleep(2)
 
-             # 下发智能按钮数据
 
-             smart_button_qs = GatewaySubDevice.objects.filter(device_id=device_id,
 
-                                                               device_type=SENSOR_TYPE['smart_button']).values(
 
-                 'ieee_addr', 'is_tampered')
 
-             if smart_button_qs.exists():
 
-                 sos_count = smart_button_qs.count()
 
-                 for index, smart_button in enumerate(smart_button_qs):
 
-                     msg = {
 
-                         'sos_count': sos_count,  # 该网关下的智能按钮数量
 
-                         'sos_num': index + 1,  # 第几个按钮
 
-                         'sensor_ieee_addr': smart_button['ieee_addr'],
 
-                         'sos_select': smart_button['is_tampered']
 
-                     }
 
-                     success = CommonService.req_publish_mqtt_msg(serial_number, topic_name, msg)
 
-                     try:
 
-                         assert success
 
-                     except AssertionError:
 
-                         return response.json(10044)
 
-                     time.sleep(2)
 
-             return response.json(0)
 
-         except Exception as e:
 
-             return response.json(500, 'error_line:{}, error_msg:{}'.format(e.__traceback__.tb_lineno, repr(e)))
 
-     @staticmethod
 
-     def voice_audition(request_dict, response):
 
-         """
 
-         智能场景音频试听
 
-         @param request_dict: 请求参数
 
-         @request_dict serial_number: 序列号
 
-         @request_dict voiceId: 音频id
 
-         @param response: 响应对象
 
-         @return: response
 
-         """
 
-         serial_number = request_dict.get('serial_number', None)
 
-         voice_id = request_dict.get('voiceId', None)
 
-         if not all([serial_number, voice_id]):
 
-             return response.json(444)
 
-         try:
 
-             topic_name = VOICE_AUDITION_TOPIC.format(serial_number)
 
-             msg = {'voice_id': int(voice_id)}
 
-             success = CommonService.req_publish_mqtt_msg(serial_number, topic_name, msg)
 
-             try:
 
-                 assert success
 
-             except AssertionError:
 
-                 return response.json(10044)
 
-             return response.json(0)
 
-         except Exception as e:
 
-             return response.json(500, 'error_line:{}, error_msg:{}'.format(e.__traceback__.tb_lineno, repr(e)))
 
-     @classmethod
 
-     def get_task_list_and_scene_data(
 
-             cls, conditions_dict, is_set_time, minutes, repeat, tz, now_time, tasks_list, scene_id,
 
-             scene_data_dict=None):
 
-         """
 
-         获取设备任务数据和场景数据
 
-         @param tasks_list: app任务列表
 
-         @param conditions_dict: 条件
 
-         @param is_set_time: 条件是否为设置时间
 
-         @param minutes: 时间分钟数
 
-         @param repeat: 重复星期周期的十进制
 
-         @param tz: 时区
 
-         @param now_time: 当前时间
 
-         @param scene_id: 场景id
 
-         @param scene_data_dict: 场景数据
 
-         @return: task_list, scene_data
 
-         """
 
-         # 删除旧的定时任务
 
-         cls.del_aps_job(scene_data_dict)
 
-         task_list = []
 
-         scene_task_list = []
 
-         # 不需要设备触发的任务
 
-         no_device_task = True if is_set_time else False
 
-         is_last_task = False
 
-         tasks_len = len(tasks_list)
 
-         # 组织条件数据
 
-         condition = cls.get_condition(conditions_dict, is_set_time, minutes, repeat, now_time, tz)
 
-         for index, task in enumerate(tasks_list):
 
-             # 判断是否为列表的最后一个元素
 
-             if index == tasks_len - 1:
 
-                 is_last_task = True
 
-             sensor_type = int(task['device_type'])
 
-             # 处理插座数据
 
-             # 不用添加到设备的任务列表,添加到mqtt任务列表
 
-             if sensor_type == DEVICE_TYPE['socket']:
 
-                 serial_number = task['serial_number']
 
-                 event_type = int(task['event_type'])
 
-                 delay_time = task['delay_time']
 
-                 task_temp = {
 
-                     'device_type': sensor_type,
 
-                     'event_type': event_type,
 
-                     'delay_time': delay_time,
 
-                     'serial_number': serial_number
 
-                 }
 
-                 # 如果条件为设置时间,创建定时任务
 
-                 if is_set_time:
 
-                     smart_scene_id = 0
 
-                     # 无设备任务,且是最后一个任务,需要通过scene_id上报场景日志
 
-                     if no_device_task and is_last_task:
 
-                         smart_scene_id = scene_id
 
-                     task_temp['task_id'] = cls.create_celery_task(
 
-                         condition, minutes, delay_time, tz, repeat, sensor_type, event_type, serial_number,
 
-                         smart_scene_id)
 
-                 scene_task_list.append(task_temp)
 
-             else:
 
-                 no_device_task = False
 
-                 task_temp = {
 
-                     'sensor_type': sensor_type,
 
-                     'sensor_delay': 0
 
-                 }
 
-                 # 延时
 
-                 if 'delay_time' in task and task['delay_time'] != 0:
 
-                     task_temp['sensor_delay'] = task['delay_time']
 
-                 # 不为-1时需要其他数据
 
-                 if sensor_type != -1:
 
-                     task_temp['sensor_action'] = int(task['event_type'])
 
-                     # 子设备返回长地址
 
-                     sub_device_id = task.get('subDeviceId', None)
 
-                     if sub_device_id:
 
-                         sub_device_qs = GatewaySubDevice.objects.filter(id=sub_device_id).values('ieee_addr').first()
 
-                         task_temp['sensor_ieee_addr'] = sub_device_qs['ieee_addr']
 
-                     # 网关添加报警类型数据
 
-                     else:
 
-                         task_temp['voice_type'] = task.get('voice_type')
 
-                         task_temp['voice_id'] = task.get('voice_id')
 
-                         task_temp['count'] = task.get('count')
 
-                         task_temp['delay_time'] = task.get('delay_time')
 
-                         task_temp['duration'] = task.get('duration')
 
-                         task_temp['value_type'] = task.get('value_type')
 
-                 task_list.append(task_temp)
 
-         # 组织完整场景数据
 
-         scene_data = {
 
-             'condition': condition,
 
-             'task_list': scene_task_list,
 
-             'no_device_task': no_device_task
 
-         }
 
-         return task_list, scene_data
 
-     @classmethod
 
-     def get_condition(cls, conditions_dict, is_set_time, minutes, repeat, now_time, tz):
 
-         """
 
-         获取场景条件
 
-         @param conditions_dict: 条件数据
 
-         @param is_set_time: 条件是否为设置时间
 
-         @param minutes: 时间分钟数
 
-         @param repeat: 重复星期周期的十进制
 
-         @param now_time: 当前时间
 
-         @param tz: 时区
 
-         @return: condition
 
-         """
 
-         condition = {}
 
-         # 条件为设置时间
 
-         if is_set_time:
 
-             hour, minute = divmod(minutes, 60)
 
-             # 一次性任务
 
-             if repeat == 0:
 
-                 condition['time'] = 'date'
 
-                 # 根据时间戳和时区获取年月日,拼接由分钟转换出来的时间
 
-                 time_string = CommonService.get_date_from_timestamp(now_time, tz)
 
-                 time_string = time_string[:time_string.index(' ')]
 
-                 time_string += ' {:02d}:{:02d}:00'.format(hour, minute)
 
-                 time_stamp = CommonService.convert_to_timestamp(tz, time_string)
 
-                 if time_stamp < now_time:
 
-                     time_stamp += 24 * 60 * 60
 
-                 time_dict = {
 
-                     'time_stamp': time_stamp
 
-                 }
 
-                 condition['time_dict'] = time_dict
 
-             else:
 
-                 condition['time'] = 'cron'
 
-                 weeks = cls.int_to_weeks(repeat)
 
-                 time_dict = {
 
-                     'weeks': weeks,
 
-                     'hour': hour,
 
-                     'minute': minute
 
-                 }
 
-                 condition['time_dict'] = time_dict
 
-         else:
 
-             device_type = int(conditions_dict['sensor']['device_type'])
 
-             event_type = int(conditions_dict['sensor']['eventValues'][0]['event_type'])
 
-             condition['event_type'] = event_type
 
-             # 温湿度传感器,取值
 
-             if device_type == SENSOR_TYPE['tem_hum_sensor']:
 
-                 condition['value'] = event_type = conditions_dict['sensor']['eventValues'][0]['value']
 
-         return condition
 
-     @staticmethod
 
-     def time_conflict(sub_device_id, conditions, is_all_day, request_dict, smart_scene_id=None):
 
-         """
 
-         判断传感器是否创建过条件相同且生效时间冲突的场景
 
-         @param sub_device_id: 传感器设备id
 
-         @param conditions: 场景条件
 
-         @param is_all_day: 全天标识
 
-         @param request_dict:
 
-         @param smart_scene_id: 场景id,编辑场景时传
 
-         @return: bool, True: 冲突, False: 不冲突
 
-         """
 
-         # 不设置时间不会冲突
 
-         if is_all_day is None:
 
-             return False
 
-         # 查询设置过时间的数据
 
-         if smart_scene_id is None:  # 创建场景
 
-             smart_scene_qs = SmartScene.objects.filter(
 
-                 ~Q(is_all_day=0),
 
-                 sub_device_id=sub_device_id,
 
-                 conditions=conditions).values('effective_time_id')
 
-         else:   # 编辑场景,过滤本身场景数据
 
-             smart_scene_qs = SmartScene.objects.filter(
 
-                 ~Q(id=smart_scene_id),
 
-                 ~Q(is_all_day=0),
 
-                 sub_device_id=sub_device_id,
 
-                 conditions=conditions).values('effective_time_id')
 
-         if not smart_scene_qs.exists():
 
-             return False
 
-         # 再设置全天必冲突
 
-         if is_all_day == 1:
 
-             return True
 
-         # 非全天
 
-         elif is_all_day == 2:
 
-             start_time = int(request_dict.get('startTime'))
 
-             end_time = int(request_dict.get('endTime'))
 
-             repeat = int(request_dict.get('repeat'))
 
-             for smart_scene in smart_scene_qs:
 
-                 effective_time_id = smart_scene['effective_time_id']
 
-                 effective_time_qs = EffectiveTime.objects.filter(id=effective_time_id).\
 
-                     values('start_time', 'end_time', 'repeat')
 
-                 if effective_time_qs.exists():
 
-                     old_start_time = effective_time_qs[0]['start_time']
 
-                     old_end_time = effective_time_qs[0]['end_time']
 
-                     old_repeat = effective_time_qs[0]['repeat']
 
-                     # 每天重复
 
-                     if repeat == 127:
 
-                         # 判断时间是否在已设置过的时间范围之内
 
-                         if old_start_time <= start_time <= old_end_time or \
 
-                                 old_start_time <= end_time <= old_end_time:
 
-                             return True
 
-                     else:
 
-                         # 有相同的重复天
 
-                         if repeat & old_repeat != 0:
 
-                             # 判断时间是否在已设置过的时间范围之内
 
-                             if old_start_time <= start_time <= old_end_time or \
 
-                                     old_start_time <= end_time <= old_end_time:
 
-                                 return True
 
-             return False
 
-     @classmethod
 
-     def create_celery_task(
 
-             cls, condition, minutes, delay_time, tz, repeat, device_type, event_type, serial_number, scene_id):
 
-         """
 
-         创建定时任务
 
-         返回任务id和时间
 
-         @param condition: 条件数据
 
-         @param minutes: 分钟时间
 
-         @param delay_time: 延迟时间
 
-         @param tz: 时区
 
-         @param repeat: 星期周期的十进制数
 
-         @param device_type: 设备类型
 
-         @param event_type: 事件类型
 
-         @param serial_number: 序列号
 
-         @param scene_id: 场景id
 
-         @return: task_id
 
-         """
 
-         celery_beat_obj = CeleryBeatObj()
 
-         name = serial_number + '_'
 
-         task = 'Controller.CeleryTasks.tasks.loocam_smart_scene'
 
-         kwargs = {
 
-             'device_type': device_type,
 
-             'event_type': event_type,
 
-             'serial_number': serial_number,
 
-             'scene_id': scene_id
 
-         }
 
-         # 一次性任务
 
-         if repeat == 0:
 
-             time_stamp = condition['time_dict']['time_stamp'] + delay_time
 
-             name += str(time_stamp)
 
-             celery_beat_obj.creat_clocked_task(
 
-                 name=name, task=task, time_stamp=time_stamp, timezone_offset=tz, kwargs=kwargs)
 
-         # 周期任务
 
-         else:
 
-             hour, minute, second, is_next_day = cls.handle_delay_time(minutes, delay_time)
 
-             # 加上延时,如果执行时间超过23:59,隔天执行
 
-             weeks = cls.int_to_weeks(repeat, is_next_day)
 
-             time_str = weeks + '_{:02d}{:02d}{:02d}'.format(hour, minute, second)
 
-             name += time_str
 
-             celery_beat_obj.creat_crontab_task(
 
-                 timezone_offset=tz, name=name, task=task, minute=minute, hour=hour, day_of_week=weeks, kwargs=kwargs)
 
-         return name
 
-     @staticmethod
 
-     def handle_delay_time(minutes, delay_time):
 
-         """
 
-         处理延迟时间
 
-         @param minutes: 时间分钟数,如1439代表23:59
 
-         @param delay_time: 延迟时间,单位:秒
 
-         @return: hour, minute, second, is_next_day
 
-         """
 
-         is_next_day = False
 
-         hour, minute = divmod(minutes, 60)
 
-         # 延迟时间转为分钟,如果加上时间分钟数大于1439,隔天执行
 
-         minute, second = divmod(delay_time, 60)
 
-         total_minutes = minutes + minute
 
-         ex_min = total_minutes - 1439
 
-         if ex_min <= 0:
 
-             hour, minute = divmod(total_minutes, 60)
 
-         else:
 
-             hour, minute = divmod(ex_min, 60)
 
-             is_next_day = True
 
-         return hour, minute, second, is_next_day
 
-     @staticmethod
 
-     def int_to_weeks(repeat, is_next_day=False):
 
-         """
 
-         十进制转星期周期
 
-         @param repeat: 星期周期的十进制数,如127 -> 1,2,3,4,5,6,7
 
-         @param is_next_day: 是否隔天
 
-         @return: weeks
 
-         """
 
-         # 十进制转为7位的二进制并倒序
 
-         bin_str = bin(repeat)[2:].zfill(7)[::-1]
 
-         # 生成星期周期字符串
 
-         weeks = ''
 
-         next_day = 1 if is_next_day else 0
 
-         for i, bit in enumerate(bin_str):
 
-             if bit == '1':
 
-                 week = i + next_day
 
-                 weeks += str(week) + ','
 
-         # 删除最后一个逗号并返回结果
 
-         return weeks[:-1]
 
-     @staticmethod
 
-     def pub_mqtt(device_type, event_type, serial_number, scene_id=0):
 
-         """
 
-         发布mqtt消息
 
-         @param device_type: 设备类型
 
-         @param event_type: 事件类型
 
-         @param serial_number: 序列号
 
-         @param scene_id: 场景id
 
-         @return:
 
-         """
 
-         if device_type == DEVICE_TYPE['socket']:
 
-             topic_name = SMART_SOCKET_TOPIC.format(serial_number)
 
-             status = 1 if event_type == EVENT_TYPE['socket_power_on'] else 0
 
-             msg = {
 
-                 'type': 1,
 
-                 'data': {'deviceSwitch': status}
 
-             }
 
-             CommonService.req_publish_mqtt_msg(serial_number, topic_name, msg)
 
-         # 没有设备任务时,最后一个任务上报场景日志
 
-         if scene_id:
 
-             data = {
 
-                 'sceneId': scene_id,
 
-                 'status': 1
 
-             }
 
-             url = DETECT_PUSH_DOMAINS + 'gatewayService/sceneLogPush'
 
-             requests.post(url=url, data=data, timeout=8)
 
-     @staticmethod
 
-     def del_aps_job(scene_data_dict):
 
-         """
 
-         删除定时任务
 
-         @param scene_data_dict: 场景数据
 
-         @return:
 
-         """
 
-         if scene_data_dict is not None:
 
-             if scene_data_dict['condition'].get('time'):
 
-                 apscheduler_obj = ApschedulerObject()
 
-                 time_task_list = scene_data_dict['task_list']
 
-                 for time_task in time_task_list:
 
-                     # 存在任务则删除
 
-                     task_id = time_task.get('task_id')
 
-                     if task_id:
 
-                         apscheduler_obj.del_job(task_id)
 
-     @classmethod
 
-     def pause_or_resume_job(cls, scene_data_dict, scene_status, scene_id):
 
-         """
 
-         暂停或恢复定时任务
 
-         @param scene_data_dict: 场景数据
 
-         @param scene_status: 场景状态: SCENE_STATUS_ON, SCENE_STATUS_OFF
 
-         @param scene_id: 场景id
 
-         @return: None or scene_data_dict
 
-         """
 
-         # 判断条件是否为设置时间
 
-         time_type = scene_data_dict['condition'].get('time')
 
-         if time_type:
 
-             apscheduler_obj = ApschedulerObject()
 
-             # 关闭: 暂停任务, 打开: 恢复任务
 
-             if scene_status == SCENE_STATUS_OFF:
 
-                 cls.pause_time_job(apscheduler_obj, scene_data_dict)
 
-             else:
 
-                 return cls.resume_time_job(apscheduler_obj, time_type, scene_data_dict, scene_id)
 
-     @staticmethod
 
-     def pause_time_job(apscheduler_obj, scene_data_dict):
 
-         """
 
-         暂停定时任务
 
-         @param apscheduler_obj: apscheduler对象
 
-         @param scene_data_dict: 场景数据
 
-         @return:
 
-         """
 
-         time_task_list = scene_data_dict['task_list']
 
-         for time_task in time_task_list:
 
-             task_id = time_task.get('task_id')
 
-             if task_id:
 
-                 try:
 
-                     apscheduler_obj.pause_job(task_id)
 
-                 except Exception:
 
-                     continue
 
-     @classmethod
 
-     def resume_time_job(cls, apscheduler_obj, time_type, scene_data_dict, scene_id):
 
-         """
 
-         恢复定时任务
 
-         @param apscheduler_obj: apscheduler对象
 
-         @param time_type: 时间类型: date, cron
 
-         @param scene_data_dict: 场景数据
 
-         @param scene_id: 场景id
 
-         @return: None or time_stamp
 
-         """
 
-         time_task_list = scene_data_dict['task_list']
 
-         if time_type == 'cron':
 
-             for time_task in time_task_list:
 
-                 task_id = time_task.get('task_id')
 
-                 if task_id:
 
-                     try:
 
-                         apscheduler_obj.resume_job(task_id)
 
-                     except Exception:
 
-                         continue
 
-             return None
 
-         else:
 
-             now_time = int(time.time())
 
-             time_stamp = scene_data_dict['condition']['time_dict']['time_stamp']
 
-             # 恢复定时任务
 
-             if time_stamp > now_time:
 
-                 for time_task in time_task_list:
 
-                     task_id = time_task.get('task_id')
 
-                     if task_id:
 
-                         try:
 
-                             apscheduler_obj.resume_job(task_id)
 
-                         except Exception:
 
-                             continue
 
-                 return None
 
-             else:
 
-                 # 创建新的定时任务,时间+24小时
 
-                 time_stamp += 24 * 60 * 60
 
-                 task_list_len = len(time_task_list)
 
-                 for index, time_task in enumerate(time_task_list):
 
-                     device_type = time_task['device_type']
 
-                     event_type = time_task['event_type']
 
-                     serial_number = time_task['serial_number']
 
-                     task_id = serial_number + '_' + str(time_stamp)
 
-                     # 最后一个任务使用传入的scene_id
 
-                     smart_scene_id = scene_id if index == task_list_len-1 else 0
 
-                     apscheduler_obj.create_date_job(func=cls.pub_mqtt, task_id=task_id, time_stamp=time_stamp,
 
-                                                     args=(device_type, event_type, serial_number, smart_scene_id))
 
-                     # 更新task_id
 
-                     time_task['task_id'] = task_id
 
-                 # 更新场景数据的时间戳
 
-                 scene_data_dict['condition']['time_dict']['time_stamp'] = time_stamp
 
-                 return scene_data_dict
 
- #
 
- #                   ___====-_  _-====___
 
- #             _--^^^#####//      \\#####^^^--_
 
- #          _-^##########// (    ) \\##########^-_
 
- #         -############//  |\^^/|  \\############-
 
- #       _/############//   (@::@)   \\############\_
 
- #      /#############((     \\//     ))#############\
 
- #     -###############\\    (oo)    //###############-
 
- #    -#################\\  / VV \  //#################-
 
- #   -###################\\/      \//###################-
 
- #  _#/|##########/\######(   /\   )######/\##########|\#_
 
- #  |/ |#/\#/\#/\/  \#/\##\  |  |  /##/\#/  \/\#/\#/\#| \|
 
- #  `  |/  V  V  `   V  \#\| |  | |/#/  V   '  V  V  \|  '
 
- #     `   `  `      `   / | |  | | \   '      '  '   '
 
- #                      (  | |  | |  )
 
- #                     __\ | |  | | /__
 
- #                    (vvv(VVV)(VVV)vvv)
 
- #                         神兽保佑
 
- #                        代码无BUG!
 
- #
 
 
  |