SmartSceneController.py 18 KB

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