Du kan inte välja fler än 25 ämnen Ämnen måste starta med en bokstav eller siffra, kan innehålla bindestreck ('-') och vara max 35 tecken långa.

131 lines
6.8KB

  1. import oss2
  2. import time
  3. from exception.CustomerException import ServiceException
  4. from enums.ExceptionEnum import ExceptionType
  5. import json
  6. from aliyunsdkcore.client import AcsClient
  7. from aliyunsdkvod.request.v20170321 import GetPlayInfoRequest
  8. from voduploadsdk.AliyunVodUtils import *
  9. from voduploadsdk.AliyunVodUploader import AliyunVodUploader
  10. from voduploadsdk.UploadVideoRequest import UploadVideoRequest
  11. class AliyunOssSdk():
  12. def __init__(self, content, logger, requestId):
  13. self.content = content
  14. self.bucket = None
  15. self.logger = logger
  16. self.requestId = requestId
  17. def get_oss_bucket(self):
  18. if self.bucket is None:
  19. self.logger.info("初始化oss桶, requestId:{}", self.requestId)
  20. auth = oss2.Auth(self.content["aliyun"]["access_key"], self.content["aliyun"]["access_secret"])
  21. self.bucket = oss2.Bucket(auth, self.content["aliyun"]["oss"]["endpoint"],
  22. self.content["aliyun"]["oss"]["bucket"],
  23. connect_timeout=self.content["aliyun"]["oss"]["connect_timeout"])
  24. async def put_object(self, updatePath, fileByte):
  25. self.bucket.put_object(updatePath, fileByte)
  26. async def upload_file(self, updatePath, fileByte):
  27. self.logger.info("开始上传文件到oss, requestId:{}", self.requestId)
  28. self.get_oss_bucket()
  29. MAX_RETRIES = 3
  30. retry_count = 0
  31. while True:
  32. try:
  33. await self.put_object(updatePath, fileByte)
  34. self.logger.info("上传文件到oss成功! requestId:{}", self.requestId)
  35. break
  36. except Exception as e:
  37. retry_count += 1
  38. time.sleep(1)
  39. self.logger.info("上传文件到oss失败, 重试次数:{}, requestId:{}", retry_count, self.requestId)
  40. if retry_count > MAX_RETRIES:
  41. self.logger.exception("上传文件到oss重试失败:{}, requestId:{}", e, self.requestId)
  42. raise e
  43. class ThAliyunVodSdk():
  44. def __init__(self, content, logger, requestId):
  45. self.content = content
  46. self.logger = logger
  47. self.requestId = requestId
  48. def init_vod_client(self, accessKeyId, accessKeySecret):
  49. regionId = self.content["aliyun"]["vod"]["ecsRegionId"]
  50. return AcsClient(accessKeyId, accessKeySecret, regionId, auto_retry=True, max_retry_time=3, timeout=5)
  51. def get_play_info(self, videoId):
  52. self.logger.info("开始获取视频地址,videoId:{}, requestId:{}", videoId, self.requestId)
  53. start = time.time()
  54. while True:
  55. try:
  56. clt = self.init_vod_client(self.content["aliyun"]["access_key"],
  57. self.content["aliyun"]["access_secret"])
  58. request = GetPlayInfoRequest.GetPlayInfoRequest()
  59. request.set_accept_format('JSON')
  60. request.set_VideoId(videoId)
  61. request.set_AuthTimeout(3600 * 5)
  62. response = json.loads(clt.do_action_with_exception(request))
  63. play_url = response["PlayInfoList"]["PlayInfo"][0]["PlayURL"]
  64. self.logger.info("获取视频地址成功,视频地址: {}, requestId: {}", play_url, self.requestId)
  65. return play_url
  66. except Exception as e:
  67. self.logger.error("获取视频地址失败,5秒后重试, requestId: {}", self.requestId)
  68. time.sleep(5)
  69. current_time = time.time()
  70. if "HTTP Status: 403" not in str(e):
  71. self.logger.exception("获取视频地址失败: {}, requestId: {}", e, self.requestId)
  72. raise ServiceException(ExceptionType.GET_VIDEO_URL_EXCEPTION.value[0],
  73. ExceptionType.GET_VIDEO_URL_EXCEPTION.value[1])
  74. if "HTTP Status: 403" in str(e) and ("UploadFail" in str(e) or "TranscodeFail" in str(e)):
  75. self.logger.exception("获取视频地址失败: {}, requestId: {}", e, self.requestId)
  76. raise ServiceException(ExceptionType.GET_VIDEO_URL_EXCEPTION.value[0],
  77. ExceptionType.GET_VIDEO_URL_EXCEPTION.value[1])
  78. diff_time = current_time - start
  79. if diff_time > 60 * 60 * 2:
  80. self.logger.exception("获取视频地址失败超时异常: {},超时时间:{}, requestId: {}", e, diff_time, self.requestId)
  81. raise ServiceException(ExceptionType.GET_VIDEO_URL_TIMEOUT_EXCEPTION.value[0],
  82. ExceptionType.GET_VIDEO_URL_TIMEOUT_EXCEPTION.value[1])
  83. def upload_local_video(self, filePath, file_title, storageLocation=None):
  84. self.logger.info("开始执行vod视频上传, filePath: {}, requestId: {}", filePath, self.requestId)
  85. uploader = AliyunVodUploader(self.content["aliyun"]["access_key"], self.content["aliyun"]["access_secret"])
  86. uploadVideoRequest = UploadVideoRequest(filePath, file_title)
  87. self.logger.info("视频分类:{}", self.content["aliyun"]["vod"][self.content["dsp"]["active"]]["CateId"])
  88. uploadVideoRequest.setCateId(self.content["aliyun"]["vod"][self.content["dsp"]["active"]]["CateId"])
  89. # 可以设置视频封面,如果是本地或网络图片可使用UploadImageRequest上传图片到视频点播,获取到ImageURL
  90. # ImageURL示例:https://example.com/sample-****.jpg
  91. # uploadVideoRequest.setCoverURL('<your Image URL>')
  92. # 标签
  93. # uploadVideoRequest.setTags('tag1,tag2')
  94. if storageLocation:
  95. uploadVideoRequest.setStorageLocation(storageLocation)
  96. MAX_RETRIES = 3
  97. retry_count = 0
  98. while True:
  99. try:
  100. result = uploader.uploadLocalVideo(uploadVideoRequest)
  101. self.logger.info("vod视频上传成功, videoId:{}, requestId:{}", result.get("VideoId"), self.requestId)
  102. return result.get("VideoId")
  103. except AliyunVodException as e:
  104. retry_count += 1
  105. time.sleep(3)
  106. self.logger.error("vod视频上传失败,重试次数:{}, requestId:{}", retry_count, self.requestId)
  107. if retry_count >= MAX_RETRIES:
  108. self.logger.exception("vod视频上传重试失败: {}, requestId:{}", e, self.requestId)
  109. raise ServiceException(ExceptionType.VIDEO_UPDATE_EXCEPTION.value[0],
  110. ExceptionType.VIDEO_UPDATE_EXCEPTION.value[1])
  111. def get_play_url(args):
  112. thAliyunVodSdk = ThAliyunVodSdk(args[2], args[3], args[4])
  113. videoId = thAliyunVodSdk.upload_local_video(args[0], args[1])
  114. if videoId is None or len(videoId) == 0:
  115. return None
  116. return thAliyunVodSdk.get_play_info(videoId)