04: 使用七牛雲進行視頻點播


1.1 七牛雲存儲介紹

  1、七牛雲存儲使用參考文檔

# 七牛雲管理后台地址
https://portal.qiniu.com/kodo/bucket # 七牛雲使用
https://developer.qiniu.com/kodo/manual/1233/console-quickstart # pythonSDK
https://developer.qiniu.com/kodo/sdk/1242/python # Node.js SDK V6
https://developer.qiniu.com/kodo/sdk/3828/node-js-v6 # JavaScript SDK歷史文檔1.x
https://developer.qiniu.com/kodo/sdk/4244/the-javascript-sdk-historical-documents-1-x

  2、七牛雲介紹

      1. 以前看見過FastDfs+FFmpeg進行視頻存儲等操作,但是這種方式穩定性會低一些,而且成本也沒有降低。

      2. 市面上關於雲存儲的第三方服務比比皆是,最著名的無疑就是七牛雲存儲,本次我們將演示用django+Vue+七牛雲進行視頻存儲與播放。      

  3、七牛雲上傳邏輯

      1. 在做七牛雲的文件上傳時,很多人有一個誤區,就是以為是前端先上傳到后台服務器,然后后台服務器再將文件上傳到七牛雲。

      2. 這個邏輯本身沒有問題,但是會遇到一個問題,如果文件大會導致上傳很慢

      3. 正確邏輯應該是前端直接上傳七牛,而后台只承擔生成token和存儲七牛雲返回的hash的任務。

1.2 django+JavaScript上傳視頻

  1、參考七牛雲SDK

# pythonSDK
https://developer.qiniu.com/kodo/sdk/1242/python
# JavaScript SDK歷史文檔1.x https://developer.qiniu.com/kodo/sdk/4244/the-javascript-sdk-historical-documents-1-x
# JavaScript 官方demo
https://github.com/qiniu/js-sdk/tree/1.x#demo

  2、代碼

    參考代碼:https://gitee.com/edushiyanlou/QiniuUploader

# 配置模板的路徑
TEMPLATES = [
    {
       'DIRS': [os.path.join(BASE_DIR,'templates')],
    },
]
  2、 配置靜態目錄

# 像ccs和js這些靜態文件如果想要使用必須在這里配置路徑
STATICFILES_DIRS = (
    os.path.join(BASE_DIR,'static'),
)
settings.py
from django.contrib import admin
from django.urls import path
from app01 import views

urlpatterns = [
    path('admin/', admin.site.urls),
    path('uptoken/', views.QNYTokenView.as_view()),
    path('upload/', views.UploadView.as_view()),
    path('vod/', views.VODView.as_view()),
]
urls.py 定義總路由
from django.shortcuts import render,HttpResponse
from django.views import View
import json

'''獲取上傳token'''
class QNYTokenView(View):
    def get(self,request):
        from qiniu import Auth, put_file, etag
        import qiniu.config
        # 需要填寫你的 Access Key 和 Secret Key
        access_key = "PwyTqrclbus4ntRct1o8G2V-qkR1rI7hbd_5Gx29"
        secret_key = "IuvSm1vJh2YUiYWFwV-kGmHAJF9R9iGuH2Q1ifea"
        # 構建鑒權對象
        q = Auth(access_key, secret_key)
        # 要上傳的空間
        bucket_name = 'imagesstatic'
        # 生成上傳 Token,可以指定過期時間等
        token = q.upload_token(bucket_name, expires=3600)
        return HttpResponse(json.dumps({'uptoken': token}, ensure_ascii=False))

'''上傳頁面'''
class UploadView(View):
    def get(self,request):
        return render(request,'upload.html')

'''播放頁面'''
class VODView(View):
    def get(self,request):
        return render(request,'vod.html')
