KVSController.py 16 KB

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