app.py 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138
  1. import logging
  2. import boto3
  3. import requests
  4. import yt_dlp
  5. from flask import Flask, request, Response
  6. logging.basicConfig(format='%(asctime)s [%(levelname)s] %(filename)s:%(lineno)d %(message)s',
  7. datefmt='%Y-%m-%d %H:%M:%S', level=logging.INFO)
  8. app = Flask(__name__)
  9. def get_key(url: str) -> str:
  10. return f"v2:{url}"
  11. def fetch_info(url):
  12. logging.info(f"fetching: {url}")
  13. with yt_dlp.YoutubeDL({
  14. "flat-playlist": True,
  15. "extract_flat": "flat-playlist"
  16. }) as ydl:
  17. info = ydl.extract_info(url, download=False)
  18. return info
  19. @app.route("/p", methods=["GET", "POST"])
  20. def p():
  21. video_id = request.args.get("id")
  22. format_id = request.args.get("fId")
  23. if video_id and format_id:
  24. info = fetch_info(url=f"https://www.youtube.com/watch?v={video_id}")
  25. if info:
  26. for item in info.get("formats", []):
  27. if item.get("format_id") == format_id:
  28. url = item.get("url")
  29. response = requests.get(url, stream=True)
  30. logging.info(f"videoId: {video_id} status code: {response.status_code}")
  31. def generate():
  32. for chunk in response.iter_content(chunk_size=1024):
  33. yield chunk
  34. return Response(generate(), status=response.status_code, headers=dict(response.headers))
  35. return {"status": 0}
  36. def convert_dto(info):
  37. thumbnails = []
  38. for item in info.get("thumbnails", []):
  39. if item.get("width"):
  40. thumbnails.append({
  41. "url": item.get("url", ""),
  42. "width": f"{item.get('width', 0)}",
  43. "height": f"{item.get('height', 0)}",
  44. })
  45. formats = []
  46. for item in info.get("formats", []):
  47. if item.get("resolution") != "audio only" and item.get("url") and item.get("acodec") and item.get(
  48. "acodec") != "none" and item.get("vcodec"):
  49. format_id = item.get("format_id")
  50. if format_id:
  51. url = f"http://d1boedwd1gi87x.cloudfront.net/p?id={info['display_id']}&fId={format_id}&videoId={info['display_id']}"
  52. formats.append({
  53. "width": f"{item.get('width', 0)}",
  54. "height": f"{item.get('height', 0)}",
  55. "type": item.get("format", ""),
  56. "quality": f'{item.get("format_note", "")}',
  57. "itag": 0,
  58. "fps": "0",
  59. "bitrate": "0",
  60. "url": url,
  61. "ext": item.get("ext"),
  62. "vcodec": item.get("vcodec", ""),
  63. "acodec": item.get("acodec", ""),
  64. "vbr": "0",
  65. "abr": "0",
  66. "container": item.get("container")
  67. })
  68. result = {
  69. "code": 200,
  70. "msg": "",
  71. "data": {
  72. "videoDetails": {
  73. "isLiveContent": info.get("is_live", False),
  74. "title": info.get("title", ""),
  75. "thumbnails": thumbnails,
  76. "description": info.get("description", ""),
  77. "lengthSeconds": f"{int(info.get('duration', 0) / 100)}",
  78. "viewCount": f"{info.get('view_count', 0)}",
  79. "keywords": [],
  80. "author": info.get("uploader", info.get("channel", "")),
  81. "channelID": info.get("channel_id", ""),
  82. "recommendInfo": [],
  83. "channelURL": info.get("channel_url", ""),
  84. "videoId": info.get("display_id", "")
  85. },
  86. "streamingData": {
  87. "formats": formats
  88. }
  89. },
  90. "id": "MusicDetailViewModel_detail_url"
  91. }
  92. return result
  93. @app.route("/extract", methods=["GET", "POST"])
  94. def extract():
  95. url: str = request.json.get("url")
  96. logging.info(f"url: {url}")
  97. info = fetch_info(url=url)
  98. return convert_dto(info=info)
  99. @app.route("/health", methods=["GET"])
  100. def health():
  101. return {
  102. "status": 1
  103. }
  104. @app.route("/refresh", methods=["GET"])
  105. def refresh():
  106. autoscaling = boto3.client(
  107. 'autoscaling',
  108. region_name="us-east-1",
  109. aws_access_key_id="AKIA2TBT2JUNG6X3W737",
  110. aws_secret_access_key="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")
  111. response = autoscaling.start_instance_refresh(
  112. AutoScalingGroupName="be-ytb-as")
  113. return {
  114. "response": response
  115. }
  116. if __name__ == '__main__':
  117. app.run(host='0.0.0.0', port=80, debug=True)