app01/views.py
<!doctype html>
<html lang="zh">
<head>
    <meta charset="UTF-8">
    <meta name="viewport"
          content="width=device-width, user-scalable=no, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0">
    <meta http-equiv="X-UA-Compatible" content="ie=edge">
    <title>js上傳文件到七牛</title>
    <style>
        #container{
            width:200px;
            height:200px;
            border:1px solid #9d9d9d;
            border-radius: 6px;
            margin:50px auto;
            position: relative;
            overflow: hidden;
        }
        .upload-progress{
            width:100%;
            height:100%;
            position: absolute;
            top:0;
            left:0;
            background: rgba(0,0,0,0.5);
            z-index: 5;
            color:#fff;
            line-height: 200px;
            text-align: center;
            display: none;
        }
        #uploadImage{
            width:100%;
            height:100%;
            position: absolute;
            top:0;
            left:0;
            z-index: 2;
            text-align: center;
            line-height: 200px;
            cursor: pointer;
        }
        #container img{
            width:100%;
            position: absolute;
            top:0;
            left:0;
            z-index: 1;
        }
    </style>
</head>
<body>
<div id="container">
    <div id="uploadImage">選擇文件</div>
    <div class="upload-progress"></div>
</div>
<script src="/static/qiniu.min.js"></script>
<script src="/static/jquery-1.12.1.min.js"></script>
<script>
    var uploader = Qiniu.uploader({
        disable_statistics_report: false,                                   // 禁止自動發送上傳統計信息到七牛,默認允許發送
        runtimes: 'html5,flash,html4',                                      // 上傳模式,依次退化
        browse_button: 'uploadImage',                                       // 上傳選擇的點選按鈕,必需
        container: 'container',                                             // 上傳區域DOM ID,默認是browser_button的父元素
        max_file_size: '500mb',                                             // 最大文件體積限制
        flash_swf_url: 'Moxie.swf',                                         // 引入flash,相對路徑
        dragdrop: false,                                                    // 關閉可拖曳上傳
        chunk_size: '4mb',                                                  // 分塊上傳時,每塊的體積
        multi_selection: !(moxie.core.utils.Env.OS.toLowerCase() === "ios"),
        uptoken_url: 'http://127.0.0.1:8000/uptoken',                                                 // 在初始化時,uptoken,uptoken_url,uptoken_func三個參數中必須有一個被設置,uptoken是上傳憑證,由其他程序生成;uptoken_url是提供了獲取上傳憑證的地址,如果需要定制獲取uptoken的過程則可以設置uptoken_func;其優先級為uptoken > uptoken_url > uptoken_func
        //uptoken:'q06bq54Ps5JLfZyP8Ax-qvByMBdu8AoIVJpMco2m:kyTiuN6GBUpfNt1nJIA7C8CCStY=:eyJzY29wZSI6IjEzMTIzMTIzMTIzIiwiZGVhZGxpbmUiOjE1NzY0MTM3MTB9',
        domain: 'redinnovation.s3-cn-north-1.qiniucs.com',               // bucket域名,下載資源時用到,必需
        get_new_uptoken: false,                                              // 設置上傳文件的時候是否每次都重新獲取新的uptoken
        auto_start: true,                                                   // 選擇文件后自動上傳,若關閉需要自己綁定事件觸發上傳
        max_retries: 3,                                                     // 上傳失敗最大重試次數
        save_key: true,
        resize: {                                                           // 想限制上傳圖片尺寸,直接用resize這個屬性
            width: 300,
            height: 300
        },
        init: {
            'FilesAdded': function(up, files) {                             // 文件添加進隊列后,處理相關的事情
                plupload.each(files, function(file) {
                    console.log(file)
                });
            },
            'BeforeUpload': function(up, file) {                            // 每個文件上傳前,處理相關的事情
                console.log("開始上傳之前");
                $(".upload-progress").show();
            },
            'UploadProgress': function(up, file) {                          // 每個文件上傳時,處理相關的事情
                console.log("上傳中");
                $(".upload-progress").html("上傳進度:"+file.percent + "%");
            },
            'FileUploaded': function(up, file, info) {                       // 每個文件上傳成功后,處理相關的事情
                console.log("上傳成功");
                console.log(info);
                //$(".upload-progress").hide();
                //var img = new Image();                                      //創建一個Image對象,實現圖片的預下載
                //img.src = "http://qiniu.com/"+res.key;
                //$("#container").append(img);
            },
            'Error': function(up, err, errTip) {
                console.log("上傳出錯")
            },
            'UploadComplete': function() {
                //隊列文件處理完畢后,處理相關的事情
            }
        }
    });
