KVSController.py 17 KB

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