SmartSceneController.py 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559
  1. # -*- coding: utf-8 -*-
  2. """
  3. @Author : Rocky
  4. @Time : 2022/6/29 9:31
  5. @File :SmartSceneController.py
  6. """
  7. import json
  8. import time
  9. from django.core.exceptions import ObjectDoesNotExist
  10. from django.db import transaction
  11. from django.db.models import F, Q
  12. from django.views import View
  13. from Model.models import FamilyRoomDevice, GatewaySubDevice, FamilyRoom, SmartScene, EffectiveTime, Device_Info
  14. from Service.CommonService import CommonService
  15. class SmartSceneView(View):
  16. def get(self, request, *args, **kwargs):
  17. request.encoding = 'utf-8'
  18. operation = kwargs.get('operation')
  19. return self.validation(request.GET, request, operation)
  20. def post(self, request, *args, **kwargs):
  21. request.encoding = 'utf-8'
  22. operation = kwargs.get('operation')
  23. return self.validation(request.POST, request, operation)
  24. def validation(self, request_dict, request, operation):
  25. token_code, user_id, response = CommonService.verify_token_get_user_id(request_dict, request)
  26. if token_code != 0:
  27. return response.json(token_code)
  28. if operation == 'condition-devices': # 添加条件-查询设备
  29. return self.condition_devices(request_dict, response)
  30. elif operation == 'task-devices': # 添加任务-查询设备
  31. return self.task_devices(request_dict, response)
  32. elif operation == 'create': # 创建智能场景
  33. return self.create_smart_scene(request_dict, user_id, response)
  34. elif operation == 'scene-list': # 查询智能场景列表
  35. return self.scene_list(request_dict, user_id, response)
  36. elif operation == 'smart-button-scene-list': # 查询智能按钮场景列表
  37. return self.smart_button_scene_list(request_dict, user_id, response)
  38. elif operation == 'update-status': # 更新智能场景状态
  39. return self.update_status(request_dict, response)
  40. elif operation == 'detail': # 查询智能场景详情
  41. return self.scene_detail(request_dict, response)
  42. elif operation == 'edit': # 编辑智能场景
  43. return self.edit_smart_scene(request_dict, user_id, response)
  44. elif operation == 'delete': # 删除智能场景
  45. return self.delete_smart_scene(request_dict, response)
  46. else:
  47. return response.json(414)
  48. @classmethod
  49. def condition_devices(cls, request_dict, response):
  50. """
  51. 添加条件-查询设备
  52. @param request_dict: 请求参数
  53. @request_dict deviceId: 网关设备id
  54. @request_dict subDeviceId: 子设备id
  55. @param response: 响应对象
  56. @return: response
  57. """
  58. device_id = request_dict.get('deviceId', None)
  59. sub_device_id = request_dict.get('subDeviceId', None)
  60. if not any([device_id, sub_device_id]):
  61. return response.json(444, {'error param': 'deviceId or subDeviceId'})
  62. try:
  63. if sub_device_id:
  64. device_id = GatewaySubDevice.objects.get(id=sub_device_id).device_id
  65. gateway_sub_device_qs = GatewaySubDevice.objects.filter(device_id=device_id)
  66. if not gateway_sub_device_qs.exists():
  67. return response.json(173)
  68. res = cls.get_sub_device_room_name(gateway_sub_device_qs)
  69. return response.json(0, res)
  70. except Exception as e:
  71. return response.json(500, repr(e))
  72. @classmethod
  73. def task_devices(cls, request_dict, response):
  74. """
  75. 添加任务-查询设备
  76. @param request_dict: 请求参数
  77. @request_dict deviceId: 网关设备id
  78. @param response: 响应对象
  79. @return: response
  80. """
  81. sub_device_id = request_dict.get('subDeviceId', None)
  82. device_id = request_dict.get('deviceId', None)
  83. if not any([device_id, sub_device_id]):
  84. return response.json(444, {'error param': 'deviceId or subDeviceId'})
  85. try:
  86. if device_id:
  87. res = [cls.get_gateway_data(device_id)]
  88. else:
  89. sub_device_qs = GatewaySubDevice.objects.filter(id=sub_device_id).values('device_id', 'device_type')
  90. device_id = sub_device_qs[0]['device_id']
  91. device_type = sub_device_qs[0]['device_type']
  92. if device_type != 216: # 非智能按钮只返回网关
  93. res = [cls.get_gateway_data(device_id)]
  94. else:
  95. gateway_data = cls.get_gateway_data(device_id)
  96. sub_device_qs = GatewaySubDevice.objects.filter(
  97. Q(device_id=device_id) & Q(device_type=215) | Q(device_type=219)).values('id', 'nickname',
  98. 'status',
  99. 'device_type')
  100. if not sub_device_qs.exists():
  101. return response.json(173)
  102. res = cls.get_sub_device_room_name(sub_device_qs, gateway_data)
  103. return response.json(0, res)
  104. except Exception as e:
  105. return response.json(500, repr(e))
  106. @staticmethod
  107. def get_sub_device_room_name(sub_device_qs, gateway_data=None):
  108. """
  109. 获取房间名称
  110. @param sub_device_qs: 子设备信息
  111. @param gateway_data: 网关参数
  112. @return: sub_device_list
  113. """
  114. sub_device_list = []
  115. if gateway_data:
  116. sub_device_list.append(gateway_data)
  117. sub_device_qs = sub_device_qs.annotate(gatewaySubId=F('id'),
  118. deviceType=F('device_type'),
  119. deviceNickName=F('nickname')). \
  120. values('gatewaySubId', 'deviceType', 'deviceNickName', 'status')
  121. for sub_device in sub_device_qs:
  122. family_room_device_qs = FamilyRoomDevice.objects.filter(sub_device=sub_device['gatewaySubId']). \
  123. values('room_id')
  124. if not family_room_device_qs.exists():
  125. sub_device['roomName'] = ''
  126. else:
  127. room_id = family_room_device_qs[0]['room_id']
  128. try:
  129. sub_device['roomName'] = FamilyRoom.objects.get(id=room_id).name
  130. except ObjectDoesNotExist:
  131. sub_device['roomName'] = ''
  132. sub_device_list.append(sub_device)
  133. return sub_device_list
  134. @staticmethod
  135. def get_gateway_data(device_id):
  136. """
  137. 获取网关数据
  138. @param device_id: 网关设备id
  139. @return: res
  140. """
  141. device_info_qs = Device_Info.objects.filter(id=device_id).values('NickName', 'Type')
  142. nickname = device_info_qs[0]['NickName']
  143. device_type = device_info_qs[0]['Type']
  144. room_id = FamilyRoomDevice.objects.filter(device_id=device_id).values('room_id')[0]['room_id']
  145. room_id_qs = FamilyRoom.objects.filter(id=room_id).values('name')
  146. room_name = room_id_qs.first()['name'] if room_id_qs.exists() else ''
  147. res = {
  148. 'deviceNickName': nickname,
  149. 'deviceType': device_type,
  150. 'roomName': room_name,
  151. 'status': 1,
  152. }
  153. return res
  154. @staticmethod
  155. def create_smart_scene(request_dict, user_id, response):
  156. """
  157. 创建智能场景
  158. @param request_dict: 请求参数
  159. @param user_id: 用户id
  160. @request_dict deviceId: 网关设备id
  161. @request_dict subDeviceId: 子设备id
  162. @request_dict sceneName: 场景名称
  163. @request_dict conditions: 条件
  164. @request_dict tasks: 任务
  165. @request_dict isAllDay: 是否全天执行
  166. @request_dict startTime: 开始时间
  167. @request_dict endTime: 结束时间
  168. @request_dict repeat: 重复周期
  169. @param response: 响应对象
  170. @return: response
  171. """
  172. device_id = request_dict.get('deviceId', None)
  173. sub_device_id = request_dict.get('subDeviceId', None)
  174. scene_name = request_dict.get('sceneName', None)
  175. conditions = request_dict.get('conditions', None)
  176. tasks = request_dict.get('tasks', None)
  177. is_all_day = request_dict.get('isAllDay', None)
  178. if not all([conditions, tasks]):
  179. return response.json(444, {'error param': 'conditions and tasks'})
  180. now_time = int(time.time())
  181. conditions_dict = eval(conditions)
  182. tasks_list = eval(tasks)
  183. try:
  184. # 判断是否已存在该场景名
  185. smart_scene_qs = SmartScene.objects.filter(user_id=user_id, scene_name=scene_name)
  186. if smart_scene_qs.exists():
  187. return response.json(174)
  188. smart_scene_dict = {
  189. 'user_id': user_id,
  190. 'scene_name': scene_name,
  191. 'conditions': conditions,
  192. 'tasks': tasks,
  193. 'created_time': now_time,
  194. 'updated_time': now_time,
  195. }
  196. msg = {}
  197. # 处理设置时间
  198. if is_all_day is not None:
  199. is_all_day = int(is_all_day)
  200. smart_scene_dict['is_all_day'] = is_all_day
  201. # 处理传网关设备id和子设备id的情况
  202. if conditions_dict['type'] == 1: # 网关设置时间
  203. if not device_id:
  204. return response.json(444, {'error param': 'deviceId'})
  205. smart_scene_dict['device_id'] = device_id
  206. device_info_qs = Device_Info.objects.filter(id=device_id).values('serial_number')
  207. if not device_info_qs.exists():
  208. return response.json(173)
  209. serial_number = device_info_qs[0]['serial_number']
  210. else: # 子设备设置场景
  211. if not sub_device_id:
  212. return response.json(444, {'error param': 'subDeviceId'})
  213. smart_scene_dict['sub_device_id'] = sub_device_id
  214. sub_device_qs = GatewaySubDevice.objects.filter(id=sub_device_id).values('device__serial_number',
  215. 'src_addr')
  216. if not sub_device_qs.exists():
  217. return response.json(173)
  218. serial_number = sub_device_qs[0]['device__serial_number']
  219. msg['scene_status'] = 1
  220. msg['sensor_type'] = conditions_dict['sensor']['device_type']
  221. msg['sensor_src'] = int(sub_device_qs[0]['src_addr'], 16)
  222. msg['sensor_status'] = int(conditions_dict['sensor']['eventValues'][0]['event_type'])
  223. with transaction.atomic():
  224. if is_all_day is None: # 不设置时间
  225. smart_scene_qs = SmartScene.objects.create(**smart_scene_dict)
  226. # 设备的time数据,分钟转为秒
  227. time_dict = {
  228. 'start_time': conditions_dict['time']['minutes'] * 60,
  229. 'repeat': conditions_dict['time']['repeat']
  230. }
  231. elif is_all_day == 1: # 全天
  232. smart_scene_qs = SmartScene.objects.create(**smart_scene_dict)
  233. # 设备的time数据
  234. time_dict = {
  235. 'is_all_day': is_all_day
  236. }
  237. elif is_all_day == 2: # 非全天
  238. start_time = int(request_dict.get('startTime', None))
  239. end_time = int(request_dict.get('endTime', None))
  240. repeat = int(request_dict.get('repeat', None))
  241. effective_time_qs = EffectiveTime.objects.filter(start_time=start_time, end_time=end_time,
  242. repeat=repeat).values('id')
  243. if effective_time_qs.exists():
  244. effective_time_id = effective_time_qs[0]['id']
  245. else:
  246. effective_time_id = EffectiveTime.objects.create(start_time=start_time, end_time=end_time,
  247. repeat=repeat).id
  248. smart_scene_dict['effective_time_id'] = effective_time_id
  249. smart_scene_qs = SmartScene.objects.create(**smart_scene_dict)
  250. # 设备的time数据,分钟转为秒
  251. time_dict = {
  252. 'is_all_day': is_all_day,
  253. 'start_time': start_time * 60,
  254. 'end_time': end_time * 60,
  255. 'repeat': repeat
  256. }
  257. else:
  258. return response.json(444, {'error param': 'invalid isAllDay'})
  259. msg['time'] = time_dict
  260. msg['smart_scene_id'] = smart_scene_qs.id
  261. task_list = []
  262. for task in tasks_list:
  263. task_temp = {
  264. 'sensor_type': task['device_type'],
  265. 'sensor_action': task['event_type']
  266. }
  267. sub_device_id = task.get('subDeviceId', None)
  268. if sub_device_id:
  269. sub_device_qs = GatewaySubDevice.objects.filter(id=sub_device_id).values('src_addr').first()
  270. task_temp['sensor_src'] = int(sub_device_qs['src_addr'], 16)
  271. task_list.append(task_temp)
  272. msg['task'] = task_list
  273. smart_scene_qs.device_data = json.dumps(msg)
  274. smart_scene_qs.save()
  275. # 发布MQTT消息通知网关设备
  276. thing_name = serial_number
  277. topic_name = 'loocam/gateway_sensor/smart_scene/{}'.format(serial_number)
  278. success = CommonService.req_publish_mqtt_msg(thing_name, topic_name, msg)
  279. try:
  280. assert success
  281. except AssertionError:
  282. return response.json(10044)
  283. return response.json(0)
  284. except Exception as e:
  285. return response.json(500, repr(e))
  286. @staticmethod
  287. def scene_list(request_dict, user_id, response):
  288. """
  289. 查询智能场景列表
  290. @param request_dict: 请求参数
  291. @param user_id: 用户id
  292. @request_dict deviceId: 网关设备id
  293. @request_dict subDeviceId: 子设备id
  294. @param response: 响应对象
  295. @return: response
  296. """
  297. device_id = request_dict.get('deviceId', None)
  298. sub_device_id = request_dict.get('subDeviceId', None)
  299. if not any([device_id, sub_device_id]):
  300. return response.json(444, {'error param': 'deviceId or subDeviceId'})
  301. try:
  302. if device_id:
  303. sub_device_id = GatewaySubDevice.objects.filter(device_id=device_id).values('id')
  304. smart_scene_qs = SmartScene.objects.filter(
  305. Q(user_id=user_id) & Q(device_id=device_id) | Q(sub_device_id__in=sub_device_id))
  306. else:
  307. smart_scene_qs = SmartScene.objects.filter(user_id=user_id, sub_device_id=sub_device_id)
  308. if not smart_scene_qs.exists():
  309. return response.json(173)
  310. smart_scene_qs = smart_scene_qs.values('id', 'scene_name', 'is_enable')
  311. return response.json(0, list(smart_scene_qs))
  312. except Exception as e:
  313. return response.json(500, repr(e))
  314. @staticmethod
  315. def smart_button_scene_list(request_dict, user_id, response):
  316. """
  317. 查询智能按钮场景列表
  318. @param request_dict: 请求参数
  319. @param user_id: 用户id
  320. @request_dict subDeviceId: 子设备id
  321. @param response: 响应对象
  322. @return: response
  323. """
  324. sub_device_id = request_dict.get('subDeviceId', None)
  325. if not sub_device_id:
  326. return response.json(444, {'error param': 'subDeviceId'})
  327. try:
  328. click_scene_qs = SmartScene.objects.filter(user_id=user_id, sub_device_id=sub_device_id,
  329. conditions__contains='2161')
  330. double_click_scene_qs = SmartScene.objects.filter(user_id=user_id, sub_device_id=sub_device_id,
  331. conditions__contains='2162')
  332. double_click_scene_qs = SmartScene.objects.filter(user_id=user_id, sub_device_id=sub_device_id,
  333. conditions__contains='2162')
  334. if not click_scene_qs.exists():
  335. return response.json(173)
  336. smart_scene_qs = click_scene_qs.values('id', 'scene_name', 'is_enable')
  337. return response.json(0, list(smart_scene_qs))
  338. except Exception as e:
  339. return response.json(500, repr(e))
  340. @staticmethod
  341. def update_status(request_dict, response):
  342. """
  343. 更新智能场景状态
  344. @param request_dict: 请求参数
  345. @request_dict smartSceneId: 智能场景id
  346. @request_dict isEnable: 状态,True or False
  347. @param response: 响应对象
  348. @return: response
  349. """
  350. smart_scene_id = request_dict.get('smartSceneId', None)
  351. is_enable = request_dict.get('isEnable', None)
  352. if not all([smart_scene_id, is_enable]):
  353. return response.json(444, {'error param': 'smartSceneId and status'})
  354. try:
  355. SmartScene.objects.filter(id=smart_scene_id).update(is_enable=is_enable)
  356. return response.json(0)
  357. except Exception as e:
  358. return response.json(500, repr(e))
  359. @staticmethod
  360. def scene_detail(request_dict, response):
  361. """
  362. 查询智能场景详情
  363. @param request_dict: 请求参数
  364. @request_dict smartSceneId: 智能场景id
  365. @param response: 响应对象
  366. @return: response
  367. """
  368. smart_scene_id = request_dict.get('smartSceneId', None)
  369. if not smart_scene_id:
  370. return response.json(444, {'error param': 'smartSceneId'})
  371. try:
  372. smart_scene_qs = SmartScene.objects.filter(id=smart_scene_id).values('id', 'scene_name', 'conditions',
  373. 'tasks', 'effective_time_id',
  374. 'is_all_day')
  375. if not smart_scene_qs.exists():
  376. return response.json(173)
  377. res = {
  378. 'scene_name': smart_scene_qs[0]['scene_name'],
  379. 'condition': eval(smart_scene_qs[0]['conditions']),
  380. 'task': eval(smart_scene_qs[0]['tasks']),
  381. }
  382. # 如果存在关联的时间数据,组织时间数据
  383. is_all_day = smart_scene_qs[0]['is_all_day']
  384. effectiveTime = {}
  385. if is_all_day != 0:
  386. effectiveTime['isAllDay'] = is_all_day
  387. if is_all_day == 2:
  388. try:
  389. effective_time_qs = EffectiveTime.objects.get(id=smart_scene_qs[0]['effective_time_id'])
  390. effectiveTime['startTime'] = effective_time_qs.start_time
  391. effectiveTime['endTime'] = effective_time_qs.end_time
  392. effectiveTime['repeat'] = effective_time_qs.repeat
  393. except ObjectDoesNotExist:
  394. return response.json(0, res)
  395. res['effectiveTime'] = effectiveTime
  396. return response.json(0, res)
  397. except Exception as e:
  398. return response.json(500, repr(e))
  399. @staticmethod
  400. def edit_smart_scene(request_dict, user_id, response):
  401. """
  402. 编辑智能场景
  403. @param request_dict: 请求参数
  404. @request_dict smartSceneId: 智能场景id
  405. @param response: 响应对象
  406. @return: response
  407. """
  408. smart_scene_id = request_dict.get('smartSceneId', None)
  409. device_id = request_dict.get('deviceId', None)
  410. sub_device_id = request_dict.get('subDeviceId', None)
  411. scene_name = request_dict.get('sceneName', None)
  412. conditions = request_dict.get('conditions', None)
  413. tasks = request_dict.get('tasks', None)
  414. is_all_day = request_dict.get('isAllDay', None)
  415. conditions_dict = eval(conditions)
  416. tasks_list = eval(tasks)
  417. now_time = int(time.time())
  418. smart_scene_qs = SmartScene.objects.filter(user_id=user_id, scene_name=scene_name)
  419. if smart_scene_qs.exists():
  420. return response.json(174)
  421. res = {
  422. 'scene_name': scene_name,
  423. 'conditions': conditions_dict,
  424. 'tasks': tasks_list
  425. }
  426. effective_time = {}
  427. if is_all_day:
  428. is_all_day = int(is_all_day)
  429. effective_time['is_all_day'] = is_all_day
  430. else:
  431. is_all_day = 0
  432. if not all([smart_scene_id, scene_name, conditions, tasks]):
  433. return response.json(444, {'error param': 'smartSceneId,sceneName,conditions or tasks'})
  434. try:
  435. smart_scene_qs = SmartScene.objects.filter(id=smart_scene_id)
  436. if not smart_scene_qs.exists():
  437. return response.json(173)
  438. msg = {}
  439. if conditions_dict['type'] == 2: # 条件为选择子设备
  440. if not sub_device_id:
  441. return response.json(444, {'error param': 'subDeviceId'})
  442. device_id = ''
  443. sub_device_qs = GatewaySubDevice.objects.filter(id=sub_device_id).values('src_addr').first()
  444. msg['smart_scene_id'] = smart_scene_id
  445. msg['scene_status'] = 1
  446. msg['sensor_type'] = conditions_dict['sensor']['device_type']
  447. msg['sensor_src'] = int(sub_device_qs['src_addr'], 16)
  448. msg['sensor_status'] = int(conditions_dict['sensor']['eventValues'][0]['event_type'])
  449. else:
  450. if not device_id:
  451. return response.json(444, {'error param': 'deviceId'})
  452. sub_device_id = 0
  453. task_list = []
  454. for task in tasks_list:
  455. task_temp = {
  456. 'sensor_type': task['device_type'],
  457. 'sensor_action': task['event_type']
  458. }
  459. task_sub_device_id = task.get('subDeviceId', None)
  460. if task_sub_device_id:
  461. sub_device_qs = GatewaySubDevice.objects.filter(id=task_sub_device_id).values('src_addr').first()
  462. task_temp['sensor_src'] = int(sub_device_qs['src_addr'], 16)
  463. task_list.append(task_temp)
  464. msg['task'] = task_list
  465. with transaction.atomic():
  466. smart_scene_qs.update(scene_name=scene_name, conditions=conditions, tasks=tasks,
  467. device_data=json.dumps(msg), updated_time=now_time, device_id=device_id,
  468. sub_device_id=sub_device_id)
  469. if is_all_day == 0 or is_all_day == 1: # 不设置时间或全天
  470. smart_scene_qs.update(effective_time_id=0, is_all_day=is_all_day)
  471. else:
  472. start_time = int(request_dict.get('startTime', None))
  473. end_time = int(request_dict.get('endTime', None))
  474. repeat = int(request_dict.get('repeat', None))
  475. effective_time_qs = EffectiveTime.objects.filter(start_time=start_time, end_time=end_time,
  476. repeat=repeat).values('id')
  477. if effective_time_qs.exists():
  478. effective_time_id = effective_time_qs[0]['id']
  479. else:
  480. effective_time_id = EffectiveTime.objects.create(start_time=start_time, end_time=end_time,
  481. repeat=repeat).id
  482. smart_scene_qs.update(effective_time_id=effective_time_id, is_all_day=is_all_day)
  483. effective_time = {
  484. 'isAllDay': is_all_day,
  485. 'startTime': start_time,
  486. 'endTime': end_time,
  487. 'repeat': repeat
  488. }
  489. res['effectiveTime'] = effective_time
  490. return response.json(0, res)
  491. except Exception as e:
  492. return response.json(500, repr(e))
  493. @staticmethod
  494. def delete_smart_scene(request_dict, response):
  495. """
  496. 删除智能场景
  497. @param request_dict: 请求参数
  498. @request_dict smartSceneIds: 智能场景id
  499. @param response: 响应对象
  500. @return: response
  501. """
  502. smart_scene_ids = request_dict.get('smartSceneIds', None)
  503. if not smart_scene_ids:
  504. return response.json(444, {'error param': 'smartSceneIds'})
  505. try:
  506. SmartScene.objects.filter(id__in=smart_scene_ids.split(',')).delete()
  507. except Exception as e:
  508. return response.json(500, repr(e))
  509. else:
  510. return response.json(0)