</script>
</body>
</html>
templates/upload.html 上傳視頻
<!doctype html>
<html lang="zh">
<head>
    <meta charset="UTF-8">
    <meta name="viewport"
          content="width=device-width, user-scalable=no, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0">
    <meta http-equiv="X-UA-Compatible" content="ie=edge">
    <title>xxx</title>
    <style></style>
</head>
<body>
    <div>
        <video id="video"  width="320" height="240" src="http://q5gq3qy79.bkt.clouddn.com/lucclaPl4-Hi4LcepzKNVWpfN4u7" controls="controls" autoplay="autoplay" muted loop="loop" >
                    您的瀏覽器不支持 video 標簽。
         </video>
    </div>
</body>
</html>
templates/vod.html 播放視頻

     

 

 1.3 vue+django上傳視頻

     參考代碼:https://gitee.com/edushiyanlou/QiniuUploader

  1、前端Vue代碼

vue init webpack qiniuVue              # 初始化一個vue前端項目
npm install --save axios               # 安裝axios
npm install  --save  jquery@1.12.1     # 安裝jquery
<!DOCTYPE html>
<html>
  <head>
    <meta charset="utf-8">
    <meta name="viewport" content="width=device-width,initial-scale=1.0">
    <title>qiniu_vue</title>
    <script src="./static/qiniu.min.js"></script>
  </head>
  <body>
    <div id="app"></div>
    <!-- built files will be auto injected -->
  </body>
</html>
index.html 引入qiniu.min.js
import Vue from 'vue'
import Router from 'vue-router'
import HelloWorld from '@/components/HelloWorld'
import Upload from '@/components/Upload'
import Vod from '@/components/Vod'

Vue.use(Router)

export default new Router({
  routes: [
    { path: '/', name: 'HelloWorld', component: HelloWorld },
    { path: '/upload', name: 'Upload', component: Upload },
    { path: '/vod', name: 'Vod', component: Vod },
  ]
})
src/router/index.js
<template>
  <div id="container">
    <div>
      <div id="uploadImage">選擇文件</div>
      <div class="upload-progress"></div>
    </div>
  </div>
</template>

