gatewayController.py 29 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709
  1. # -*- coding: utf-8 -*-
  2. """
  3. @Author : Rocky
  4. @Time : 2022/5/9 10:51
  5. @File :gatewayController.py
  6. """
  7. import datetime
  8. import threading
  9. import time
  10. import pytz
  11. import requests
  12. from django.views.generic.base import View
  13. from AnsjerPush.Config.gatewaySensorConfig import SENSOR_TYPE, EVENT_TYPE, DEVICE_TYPE, SMART_SOCKET_TOPIC, \
  14. ANSJER_GENERIC_TOPIC
  15. from AnsjerPush.config import LOGGER, XM_PUSH_CHANNEL_ID
  16. from Model.models import SensorRecord, GatewaySubDevice, GatewayPush, Device_Info, SceneLog, SmartScene, CountryModel, \
  17. EffectiveTime
  18. from Object.ResponseObject import ResponseObject
  19. from Service.CommonService import CommonService
  20. from AnsjerPush.config import DETECT_PUSH_DOMAIN
  21. from Service.EquipmentInfoService import EquipmentInfoService
  22. from Service.HuaweiPushService.HuaweiPushService import HuaweiPushObject
  23. from Service.PushService import PushObject
  24. class GatewayView(View):
  25. def get(self, request, *args, **kwargs):
  26. request.encoding = 'utf-8'
  27. operation = kwargs.get('operation')
  28. return self.validation(request.GET, operation)
  29. def post(self, request, *args, **kwargs):
  30. request.encoding = 'utf-8'
  31. operation = kwargs.get('operation')
  32. return self.validation(request.POST, operation)
  33. def validation(self, request_dict, operation):
  34. response = ResponseObject()
  35. if operation == 'gatewayPush': # 网关推送
  36. return self.gateway_push(request_dict, response)
  37. elif operation == 'sceneLogPush': # 场景日志推送
  38. return self.scene_log_push(request_dict, response)
  39. elif operation == 'socketPush': # 插座推送
  40. return self.socket_msg_push(request_dict, response)
  41. else:
  42. return response.json(414)
  43. @classmethod
  44. def gateway_push(cls, request_dict, response):
  45. """
  46. 网关推送
  47. @param request_dict: 请求参数
  48. @request_dict serial_number: 序列号
  49. @request_dict ieee_addr: 长地址
  50. @request_dict sensor_type: 传感器类型
  51. @request_dict event_type: 事件类型
  52. @request_dict defense: 防御状态,0:撤防,1:防御
  53. @request_dict sensor_status: 拆动状态,拆动时传参
  54. @param response: 响应对象
  55. @return: response
  56. """
  57. serial_number = request_dict.get('serial_number', None)
  58. ieee_addr = request_dict.get('ieee_addr', None)
  59. sensor_type = int(request_dict.get('sensor_type', None))
  60. event_type = int(request_dict.get('event_type', None))
  61. defense = int(request_dict.get('defense', None))
  62. LOGGER.info('---调用网关推送接口--- request_dict:{}'.format(request_dict))
  63. if not all([serial_number, ieee_addr, sensor_type, event_type]):
  64. return response.json(444)
  65. num = None
  66. n_time = int(time.time())
  67. try:
  68. # 获取温湿度
  69. if sensor_type == SENSOR_TYPE['tem_hum_sensor'] and (
  70. event_type == EVENT_TYPE['temperature'] or event_type == EVENT_TYPE['humidity']):
  71. num = request_dict.get('num')
  72. if num is None:
  73. return response.json(444)
  74. num = int(num) / 100
  75. # 查询子设备表id
  76. gateway_sub_device_qs = GatewaySubDevice.objects.filter(device__serial_number=serial_number,
  77. device_type=sensor_type, ieee_addr=ieee_addr). \
  78. values('id', 'nickname', 'device__userID__region_country')
  79. if not gateway_sub_device_qs.exists():
  80. return response.json(173)
  81. gateway_sub_device_id = gateway_sub_device_qs[0]['id']
  82. # 多线程执行场景任务
  83. task_kwargs = {
  84. 'num': num,
  85. 'n_time': n_time,
  86. 'event_type': event_type,
  87. 'gateway_sub_device_id': gateway_sub_device_id
  88. }
  89. execute_task_thread = threading.Thread(target=cls.execute_scene_tasks, kwargs=task_kwargs)
  90. execute_task_thread.start()
  91. country_id = gateway_sub_device_qs[0]['device__userID__region_country']
  92. lang = cls.confirm_lang(country_id)
  93. alarm = cls.get_alarm(lang, event_type)
  94. nickname = gateway_sub_device_qs[0]['nickname']
  95. sensor_record_dict = {
  96. 'gateway_sub_device_id': gateway_sub_device_id,
  97. 'alarm': alarm,
  98. 'event_type': event_type,
  99. 'created_time': n_time,
  100. }
  101. # 温湿度上报不推送
  102. if num is not None:
  103. sensor_record_dict['alarm'] = str(num)
  104. SensorRecord.objects.create(**sensor_record_dict)
  105. return response.json(0)
  106. SensorRecord.objects.create(**sensor_record_dict)
  107. # 门磁被拆动/拆动恢复,修改拆动状态
  108. if event_type == 2156:
  109. gateway_sub_device_qs.update(is_tampered=1)
  110. elif event_type == 2152:
  111. gateway_sub_device_qs.update(is_tampered=0)
  112. # 撤防状态不推送
  113. if defense == 0:
  114. return response.json(0)
  115. device_info_qs = Device_Info.objects.filter(serial_number=serial_number).values('userID_id')
  116. if not device_info_qs.exists():
  117. return response.json(173)
  118. equipment_info_list = []
  119. equipment_info_model = EquipmentInfoService.randoms_choice_equipment_info()
  120. # 推送表存储数据
  121. equipment_info_kwargs = {
  122. 'device_uid': serial_number,
  123. 'device_nick_name': nickname,
  124. 'event_type': event_type,
  125. 'event_time': n_time,
  126. 'add_time': n_time,
  127. 'alarm': alarm
  128. }
  129. for device_info in device_info_qs:
  130. user_id = device_info['userID_id']
  131. equipment_info_kwargs['device_user_id'] = user_id
  132. equipment_info_list.append(equipment_info_model(**equipment_info_kwargs))
  133. # 开启异步推送
  134. push_kwargs = {
  135. 'user_id': user_id,
  136. 'n_time': n_time,
  137. 'event_type': event_type,
  138. 'nickname': nickname,
  139. 'alarm': alarm,
  140. }
  141. push_thread = threading.Thread(
  142. target=cls.gateway_push_msg,
  143. kwargs=push_kwargs)
  144. push_thread.start()
  145. if equipment_info_list:
  146. equipment_info_model.objects.bulk_create(equipment_info_list)
  147. return response.json(0)
  148. except Exception as e:
  149. LOGGER.info('---网关推送接口异常--- {}'.format(repr(e)))
  150. return response.json(500, 'error_ine:{}, error_msg:{}'.format(e.__traceback__.tb_lineno, repr(e)))
  151. @staticmethod
  152. def confirm_lang(country_id):
  153. """
  154. 根据country_id确定语言
  155. @param country_id: 国家id
  156. @return lang: 语言
  157. """
  158. country_qs = CountryModel.objects.filter(id=country_id).values('country_code')
  159. if not country_qs.exists():
  160. lang = 'NA'
  161. else:
  162. lang = country_qs[0]['country_code']
  163. return lang
  164. @staticmethod
  165. def get_alarm(lang, event_type):
  166. """
  167. 根据语言和事件类型确定警报内容
  168. @param lang: 语言
  169. @param event_type: 事件类型
  170. @return alarm: 警报内容
  171. """
  172. alarm = ''
  173. if lang == 'CN':
  174. # 门磁
  175. if event_type == 2150:
  176. alarm = '门磁开'
  177. elif event_type == 2151:
  178. alarm = '门磁关'
  179. elif event_type == 2156:
  180. alarm = '被拆动'
  181. elif event_type == 2152:
  182. alarm = '拆动恢复'
  183. # 智能按钮
  184. elif event_type == 2160:
  185. alarm = '紧急按钮按下'
  186. elif event_type == 2161:
  187. alarm = '单击'
  188. elif event_type == 2162:
  189. alarm = '双击'
  190. elif event_type == 2163:
  191. alarm = '长按'
  192. # 水浸
  193. elif event_type == 2170:
  194. alarm = '水浸触发'
  195. elif event_type == 2171:
  196. alarm = '水浸恢复'
  197. # 烟雾
  198. elif event_type == 2180:
  199. alarm = '烟雾触发'
  200. elif event_type == 2181:
  201. alarm = '烟雾恢复'
  202. # 人体红外
  203. elif event_type == 2190:
  204. alarm = '有人移动'
  205. elif event_type == 2191:
  206. alarm = '无人移动'
  207. # 低电量
  208. elif event_type in (2153, 2164, 2172, 2182, 2193):
  209. alarm = '低电量'
  210. else:
  211. # 门磁
  212. if event_type == 2150:
  213. alarm = 'Door magnetic opening'
  214. elif event_type == 2151:
  215. alarm = 'Door magnetic closing'
  216. elif event_type == 2156:
  217. alarm = 'Be dismantled'
  218. elif event_type == 2152:
  219. alarm = 'Dismantling recovery'
  220. # 智能按钮
  221. elif event_type == 2160:
  222. alarm = 'Emergency button pressed'
  223. elif event_type == 2161:
  224. alarm = 'Single click'
  225. elif event_type == 2162:
  226. alarm = 'Double click'
  227. elif event_type == 2163:
  228. alarm = 'Long press'
  229. # 水浸
  230. elif event_type == 2170:
  231. alarm = 'Water immersion trigger'
  232. elif event_type == 2171:
  233. alarm = 'Water immersion recovery'
  234. # 烟雾
  235. elif event_type == 2180:
  236. alarm = 'Smoke triggering'
  237. elif event_type == 2181:
  238. alarm = 'Smoke recovery'
  239. # 人体红外
  240. elif event_type == 2190:
  241. alarm = 'Someone moving'
  242. elif event_type == 2191:
  243. alarm = 'Unmanned movement'
  244. # 低电量
  245. elif event_type in (2153, 2164, 2172, 2182, 2193):
  246. alarm = 'LOW BATTERY'
  247. return alarm
  248. @classmethod
  249. def scene_log_push(cls, request_dict, response):
  250. """
  251. 网关智能场景日志推送
  252. @param request_dict: 请求参数
  253. @request_dict sceneId: 场景id
  254. @request_dict status: 状态
  255. @param response: 响应对象
  256. @return: response
  257. """
  258. scene_id = request_dict.get('sceneId', None)
  259. status = request_dict.get('status', None)
  260. LOGGER.info('---场景日志推送接口--- request_dict:{}'.format(request_dict))
  261. if not all([scene_id, status]):
  262. return response.json(444)
  263. smart_scene_qs = SmartScene.objects.filter(id=scene_id).values('scene_name', 'conditions', 'tasks', 'device_id',
  264. 'sub_device_id', 'user_id', 'scene_data')
  265. if not smart_scene_qs.exists():
  266. return response.json(173)
  267. scene_name = smart_scene_qs[0]['scene_name']
  268. tasks = smart_scene_qs[0]['tasks']
  269. device_id = smart_scene_qs[0]['device_id']
  270. sub_device_id = smart_scene_qs[0]['sub_device_id']
  271. n_time = int(time.time())
  272. user_id = smart_scene_qs[0]['user_id']
  273. scene_data = smart_scene_qs[0]['scene_data']
  274. if sub_device_id:
  275. gateway_sub_device_qs = GatewaySubDevice.objects.filter(id=sub_device_id).values('nickname')
  276. nickname = gateway_sub_device_qs[0]['nickname'] if gateway_sub_device_qs.exists() else ''
  277. else:
  278. device_qs = Device_Info.objects.filter(id=device_id).values('NickName')
  279. nickname = device_qs[0]['NickName'] if device_qs.exists() else ''
  280. log_dict = {
  281. 'scene_id': scene_id,
  282. 'scene_name': scene_name,
  283. 'tasks': tasks,
  284. 'status': status,
  285. 'device_id': device_id,
  286. 'sub_device_id': sub_device_id,
  287. 'created_time': n_time,
  288. }
  289. tasks = eval(tasks)
  290. try:
  291. SceneLog.objects.create(**log_dict)
  292. # 如果是一次性场景,关闭场景
  293. if scene_data:
  294. scene_data_dict = eval(scene_data)
  295. condition = scene_data_dict.get('condition')
  296. if condition:
  297. time_type = condition.get('time')
  298. if time_type == 'date':
  299. smart_scene_qs.update(is_enable=False)
  300. # 推送日志
  301. gateway_push_qs = GatewayPush.objects.filter(user_id=user_id, logout=False). \
  302. values('user_id', 'app_bundle_id', 'app_type', 'push_type', 'token_val', 'm_code', 'lang', 'm_code',
  303. 'tz')
  304. if not gateway_push_qs.exists():
  305. return response.json(174)
  306. # 开启异步推送
  307. push_kwargs = {
  308. 'tasks': tasks,
  309. 'n_time': n_time,
  310. 'nickname': nickname,
  311. 'gateway_push_qs': gateway_push_qs
  312. }
  313. push_thread = threading.Thread(
  314. target=cls.scene_msg_push,
  315. kwargs=push_kwargs)
  316. push_thread.start()
  317. return response.json(0)
  318. except Exception as e:
  319. LOGGER.info('---场景日志推送接口异常--- {}'.format(repr(e)))
  320. return response.json(500, 'error_ine:{}, error_msg:{}'.format(e.__traceback__.tb_lineno, repr(e)))
  321. @classmethod
  322. def gateway_push_msg(cls, **push_kwargs):
  323. user_id = push_kwargs['user_id']
  324. # 查询推送配置数据
  325. gateway_push_qs = GatewayPush.objects.filter(user_id=user_id, logout=False). \
  326. values('user_id', 'app_bundle_id', 'app_type', 'push_type', 'token_val', 'm_code', 'lang', 'm_code',
  327. 'tz')
  328. if gateway_push_qs.exists():
  329. n_time = push_kwargs['n_time']
  330. event_type = push_kwargs['event_type']
  331. nickname = push_kwargs['nickname']
  332. alarm = push_kwargs['alarm']
  333. kwargs = {
  334. 'n_time': n_time,
  335. 'event_type': event_type,
  336. 'nickname': nickname,
  337. }
  338. # 推送到每台登录账号的手机
  339. for gateway_push in gateway_push_qs:
  340. app_bundle_id = gateway_push['app_bundle_id']
  341. push_type = gateway_push['push_type']
  342. token_val = gateway_push['token_val']
  343. lang = gateway_push['lang']
  344. tz = gateway_push['tz'] if gateway_push['tz'] else 0
  345. # 获取推送所需数据
  346. msg_title = PushObject.get_msg_title(nickname)
  347. msg_text = PushObject.get_gateway_msg_text(n_time, tz, lang, alarm)
  348. kwargs['msg_title'] = msg_title
  349. kwargs['msg_text'] = msg_text
  350. kwargs['app_bundle_id'] = app_bundle_id
  351. kwargs['token_val'] = token_val
  352. try:
  353. # 推送消息
  354. cls.push_msg(push_type, **kwargs)
  355. except Exception as e:
  356. LOGGER.info('网关推送消息异常,errLine:{}, errMsg:{}'.format(e.__traceback__.tb_lineno, repr(e)))
  357. continue
  358. @classmethod
  359. def scene_msg_push(cls, **push_kwargs):
  360. tasks = push_kwargs['tasks']
  361. n_time = push_kwargs['n_time']
  362. nickname = push_kwargs['nickname']
  363. gateway_push_qs = push_kwargs['gateway_push_qs']
  364. for task in tasks:
  365. event_type = task['event_type']
  366. if event_type == '1001':
  367. kwargs = {
  368. 'n_time': n_time,
  369. 'event_type': event_type,
  370. 'nickname': nickname,
  371. }
  372. event_info = task['value']
  373. # 推送到每台登录账号的手机
  374. for gateway_push in gateway_push_qs:
  375. app_bundle_id = gateway_push['app_bundle_id']
  376. push_type = gateway_push['push_type']
  377. token_val = gateway_push['token_val']
  378. kwargs['msg_title'] = PushObject.get_msg_title(nickname)
  379. kwargs['msg_text'] = event_info
  380. kwargs['app_bundle_id'] = app_bundle_id
  381. kwargs['token_val'] = token_val
  382. try:
  383. # 推送消息
  384. cls.push_msg(push_type, **kwargs)
  385. except Exception as e:
  386. LOGGER.info(
  387. '场景日志推送消息异常,errLine:{}, errMsg:{}'.format(e.__traceback__.tb_lineno, repr(e)))
  388. continue
  389. @staticmethod
  390. def push_msg(push_type, **kwargs):
  391. """
  392. 发送推送消息
  393. @param push_type: 推送类型
  394. @param kwargs: 推送参数
  395. @return: None
  396. """
  397. if push_type == 0: # ios apns
  398. PushObject.ios_apns_push(**kwargs)
  399. elif push_type == 1: # android gcm
  400. PushObject.android_fcm_push_v1(**kwargs)
  401. elif push_type == 2: # android 极光推送
  402. PushObject.android_jpush(**kwargs)
  403. elif push_type == 3:
  404. huawei_push_object = HuaweiPushObject()
  405. huawei_push_object.send_push_notify_message(**kwargs)
  406. elif push_type == 4: # android 小米推送
  407. channel_id = XM_PUSH_CHANNEL_ID['device_reminder']
  408. PushObject.android_xmpush(channel_id=channel_id, **kwargs)
  409. elif push_type == 5: # android vivo推送
  410. PushObject.android_vivopush(**kwargs)
  411. elif push_type == 6: # android oppo推送
  412. channel_id = 'DEVICE_REMINDER'
  413. PushObject.android_oppopush(channel_id=channel_id, **kwargs)
  414. elif push_type == 7: # android 魅族推送
  415. PushObject.android_meizupush(**kwargs)
  416. @classmethod
  417. def socket_msg_push(cls, request_dict, response):
  418. """
  419. 智能插座开关状态推送
  420. """
  421. try:
  422. serial_number = request_dict.get('serialNumber', None)
  423. device_time = request_dict.get('deviceTime', None)
  424. status = request_dict.get('status', None)
  425. if not all([serial_number, status, device_time]):
  426. return response.json(444)
  427. status = int(status)
  428. now_time = int(device_time) if device_time else int(time.time())
  429. # 获取主用户设备id
  430. log_dict = {
  431. 'status': status,
  432. 'device_id': serial_number,
  433. 'created_time': now_time,
  434. }
  435. SceneLog.objects.create(**log_dict)
  436. LOGGER.info('成功接收并保存,插座序列号{},状态:{}'.format(serial_number, status))
  437. return response.json(0)
  438. except Exception as e:
  439. LOGGER.info('插座开关日志推送接口异常, error_line:{}, error_msg:{}'.format(e.__traceback__.tb_lineno, repr(e)))
  440. return response.json(500, 'error_line:{}, error_msg:{}'.format(e.__traceback__.tb_lineno, repr(e)))
  441. @classmethod
  442. def execute_scene_tasks(cls, gateway_sub_device_id, event_type, num, n_time):
  443. """
  444. 执行场景任务
  445. @param gateway_sub_device_id: 子设备id
  446. @param event_type: 事件类型
  447. @param num: 温湿度值
  448. @param n_time: 当前时间
  449. @return:
  450. """
  451. smart_scene_qs = SmartScene.objects.filter(sub_device_id=gateway_sub_device_id, is_enable=True). \
  452. values('id', 'scene_data', 'tz', 'is_all_day', 'effective_time_id')
  453. if smart_scene_qs.exists():
  454. for smart_scene in smart_scene_qs:
  455. if smart_scene['scene_data']:
  456. scene_data_dict = eval(smart_scene['scene_data'])
  457. condition = scene_data_dict.get('condition')
  458. if condition:
  459. # None 或 event_type
  460. scene_data_event_type = condition.get('event_type')
  461. # 触发事件类型跟条件事件类型一致,且任务列表不为空
  462. task_list = scene_data_dict.get('task_list')
  463. if event_type == scene_data_event_type and task_list:
  464. # 判断温湿度是否在条件设置的范围
  465. if num is not None:
  466. condition_value = condition.get('value')
  467. space_index = condition_value.index(' ')
  468. symbol, value = condition_value[:space_index], float(condition_value[space_index + 1:])
  469. # 温湿度不在设定范围,不执行
  470. if not ((symbol == '≥' and num >= value) or (symbol == '≤' and num <= value)):
  471. continue
  472. execute_task = cls.judge_execute_task(smart_scene, n_time)
  473. # 执行任务
  474. if execute_task:
  475. scene_id = 0
  476. task_list_len = len(task_list)
  477. for index, task in enumerate(task_list):
  478. # 最后一个任务上报日志
  479. if index == task_list_len - 1:
  480. scene_id = smart_scene['id']
  481. kwargs = {
  482. 'sensor_event_type': event_type,
  483. 'device_type': task['device_type'],
  484. 'event_type': task['event_type'],
  485. 'serial_number': task['serial_number'],
  486. 'delay_time': task['delay_time'],
  487. 'scene_id': scene_id
  488. }
  489. pub_mqtt_thread = threading.Thread(target=cls.pub_mqtt, kwargs=kwargs)
  490. pub_mqtt_thread.start()
  491. @classmethod
  492. def judge_execute_task(cls, smart_scene, n_time):
  493. """
  494. 判断是否执行任务
  495. @param smart_scene: 场景数据
  496. @param n_time: 当前时间
  497. @return execute_task: bool
  498. """
  499. execute_task = False
  500. # 判断时间是否在执行时间范围内
  501. if smart_scene['is_all_day'] == 1:
  502. # 全天必执行
  503. execute_task = True
  504. elif smart_scene['is_all_day'] == 2:
  505. # 非全天,判断当前时间是否在设置的时间范围
  506. tz = smart_scene['tz']
  507. time_minute, week = cls.get_time_info(n_time, tz)
  508. effective_time_id = smart_scene['effective_time_id']
  509. effective_time_qs = EffectiveTime.objects.filter(id=effective_time_id). \
  510. values('start_time', 'end_time', 'repeat')
  511. if effective_time_qs.exists():
  512. start_time = effective_time_qs[0]['start_time']
  513. end_time = effective_time_qs[0]['end_time']
  514. repeat = effective_time_qs[0]['repeat']
  515. time_frame_dict, time_frame_next_day_dict = cls.get_time_frame_dict(
  516. start_time, end_time, repeat)
  517. # 判断当前时间是否在设置的时间范围
  518. if time_frame_dict.get(week):
  519. start_time = time_frame_dict[week]['start_time']
  520. end_time = time_frame_dict[week]['end_time']
  521. if start_time <= time_minute <= end_time:
  522. execute_task = True
  523. if time_frame_next_day_dict.get(week):
  524. start_time = time_frame_next_day_dict[week]['start_time']
  525. end_time = time_frame_next_day_dict[week]['end_time']
  526. if start_time <= time_minute <= end_time:
  527. execute_task = True
  528. return execute_task
  529. @staticmethod
  530. def get_time_info(timestamp, timezone_offset):
  531. """
  532. 根据时间戳和时区获取时间(时分转为分钟数)和星期
  533. @param timestamp: 时间戳
  534. @param timezone_offset: 时区
  535. @return: time_minute, week
  536. """
  537. # 计算时区偏移量(以分钟为单位)
  538. timezone_offset_minutes = int(timezone_offset * 60)
  539. timezone = pytz.FixedOffset(timezone_offset_minutes)
  540. # 将时间戳转换为datetime对象,并应用时区
  541. dt_object = datetime.datetime.fromtimestamp(timestamp, tz=timezone)
  542. # 获取时分,并转为分钟数
  543. hour = dt_object.hour
  544. minute = dt_object.minute
  545. time_minute = hour * 60 + minute
  546. # 获取星期(0表示星期一,6表示星期日)
  547. week = str(dt_object.weekday())
  548. return time_minute, week
  549. @staticmethod
  550. def get_time_frame_dict(start_time, end_time, repeat):
  551. """
  552. 获取时间范围字典
  553. @param start_time: 开始时间
  554. @param end_time: 结束时间
  555. @param repeat: 星期周期的十进制数,如127 -> 0,1,2,3,4,5,6
  556. @return: time_frame_dict, time_frame_next_day_dict
  557. """
  558. # 十进制转为7位的二进制并倒序
  559. bin_str = bin(repeat)[2:].zfill(7)[::-1]
  560. # 生成星期周期列表
  561. week_list = []
  562. for i, bit in enumerate(bin_str):
  563. if bit == '1':
  564. week_list.append(i)
  565. # 生成时间范围字典
  566. time_frame_dict = {}
  567. time_frame_next_day_dict = {}
  568. # 非隔天
  569. if end_time > start_time:
  570. for week in week_list:
  571. time_frame_dict[str(week)] = {
  572. 'start_time': start_time,
  573. 'end_time': end_time
  574. }
  575. # 隔天
  576. else:
  577. # time_frame_dict记录当天时间范围,time_frame_next_day_dict记录溢出到第二天的时间范围
  578. for week in week_list:
  579. time_frame_dict[str(week)] = {
  580. 'start_time': start_time,
  581. 'end_time': 1439 # 23:59
  582. }
  583. week += 1
  584. if week == 7:
  585. week = 0
  586. time_frame_next_day_dict[str(week)] = {
  587. 'start_time': 0, # 00:00
  588. 'end_time': end_time
  589. }
  590. return time_frame_dict, time_frame_next_day_dict
  591. @staticmethod
  592. def pub_mqtt(sensor_event_type, device_type, event_type, serial_number, delay_time, scene_id=0):
  593. """
  594. 发布mqtt消息
  595. @param sensor_event_type: 传感器事件类型
  596. @param device_type: 设备类型
  597. @param event_type: 事件类型
  598. @param serial_number: 序列号
  599. @param delay_time: 延迟时间
  600. @param scene_id: 场景id
  601. @return:
  602. """
  603. msg = {}
  604. # 插座
  605. if device_type == DEVICE_TYPE['socket']:
  606. topic_name = SMART_SOCKET_TOPIC.format(serial_number)
  607. status = 1 if event_type == EVENT_TYPE['socket_power_on'] else 0
  608. msg['type'] = 1
  609. msg['data'] = {
  610. 'deviceSwitch': status
  611. }
  612. # 摄像头
  613. elif device_type == DEVICE_TYPE['C516']:
  614. topic_name = ANSJER_GENERIC_TOPIC.format(serial_number)
  615. if event_type == EVENT_TYPE['detection_reminder_on']:
  616. msg['commandType'] = 'detection_reminder'
  617. msg['enable'] = 1
  618. elif event_type == EVENT_TYPE['detection_reminder_off']:
  619. msg['commandType'] = 'detection_reminder'
  620. msg['enable'] = 0
  621. elif event_type == EVENT_TYPE['snapshot']:
  622. msg['commandType'] = 'snapshot'
  623. msg['eventType'] = sensor_event_type
  624. elif event_type == EVENT_TYPE['record_video']:
  625. msg['commandType'] = 'record_video'
  626. elif event_type == EVENT_TYPE['human_tracking_on']:
  627. msg['commandType'] = 'human_tracking'
  628. msg['enable'] = 1
  629. elif event_type == EVENT_TYPE['human_tracking_off']:
  630. msg['commandType'] = 'human_tracking'
  631. msg['enable'] = 0
  632. else:
  633. return
  634. else:
  635. return
  636. if delay_time:
  637. time.sleep(delay_time)
  638. CommonService.req_publish_mqtt_msg(serial_number, topic_name, msg)
  639. # 没有设备任务时,最后一个任务上报场景日志
  640. if scene_id:
  641. data = {
  642. 'sceneId': scene_id,
  643. 'status': 1
  644. }
  645. url = DETECT_PUSH_DOMAIN + 'gatewayService/sceneLogPush'
  646. requests.post(url=url, data=data, timeout=8)