KVSController.py 19 KB

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