| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523 | # -*- coding: utf-8 -*-"""@Author : Rocky@Time : 2022/6/29 9:31@File :SmartSceneController.py"""import jsonimport timefrom django.core.exceptions import ObjectDoesNotExistfrom django.db import transactionfrom django.db.models import F, Q, Countfrom django.views import Viewfrom 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, DEVICE_TYPE, SMART_SCENE_TASK, LINKAGE_DEVICE_TYPE_LISTfrom Model.models import FamilyRoomDevice, GatewaySubDevice, FamilyRoom, SmartScene, EffectiveTime, Device_Info, \    SceneLogfrom Object.CeleryBeatObject import CeleryBeatObjfrom Object.ResponseObject import ResponseObjectfrom Service.CommonService import CommonServiceclass 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_linkage_device_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_linkage_device_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=LINKAGE_DEVICE_TYPE_LIST).\            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_celery_task(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_celery_task(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'])            # 处理联动设备数据            # 不用添加到设备的任务列表,添加到场景数据任务列表            if sensor_type in LINKAGE_DEVICE_TYPE_LIST:                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, index)                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                }            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, index):        """        创建定时任务        返回任务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        @param index: 任务下标        @return: task_id        """        celery_beat_obj = CeleryBeatObj()        # 任务名拼接当前时间戳,防止重复        name = '{}_{}_{}_'.format(serial_number, str(int(time.time())), str(index))        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=SMART_SCENE_TASK, time_stamp=time_stamp, 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=SMART_SCENE_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 -> 每天 -> 0,1,2,3,4,5,6或*        @param is_next_day: 是否隔天        @return: weeks        """        if repeat == 127:            return '*'        # 十进制转为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':                # 7 -> 0                week = i + 1 + next_day                if week == 8:                    week = 0                weeks += str(week) + ','        # 删除最后一个逗号并返回结果        return weeks[:-1]    @staticmethod    def del_celery_task(scene_data_dict):        """        删除定时任务        @param scene_data_dict: 场景数据        @return:        """        if scene_data_dict is not None:            if scene_data_dict['condition'].get('time'):                celery_beat_obj = CeleryBeatObj()                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:                            celery_beat_obj.del_task(task_id)                        except Exception:                            continue    @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:            celery_beat_obj = CeleryBeatObj()            # 关闭: 暂停任务, 打开: 恢复任务            if scene_status == SCENE_STATUS_OFF:                cls.pause_time_job(celery_beat_obj, scene_data_dict)            else:                return cls.resume_time_job(celery_beat_obj, time_type, scene_data_dict, scene_id)    @staticmethod    def pause_time_job(celery_beat_obj, scene_data_dict):        """        暂停定时任务        @param celery_beat_obj: celery_beat对象        @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:                    celery_beat_obj.disable_task(task_id)                except Exception:                    continue    @classmethod    def resume_time_job(cls, celery_beat_obj, time_type, scene_data_dict, scene_id):        """        恢复定时任务        @param celery_beat_obj: celery_beat对象        @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:                        celery_beat_obj.enable_task(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:                            celery_beat_obj.enable_task(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 = '{}_{}_{}'.format(serial_number, str(now_time), str(time_stamp))                    # 最后一个任务使用传入的scene_id                    smart_scene_id = scene_id if index == task_list_len-1 else 0                    kwargs = {                        'device_type': device_type,                        'event_type': event_type,                        'serial_number': serial_number,                        'scene_id': smart_scene_id                    }                    celery_beat_obj.creat_clocked_task(                        name=task_id, task=SMART_SCENE_TASK, time_stamp=time_stamp, kwargs=kwargs)                    # 更新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!#
 |