from datetime import timedelta from apps.vm.models import Visit from apps.ai.main import ai_analyse from apps.opm.models import Operation, Opl, OplWorker from apps.utils.sms import send_sms import requests from apps.am.models import Access, Area from apps.am.tasks import cache_areas_info from apps.ecm.models import AlgoChannel, Event, EventCate, Eventdo, NotifySetting, Remind from apps.hrm.models import ClockRecord, Employee from apps.system.models import User from apps.third.models import TDevice from apps.utils.queryset import get_child_queryset2, get_parent_queryset from django.core.cache import cache import time from shapely.geometry import Point from apps.third.clients import dhClient, spClient from django.utils import timezone from django.conf import settings import os from apps.utils.speech import generate_voice from threading import Thread from apps.utils.tools import timestamp_to_time from apps.third.clients import xxClient from apps.third.tapis import xxapis import logging myLogger = logging.getLogger('log') requests.packages.urllib3.disable_warnings() def update_remind_read(event: Event, user: User): qs = Remind.objects.filter(event=event, recipient=user, is_read=False) if qs.exists(): qs.update(is_read=True) def get_area_info_from_cache(target: str, cache: list): """从区域信息缓存里匹配到所在区域 """ for i in cache: if i['id'] == target: return i return None def save_dahua_pic(pic_url: str): """保存大华报警图片到本地(全路径) 返回本地路径 """ file_name = pic_url.split('/')[-1].split('?')[0] res = requests.get(url=pic_url, verify=False) path = '/media/' + timezone.now().strftime('%Y/%m/%d/') full_path = settings.BASE_DIR + path if not os.path.exists(full_path): os.makedirs(full_path) with open(full_path + file_name, 'wb') as f: f.write(res.content) return path + file_name def get_ep_default(): """返回人员默认位置信息 """ return { 'area_fix_id': None, # 当前所在固定区域ID 'area_temp_id': None, # 当前所在临时区域ID 'xx_detail': {}, # 寻息定位的详细信息 'time0': None, # 定位首次出现时间戳 "time1": None, # 首次在该区域时间戳 "time2": int(time.time()), # 当前时间戳 } def notify_event(event: Event, voice_msg=''): """事件后续处理: Args: event (Event): _description_ """ # 生成通知文本 ep = event.employee obj_cate = event.obj_cate area_name = event.area.name if event.area else '未知区域' params = {'area': area_name, 'employee': '', 'event': ''} if event.opl: # 主要针对的是作业人员未就位事件 params['employee'] = event.opl.cate.name + '作业' elif ep: ep_name = ep.name ep_type = '正式员工' if ep.type == 'rempoyee': ep_type = '相关方人员' elif ep.type == 'visitor': ep_type = '访客' elif ep.type == 'driver': ep_type = '货车司机' params['employee'] = ep_type + ep_name elif obj_cate == 'people': params['employee'] = '未知人员' cats_list = [] for i in event.cates.all(): cats_list.append(i.name) params['event'] = ','.join(cats_list) if event.voice_msg: # 如果已经生成通知文本了就不再处理 pass else: if params['employee']: event.voice_msg = '位于{}的{},{},请及时处理'.format(params['area'], params['employee'], params['event']) else: event.voice_msg = '在{}下,发生{},请及时处理'.format(params['area'], params['event']) if voice_msg: event.voice_msg = voice_msg event.save() # 喇叭播放(任何情况下) Thread(target=save_voice_and_speak, args=(event,), daemon=True).start() # 如果还未创建提醒 if not event.is_pushed: Thread(target=create_remind, args=(event, params), daemon=True).start() def save_voice_and_speak(event: Event): """生成语音同时喇叭播放 Args: event (Event): _description_ """ try: main_cate = event.cates.all().order_by('priority', 'create_time').first() v_p, v_num = main_cate.voice_person, main_cate.voice_num if event.voice: # 如果已经生成了报警声音不用再请求 pass else: _, event.voice, _ = generate_voice(event.voice_msg, v_p) event.save() if main_cate.speaker_on: sps = [] if event.area: # 如果事件存在发生区域 sps = list(TDevice.objects.filter(area=event.area, type=TDevice.DEVICE_SPEAKER).values_list('code', flat=True)) if len(sps) == 0: # 如果当前区域没有喇叭就找覆盖区的喇叭 sps = list(TDevice.objects.filter(areas=event.area, type=TDevice.DEVICE_SPEAKER).values_list('code', flat=True)) # 查找固定喇叭 sps2 = list(TDevice.objects.filter(ec_speakers__event_cates=event).values_list('code', flat=True)) for i in sps2: if i not in sps: sps.append(i) # myLogger.info('获取到喇叭:' + str(sps)) if sps: spClient.speak(event.voice, sps, v_num) except Exception: myLogger.error('喇叭播放失败', exc_info=True) def create_remind(event: Event, params: dict): """ 创建事件提醒并发送短信/微信 """ # 向当事人本人发送通知 # if event.employee and event.employee.phone: # t_sms = Thread(target=send_sms, args=(event.employee.phone, '1001', {'code': '5678'}), daemon=True) # t_sms.start() # 查找所有提醒配置 n_s = NotifySetting.objects.filter(event_cate__in=event.cates.all()).order_by('sort') area_level = event.area.level if event.area else Area.AREA_4 for i in n_s: if i.user and area_level >= i.filter_area_level: Remind.objects.get_or_create(event=event, recipient=i.user, defaults={ 'event': event, 'recipient': i.user, 'notify_setting': i, 'can_handle': i.can_handle, }) elif i.post and area_level >= i.filter_area_level: qs = User.objects.filter(posts=i.post) if i.filter_recipient == 20: # 当事人所在部门以上 if event.employee and event.employee.user: qs = qs.filter(depts__in=get_parent_queryset(event.employee.user.belong_dept)) elif i.filter_recipient == 40: # 属地部门以上 if event.area.belong_dept: qs = qs.filter(depts__in=get_parent_queryset(event.area.belong_dept)) for m in qs: Remind.objects.get_or_create(event=event, recipient=m, defaults={ 'event': event, 'recipient': m, 'notify_setting': i, 'can_handle': i.can_handle, }) elif i.variable and area_level >= i.filter_area_level: if i.variable == 'self': if event.employee and event.employee.user: Remind.objects.get_or_create(event=event, recipient=event.employee.user, defaults={ 'event': event, 'recipient': event.employee.user, 'notify_setting': i, 'can_handle': i.can_handle, }) elif i.variable == 'area_manager': if event.area.manager: Remind.objects.get_or_create(event=event, recipient=event.area.manager, defaults={ 'event': event, 'recipient': event.area.manager, 'notify_setting': i, 'can_handle': i.can_handle, }) elif i.variable == 'visit_receptionist': if event.employee and event.employee.type == 'visitor': # 确定是访客 visit = Visit.objects.filter(visitors__employee=event.employee, state=Visit.V_WORKING).first() if visit: Remind.objects.get_or_create(event=event, recipient=visit.receptionist, defaults={ 'event': event, 'recipient': visit.receptionist, 'notify_setting': i, 'can_handle': i.can_handle, }) # 开始发送通知 for i in Remind.objects.filter(event=event): if i.notify_setting.sms_enable and i.recipient.phone: # 发送短信通知 Thread(target=send_sms, args=(i.recipient.phone, 1003, params), daemon=True).start() if i.notify_setting.wechat_enable: pass event.is_pushed = True event.save() def check_not_in_place(opl: Opl): # 检查作业人员未就位事件(直接创建事件) area = opl.operation.area try: railId = area.third_info['xx_rail']['id'] json = { "railId": railId, "type": "" } is_ok, res = xxClient.request(**xxapis['rail_ibeacon_list'], json=json, raise_exception=False) if is_ok: blt_list = res['recordList'] macs = [] # 所有该区域在线标签 for i in blt_list: macs.append(i['mac']) tds = TDevice.objects.filter(type=TDevice.DEVICE_BLT, employee__user__opl_worker__opl=opl) # 所有工作人员的定位标签 for i in tds: if i.code not in macs: # 触发作业人员未就位事件 event = Event() event.area = area event.obj_cate = 'opl' event.employee = i.employee event.opl = opl event.happen_time = timezone.now() event.save() cate = EventCate.objects.get(code='not_in_place') Eventdo.objects.get_or_create(cate=cate, event=event, defaults={ 'cate': cate, 'event': event }) notify_event(event) break except Exception: myLogger.error('检查作业人员未就位失败', exc_info=True) def check_miss_lcard(ep: Employee, area: Area): # 获取当前区域下的定位卡列表 try: railId = area.third_info['xx_rail']['id'] json = { "railId": railId, "type": "" } is_ok, res = xxClient.request(**xxapis['rail_ibeacon_list'], json=json, raise_exception=False) if is_ok: blt_list = res['recordList'] macs = [] # 所有该区域在线标签 for i in blt_list: macs.append(i['mac']) td = TDevice.objects.filter(type=TDevice.DEVICE_BLT, employee=ep).first() if td: if td.code not in macs: return True else: return True except Exception: myLogger.error('检查未带定位卡失败', exc_info=True) def dispatch_dahua_event(data: dict): """分发大华事件进行处理 """ vchannel_code = data['info']['nodeCode'] alarm_type = data['info']['alarmType'] vchannel = TDevice.objects.filter(code=vchannel_code).first() event = None if vchannel is None: return face_img_o = None global_img_o = None area = vchannel.area obj_cate = 'other' ep = None if alarm_type in [1001003, 1001000]: obj_cate = 'people' face_img_o = dhClient.get_full_pic(data['info']['alarmPicture']) global_img_o = dhClient.get_full_pic(data['info']['extend']['globalScenePicUrl']) else: global_img_o = dhClient.get_full_pic(data['info']['alarmPicture']) happen_time = timestamp_to_time(int(data['info']['alarmDate'])) algo_channels = AlgoChannel.objects.filter(vchannel=vchannel, algo__self_algo=True).exclude( algo__code=None).order_by('algo__priority', 'algo__create_time').values('id', 'algo', 'algo__code') cates = [] # 触发的事件种类 algo_codes = [] # 需要执行的ai算法 for i in algo_channels: if i['algo__code'] == str(alarm_type): cates.append(i['algo']) else: algo_codes.append(i['algo__code']) if algo_codes: ec_codes = ai_analyse(algo_codes, global_img=global_img_o, face_img=face_img_o) # 算法处理 for m in ec_codes.keys(): for n in algo_channels: if m == n['algo__code']: cates.append(n['algo']) if alarm_type == 1001003 and area: # 内部人员需要执行未带定位卡算法 ep = Employee.objects.filter(id_number=data['info']['extend']['candidateInfo'][0]['id']).first() # 检查是否携带定位卡只针对内部员工和相关方 if 'miss_lcard' in algo_codes and ep: is_happend = check_miss_lcard(ep=ep, area=area) if is_happend: cates.append(EventCate.objects.filter(code='miss_lcard').first().id) if cates: event = Event() event.global_img = save_dahua_pic(global_img_o) if face_img_o: event.face_img = save_dahua_pic(face_img_o) event.area = area event.obj_cate = obj_cate event.vchannel = vchannel event.employee = ep event.happen_time = happen_time event.save() for i in cates: cate = EventCate.objects.get(id=i) Eventdo.objects.get_or_create(cate=cate, event=event, defaults={ 'cate': cate, 'event': event }) if event: notify_event(event) def rail_in(data): """围栏进入 """ # 找到所在围栏 area = Area.objects.filter(third_info__xx_rail__id=data['railId']).first() # 找到进入对象 blts = TDevice.objects.filter(code=data['userId']).first() if area and blts and blts.employee: # 如果是人 ep_blt = blts.employee # 标签绑定人员 if ep_blt: for i in Access.objects.filter(area=area).order_by('sort'): # 优先自定义权限过滤 if i.post: # 如果是按岗位设定的 eps_access = Employee.objects.filter(user__posts=i.post) if ep_blt in eps_access and i.type == Access.ACCESS_IN_YES: return elif ep_blt in eps_access and i.type == Access.ACCESS_IN_NO: # 触发违规进入事件 handle_xx_event_2('i_enter', ep=ep_blt, area=area) elif i.dept: # 如果是按部门设定的 if i.dept.type == 'dept': # 如果是内部部门 depts = get_child_queryset2(i.dept) if ep_blt.belong_dept in depts and i.type == Access.ACCESS_IN_YES: return elif ep_blt.belong_dept in depts and i.type == Access.ACCESS_IN_NO: # 触发违规进入事件 return handle_xx_event_2('i_enter', ep=ep_blt, area=area) elif i.dept.type == 'rparty': # 如果是相关方 if ep_blt.belong_dept == i.dept and i.type == Access.ACCESS_IN_YES: return elif ep_blt.belong_dept == i.dept and i.type == Access.ACCESS_IN_NO: # 触发违规进入事件 return handle_xx_event_2('i_enter', ep=ep_blt, area=area) elif i.employee: # 如果是按人设定的 if ep_blt == i.employee and i.type == Access.ACCESS_IN_YES: return elif ep_blt == i.employee and i.type == Access.ACCESS_IN_NO: # 触发违规进入事件 return handle_xx_event_2('i_enter', ep=ep_blt, area=area) # 通用权限设置过滤 access_list = area.access_list if ep_blt.type not in access_list: # 触发违规进入事件 return handle_xx_event_2('i_enter', ep=ep_blt, area=area) # elif area and (not blts): # # 触发未知标签进入事件 # e_i_enter(ep=ep_blt, area=area) def rail_out(data): pass def loc_change(data): blts = TDevice.objects.filter(code=data['userId']).first() if blts and blts.employee: # 从缓存查询人员位置信息 key_str = 'ep_{}'.format(blts.employee.id) ep_loc_dict = cache.get_or_set( key_str, get_ep_default(), timeout=None ) ep_loc_dict['time2'] = int(time.time()) ep_loc_dict['xx_detail'] = data area_fix, area_temp = get_area_from_point(data['longitude'], data['latitude'], data['floorNo'], ep_loc_dict['area_fix_id']) time2 = int(time.time()) ep_loc_dict['area_temp_id'] = area_temp['id'] if area_temp else None ep_loc_dict['time2'] = time2 if area_fix and ep_loc_dict['area_fix_id'] == area_fix['id']: # 如果停留在该区域 cache.set(key_str, ep_loc_dict) # 在该固定区域停留时间(分钟) stay_minute = int((ep_loc_dict['time2']-ep_loc_dict['time1'])/60) # 判断停留时间是否合理 # 先通过自定义权限过滤(暂未做) # 再经过通用设置过滤 code_name = '' # if 0 < stay_minute < area_fix['stay_minute_min']: # # 触发离岗事件 # code_name = 'leave_area' if area_fix['stay_minute_max'] > 0 and area_fix['stay_minute_max'] < stay_minute: # 触发超时滞留事件 code_name = 'stand_area' if code_name: # 2分钟不再次触发 last_event = Event.objects.filter( cates__code=code_name, employee=blts.employee).order_by('-create_time').first() if last_event and last_event.create_time + timedelta(minutes=2) > timezone.now(): pass else: handle_xx_event_2(code_name, ep=blts.employee, area=Area.objects.get(id=area_fix['id'])) else: ep_loc_dict['time1'] = time2 ep_loc_dict['area_fix_id'] = area_fix['id'] if area_fix else None cache.set(key_str, ep_loc_dict) return ep_loc_dict def handle_xx_event(name: str, data: dict): # 有绑定对象再提示事件(包括一键报警事件/低电量) blts = TDevice.objects.filter(code=data['mac']).first() ep_loc_dict = {} if blts and blts.employee: # 触发事件 cate = EventCate.objects.filter(code=name).first() # 找到最近未处理同一人发生的事件 if cate: # 5分钟内不再次触发 last_event = Event.objects.filter(cates__code=name, employee=blts.employee).order_by('-create_time').first() if last_event and last_event.create_time + timedelta(minutes=5) > timezone.now(): return event = Event() # 查询定位信息 key_str = 'ep_{}'.format(blts.employee.id) ep_loc_dict = cache.get(key_str, None) if ep_loc_dict: area = None if ep_loc_dict['area_fix_id']: area = Area.objects.get(id=ep_loc_dict['area_fix_id']) else: area = Area.objects.get(id='1569585234968711168') event.area = area event.location = ep_loc_dict event.obj_cate = 'people' event.employee = blts.employee event.happen_time = timezone.now() event.save() Eventdo.objects.get_or_create(cate=cate, event=event, defaults={ 'cate': cate, 'event': event }) notify_event(event) return ep_loc_dict def handle_xx_event_2(name: str, ep: Employee, area: Area): # 违规进入事件特殊处理 # 找寻该区域下审批和进行的作业, 本厂或相关方人员, 如是就不触发 if name == 'i_enter' and ep.type in ['employee', 'remployee']: last_event = Event.objects.filter( cates__code='i_enter', employee=ep, area=area).order_by('-create_time').first() if last_event and last_event.create_time + timedelta(minutes=2) > timezone.now(): # 2分钟不再次触发 return ops = Operation.objects.filter(area=area, state__in=[Operation.OP_AUDIT, Operation.OP_WAIT, Operation.OP_WORK]) if OplWorker.objects.filter(opl__operation__in=ops, worker__employee=ep).exists(): # 如果是作业人员 return elif ops.filter(coordinator__employee=ep).exists(): # 如果是协调员 return elif Opl.objects.filter(operation__in=ops, charger__employee=ep).exists(): # 如果是作业负责人 return elif Opl.objects.filter(operation__in=ops, monitor__employee=ep).exists(): # 如果是作业监护人 return cate = EventCate.objects.filter(code=name).first() if cate: event = Event() event.area = area # 查询定位信息 key_str = 'ep_{}'.format(ep.id) ep_loc_dict = cache.get(key_str, None) if ep_loc_dict: event.location = ep_loc_dict event.obj_cate = 'people' event.employee = ep event.happen_time = timezone.now() event.save() Eventdo.objects.get_or_create(cate=cate, event=event, defaults={ 'cate': cate, 'event': event }) notify_event(event) def blt_online(data): # 定位在线 blts = TDevice.objects.filter(code=data['userId']).first() if blts.employee: ep = blts.employee if ep.type == 'employee' and ep.is_atwork is False: # 上班打卡 now = timezone.now() cr_10 = ClockRecord.objects.filter(type=10, employee=ep, create_time__year=now.year, create_time__month=now.month, create_time__day=now.day).first() if cr_10: if now < cr_10.create_by: cr_10.create_by = now cr_10.trigger = 'location' cr_10.detail = data cr_10.save() else: cr_10 = ClockRecord() cr_10.type = 10 cr_10.employee = ep cr_10.trigger = 'location' cr_10.detail = data cr_10.save() ep.is_atwork = True ep.last_check_time = now ep.save() def blt_offline(data): blts = TDevice.objects.filter(code=data['userId']).first() # 定位离线 if blts.employee: ep = blts.employee if ep.type == 'employee' and ep.is_atwork: # 下班打卡 now = timezone.now() cr_20 = ClockRecord.objects.filter(type=20, employee=ep, create_time__year=now.year, create_time__month=now.month, create_time__day=now.day).first() if cr_20: if now > cr_20.create_time: cr_20.create_time = now cr_20.trigger = 'location' cr_20.detail = data cr_20.save() else: cr_20 = ClockRecord() cr_20.type = 20 cr_20.employee = ep cr_20.trigger = 'location' cr_20.detail = data cr_20.save() ep.is_atwork = False ep.last_check_time = now ep.save() def get_area_from_point(x: int, y: int, floorNo: str, area_fix_id): """ 从位置信息获取所在固定区域(可能有一个区域) 返回一个固定区域, 一个临时区域 """ area_fix = None area_temp = None area_list = cache.get('area_list', None) if not area_list: area_list = cache_areas_info() point = Point(x, y) for i in area_list: if floorNo == i['floor_no']: if i['polygon'].intersects(point): # 如果点在多边形中 if i['id'] == area_fix_id: area_fix = i elif i['type'] == Area.AREA_TYPE_FIX: area_fix = i elif i['type'] == Area.AREA_TYPE_TEMP: area_temp = i return area_fix, area_temp def snap_and_analyse(vchannel: TDevice, algo_codes: list, opl: Opl = None): global_img_o = dhClient.snap(vchannel.code) happen_time = timezone.now() ec_codes = ai_analyse(algo_codes, global_img=global_img_o) # 算法处理返回的事件结果 if ec_codes: # 获取本次所有发生事件种类 ecs = EventCate.objects.filter(code__in=ec_codes.keys()) obj_cate = 'opl' if opl else 'other' ep = None if 'helmet' in ec_codes or 'helmet2' in ec_codes and obj_cate == 'other': obj_cate = 'people' # 如果存在安全帽事件 # 尝试以图搜图找到当事人 # res = dhClient.face_search(path=global_img_o) # if res and res[0]: # ep = Employee.objects.filter(id_number=res[0]['identity']).first() pass event = Event() event.global_img = save_dahua_pic(global_img_o) event.vchannel = vchannel event.area = vchannel.area event.obj_cate = obj_cate event.employee = ep event.happen_time = happen_time event.opl = opl event.save() for i in ecs: Eventdo.objects.get_or_create(cate=i, event=event, defaults={ 'cate': i, 'event': event }) if event: notify_event(event) def loop_task(): # 轮询拍照进行算法识别 pass # cates = EventCate.objects.filter(loop_on=True, self_algo=True) # AlgoChannel.objects.filter(algo__loop_on=True, algo__self_algo=True).values('algo__code', 'vchannel')