KVSController.py 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478
  1. # -*- coding: utf-8 -*-
  2. """
  3. @Author : Rocky
  4. @Time : 2022/10/18 9:48
  5. @File :KVSController.py
  6. """
  7. import hashlib
  8. import time
  9. import uuid
  10. import datetime
  11. from django.http import HttpResponse
  12. from django.views import View
  13. from Model.models import KVS, Device_User, Device_Info
  14. from Object.AWS.AmazonKinesisVideoUtil import AmazonKinesisVideoObject, AmazonKVAMObject
  15. from Object.RedisObject import RedisObject
  16. from Object.ResponseObject import ResponseObject
  17. from Ansjer.config import ACCESS_KEY_ID, SECRET_ACCESS_KEY, REGION_NAME, SERVER_DOMAIN
  18. from Object.TokenObject import TokenObject
  19. class UserRelatedView(View):
  20. def get(self, request, *args, **kwargs):
  21. request.encoding = 'utf-8'
  22. operation = kwargs.get('operation')
  23. return self.validation(request.GET, operation, request)
  24. def post(self, request, *args, **kwargs):
  25. request.encoding = 'utf-8'
  26. operation = kwargs.get('operation')
  27. return self.validation(request.POST, operation, request)
  28. def validation(self, request_dict, operation, request):
  29. response = ResponseObject()
  30. if operation == 'generate-qr-code': # 网页生成二维码
  31. return self.generate_qr_code(response)
  32. elif operation == 'get-scanning-status': # 确认app是否扫码
  33. return self.get_scanning_status(request_dict, response)
  34. elif operation == 'web-login': # 网页登录
  35. return self.web_login(request_dict, response)
  36. elif operation == 'pc-login': # pc端登录
  37. return self.pc_login(request_dict, response)
  38. elif operation == 'confirm-login': # app确认登录
  39. return self.confirm_login(request_dict, response)
  40. else:
  41. tko = TokenObject(
  42. request.META.get('HTTP_AUTHORIZATION'),
  43. returntpye='pc')
  44. if tko.code != 0:
  45. return response.json(tko.code)
  46. response.lang = tko.lang
  47. user_id = tko.userID
  48. if operation == 'get-device': # 获取设备列表
  49. return self.get_device(response, user_id)
  50. else:
  51. return response.json(404)
  52. @staticmethod
  53. def get_user_info(user_id):
  54. """
  55. 获取用户信息
  56. @param user_id: 用户id
  57. @return: response
  58. """
  59. device_user_qs = Device_User.objects.filter(userID=user_id).values('NickName', 'userIconPath',
  60. 'userIconUrl')
  61. if not device_user_qs.exists():
  62. user_icon_url = ''
  63. nick_name = ''
  64. else:
  65. users = device_user_qs.first()
  66. nick_name = users['NickName']
  67. user_icon_path = str(users['userIconPath'])
  68. if user_icon_path:
  69. user_icon_path = user_icon_path.replace('static/', '').replace('\\', '/')
  70. user_icon_url = SERVER_DOMAIN + 'account/getAvatar/' + user_icon_path
  71. else:
  72. user_icon_url = ''
  73. return user_icon_url, nick_name
  74. @staticmethod
  75. def generate_qr_code(response):
  76. """
  77. 网页生成二维码
  78. @param response: 响应对象
  79. @return: response
  80. """
  81. nwo_time = time.time()
  82. redis_obj = RedisObject()
  83. try:
  84. uuid_number = hashlib.md5((str(uuid.uuid1()) + str(nwo_time)).encode('utf-8')).hexdigest()
  85. flag = redis_obj.set_ex_data(uuid_number, 0, 300) # redis记录uuid,状态为生成二维码
  86. res = {'type': 'autologin', 'id': uuid_number}
  87. if flag:
  88. return response.json(0, res)
  89. else:
  90. return response.json(119)
  91. except Exception as e:
  92. print(e)
  93. return response.json(500)
  94. @staticmethod
  95. def get_scanning_status(request_dict, response):
  96. """
  97. 获取app扫码状态
  98. @param request_dict: 请求参数
  99. @request_dict serial_number: 序列号
  100. @param response: 响应对象
  101. @return: response
  102. """
  103. uuid_number = request_dict.get('uuid', None)
  104. if not uuid_number:
  105. return response.json(444, {'error param': 'uuid'})
  106. try:
  107. redis_obj = RedisObject()
  108. status = redis_obj.get_data(uuid_number)
  109. if status is False:
  110. return response.json(119)
  111. elif status == '1' or status == '2':
  112. res = {'status': 1} # 已扫码
  113. else:
  114. res = {'status': 0} # 未扫码
  115. return response.json(0, res)
  116. except Exception as e:
  117. print(e)
  118. return response.json(500)
  119. @staticmethod
  120. def pc_login(request_dict, response):
  121. """
  122. pc端登录
  123. @param request_dict: 请求参数
  124. @request_dict serial_number: 序列号
  125. @param response: 响应对象
  126. @return: response
  127. """
  128. uuid_number = request_dict.get('uuid', None)
  129. if not uuid_number:
  130. return response.json(444, {'error param': 'uuid'})
  131. try:
  132. redis_obj = RedisObject()
  133. status = redis_obj.get_data(uuid_number)
  134. token = redis_obj.get_data(uuid_number + 'token')
  135. if status is False or token is False:
  136. return response.json(119)
  137. elif status == '2': # 已登录
  138. token_obj = TokenObject(token)
  139. response.lang = token_obj.lang
  140. if token_obj.code != 0:
  141. return response.json(token_obj.code)
  142. user_id = token_obj.userID
  143. user_icon_url, nick_name = UserRelatedView.get_user_info(user_id)
  144. res = {'status': 1, 'userIconUrl': user_icon_url, 'nickName': nick_name, 'token': token}
  145. redis_obj.del_data(uuid_number)
  146. redis_obj.del_data(uuid_number + 'token')
  147. else: # 未登录
  148. res = {'status': 0}
  149. return response.json(0, res)
  150. except Exception as e:
  151. print(e)
  152. return response.json(500)
  153. @staticmethod
  154. def web_login(request_dict, response):
  155. """
  156. 网页登录
  157. @param request_dict: 请求参数
  158. @request_dict serial_number: 序列号
  159. @param response: 响应对象
  160. @return: response
  161. """
  162. uuid_number = request_dict.get('uuid', None)
  163. confirm = request_dict.get('confirm', None)
  164. if not all([uuid_number, confirm]):
  165. return response.json(444, {'error param': 'uuid or confirm'})
  166. try:
  167. redis_obj = RedisObject()
  168. if confirm == '1': # 取消登录
  169. redis_obj.del_data(uuid_number)
  170. redis_obj.del_data(uuid_number + 'token')
  171. return response.json(0)
  172. token = redis_obj.get_data(uuid_number + 'token')
  173. ttl = redis_obj.get_ttl(uuid_number)
  174. if token is False or ttl == 0:
  175. return response.json(119)
  176. result = redis_obj.set_ex_data(uuid_number, 2, ttl) # 修改uuid状态为已登录
  177. if result is False:
  178. return response.json(119)
  179. token_obj = TokenObject(token)
  180. response.lang = token_obj.lang
  181. if token_obj.code != 0:
  182. return response.json(token_obj.code)
  183. user_id = token_obj.userID
  184. user_icon_url, nick_name = UserRelatedView.get_user_info(user_id)
  185. return response.json(0, {'token': token, 'userIconUrl': user_icon_url, 'nickName': nick_name})
  186. except Exception as e:
  187. print(e)
  188. return response.json(500)
  189. @staticmethod
  190. def confirm_login(request_dict, response):
  191. """
  192. app确认登录
  193. @param request_dict: 请求参数
  194. @request_dict serial_number: 序列号
  195. @param response: 响应对象
  196. @return: response
  197. """
  198. uuid_number = request_dict.get('uuid', None)
  199. token = request_dict.get('token', None)
  200. if not all([uuid_number, token]):
  201. return response.json(444, {'error param': 'uuid or token'})
  202. redis_obj = RedisObject()
  203. try:
  204. status = redis_obj.get_data(uuid_number)
  205. ttl = redis_obj.get_ttl(uuid_number)
  206. if status is False or ttl <= 0:
  207. return response.json(119)
  208. result1 = redis_obj.set_ex_data(uuid_number, 1, ttl) # 修改uuid状态为已扫码
  209. result2 = redis_obj.set_ex_data(uuid_number + 'token', token, ttl)
  210. if result1 is False or result2 is False:
  211. return response.json(119)
  212. return response.json(0)
  213. except Exception as e:
  214. print(e)
  215. return response.json(500)
  216. @staticmethod
  217. def get_device(response, user_id):
  218. """
  219. 获取设备列表
  220. @param response: 响应对象
  221. @param user_id: 用户id
  222. @return: response
  223. """
  224. try:
  225. device_qs = Device_Info.objects.filter(userID=user_id).values('serial_number', 'NickName')
  226. return response.json(0, list(device_qs))
  227. except Exception as e:
  228. print(e)
  229. return response.json(500)
  230. class KVSView(View):
  231. def get(self, request, *args, **kwargs):
  232. request.encoding = 'utf-8'
  233. operation = kwargs.get('operation')
  234. return self.validation(request.GET, request, operation)
  235. def post(self, request, *args, **kwargs):
  236. request.encoding = 'utf-8'
  237. operation = kwargs.get('operation')
  238. return self.validation(request.POST, request, operation)
  239. def validation(self, request_dict, request, operation):
  240. response = ResponseObject()
  241. if operation == 'create-media': # 创建视频流
  242. return self.create_media(request_dict, response)
  243. elif operation == 'update-data-retention': # 修改视频流数据保留时间
  244. return self.update_data_retention(request_dict, response)
  245. else:
  246. # tko = TokenObject(request.META.get('HTTP_AUTHORIZATION'))
  247. # if tko.code != 0:
  248. # return response.json(tko.code)
  249. # response.lang = tko.lang
  250. # user_id = tko.userID
  251. if operation == 'get-device-midea-list': # 获取设备列表
  252. return self.get_device_midea_list(request_dict, response)
  253. elif operation == 'get-hls-midea': # 获取视频播放地址
  254. return self.get_hls_midea_url(request_dict, response)
  255. elif operation == 'download-clip': # 获取视频播放地址
  256. return self.download_clip(request_dict, response)
  257. else:
  258. return response.json(404)
  259. @staticmethod
  260. def create_media(request_dict, response):
  261. """
  262. 创建视频流
  263. @param request_dict: 请求参数
  264. @request_dict serial_number: 序列号
  265. @param response: 响应对象
  266. @return: response
  267. """
  268. serial_number = request_dict.get('serial_number', None)
  269. try:
  270. kvs_qs = KVS.objects.filter(stream_name=serial_number)
  271. if kvs_qs.exists():
  272. return response.json(174)
  273. kinesis_video_obj = AmazonKinesisVideoObject(
  274. aws_access_key_id=ACCESS_KEY_ID,
  275. secret_access_key=SECRET_ACCESS_KEY,
  276. region_name=REGION_NAME
  277. )
  278. stream_arn = kinesis_video_obj.create_stream(stream_name=serial_number)
  279. if stream_arn:
  280. now_time = int(time.time())
  281. KVS.objects.create(stream_name=serial_number, stream_arn=stream_arn, created_time=now_time,
  282. updated_time=now_time)
  283. return response.json(0)
  284. else:
  285. return response.json(178)
  286. except Exception as e:
  287. print(e)
  288. return response.json(500)
  289. @staticmethod
  290. def update_data_retention(request_dict, response):
  291. """
  292. 修改视频流数据保留时间
  293. @param request_dict: 请求参数
  294. @request_dict serial_number: 序列号
  295. @request_dict operation: 操作,增加/减少
  296. @request_dict data_retention_change_in_hours: 修改的时间
  297. @param response: 响应对象
  298. @return: response
  299. """
  300. serial_number = request_dict.get('serial_number', None)
  301. operation = request_dict.get('operation', None)
  302. data_retention_change_in_hours = request_dict.get('data_retention_change_in_hours', None)
  303. try:
  304. kvs_qs = KVS.objects.filter(stream_name=serial_number)
  305. if not kvs_qs.exists():
  306. return response.json(174)
  307. kinesis_video_obj = AmazonKinesisVideoObject(
  308. aws_access_key_id=ACCESS_KEY_ID,
  309. secret_access_key=SECRET_ACCESS_KEY,
  310. region_name=REGION_NAME
  311. )
  312. now_time = int(time.time())
  313. data_retention_change_in_hours = int(data_retention_change_in_hours)
  314. kinesis_video_obj.update_data_retention(stream_name=serial_number, operation=operation,
  315. data_retention_change_in_hours=data_retention_change_in_hours)
  316. kvs_qs.update(data_retention_in_hours=data_retention_change_in_hours, updated_time=now_time)
  317. return response.json(0)
  318. except Exception as e:
  319. print(e)
  320. return response.json(500)
  321. @staticmethod
  322. def get_hls_midea_url(request_dict, response):
  323. """
  324. 获取视频播放地址
  325. @param request_dict: 请求参数
  326. @request_dict serial_number: 序列号
  327. @request_dict startTime: 开始时间
  328. @request_dict endTime: 结束时间
  329. @request_dict playMode: 播放模式
  330. @param response: 响应对象
  331. @return: response
  332. """
  333. serial_number = request_dict.get('serial_number', None)
  334. start_time = request_dict.get('startTime', None)
  335. end_time = request_dict.get('endTime', None)
  336. play_mode = request_dict.get('playMode', None)
  337. if not all([serial_number, start_time, end_time, play_mode]):
  338. return response.json(444)
  339. start_time = datetime.datetime.fromtimestamp(int(start_time)) - datetime.timedelta(hours=8)
  340. end_time = datetime.datetime.fromtimestamp(int(end_time)) - datetime.timedelta(hours=8)
  341. play_mode = int(play_mode)
  342. play_mode = 'ON_DEMAND' if play_mode == 0 else 'LIVE_REPLAY'
  343. try:
  344. # kvs_qs = KVS.objects.filter(stream_name=serial_number)
  345. # if not kvs_qs.exists():
  346. # return response.json(174)
  347. kinesis_video_obj = AmazonKVAMObject(
  348. aws_access_key_id='AKIA2E67UIMD45Y3HL53',
  349. secret_access_key='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',
  350. region_name='us-east-1',
  351. stream_name=serial_number,
  352. api_name='GET_HLS_STREAMING_SESSION_URL'
  353. )
  354. hls_streaming_session_url = kinesis_video_obj.get_hls_streaming_session_url(serial_number, start_time,
  355. end_time, play_mode)
  356. return response.json(0, {"HlsStreamingSessionUrl": hls_streaming_session_url})
  357. except Exception as e:
  358. print(e)
  359. return response.json(500, repr(e))
  360. @staticmethod
  361. def get_device_midea_list(request_dict, response):
  362. """
  363. 获取视频播放列表
  364. @param request_dict: 请求参数
  365. @request_dict serial_number: 序列号
  366. @request_dict startTime: 开始时间
  367. @request_dict endTime: 结束时间
  368. @param response: 响应对象
  369. @return: response
  370. """
  371. serial_number = request_dict.get('serial_number', None)
  372. start_time = request_dict.get('startTime', None)
  373. end_time = request_dict.get('endTime', None)
  374. page = request_dict.get('page', None)
  375. size = request_dict.get('size', None)
  376. if not all([serial_number, start_time, end_time, page, size]):
  377. return response.json(444)
  378. page = int(page)
  379. size = int(size)
  380. start_time = datetime.datetime.fromtimestamp(int(start_time)) - datetime.timedelta(hours=8)
  381. end_time = datetime.datetime.fromtimestamp(int(end_time)) - datetime.timedelta(hours=8)
  382. try:
  383. # kvs_qs = KVS.objects.filter(stream_name=serial_number)
  384. # if not kvs_qs.exists():
  385. # return response.json(174)
  386. kinesis_fragments_obj = AmazonKVAMObject(
  387. aws_access_key_id='AKIA2E67UIMD45Y3HL53',
  388. secret_access_key='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',
  389. region_name='us-east-1',
  390. stream_name=serial_number,
  391. api_name='LIST_FRAGMENTS'
  392. )
  393. kinesis_images_obj = AmazonKVAMObject(
  394. aws_access_key_id='AKIA2E67UIMD45Y3HL53',
  395. secret_access_key='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',
  396. region_name='us-east-1',
  397. stream_name=serial_number,
  398. api_name='GET_IMAGES'
  399. )
  400. stream_list = kinesis_fragments_obj.get_list_fragments(serial_number, start_time, end_time)
  401. total_page = len(stream_list)
  402. stream_list = stream_list[(page - 1) * size:page * size]
  403. for item in stream_list:
  404. temp_start_time = (item['startTime'] - datetime.timedelta(hours=8)).replace(tzinfo=datetime.timezone.utc)
  405. temp_end_time = temp_start_time + datetime.timedelta(seconds=300)
  406. item['image'] = kinesis_images_obj.get_images(serial_number, temp_start_time, temp_end_time)
  407. item['startTime'] = int(item['startTime'].timestamp())
  408. item['endTime'] = int(item['endTime'].timestamp())
  409. res = {
  410. 'totalPage': total_page,
  411. 'fragments': stream_list
  412. }
  413. return response.json(0, res)
  414. except Exception as e:
  415. print(e)
  416. return response.json(500, repr(e))
  417. @staticmethod
  418. def download_clip(request_dict, response):
  419. """
  420. 获取视频播放地址
  421. @param request_dict: 请求参数
  422. @request_dict serial_number: 序列号
  423. @request_dict startTime: 开始时间
  424. @request_dict endTime: 结束时间
  425. @param response: 响应对象
  426. @return: response
  427. """
  428. serial_number = request_dict.get('serial_number', None)
  429. start_time = request_dict.get('startTime', None)
  430. end_time = request_dict.get('endTime', None)
  431. if not all([serial_number, start_time, end_time]):
  432. return response.json(444)
  433. start_time = datetime.datetime.fromtimestamp(int(start_time)) - datetime.timedelta(hours=8)
  434. end_time = datetime.datetime.fromtimestamp(int(end_time)) - datetime.timedelta(hours=8)
  435. try:
  436. # kvs_qs = KVS.objects.filter(stream_name=serial_number)
  437. # if not kvs_qs.exists():
  438. # return response.json(174)
  439. kinesis_video_obj = AmazonKVAMObject(
  440. aws_access_key_id='AKIA2E67UIMD45Y3HL53',
  441. secret_access_key='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',
  442. region_name='us-east-1',
  443. stream_name=serial_number,
  444. api_name='GET_CLIP'
  445. )
  446. clip_obj, clip_size = kinesis_video_obj.get_clip(serial_number, start_time, end_time)
  447. res = HttpResponse(clip_obj.read())
  448. res["content_type"] = "video/mp4"
  449. res["Content-Disposition"] = "attachment;filename=video.mp4"
  450. res['Content-Length'] = str(clip_size)
  451. return res
  452. except Exception as e:
  453. print(e)
  454. return response.json(500, repr(e))