如何在 Python 中从 Youtube URL 流式传输音频(无需下载)?

新手上路,请多包涵

我一直在尝试创建一种方法来直接从 python 代码流式传输 youtube url(最好是音频,尽管这无关紧要)。我尝试了很多东西,但似乎都没有用。到目前为止,我能够使用 youtube 数据 api 搜索视频或播放列表,获取第一个视频或播放列表并将其传递到 pafy 以获取不同的流媒体 url。有谁知道无需先下载视频即可通过 python 播放 youtube 音频/视频的方法?我认为可以使用 mplayer 或 vlc 等 cmd 行工具使用子进程弹出打开 cmd 行的 cmd 并传入 url,但我被卡住了。需要任何帮助。请!这是我的以下代码:

 import argparse
import pafy
from googleapiclient.discovery import build
from googleapiclient.errors import HttpError

DEVELOPER_KEY = 'DEVELOPER KEY'
YOUTUBE_API_SERVICE_NAME = 'youtube'
YOUTUBE_API_VERSION = 'v3'

def pafy_video(video_id):
    url = 'https://www.youtube.com/watch?v={0}'.format(video_id)
    vid = pafy.new(url)

def pafy_playlist(playlist_id)
    url = "https://www.youtube.com/playlist?list={0}".format(playlist_id)
    playlist = pafy.get_playlist(url)

def youtube_search(options):
  youtube = build(YOUTUBE_API_SERVICE_NAME, YOUTUBE_API_VERSION, developerKey=DEVELOPER_KEY)

  search_response = youtube.search().list(
    q='Hello world',
    part='id,snippet',
    maxResults=options.max_results
  ).execute()

  videos = []
  playlists = []
  channels = []
  for search_result in search_response.get('items', []):
    if search_result['id']['kind'] == 'youtube#video':
      videos.append('%s' % (search_result['id']['videoId']))
    elif search_result['id']['kind'] == 'youtube#channel':
      channels.append('%s' % (search_result['id']['channelId']))
    elif search_result['id']['kind'] == 'youtube#playlist':
      playlists.append('%s' % (search_result['id']['playlistId']))

  if videos:
    print('Videos:{0}'.format(videos))
    pafy_video(videos[0])
  elif playlists:
    print('Playlists:{0}'.format(playlists))
    pafy_video(playlists[0])

  #https://www.youtube.com/watch?v=rOU4YiuaxAM
  #url = 'https://www.youtube.com/watch?v={0}'.format(videos[0])
  #print(url)

if __name__ == '__main__':
  parser = argparse.ArgumentParser()
  parser.add_argument('--q', help='Search term', default='Google')
  parser.add_argument('--max-results', help='Max results', default=3)
  args = parser.parse_args()
  youtube_search(args)

Tldr;我想直接从 python 代码流式传输 youtube 视频(使用 url 或 id)而不先下载视频

谢谢!

原文由 Nick D 发布,翻译遵循 CC BY-SA 4.0 许可协议

阅读 1.1k
2 个回答

pafy 根据其 文档,不要直接列出播放媒体(至少我没有找到)。

但是我们可以用它获取正确的url,然后使用播放器如 vlc 直接播放,无需下载。

你可以 从这里下载vlc

首先,我们从 youtube 使用 pafy

 import pafy
import vlc

url = "https://www.youtube.com/watch?v=bMt47wvK6u0"
video = pafy.new(url)
best = video.getbest()
playurl = best.url

在这里 playurl 是最好玩的网址。然后我们用VLC来播放。

 Instance = vlc.Instance()
player = Instance.media_player_new()
Media = Instance.media_new(playurl)
Media.get_mrl()
player.set_media(Media)
player.play()

这将打开一个没有控件(播放/暂停/停止等)的窗口。您可以在 repr 窗口或 python 提示符下运行这些命令(取决于您的使用方式)

您将需要使用 vlc 命令相应地构建一个,例如

