这只是部分答案,因为我目前也坚持这一点。
如果文件已经是 FLV 格式,那么 Static::Simple 插件就可以正常工作。我已经使用 root/static/ 目录中的文件和$c->serve_static_file
方法对其进行了测试。下面是我的 jwplayer 设置(包装在 JQuery 的 ready 函数中。
<script type="text/javascript" src="[% c.uri_for('/static/js/mediaplayer-5.10') %]/jwplayer.js"></script>
<script type="text/javascript">
$(function() {
jwplayer('mediaplayer').setup({
'flashplayer': "[% c.uri_for('/static/js/mediaplayer-5.10/player.swf') %]",
'id': 'playerID',
'width': '480',
'height': '270',
'file': "[% c.uri_for('/download') %]/dump/ffs/root/static/transcode_jEfhmk.flv"
});
});
</script>
如果文件是其他格式,则需要对其进行转码。以下是我目前的尝试。它可以很好地转码,您甚至可以将文件流式传输到下载,但我无法让 jwplayer 从转码流中播放。
use IPC::Open3;
#path comes in as /flv/path/to/file.avi
sub index :Path :Args {
my ( $self, $c, @path ) = @_;
@path = grep($_ ne '..', @path);
my $path = join('/',@path);
my $abs_path = $c->config->{'serve_dir'} . '/';
$abs_path .= join("/", @path);
if (-r $abs_path){
my ($stdin, $stdout, $stderr, $pid);
#avconv
# -i input_file
# -b:v video bitrate
# -s video size
# -r video framerate
# -an no audio (having trouble with the audio settings)
# -f format
# pipe:1 send transcoded video to STDOUT
$pid = open3($stdin, $stdout, $stderr, "avconv -i \"$abs_path\" -b:v 600k -s 320x240 -r 25 -f flv -an pipe:1");
$c->response->content_type("video/x-flv");
$c->response->header('Content-Disposition' => "filename=transcode.flv");
my $chunk_size = 1048576;
do {
read( $stdout, my $buffer, $chunk_size );
$c->write( $buffer );
} while (kill(0, $pid)); #loop while transcoding process is alive
}
}
要添加伪流,您需要创建一个控制器来查找开始查询参数,然后查找它,并将其发送$c->response->body
到文件句柄。
open(my $fh, "<", $path);
binmode($fh);
if ($c->req->param('start')){
seek($fh, $c->req->param('start'), 0);
}
$c->response->body($fh);
一个警告:视频必须在元数据中包含关键帧,jwplayer 才能进行搜索。
从 jwplayer 的HTTP 流页面:
注意:某些 FLV 编码器在编码视频时不包含搜索点元数据。如果没有这些数据,HTTP 伪流将无法工作。如果您怀疑您的视频没有元数据,请使用我们的Metaviewer 插件检查视频。应该有一个搜索点或关键帧列表。如果不存在,请使用 FLVMDI 工具解析您的 FLV 视频并注入此元数据。