<script>
  import $ from 'jquery'
  import axios from 'axios'
  export default {
    data() {
      return {
        uptoken: '',
      };
    },
    methods: {
      getQiniuToken: function (callback){
        axios({
          url: 'http://127.0.0.1:8000/uptoken',
          method: 'get',
        }).then( ret =>{
          this.uptoken = ret.data.uptoken;
          callback()  // callback 出入的是回調函數 initQiniu() 用來初始化Qiniu.uploader()
        }).catch( err=>{
          console.log(err,'err')
        })
      },

      initQiniu: function () {
        var uploader = Qiniu.uploader({
          disable_statistics_report: false,                                   // 禁止自動發送上傳統計信息到七牛,默認允許發送
          runtimes: 'html5,flash,html4',                                      // 上傳模式,依次退化
          browse_button: 'uploadImage',                                       // 上傳選擇的點選按鈕,必需
          container: 'container',                                             // 上傳區域DOM ID,默認是browser_button的父元素
          max_file_size: '500mb',                                             // 最大文件體積限制
          flash_swf_url: 'Moxie.swf',                                         // 引入flash,相對路徑
          dragdrop: false,                                                    // 關閉可拖曳上傳
          chunk_size: '4mb',                                                  // 分塊上傳時,每塊的體積
          multi_selection: !(moxie.core.utils.Env.OS.toLowerCase() === "ios"),
          uptoken: this.uptoken,     // 在初始化時,uptoken,uptoken_url,uptoken_func三個參數中必須有一個被設置,uptoken是上傳憑證,由其他程序生成;uptoken_url是提供了獲取上傳憑證的地址,如果需要定制獲取uptoken的過程則可以設置uptoken_func;其優先級為uptoken > uptoken_url > uptoken_func
          // uptoken_url: 'http://127.0.0.1:8000/uptoken',                                                 // 在初始化時,uptoken,uptoken_url,uptoken_func三個參數中必須有一個被設置,uptoken是上傳憑證,由其他程序生成;uptoken_url是提供了獲取上傳憑證的地址,如果需要定制獲取uptoken的過程則可以設置uptoken_func;其優先級為uptoken > uptoken_url > uptoken_func
          // uptoken:'q06bq54Ps5JLfZyP8Ax-qvByMBdu8AoIVJpMco2m:kyTiuN6GBUpfNt1nJIA7C8CCStY=:eyJzY29wZSI6IjEzMTIzMTIzMTIzIiwiZGVhZGxpbmUiOjE1NzY0MTM3MTB9',
          domain: 'redinnovation.s3-cn-north-1.qiniucs.com',               // bucket域名,下載資源時用到,必需
          get_new_uptoken: false,                                              // 設置上傳文件的時候是否每次都重新獲取新的uptoken
          auto_start: true,                                                   // 選擇文件后自動上傳,若關閉需要自己綁定事件觸發上傳
          max_retries: 3,                                                     // 上傳失敗最大重試次數
          save_key: true,
          resize: {                                                           // 想限制上傳圖片尺寸,直接用resize這個屬性
            width: 300,
            height: 300
          },
          init: {
            'FilesAdded': function(up, files) {                             // 文件添加進隊列后,處理相關的事情
              plupload.each(files, function(file) {
                console.log(file)
              });
            },
            'BeforeUpload': function(up, file) {                            // 每個文件上傳前,處理相關的事情
              console.log("開始上傳之前");
              $(".upload-progress").show();
            },
            'UploadProgress': function(up, file) {                          // 每個文件上傳時,處理相關的事情
              console.log("上傳中");
              $(".upload-progress").html("上傳進度:"+file.percent + "%");
            },
            'FileUploaded': function(up, file, info) {                       // 每個文件上傳成功后,處理相關的事情
              console.log("上傳成功");
              console.log(info);
              //$(".upload-progress").hide();
              //var img = new Image();                                      //創建一個Image對象,實現圖片的預下載
              //img.src = "http://qiniu.com/"+res.key;
              //$("#container").append(img);
            },
            'Error': function(up, err, errTip) {
              console.log("上傳出錯")
            },
            'UploadComplete': function() {
              //隊列文件處理完畢后,處理相關的事情
            }
          }
        });
      }
    },
    mounted(){
      this.getQiniuToken(() => {
        this.initQiniu()  // 將initQiniu()當做callback回調函數傳入給getQiniuToken函數
      })
    }
  }
</script>

<style>
  #container{
    width:200px;
    height:200px;
    border:1px solid #9d9d9d;
    border-radius: 6px;
    margin:50px auto;
    position: relative;
    overflow: hidden;
  }
  .upload-progress{
    width:100%;
    height:100%;
    position: absolute;
    top:0;
    left:0;
    background: rgba(0,0,0,0.5);
    z-index: 5;
    color:#fff;
    line-height: 200px;
    text-align: center;
    display: none;
  }
  #uploadImage{
    width:100%;
    height:100%;
    position: absolute;
    top:0;
    left:0;
    z-index: 2;
    text-align: center;
    line-height: 200px;
    cursor: pointer;
  }
  #container img{
    width:100%;
    position: absolute;
    top:0;
    left:0;
    z-index: 1;
  }