>>> player.pause() #-- to pause video
>>> player.play()  #-- resume paused video. On older versions,
                   #     this function was called resume
>>> player.stop()  #-- to stop/end video

原文由 Anil_M 发布,翻译遵循 CC BY-SA 4.0 许可协议

实现此目的的最佳方法是将 mpvyoutube-dl (一种带有命令行控制台的轻量级媒体播放器,允许您通过将 url 复制并粘贴到 mpv 控制台(适用于cmd 和终端)。 用法:path\to\mpv https://your/video/audioURL.com/

首先导入这些模块 - 通过 pip install bs4 requests 安装

import re, requests, subprocess, urllib.parse, urllib.request
from bs4 import BeautifulSoup

然后创建一个变量来存储您选择的音乐标题

music_name = "Linkin Park Numb"
query_string = urllib.parse.urlencode({"search_query": music_name})

这里的目标是使用此搜索查询来提取第一个视频结果的输出标题和链接。然后我们可以使用标识符对 youtube url "https://www.youtube.com/results?search_query=" 进行 urllib (例如 "https://www.youtube.com/results?search_query=linkin+park+numb"

 formatUrl = urllib.request.urlopen("https://www.youtube.com/results?" + query_string)

This right here re.findall(r"watch\?v=(\S{11})" 查看我们查询的所有视频结果的 11 个字符标识符

search_results = re.findall(r"watch\?v=(\S{11})", formatUrl.read().decode())

解码内容后,我们可以通过将主要的 youtube url 与 11 字符标识符连接起来来提取完整的 url

 clip = requests.get("https://www.youtube.com/watch?v=" + "{}".format(search_results[0]))
clip2 = "https://www.youtube.com/watch?v=" + "{}".format(search_results[0])

print(clip2)

  • 输出==> https://www.youtube.com/watch?v=kXYiU_JCYtU

为了进一步检查内容,我们可以使用 beautifulsoup 来抓取视频的确切标题

inspect = BeautifulSoup(clip.content, "html.parser")
yt_title = inspect.find_all("meta", property="og:title")

for concatMusic1 in yt_title:
    pass

print(concatMusic1['content'])

  • 输出==> Numb (Official Video) - Linkin Park

最后,为了播放我们的音乐,我们将其输入命令行

subprocess.Popen(
"start /b " + "path\\to\\mpv.exe" + clip2 + " --no-video --loop=inf --input-ipc-server=\\\\.\\pipe\\mpv-pipe > output.txt",
shell=True)

完整脚本

================================================ =================

 import re, requests, subprocess, urllib.parse, urllib.request
from bs4 import BeautifulSoup

music_name = "Linkin Park Numb"
query_string = urllib.parse.urlencode({"search_query": music_name})
formatUrl = urllib.request.urlopen("https://www.youtube.com/results?" + query_string)

search_results = re.findall(r"watch\?v=(\S{11})", formatUrl.read().decode())
clip = requests.get("https://www.youtube.com/watch?v=" + "{}".format(search_results[0]))
clip2 = "https://www.youtube.com/watch?v=" + "{}".format(search_results[0])

inspect = BeautifulSoup(clip.content, "html.parser")
yt_title = inspect.find_all("meta", property="og:title")

for concatMusic1 in yt_title:
    pass

print(concatMusic1['content'])

subprocess.Popen(
"start /b " + "path\\to\\mpv.exe " + clip2 + " --no-video --loop=inf --input-ipc-server=\\\\.\\pipe\\mpv-pipe > output.txt",
shell=True)

# Alternatively, you can do this for simplicity sake:
# subprocess.Popen("start /b " + "path\\to\\mpv.exe " + clip2 + "--no-video", shell=True)

原文由 Rook Crossfield 发布,翻译遵循 CC BY-SA 4.0 许可协议

撰写回答
你尚未登录,登录后可以
  • 和开发者交流问题的细节
  • 关注并接收问题和回答的更新提醒
  • 参与内容的编辑和改进,让解决方法与时俱进
推荐问题