SmartSceneController.py 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427
  1. # -*- coding: utf-8 -*-
  2. """
  3. @Author : Rocky
  4. @Time : 2022/6/29 9:31
  5. @File :SmartSceneController.py
  6. """
  7. import time
  8. import json
  9. from django.core.exceptions import ObjectDoesNotExist
  10. from django.db.models import F, Q
  11. from django.views import View
  12. from django.db import transaction
  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 == 'update-status': # 更新智能场景状态
  37. return self.update_status(request_dict, response)
  38. elif operation == 'detail': # 查询智能场景详情
  39. return self.scene_detail(request_dict, response)
  40. elif operation == 'edit': # 编辑智能场景
  41. return self.edit_smart_scene(request_dict, response)
  42. elif operation == 'delete': # 删除智能场景
  43. return self.delete_smart_scene(request_dict, response)
  44. else:
  45. return response.json(414)
  46. @staticmethod
  47. def condition_devices(request_dict, response):
  48. """
  49. 添加条件-查询设备
  50. @param request_dict: 请求参数
  51. @request_dict deviceId: 网关设备id
  52. @request_dict subDeviceId: 子设备id
  53. @param response: 响应对象
  54. @return: response
  55. """
  56. device_id = request_dict.get('deviceId', None)
  57. sub_device_id = request_dict.get('subDeviceId', None)
  58. if not any([device_id, sub_device_id]):
  59. return response.json(444, {'error param': 'deviceId or subDeviceId'})
  60. try:
  61. if sub_device_id:
  62. device_id = GatewaySubDevice.objects.get(id=sub_device_id).device_id
  63. gateway_sub_device_qs = GatewaySubDevice.objects.filter(device_id=device_id)
  64. if not gateway_sub_device_qs.exists():
  65. return response.json(173)
  66. gateway_sub_device_qs = gateway_sub_device_qs.annotate(gatewaySubId=F('id'), deviceType=F('device_type'),
  67. deviceNickName=F('nickname')). \
  68. values('gatewaySubId', 'deviceType', 'deviceNickName', 'status')
  69. sub_device_list = []
  70. for sub_device in gateway_sub_device_qs:
  71. family_room_device_qs = FamilyRoomDevice.objects.filter(sub_device=sub_device['gatewaySubId']). \
  72. values('room_id')
  73. if not family_room_device_qs.exists():
  74. sub_device['roomName'] = ''
  75. else:
  76. room_id = family_room_device_qs[0]['room_id']
  77. try:
  78. sub_device['roomName'] = FamilyRoom.objects.get(id=room_id).name
  79. except ObjectDoesNotExist:
  80. sub_device['roomName'] = ''
  81. sub_device_list.append(sub_device)
  82. return response.json(0, sub_device_list)
  83. except Exception as e:
  84. return response.json(500, repr(e))
  85. @classmethod
  86. def task_devices(cls, request_dict, response):
  87. """
  88. 添加任务-查询设备
  89. @param request_dict: 请求参数
  90. @request_dict deviceId: 网关设备id
  91. @param response: 响应对象
  92. @return: response
  93. """
  94. sub_device_id = request_dict.get('subDeviceId', None)
  95. device_id = request_dict.get('deviceId', None)
  96. if not any([device_id, sub_device_id]):
  97. return response.json(444, {'error param': 'deviceId or subDeviceId'})
  98. try:
  99. if device_id:
  100. res = cls.get_gateway_data(device_id)
  101. else:
  102. sub_device_qs = GatewaySubDevice.objects.filter(id=sub_device_id).values('device_id', 'device_type')
  103. device_id = sub_device_qs[0]['device_id']
  104. device_type = sub_device_qs[0]['device_type']
  105. if device_type != 216:
  106. res = cls.get_gateway_data(device_id)
  107. else:
  108. sub_device_list = []
  109. gateway_data = cls.get_gateway_data(device_id)
  110. sub_device_list.append(gateway_data)
  111. sub_device_qs = GatewaySubDevice.objects.filter(
  112. Q(device_id=device_id) & Q(device_type=215) | Q(device_type=219)).values('id', 'nickname',
  113. 'status',
  114. 'device_type')
  115. sub_device_qs = sub_device_qs.annotate(gatewaySubId=F('id'),
  116. deviceType=F('device_type'),
  117. deviceNickName=F('nickname')). \
  118. values('gatewaySubId', 'deviceType', 'deviceNickName', 'status')
  119. for sub_device in sub_device_qs:
  120. sub_device_id = sub_device['gatewaySubId']
  121. room_qs = FamilyRoomDevice.objects.filter(sub_device=sub_device_id).values('room_id')
  122. room_qs.exists()
  123. room_qs = room_qs.first()
  124. room_id = room_qs['room_id']
  125. room_id_qs = FamilyRoom.objects.filter(id=room_id).values('name')
  126. room_name = room_id_qs.first()['name'] if room_id_qs.exists() else ''
  127. try:
  128. if sub_device_qs.exists():
  129. sub_device['roomName'] = FamilyRoom.objects.get(id=room_id).name
  130. except ObjectDoesNotExist:
  131. sub_device['roomName'] = room_name
  132. sub_device.pop('id')
  133. sub_device_list.append(sub_device)
  134. res = sub_device_list
  135. return response.json(0, res)
  136. except Exception as e:
  137. return response.json(500, repr(e))
  138. @staticmethod
  139. def get_gateway_data(device_id):
  140. """
  141. 获取网关数据
  142. @param device_id: 网关设备id
  143. @return: res
  144. """
  145. device_info_qs = Device_Info.objects.filter(id=device_id).values('NickName', 'Type')
  146. nickname = device_info_qs[0]['NickName']
  147. device_type = device_info_qs[0]['Type']
  148. room_id = FamilyRoomDevice.objects.filter(device_id=device_id).values('room_id')[0]['room_id']
  149. room_id_qs = FamilyRoom.objects.filter(id=room_id).values('name')
  150. room_name = room_id_qs.first()['name'] if room_id_qs.exists() else ''
  151. res = {
  152. 'deviceNickName': nickname,
  153. 'deviceType': device_type,
  154. 'roomName': room_name,
  155. 'status': 1,
  156. }
  157. return res
  158. @staticmethod
  159. def create_smart_scene(request_dict, user_id, response):
  160. """
  161. 创建智能场景
  162. @param request_dict: 请求参数
  163. @param user_id: 用户id
  164. @request_dict deviceId: 网关设备id
  165. @request_dict subDeviceId: 子设备id
  166. @request_dict sceneName: 场景名称
  167. @request_dict conditions: 条件
  168. @request_dict tasks: 任务
  169. @request_dict isAllDay: 是否全天执行
  170. @request_dict startTime: 开始时间
  171. @request_dict endTime: 结束时间
  172. @request_dict repeat: 重复周期
  173. @param response: 响应对象
  174. @return: response
  175. """
  176. device_id = request_dict.get('deviceId', None)
  177. sub_device_id = request_dict.get('subDeviceId', None)
  178. scene_name = request_dict.get('sceneName', None)
  179. conditions = request_dict.get('conditions', None)
  180. tasks = request_dict.get('tasks', None)
  181. is_all_day = request_dict.get('isAllDay', None)
  182. if not any([device_id, sub_device_id]):
  183. return response.json(444, {'error param': 'deviceId or subDeviceId'})
  184. if not all([conditions, tasks]):
  185. return response.json(444, {'error param': 'conditions and tasks'})
  186. now_time = int(time.time())
  187. conditions_dict = eval(conditions)
  188. tasks_list = eval(tasks)
  189. try:
  190. # 判断是否已存在该场景名
  191. smart_scene_qs = SmartScene.objects.filter(user_id=user_id, scene_name=scene_name)
  192. if smart_scene_qs.exists():
  193. return response.json(174)
  194. smart_scene_dict = {
  195. 'user_id': user_id,
  196. 'scene_name': scene_name,
  197. 'conditions': conditions,
  198. 'tasks': tasks,
  199. 'created_time': now_time,
  200. 'updated_time': now_time,
  201. }
  202. # 处理传网关设备id和子设备id的情况
  203. if device_id:
  204. smart_scene_dict['device_id'] = device_id
  205. device_info_qs = Device_Info.objects.filter(id=device_id).values('serial_number')
  206. if not device_info_qs.exists():
  207. return response.json(173)
  208. serial_number = device_info_qs[0]['serial_number']
  209. else:
  210. smart_scene_dict['sub_device_id'] = sub_device_id
  211. sub_device_qs = GatewaySubDevice.objects.filter(id=sub_device_id).values('device__serial_number')
  212. if not sub_device_qs.exists():
  213. return response.json(173)
  214. serial_number = sub_device_qs[0]['device__serial_number']
  215. with transaction.atomic():
  216. if not is_all_day: # 没传时间
  217. smart_scene_qs = SmartScene.objects.create(**smart_scene_dict)
  218. else:
  219. if is_all_day == '0':
  220. effective_time_qs = EffectiveTime.objects.filter(is_all_day=True).values('id')
  221. if effective_time_qs.exists():
  222. effective_time_id = effective_time_qs[0]['id']
  223. else:
  224. effective_time_id = EffectiveTime.objects.create(is_all_day=True).id
  225. smart_scene_dict['effective_time_id'] = effective_time_id
  226. smart_scene_qs = SmartScene.objects.create(**smart_scene_dict)
  227. else:
  228. start_time = int(request_dict.get('startTime', None))
  229. end_time = int(request_dict.get('endTime', None))
  230. repeat = int(request_dict.get('repeat', None))
  231. if not all([start_time, end_time, repeat]):
  232. return response.json(444, {'error param': 'startTime and endTime and repeat'})
  233. effective_time_qs = EffectiveTime.objects.filter(start_time=start_time, end_time=end_time,
  234. repeat=repeat).values('id')
  235. if effective_time_qs.exists():
  236. effective_time_id = effective_time_qs[0]['id']
  237. else:
  238. effective_time_id = EffectiveTime.objects.create(start_time=start_time, end_time=end_time,
  239. repeat=repeat).id
  240. smart_scene_dict['effective_time_id'] = effective_time_id
  241. smart_scene_qs = SmartScene.objects.create(**smart_scene_dict)
  242. sub_device_id = conditions_dict['sensor']['subDeviceId']
  243. sub_device_qs = GatewaySubDevice.objects.filter(id=sub_device_id).values('src_addr').first()
  244. msg = {
  245. 'smart_scene_id': smart_scene_qs.id,
  246. 'sensor_type': conditions_dict['sensor']['device_type'],
  247. 'sensor_src': int(sub_device_qs['src_addr'], 16),
  248. 'sensor_status': conditions_dict['sensor']['eventValues'][0]['value'],
  249. }
  250. task_list = []
  251. for task in tasks_list:
  252. task_temp = {
  253. 'sensor_type': task['device_type'],
  254. 'sensor_action': task['event_type']
  255. }
  256. if 'subDeviceId' in task:
  257. sub_device_id = task['subDeviceId']
  258. sub_device_qs = GatewaySubDevice.objects.filter(id=sub_device_id).values('src_addr').first()
  259. task_temp['sensor_src'] = int(sub_device_qs['src_addr'], 16)
  260. task_list.append(task_temp)
  261. msg['task'] = task_list
  262. # smart_scene_qs.update(device_data=json.dumps(msg))
  263. smart_scene_qs.device_data = json.dumps(msg)
  264. smart_scene_qs.save()
  265. # 发布MQTT消息通知网关设备
  266. thing_name = serial_number
  267. topic_name = 'loocam/gateway_sensor/{}/smart_scene'.format(serial_number)
  268. success = CommonService.req_publish_mqtt_msg(thing_name, topic_name, msg)
  269. if not success:
  270. return response.json(10044)
  271. return response.json(0)
  272. except Exception as e:
  273. return response.json(500, repr(e))
  274. @staticmethod
  275. def scene_list(request_dict, user_id, response):
  276. """
  277. 查询智能场景列表
  278. @param request_dict: 请求参数
  279. @param user_id: 用户id
  280. @request_dict deviceId: 网关设备id
  281. @request_dict subDeviceId: 子设备id
  282. @param response: 响应对象
  283. @return: response
  284. """
  285. device_id = request_dict.get('deviceId', None)
  286. sub_device_id = request_dict.get('subDeviceId', None)
  287. if not any([device_id, sub_device_id]):
  288. return response.json(444, {'error param': 'deviceId or subDeviceId'})
  289. try:
  290. if device_id:
  291. smart_scene_qs = SmartScene.objects.filter(user_id=user_id, device_id=device_id)
  292. else:
  293. smart_scene_qs = SmartScene.objects.filter(user_id=user_id, sub_device_id=sub_device_id)
  294. if not smart_scene_qs.exists():
  295. return response.json(173)
  296. smart_scene_qs = smart_scene_qs.values('id', 'scene_name', 'is_enable')
  297. return response.json(0, list(smart_scene_qs))
  298. except Exception as e:
  299. return response.json(500, repr(e))
  300. @staticmethod
  301. def update_status(request_dict, response):
  302. """
  303. 更新智能场景状态
  304. @param request_dict: 请求参数
  305. @request_dict smartSceneId: 智能场景id
  306. @request_dict isEnable: 状态,True or False
  307. @param response: 响应对象
  308. @return: response
  309. """
  310. smart_scene_id = request_dict.get('smartSceneId', None)
  311. is_enable = request_dict.get('isEnable', None)
  312. if not all([smart_scene_id, is_enable]):
  313. return response.json(444, {'error param': 'smartSceneId and status'})
  314. try:
  315. SmartScene.objects.filter(id=smart_scene_id).update(is_enable=is_enable)
  316. return response.json(0)
  317. except Exception as e:
  318. return response.json(500, repr(e))
  319. @staticmethod
  320. def scene_detail(request_dict, response):
  321. """
  322. 查询智能场景详情
  323. @param request_dict: 请求参数
  324. @request_dict smartSceneId: 智能场景id
  325. @param response: 响应对象
  326. @return: response
  327. """
  328. smart_scene_id = request_dict.get('smartSceneId', None)
  329. if not smart_scene_id:
  330. return response.json(444, {'error param': 'smartSceneId'})
  331. try:
  332. smart_scene_qs = SmartScene.objects.filter(id=smart_scene_id).values('id', 'scene_name', 'conditions',
  333. 'tasks', 'effective_time_id')
  334. if not smart_scene_qs.exists():
  335. return response.json(173)
  336. res = {
  337. 'scene_name': smart_scene_qs[0]['scene_name'],
  338. 'condition': eval(smart_scene_qs[0]['conditions']),
  339. 'task': eval(smart_scene_qs[0]['tasks']),
  340. }
  341. # 如果存在关联的时间数据,组织时间数据
  342. try:
  343. effective_time_qs = EffectiveTime.objects.get(id=smart_scene_qs[0]['effective_time_id'])
  344. except ObjectDoesNotExist:
  345. return response.json(0, res)
  346. if effective_time_qs.is_all_day: # 全天
  347. time_dict = {'all_day': 1}
  348. else:
  349. time_dict = {
  350. 'start_time': effective_time_qs.start_time,
  351. 'end_time': effective_time_qs.end_time,
  352. 'repeat': effective_time_qs.repeat,
  353. }
  354. res['time'] = time_dict
  355. return response.json(0, res)
  356. except Exception as e:
  357. return response.json(500, repr(e))
  358. @staticmethod
  359. def edit_smart_scene(request_dict, response):
  360. """
  361. 编辑智能场景
  362. @param request_dict: 请求参数
  363. @request_dict smartSceneId: 智能场景id
  364. @param response: 响应对象
  365. @return: response
  366. """
  367. smart_scene_id = request_dict.get('smartSceneId', None)
  368. if not smart_scene_id:
  369. return response.json(444, {'error param': 'smartSceneId'})
  370. try:
  371. smart_scene_qs = SmartScene.objects.filter(id=smart_scene_id).values('id', 'scene_name', 'conditions',
  372. 'tasks', 'effective_time_id')
  373. if not smart_scene_qs.exists():
  374. return response.json(173)
  375. return response.json(0, list(smart_scene_qs))
  376. except Exception as e:
  377. return response.json(500, repr(e))
  378. @staticmethod
  379. def delete_smart_scene(request_dict, response):
  380. """
  381. 删除智能场景
  382. @param request_dict: 请求参数
  383. @request_dict smartSceneId: 智能场景id
  384. @param response: 响应对象
  385. @return: response
  386. """
  387. smart_scene_id = request_dict.get('smartSceneId', None)
  388. if not smart_scene_id:
  389. return response.json(444, {'error param': 'smartSceneId'})
  390. try:
  391. SmartScene.objects.filter(id=smart_scene_id).delete()
  392. return response.json(0)
  393. except Exception as e:
  394. return response.json(500, repr(e))