KVSController.py 22 KB

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