PHP + NGINX 控制視頻文件播放,并防止文件下載

最簡單的方法是使用NGINX的 internal 功能

server {
?? ?listen 80;
?? ?server_name www.xxx.com;
 
?? ?location / {
?? ??? ?index index.php index.html index.htm;
?? ??? ?root? /xxx;

?? ??? ?if (!-e $request_filename) {
?? ??? ? rewrite ^/index.php(.*)$ /index.php?s=$1 last;
?? ??? ? rewrite ^(.*)$ /index.php?s=$1 last;
?? ??? ? break;
?? ??? ?}
?? ?}

 # 這里使用internal做下載防護,只允許內部程序(PHP等)訪問,這樣外部直接訪問這個地址就會提示404錯誤
?? ?location ~ \.mp4$ {
?? ??? ?internal;
  # 這里的路徑配置是可選的,可以配置到網站外部,和其他location里的配置路徑是一個意思,可以更好的防止文件被通過網址下載
?? ??? ?root /bbb;?
?? ?}

?? ?location ~ \.php$ {
?? ??? ?root? /xxx;
?? ??? ?try_files $uri =404;
?? ??? ?fastcgi_pass? unix:/var/run/php-fpm/php-fpm.sock;
?? ??? ?fastcgi_index index.php;
?? ??? ?fastcgi_param SCRIPT_FILENAME $document_root$fastcgi_script_name;
?? ??? ?include fastcgi_params;
?? ?}
}

然后在PHP中通過header('Location: /xxx/aa.mp4')訪問,但這個只適合于小文件,因為這種方式不支持Content-Range和HTTP 206,導致不支持斷點續傳和視頻文件的邊下邊播

最好的方式是通過header('X-Accel-Redirect: /xxx/aa.mp4')訪問,X-Accel-Redirect支持Content-Range和HTTP 206,Apache里面是X-Sendfile

?

如果不能配置nginx的internal選項,或者nginx不支持X-Accel-Redirect,卻要使用斷點續傳、邊下邊播和下載防護,那么就要通過PHP代碼來控制

思路上是使用token鑒權,首先下載文件前由前端請求一個token,這個token由用戶ID+時間戳+視頻ID組成,然后存儲到session并加密后發送給前端

$token = $_SESSION['uid'] . '|' . time() . '|' . $id;
// 保存未加密token到seesion
$_SESSION['video_token'] = $token;
// 加密token以便發送到客戶端
$token = Mcrypt::encode($token, 'lbnnbs');

?

前端收到token,向下載控制器網址發送token請求文件下載,控制器對token解碼,判斷Session是否一致、用戶ID是否正確、是否超時、視頻ID是否正確

然后將視頻ID轉換為文件地址,通過PHP讀取文件發送給前端下載。并通過發送Content-Range和HTTP 206來支持斷點續傳、邊下邊播等操作

$id = decodeIdFromToken($token);$file = getFilePath($id);SendVideo($file);

?