</style>
src/components/upload.vue
<template>
  <div>
    <div>
      <video id="video"  width="320" height="240" src="http://q5gq3qy79.bkt.clouddn.com/lucclaPl4-Hi4LcepzKNVWpfN4u7" controls="controls" autoplay="autoplay" muted loop="loop" >
        您的瀏覽器不支持 video 標簽。
      </video>
    </div>
  </div>
</template>

<script>

</script>

<style>

</style>
src/components/vod.vue

  2、django代碼

INSTALLED_APPS = [

    'corsheaders',
]


'''配置模板的路徑'''
TEMPLATES = [
    {
       'DIRS': [os.path.join(BASE_DIR,'templates')],
    },
]
  2、 配置靜態目錄


'''像ccs和js這些靜態文件如果想要使用必須在這里配置路徑'''
STATICFILES_DIRS = (
    os.path.join(BASE_DIR,'static'),
)

#

'''配置cors'''
CORS_ALLOW_CREDENTIALS = True
CORS_ORIGIN_ALLOW_ALL = True
settings.py
from django.contrib import admin
from django.urls import path
from app01 import views

urlpatterns = [
    path('admin/', admin.site.urls),
    path('uptoken/', views.QNYTokenView.as_view()),
    path('upload/', views.UploadView.as_view()),
    path('vod/', views.VODView.as_view()),
]
urls.py 定義總路由
from django.shortcuts import render,HttpResponse
from django.views import View
import json

'''獲取上傳token'''
class QNYTokenView(View):
    def get(self,request):
        from qiniu import Auth, put_file, etag
        import qiniu.config
        # 需要填寫你的 Access Key 和 Secret Key
        access_key = "PwyTqrclbus4ntRct1o8G2V-qkR1rI7hbd_5Gx29"
        secret_key = "IuvSm1vJh2YUiYWFwV-kGmHAJF9R9iGuH2Q1ifea"
        # 構建鑒權對象
        q = Auth(access_key, secret_key)
        # 要上傳的空間
        bucket_name = 'imagesstatic'
        # 生成上傳 Token,可以指定過期時間等
        token = q.upload_token(bucket_name, expires=3600)
        return HttpResponse(json.dumps({'uptoken': token}, ensure_ascii=False))

'''上傳頁面'''
class UploadView(View):
    def get(self,request):
        return render(request,'upload.html')

'''播放頁面'''
class VODView(View):
    def get(self,request):
        return render(request,'vod.html')
app01/views.py
<!doctype html>
<html lang="zh">
<head>
    <meta charset="UTF-8">
    <meta name="viewport"
          content="width=device-width, user-scalable=no, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0">
    <meta http-equiv="X-UA-Compatible" content="ie=edge">
    <title>js上傳文件到七牛</title>
    <style>
        #container{
            width:200px;
            height:200px;
            border:1px solid #9d9d9d;
            border-radius: 6px;
            margin:50px auto;
            position: relative;
            overflow: hidden;
        }
        .upload-progress{
            width:100%;
            height:100%;
            position: absolute;
            top:0;
            left:0;
            background: rgba(0,0,0,0.5);
            z-index: 5;
            color:#fff;
            line-height: 200px;
            text-align: center;
            display: none;
        }
        #uploadImage{
            width:100%;
            height:100%;
            position: absolute;
            top:0;
            left:0;
            z-index: 2;
            text-align: center;
            line-height: 200px;
            cursor: pointer;
        }
        #container img{
            width:100%;
            position: absolute;
            top:0;
            left:0;
            z-index: 1;
        }
    </style>
</head>
<body>
<div id="container">
    <div id="uploadImage">選擇文件</div>
    <div class="upload-progress"></div>
