使用LibVLC ,我想保存流同时播放它。 这是Python代码:
import os
import sys
import vlc
if __name__ == '__main__':
filepath = <either-some-url-or-local-path>
movie = os.path.expanduser(filepath)
if 'http://' not in filepath:
if not os.access(movie, os.R_OK):
print ( 'Error: %s file is not readable' % movie )
sys.exit(1)
instance = vlc.Instance("--sub-source marq --sout=file/ps:example.mpg")
try:
media = instance.media_new(movie)
except NameError:
print ('NameError: % (%s vs Libvlc %s)' % (sys.exc_info()[1],
vlc.__version__, vlc.libvlc_get_version()))
sys.exit(1)
player = instance.media_player_new()
player.set_media(media)
player.play()
#dont exit!
while(1):
continue
它保存的视频流文件example.mpg
。 按照这个文档,所述命令来保存一个流是这样的:
--sout=file/ps:example.mpg
我已经创建的实例时使用vlc.Instance
:
instance = vlc.Instance("--sub-source marq --sout=file/ps:example.mpg")
但问题是,它不仅节省了流,它不会同时播放的流。
有什么办法(以LibVLC)我可以保存流(本地文件),同时支付呢?
虽然,我正在寻找一个解决方案Python 3.3.1
,但它是好的,如果有任何C或C ++的解决方案。
我创建了一个类似的,但不是重复的话题昨天。
理念:
其基本思想是很简单的。 你要复制的输出流,并将其重定向到一个文件。 这样做,因为Maresh正确地指出,使用SOUT =#重复{...}指令。
工作方案:
下面的解决方案适用于我的机器™。 我测试过它在Ubuntu 12.10使用VLC V2.0.3(TwoFlower)和Python 2.7.1。 我想这也应该对Python 3的工作,因为最繁重的工作是由libVlc反正做。
import os
import sys
import vlc
if __name__ == '__main__':
#filepath = <either-some-url-or-local-path>
movie = os.path.expanduser(filepath)
if 'http://' not in filepath:
if not os.access(movie, os.R_OK):
print ( 'Error: %s file is not readable' % movie )
sys.exit(1)
instance = vlc.Instance("--sout=#duplicate{dst=file{dst=example.mpg},dst=display}")
try:
media = instance.media_new(movie)
except NameError:
print ('NameError: % (%s vs Libvlc %s)' % (sys.exc_info()[1],
vlc.__version__, vlc.libvlc_get_version()))
sys.exit(1)
player = instance.media_player_new()
player.set_media(media)
player.play()
#dont exit!
while(1):
continue
有用的网址
- 在命令行帮助是必不可少的破译的VLC命令行选项过多。
- VLC流方法文档的第3章 。 解释了流输出,其指令的结构和各种可用的模块的描述。 第4章给出了一些例子。
- LibVLC API文档如果你想在运行时改变媒体的选择
更新 - 保存YouTube视频:
上面的代码不与YouTube发挥好。 我四处搜寻,发现一个额外的transcode
指令可用于YouTube的视频流转换为常规视频格式。 我用#transcode{vcodec=mp4v,acodec=mpga,vb=800,ab=128,deinterlace}
- 了vcodec = MP4V是要在编码视频格式(MP4V是MPEG-4,是mpgv MPEG-1,和也有H263,DIV1,DIV2,DIV3,I420,I422,I444,RV24,YUY2)。
- acodec = MPGA是要在编码(MPGA是MPEG音频层2,A52是A52即AC3声音)的音频格式。
- VB = 800处于千比特/秒的视频比特率。
- AB = 128是在千比特/秒的音频比特率。
- 逐行扫描告诉VLC去隔行的飞行视频。
更新后的代码如下所示:
import os
import sys
import vlc
if __name__ == '__main__':
#filepath = <either-some-url-or-local-path>
filepath = "http://r1---sn-nfpnnjvh-1gil.c.youtube.com/videoplayback?source=youtube&newshard=yes&fexp=936100%2C906397%2C928201%2C929117%2C929123%2C929121%2C929915%2C929906%2C929907%2C929125%2C929127%2C925714%2C929917%2C929919%2C912512%2C912515%2C912521%2C906838%2C904485%2C906840%2C931913%2C904830%2C919373%2C933701%2C904122%2C932216%2C936303%2C909421%2C912711%2C907228%2C935000&sver=3&expire=1373237257&mt=1373214031&mv=m&ratebypass=yes&id=1907b7271247a714&ms=au&ipbits=48&sparams=cp%2Cid%2Cip%2Cipbits%2Citag%2Cratebypass%2Csource%2Cupn%2Cexpire&itag=45&key=yt1&ip=2a02%3A120b%3Ac3c6%3A7190%3A6823%3Af2d%3A732c%3A3577&upn=z3zzcrvPC0U&cp=U0hWSFJOVV9KUUNONl9KSFlDOmt4Y3dEWFo3dDFu&signature=D6049FD7CD5FBD2CC6CD4D60411EE492AA0E9A77.5D0562CCF4E10A6CC53B62AAFFF6CB3BB0BA91C0"
movie = os.path.expanduser(filepath)
savedcopy = "yt-stream.mpg"
if 'http://' not in filepath:
if not os.access(movie, os.R_OK):
print ( 'Error: %s file is not readable' % movie )
sys.exit(1)
instance = vlc.Instance("--sout=#transcode{vcodec=mp4v,acodec=mpga,vb=800,ab=128,deinterlace}:duplicate{dst=file{dst=%s},dst=display}" % savedcopy)
try:
media = instance.media_new(movie)
except NameError:
print ('NameError: % (%s vs Libvlc %s)' % (sys.exc_info()[1],
vlc.__version__, vlc.libvlc_get_version()))
sys.exit(1)
player = instance.media_player_new()
player.set_media(media)
player.play()
#dont exit!
while(1):
continue
几个重要的点:
我使用MPEG音频和视频编解码器的转码指令。 看来使用输出文件匹配的扩展名(MPG在这种情况下)是重要的。 否则VLC打开保存的文件进行播放时,就会犯糊涂。 记住这一点,如果你决定要切换到另一种视频格式。
您不能添加普通YouTube网址为文件路径。 相反,你必须指定视频本身的位置。 这就是为什么,我用的文件路径看起来很神秘的原因。 这文件路径对应的视频在http://www.youtube.com/watch?v=GQe3JxJHpxQ
。 VLC本身能够提取从给定的YouTube网址视频定位,但libVLC不这样做开箱。 你必须写自己的解析器来做到这一点。 看到这个相关的SO问题 。 我采取了这一做法手动解决视频定位为我的测试。
我认为你需要复制的输出,以便在同一时间播放和录制它:
vlc.Instance("--sub-source marq --sout=#stream_out_duplicate{dst=display,dst=std{access=file,mux=ts,dst=/path/file.mpg}}")
要么
libvlc_media_add_option(media, ":sout=#stream_out_duplicate{dst=display,dst=std{access=file,mux=ts,dst=/path/file.mpg}}")
你尝试加入的选项列表下面的选项?
--sout显示
即
instance = vlc.Instance("--sub-source marq --sout=file/ps:example.mpg --sout-display")
在活动状态的网站我看到有人示例代码前段时间发挥和使用VLC使用vlc.py模块记录的MP3文件。 你可以看看它的示例代码,看看如何复制流。 我这里抄个代码为你(我是从拷贝它http://code.activestate.com/recipes/577802-using-vlcpy-to-record-an-mp3-and-save-a-cue-file/ ):
import vlc
import time
import os
def new_filename(ext = '.mp3'):
"find a free filename in 00000000..99999999"
D = set(x[:8] for x in os.listdir('.')
if (x.endswith(ext) or x.endswith('.cue')) and len(x) == 12)
for i in xrange(10**8):
s = "%08i" %i
if s not in D:
return s
def initialize_cue_file(name,instream,audiofile):
"create a cue file and write some data, then return it"
cueout = '%s.cue' %name
outf = file(cueout,'w')
outf.write('PERFORMER "%s"\n' %instream)
outf.write('TITLE "%s"\n' %name)
outf.write('FILE "%s" WAVE\n' %audiofile)
outf.flush()
return outf
def initialize_player(instream, audiofile):
"initialize a vlc player which plays locally and saves to an mp3file"
inst = vlc.Instance()
p = inst.media_player_new()
cmd1 = "sout=#duplicate{dst=file{dst=%s},dst=display}" %audiofile
cmd2 ="no-sout-rtp-sap"
cmd3 = "no-sout-standard-sap"
cmd4 ="sout-keep"
med=inst.media_new(instream,cmd1,cmd2,cmd3,cmd4)
med.get_mrl()
p.set_media(med)
return p, med
def write_track_meta_to_cuefile(outf,instream,idx,meta,millisecs):
"write the next track info to the cue file"
outf.write(' TRACK %02i AUDIO\n' %idx)
outf.write(' TITLE "%s"\n' %meta)
outf.write(' PERFORMER "%s"\n' %instream)
m = millisecs // 60000
s = (millisecs - (m*60000)) // 1000
hs = (millisecs - (m*60000) - (s*1000)) //10
ts = '%02i:%02i:%02i' %(m,s,hs)
outf.write(' INDEX 01 %s\n' %ts)
outf.flush()
def test():
#some online audio stream for which this currently works ....
instream = 'http://streamer-mtc-aa05.somafm.com:80/stream/1018'
#if the output filename ends with mp3 vlc knows which mux to use
ext = '.mp3'
name = new_filename(ext)
audiofile = '%s%s' %(name,ext)
outf = initialize_cue_file(name,instream,audiofile)
p,med = initialize_player(instream, audiofile)
p.play()
np = None
i = 0
while 1:
time.sleep(.1)
new = med.get_meta(12)
if new != np:
i +=1
t = p.get_time()
print "millisecs: %i" %t
write_track_meta_to_cuefile(outf,instream,i,new,t)
np = new
print "now playing: %s" %np
if __name__=='__main__':
test()