復制代碼
    private function SendVideo($file) {header("Content-type: video/mp4");header("Accept-Ranges: bytes");$size = filesize($file);if (isset($_SERVER['HTTP_RANGE'])) {header("HTTP/1.1 206 Partial Content");list($name, $range) = explode("=", $_SERVER['HTTP_RANGE']);list($begin, $end) = explode("-", $range);if ($end == 0)$end = $size - 1;}else {$begin = 0;$end = $size - 1;}header("Content-Length: " . ($end - $begin + 1));header("Content-Disposition: filename=" . basename($file));header("Content-Range: bytes " . $begin . "-" . $end . "/" . $size);$fp = fopen($file, 'rb');fseek($fp, $begin);while (!feof($fp)) {$p = min(1024, $end - $begin + 1);$begin += $p;echo fread($fp, $p);}fclose($fp);}private function decodeIdFromToken($token) {if (empty($_SESSION['uid'])) {return false;}$token = Mcrypt::decode($token, 'lbnnbs');if ($token != $_SESSION['video_token']) {//token解密失敗,判定失效return false;}$token_arr = explode('|', $token);if (intval($token_arr[0]) != $_SESSION['uid']) {// token不是當前用戶,判定失效return false;}if (intval($token_arr[1]) < time() - 10) {// token生成于10秒前,判定失效return false;}unset($_SESSION['video_token']);return $token_arr[2]; // 返回id}
復制代碼

?

另外需要注意的是,很多時候mp4文件會無法支持邊下邊播,這是因為在轉碼或者壓縮視頻文件到mp4的時候把文件的元數據移除了或者放到了文件的末尾,導致前端播放器不能第一時間獲取到視頻文件的播放時長等信息,只能全部下載完畢后或者直到讀取到了元數據后才能播放。這個時候可以使用qt-faststart.exe工具進行處理,把元數據放回文件頭部即可。

轉載于:https://www.cnblogs.com/liliuguang/p/10576000.html

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/252091.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/252091.shtml
英文地址,請注明出處:http://en.pswp.cn/news/252091.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

可視化調試工具

rosrun rqt_console rqt_console # 查看日志消息&#xff0c;可filter、highlight指定級別。 rosrun rqt_logger_level rqt_logger_level # 可設在日志記錄器的嚴重級別 rosrun rqt_topic rqt_topic # 顯示topic調試信息 rosrun rqt_publisher rqt_publisher # 在界面中管理ro…

C#:委托基礎與事件

通過以下思維導圖&#xff0c;學習委托的基本概念&#xff0c;后面著重講解委托的運用&#xff0c;希望通過最簡單的方式收獲更多的知識。 1.委托的各種寫法 1、委托 委托名new 委托&#xff08;會調用的方法名); 委托名&#xff08;參數&#xff09;; 2、委托 委托名 會調用…

Git Bash關鍵命令

1.默認目錄是C:\Users\用戶名 2.切換目錄&#xff1a;$cd c:\\windows 3.切換到上級目錄&#xff1a;cd ..&#xff0c;中間有空格 4.列出某目錄所有文件&#xff0c;相當于DOS下的dir&#xff1a;ls c:\\windows 5.查看配置信息&#xff1a;git config --list 以下是顯示信息 …

C#:invoke 與 BeginInvoke使用區別

invoke和begininvoke 區別 一直對invoke和begininvoke的使用和概念比較混亂&#xff0c;這兩天看了些資料&#xff0c;對這兩個的用法和原理有了些新的認識和理解。 首先說下&#xff0c;invoke和begininvoke的使用有兩種情況&#xff1a; 1. control中的invoke、begininvoke。…

Django基本命令

Django基本命令 1.創建一個Django 項目 django_admin.py startproject mysite當前目錄下會生成mysite的工程&#xff0c;目錄結構如下&#xff1a; manage.py ----- Django項目里面的工具&#xff0c;通過它可以調用django shell和數據庫等。settings.py ---- 包含了項目的默認…

Git忽略規則.gitignore梳理

對于經常使用Git的朋友來說&#xff0c;.gitignore配置一定不會陌生。廢話不說多了&#xff0c;接下來就來說說這個.gitignore的使用。首先要強調一點&#xff0c;這個文件的完整文件名就是".gitignore"&#xff0c;注意最前面有個“.”。 一般來說每個Git項目中都需…

第二周CoreIDRAW課總結

1.這節課學到了什么知識&#xff1f; 學到了圖像的復制&#xff0c;再制鼠標復制&#xff0c;重復&#xff0c;還有對象的對齊&#xff0c;對象的分布順序。 2.有哪些心得體會&#xff1f; 做了課本的練習&#xff0c;會用窗口里面的泊塢窗造型命令也作出了一個作品。 3.這節課…

axios關于針對請求時長策略設計的思考

前言 在我們的業務請求中&#xff0c;有很多時候會針對有不同時長的需求策略性設置。這里針對這個需求進行詳細的展開。 針對這種情況&#xff0c;我們的timout的一般是根據請求地址來的&#xff0c;所以核心處理技巧便是如何根據不同的request地址去設置不同的timeout. 我們之…

C#:WinForm無邊框窗體移動方法、模仿鼠標單擊標題欄移動窗體位置

方法一&#xff1a;直接通過修改窗體位置從而達到移動窗體的效果 方法二&#xff1a;直接偽裝發送單擊任務欄消息&#xff0c;讓應用程序誤以為單擊任務欄從而移動窗體 方法一 1.定義一個位置信息Point用于存儲鼠標位置 private Point mPoint; 2.給窗體等控件增加MouseDown…

Python 字典刪除元素clear、pop、popitem

同其它python內建數據類型一樣&#xff0c;字典dict也是有一些實用的操作方法。這里我們要說的是字典刪除方法&#xff1a;clear()、pop()和popitem()&#xff0c;這三種方法的作用不同&#xff0c;操作方法及返回值都不相同。接下來就來查看下這些字典特定方法的具體用法是什么…

reactor模式:多線程的reactor模式

上文說到單線程的reactor模式 reactor模式&#xff1a;單線程的reactor模式 單線程的reactor模式并沒有解決IO和CPU處理速度不匹配問題&#xff0c;所以多線程的reactor模式引入線程池的概念&#xff0c;把耗時的IO操作交由線程池處理&#xff0c;處理完了之后再同步到selecti…

Elasticsearch實戰篇——Spring Boot整合ElasticSearch

2019獨角獸企業重金招聘Python工程師標準>>> 當前Spring Boot很是流行&#xff0c;包括我自己&#xff0c;也是在用Spring Boot集成其他框架進行項目開發&#xff0c;所以這一節&#xff0c;我們一起來探討Spring Boot整合ElasticSearch的問題。 本文主要講以下內容…

C#:Dockpanel的一些入門的基本操作

原文鏈接&#xff1a; 一、引用&#xff1a; 1.建立一個WinForm工程&#xff0c;默認生成了一個WinForm窗體Form1&#xff08;此處默認為主窗體&#xff09;。 2.引用—>添加引用—>瀏覽—>weiFenLuo.winFormsUI.Docking.dll。 3.設置Form1窗體屬性IsMdiContainer…

MyBatis中if,where,set標簽

<if>標簽 <select id"findActiveBlogWithTitleLike"resultType"Blog">SELECT * FROM BLOG WHERE state ‘ACTIVE’ <if test"title ! null">AND title like #{title}</if> </select> if標簽通常伴隨著where,set…

Python3基礎 __repr__ 類的實例對象的名字 可以打印文字(1)

引用自&#xff1a;http://www.bubuko.com/infodetail-1918622.html 這個__repr__的作用從下邊的例子中可以看出,返回實例化對象的表達 code: class MyClass() :def __str__(self) :return "我是MyClass的一個實例"def __repr__(self) :return "這回連print都省…

Day03:文件打開;錯誤處理

錯誤處理 try: #要執行的代碼 except 錯誤的類型&#xff08;可選&#xff09;: #發生錯誤時執行的代碼 finally: #有沒有發生錯誤都執行的代碼 復制代碼with open() as 變量名&#xff1a; with提供一種叫上下文管理協議的python技術&#xff0c;系統會自動關閉文件 open() 默…

Python: pip升級報錯了:You are using pip version 10.0.1, however version 20.3.3 is available.

1,Python使用命令&#xff1a;python -m pip install --upgrade pip升級pip的時候報了下面這個錯 2,換了個命令&#xff1a; python -m pip install --upgrade pip -i https://pypi.douban.com/simple 更新成功了&#xff0c;但又報了一個新的錯誤&#xff1a; AttributeError:…

新手上路之Hibernate:第一個Hibernate例子

一、Hibernate概述 &#xff08;一&#xff09;什么是Hibernate&#xff1f; Hibernate核心內容是ORM&#xff08;關系對象模型&#xff09;。可以將對象自動的生成數據庫中的信息&#xff0c;使得開發更加的面向對象。這樣作為程序員就可以使用面向對象的思想來操作數據庫&…

模板標簽及模板的繼承與引用

1.常用的模板標簽 - 作用是什么:提供各種邏輯 view.py: def index(request):#模板標簽 --常用標簽 總結&#xff1a;語法 {% tag %} {% endtag %} {% tag 參數 參數 %} 示例 展示頁index.html&#xff0c;包含for標簽&#xff0c;if標簽&#xff0c;url標簽 {% extends teacher…

文件夾操作之創建

創建文件夾可通過Directory類的CreateDirectory方法來實現格式為&#xff1a;Directory.CreateDirectory(“文件路徑”)&#xff1b;String path“C:\Users\Administrator\Desktop\51zxw”&#xff1b; If&#xff08;Directory.exists&#xff08;path&#xff09;&#xff09…