- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在用 symfony2
编写应用程序,但我遇到了视频流问题。
如果一个 Action 需要很长时间才能执行 - 例如 1 分钟
,整个应用程序将被卡住(当在第二个选项卡中打开时)并且必须等待该执行结束。
问题出在哪里?
//编辑
public function streamAction($fileName) {
$user = $this->get('security.context')->getToken()->getUser();
$request = $this->getRequest();
$uid = $request->get('uid') != 'null' ? $user->getId() : $request->get('uid');
$libPath = $this->_libPath('Users', 'uid' . str_pad($uid, 6, "0", STR_PAD_LEFT));
$file = pathinfo($fileName);
$fileName = $file['basename'];
$fileExt = $file['extension'];
$filePath = realpath($libPath . $fileName);
if (in_array($fileExt, $this->formats['video'])) {
$mime = 'video';
}
if (in_array($fileExt, $this->formats['audio'])) {
$mime = 'audio';
}
$mime .= '/' . $fileExt;
header("Accept-Ranges: bytes");
if (is_file($filePath)) {
header("Content-type: $mime");
if (isset($_SERVER['HTTP_RANGE'])) {
$fp = fopen($filePath, 'rb');
$size = filesize($filePath);
$length = $size;
$start = 0;
$end = $size - 1;
if (isset($_SERVER['HTTP_RANGE'])) {
$c_start = $start;
$c_end = $end;
list(, $range) = explode('=', $_SERVER['HTTP_RANGE'], 2);
if (strpos($range, ',') !== false) {
header('HTTP/1.1 416 Requested Range Not Satisfiable');
header("Content-Range: bytes $start-$end/$size");
exit;
}
if ($range == '-') {
$c_start = $size - substr($range, 1);
} else {
$range = explode('-', $range);
$c_start = $range[0];
$c_end = ( isset($range[1]) && is_numeric($range[1]) ) ? $range[1] : $size;
}
$c_end = ($c_end > $end) ? $end : $c_end;
if ($c_start > $c_end || $c_start > $size - 1 || $c_end >= $size) {
header('HTTP/1.1 416 Requested Range Not Satisfiable');
header("Content-Range: bytes $start-$end/$size");
exit;
}
$start = $c_start;
$end = $c_end;
$length = $end - $start + 1;
fseek($fp, $start);
header('HTTP/1.1 206 Partial Content');
header('ETag: "' . md5(microtime()) . '"');
}
header("Content-Range: bytes $start-$end/$size");
header("Content-Length: $length");
header('Connection: Close');
$buffer = 1024 * 8;
while (!feof($fp) && ($p = ftell($fp)) <= $end) {
if ($p + $buffer > $end) {
$buffer = $end - $p + 1;
}
set_time_limit(0);
echo fread($fp, $buffer);
flush();
}
fclose($fp);
} else {
header("Content-Length: " . filesize($filePath));
readfile($filePath);
}
} else {
echo "error";
}
die();
}
问题不在于代码,因为 symfony2 不仅会阻止流式传输,还会阻止其他长时间操作,例如下载文件。
最佳答案
如果您使用 Symfony 等 OOP 框架,为什么还要使用过程方法而不是 Symfony 必须提供的方法?例如 StreamedResponse
类。
我在下面为您提供了更多面向 OOP/Symfony 的内容,但是,如果提供的范围只是 -
,您当前将范围开始重置为...文件大小?我很确定这不是你应该做的。我认为您应该仔细检查一下!
这是 streamAction() 的审查版本:
use SplFileInfo;
use RuntimeException;
// Symfony >= 2.1
use Symfony\Component\HttpFoundation\StreamedResponse;
public function streamAction($fileName) {
$user = $this->getUser();
$request = $this->getRequest();
// Create the StreamedResponse object
$response = new StreamedResponse();
$uid = $request->get('uid') != 'null' ? $user->getId() : $request->get('uid');
$libPath = $this->_libPath('Users', 'uid' . str_pad($uid, 6, "0", STR_PAD_LEFT));
try {
$file = new SplFileObject($libPath . $fileName);
}
catch (RuntimeException $runtimeException) {
// The file cannot be opened (permissions?)
// throw new AnyCustomFileErrorException() maybe?
}
// Check file existence
if (!($file->isFile())) {
// The file does not exists, or is not a file.
throw $this->createNotFoundException('This file does not exists, or is not a valid file.');
}
// Retrieve file informations
$fileName = $file->getBasename();
$fileExt = $file->getExtension();
$filePath = $file->getRealPath();
$fileSize = $file->getSize();
// Guess MIME Type from file extension
if (in_array($fileExt, $this->formats['video'])) {
$mime = 'video';
} elseif (in_array($fileExt, $this->formats['audio'])) {
$mime = 'audio';
}
$mime .= '/' . $fileExt;
$response->headers->set('Accept-Ranges', 'bytes');
$response->headers->set('Content-Type', $mime);
// Prepare File Range to read [default to the whole file content]
$rangeMin = 0;
$rangeMax = $fileSize - 1;
$rangeStart = $rangeMin;
$rangeEnd = $rangeMax;
$httpRange = $request->server->get('HTTP_RANGE');
// If a Range is provided, check its validity
if ($httpRange) {
$isRangeSatisfiable = true;
if (preg_match('/bytes=\h*(\d+)-(\d*)[\D.*]?/i', $httpRange, $matches)) {
$rangeStart = intval($matches[1]);
if (!empty($matches[2])) {
$rangeEnd = intval($matches[2]);
}
} else {
// Requested HTTP-Range seems invalid.
$isRangeSatisfiable = false;
}
if ($rangeStart <= $rangeEnd) {
$length = $rangeEnd - $rangeStart + 1;
} else {
// Requested HTTP-Range seems invalid.
$isRangeSatisfiable = false;
}
if ($file->fseek($rangeStart) !== 0) {
// Could not seek the file to the requested range: it might be out-of-bound, or the file is corrupted?
// Assume the range is not satisfiable.
$isRangeSatisfiable = false;
// NB : You might also wish to throw an Exception here...
// Depending the server behaviour you want to set-up.
// throw new AnyCustomFileErrorException();
}
if ($isRangeSatisfiable) {
// Now the file is ready to be read...
// Set additional headers and status code.
// Symfony < 2.4
// $response->setStatusCode(206);
// Or using Symfony >= 2.4 constants
$response->setStatusCode(StreamedResponse::HTTP_PARTIAL_CONTENT);
$response->headers->set('Content-Range', sprintf('bytes %d/%d', $rangeStart - $rangeEnd, $fileSize));
$response->headers->set('Content-Length', $length);
$response->headers->set('Connection', 'Close');
} else {
$response = new Response();
// Symfony < 2.4
// $response->setStatusCode(416);
// Or using Symfony >= 2.4 constants
$response->setStatusCode(StreamedResponse::HTTP_REQUESTED_RANGE_NOT_SATISFIABLE);
$response->headers->set('Content-Range', sprintf('bytes */%d', $fileSize));
return $response;
}
} else {
// No range has been provided: the whole file content can be sent
$response->headers->set('Content-Length', $fileSize);
}
// At this step, the request headers are ready to be sent.
$response->prepare($request);
$response->sendHeaders();
// Prepare the StreamCallback
$response->setCallback(function () use ($file, $rangeEnd) {
$buffer = 1024 * 8;
while (!($file->eof()) && (($offset = $file->ftell()) < $rangeEnd)) {
set_time_limit(0);
if ($offset + $buffer > $rangeEnd) {
$buffer = $rangeEnd + 1 - $offset;
}
echo $file->fread($buffer);
}
// Close the file handler
$file = null;
});
// Then everything should be ready, we can send the Response content.
$response->sendContent();
// DO NOT RETURN $response !
// It has already been sent, both headers and body.
}
关于Symfony2 视频流,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14559371/
我正在用 symfony2 编写应用程序,但我遇到了视频流问题。 如果一个 Action 需要很长时间才能执行 - 例如 1 分钟,整个应用程序将被卡住(当在第二个选项卡中打开时)并且必须等待该执行结
我正在用 symfony2 编写应用程序,但我遇到了视频流问题。 如果一个 Action 需要很长时间才能执行 - 例如 1 分钟,整个应用程序将被卡住(当在第二个选项卡中打开时)并且必须等待该执行结
我正在尝试用视频流做一个应用程序,现在我只能从服务器向客户端发送一张图像。当我尝试在客户端发送不止一张图像时,我收到以下错误:“参数无效。”在 pictureBox1.Image = new Bitm
我正在使用 mediaViews 和 html5 在我们的 cakephp 网站上流式传输视频截屏。我正在为播放器使用 videojs,但遇到了一些问题: 不会在 safari/ipad/iphone
我已经安装了fluent-ffmpeg和ffmpeg-static以将流转换为HTML可复制的视频。如何使用这些软件包在客户端中显示流? 服务器端( main.js ): const { app, B
我在我的应用程序中实现了 TTS(文本转语音)功能。一切都很顺利,但现在我遇到了一个无法找到解决方案的问题。 基本上,我在激活 TTS 功能之前获得音频焦点。在实际的应用程序中,当我这样做时音乐就会停
我已经执行了开源社区链接发布的 AudioPlayer.java 示例 Audio Player Tutorial 我得到以下异常 Exception in thread "main" java.la
有什么方法可以在iOS原生App中通过URL播放无法下载或保存到本地的视频流。我想让用户观看视频,但他不能在本地下载或保存视频(通过第 3 方下载器应用程序)。 在 MPMoviePlayerCont
我正在开发用于网络吞吐量测试的点对点视频流。两个对等点之间的网络支持低比特率传输。 我使用 OpenCV 为视频流制作了简单的 python 程序。 该程序从网络摄像头抓取视频帧并将其编码为 JPEG
我正在我的应用程序中流式传输实时视频。我有一个 .m3u8 链接,它在 vlc 播放器中完美运行。但是当我在我的应用程序中播放这个流时,视频的可视化被破坏了(见截图)。有谁知道,这可能是什么原因造成的
好的。因此,有数十亿种不同的 Android 设备。我有一个视频流服务,非常适用于 iOS。我的应用程序具有实时视频功能和保存的视频剪辑播放功能(也可以流式传输到设备)。我在不同的 Android 设
我有一个屏幕共享应用程序 (WebRTC)。一个用户想与另一个用户共享他的屏幕。就像一个应用程序在用户 1 机器上运行而另一个应用程序在用户 2 机器上运行。用户 1 想要共享他的屏幕,现在如何在用户
我正在尝试提供即时转码的视频。不幸的是,这意味着寻求不起作用。我假设这是因为浏览器不知道视频有多长,因此无法正确显示搜索栏。 有谁知道是否可以对视频的时长进行硬编码? 我想到的另一个选择可能是创建我自
在将解复用的 h264 解码输出发送到 gstreamer 管道中的 autovideosink 之前,是否可以在两者之间提供一些延迟。如果是这样,任何人都可以发布示例管道来做到这一点。 我使用的管道
一直在把我的头发拉出来,这本来应该是一项快速而简单的任务。 我有一个自托管的 WCF 服务,我需要在其中实现实时视频转码,转码本身不是问题,使用 FFMpeg 到本地临时文件。 我的代码的快速示例;
我想在 C# 应用程序和浏览器之间建立视频流。 我使用 Websockets 进行通信。 视频源是网络摄像头。 我可以请求单个 PNG 帧,但速度太慢了。 websocket 服务器( Ratchet
关闭。这个问题是opinion-based .它目前不接受答案。 想要改进这个问题? 更新问题,以便 editing this post 可以用事实和引用来回答它. 关闭 9 年前。 Improve
是否可以在 lync 2010 或 2013 中捕获 lync session 的视频流?我已经在我们的实验室中运行了测试版。 UCMA 为 AudioVideoFlow 提供了钩子(Hook),但似
我有一个 ffmpeg 命令,它接收帧流并输出 ogg 视频。如何从另一个文件中添加音频? ffmpeg -f rawvideo -pix_fmt bgr24 -s WIDTHxHEIGHT -r 3
我正在使用全息镜头,并且我正在尝试保存具有可用世界/投影矩阵的视频流。我一直在尝试拍摄一系列照片并保存数据,但我找不到保存图像和矩阵的方法。 保存到磁盘时,没有获取照片捕获帧(包含矩阵数据)的选项,保
我是一名优秀的程序员,十分优秀!