基本的に、私はビデオをユーザーから非表示にする必要があるプロジェクトを持っていますが、それでも(phpを使用して)それらを見ることができます。ここに私がこれまでに得たものがあります:
Video.phpファイルには次のものがあります。
<?php
$ch = curl_init();
curl_setopt($ch, CURLOPT_URL, 'path/to/movie.mp4');
curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
curl_setopt($ch, CURLOPT_HEADER, 0);
$out = curl_exec($ch);
curl_close($ch);
header('Content-type: video/mp4');
header('Content-type: video/mpeg');
header('Content-disposition: inline');
header("Content-Transfer-Encoding: binary");
header("Content-Length: ".filesize($out));
echo $out;
exit();
?>
そして、これを表示するはずのhtmlファイルは、予想どおりhtml5を使用しています。今ここにあります..私はこれを(ではなく)まっすぐに埋め込むと動作します。しかし、それは私のiPhoneでは動作せず、タグでは動作しません... PHPラッパーの代わりに直接ファイルを使用すると、iPhoneでもすべて正常に動作します...
だから、これについての私の質問はこれだと思います:iPhoneとHMTL5経由でストリーミングできるmp4を完全に複製するための適切なheader()情報は何ですか?
派生ソリューション: http://mobiforge.com/developing/story/content-delivery-mobile-devices
video.phpファイル:
<?php
$file = 'path/to/videofile.mp4';
$fp = @fopen($file, 'rb');
$size = filesize($file); // File size
$length = $size; // Content length
$start = 0; // Start byte
$end = $size - 1; // End byte
header('Content-type: video/mp4');
header("Accept-Ranges: 0-$length");
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("Content-Range: bytes $start-$end/$size");
header("Content-Length: ".$length);
$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);
exit();
?>
Iphoneは、オーディオとビデオの要求にバイト範囲と呼ばれるものを使用します。ソリューションについては、このリンクを参照してください。付録Aにあります。
http://mobiforge.com/developing/story/content-delivery-mobile-devices
ここに、あなたが望むことをするコードがあります( この質問から )。 PHPソリューションはより洗練されているように見え、Webサーバーを使用してコンテンツを提供する、より効率的なソリューションが追加されます。
<?php
$path = 'file.mp4';
$size=filesize($path);
$fm=@fopen($path,'rb');
if(!$fm) {
// You can also redirect here
header ("HTTP/1.0 404 Not Found");
die();
}
$begin=0;
$end=$size;
if(isset($_SERVER['HTTP_RANGE'])) {
if(preg_match('/bytes=\h*(\d+)-(\d*)[\D.*]?/i', $_SERVER['HTTP_RANGE'], $matches)) {
$begin=intval($matches[0]);
if(!empty($matches[1])) {
$end=intval($matches[1]);
}
}
}
if($begin>0||$end<$size)
header('HTTP/1.0 206 Partial Content');
else
header('HTTP/1.0 200 OK');
header("Content-Type: video/mp4");
header('Accept-Ranges: bytes');
header('Content-Length:'.($end-$begin));
header("Content-Disposition: inline;");
header("Content-Range: bytes $begin-$end/$size");
header("Content-Transfer-Encoding: binary\n");
header('Connection: close');
$cur=$begin;
fseek($fm,$begin,0);
while(!feof($fm)&&$cur<$end&&(connection_status()==0))
{ print fread($fm,min(1024*16,$end-$cur));
$cur+=1024*16;
usleep(1000);
}
die();
パフォーマンスの向上
ファイル全体がPHPを通過する必要があるため、これが最も効率的な方法ではないことに注意してください。
これを行う理由は、アクセスを制限することであり、後でより効率が必要な場合、Webサーバーのフラグを使用できます。
X-Sendfileモジュールまたは軽量のApache( nginx info here )
$path = 'file.mp4';
header("X-Sendfile: $path");
die();
これはもう少し高度であり、必要な場合にのみ使用する必要がありますが、かなり簡単で平凡なパフォーマンスを備えたものから始める場合、アップグレードオプションがあることを知ってリラックスしています。
このコードは私にとって非常に便利でしたが、セッション変数を使用し、PHPセッションへのキューアクセス。ビデオがロードされている場合、すべてAJAXリクエストは不可能など。したがって、出力を開始する前にsession_write_close()
を呼び出してください。
はい、簡単です。これらのヘッダーを手動で設定する必要はありません。サーバーに自動的に実行させます。
動作スクリプト-
ob_start();
if( isset($_SERVER['HTTP_RANGE']) )
$opts['http']['header']="Range: ".$_SERVER['HTTP_RANGE'];
$opts['http']['method']= "HEAD";
$conh=stream_context_create($opts);
$opts['http']['method']= "GET";
$cong= stream_context_create($opts);
$out[]= file_get_contents($real_file_location_path_or_url,false,$conh);
$out[]= $http_response_header;
ob_end_clean();
array_map("header",$http_response_header);
readfile($real_file_location_path_or_url,false,$cong);