</div>
<script src="/static/qiniu.min.js"></script>
<script src="/static/jquery-1.12.1.min.js"></script>
<script>
    var uploader = Qiniu.uploader({
        disable_statistics_report: false,                                   // 禁止自動發送上傳統計信息到七牛,默認允許發送
        runtimes: 'html5,flash,html4',                                      // 上傳模式,依次退化
        browse_button: 'uploadImage',                                       // 上傳選擇的點選按鈕,必需
        container: 'container',                                             // 上傳區域DOM ID,默認是browser_button的父元素
        max_file_size: '500mb',                                             // 最大文件體積限制
        flash_swf_url: 'Moxie.swf',                                         // 引入flash,相對路徑
        dragdrop: false,                                                    // 關閉可拖曳上傳
        chunk_size: '4mb',                                                  // 分塊上傳時,每塊的體積
        multi_selection: !(moxie.core.utils.Env.OS.toLowerCase() === "ios"),
        uptoken_url: 'http://127.0.0.1:8000/uptoken',                                                 // 在初始化時,uptoken,uptoken_url,uptoken_func三個參數中必須有一個被設置,uptoken是上傳憑證,由其他程序生成;uptoken_url是提供了獲取上傳憑證的地址,如果需要定制獲取uptoken的過程則可以設置uptoken_func;其優先級為uptoken > uptoken_url > uptoken_func
        //uptoken:'q06bq54Ps5JLfZyP8Ax-qvByMBdu8AoIVJpMco2m:kyTiuN6GBUpfNt1nJIA7C8CCStY=:eyJzY29wZSI6IjEzMTIzMTIzMTIzIiwiZGVhZGxpbmUiOjE1NzY0MTM3MTB9',
        domain: 'redinnovation.s3-cn-north-1.qiniucs.com',               // bucket域名,下載資源時用到,必需
        get_new_uptoken: false,                                              // 設置上傳文件的時候是否每次都重新獲取新的uptoken
        auto_start: true,                                                   // 選擇文件后自動上傳,若關閉需要自己綁定事件觸發上傳
        max_retries: 3,                                                     // 上傳失敗最大重試次數
        save_key: true,
        resize: {                                                           // 想限制上傳圖片尺寸,直接用resize這個屬性
            width: 300,
            height: 300
        },
        init: {
            'FilesAdded': function(up, files) {                             // 文件添加進隊列后,處理相關的事情
                plupload.each(files, function(file) {
                    console.log(file)
                });
            },
            'BeforeUpload': function(up, file) {                            // 每個文件上傳前,處理相關的事情
                console.log("開始上傳之前");
                $(".upload-progress").show();
            },
            'UploadProgress': function(up, file) {                          // 每個文件上傳時,處理相關的事情
                console.log("上傳中");
                $(".upload-progress").html("上傳進度:"+file.percent + "%");
            },
            'FileUploaded': function(up, file, info) {                       // 每個文件上傳成功后,處理相關的事情
                console.log("上傳成功");
                console.log(info);
                //$(".upload-progress").hide();
                //var img = new Image();                                      //創建一個Image對象,實現圖片的預下載
                //img.src = "http://qiniu.com/"+res.key;
                //$("#container").append(img);
            },
            'Error': function(up, err, errTip) {
                console.log("上傳出錯")
            },
            'UploadComplete': function() {
                //隊列文件處理完畢后,處理相關的事情
            }
        }
    });
</script>
</body>
</html>
templates/upload.html 上傳視頻
<!doctype html>
<html lang="zh">
<head>
    <meta charset="UTF-8">
    <meta name="viewport"
          content="width=device-width, user-scalable=no, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0">
    <meta http-equiv="X-UA-Compatible" content="ie=edge">
    <title>xxx</title>
    <style></style>
</head>
<body>
    <div>
        <video id="video"  width="320" height="240" src="http://q5gq3qy79.bkt.clouddn.com/lucclaPl4-Hi4LcepzKNVWpfN4u7" controls="controls" autoplay="autoplay" muted loop="loop" >
                    您的瀏覽器不支持 video 標簽。
         </video>
    </div>
</body>
</html>
templates/vod.html 播放視頻

  

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM