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'), )

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()), ]

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')

<!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>

<!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>
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>

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 }, ] })

<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>

<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>
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

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()), ]

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')

<!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>

<!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>