1、上傳視頻javascript
1:使用百度雲:html
2:自定義轉碼設置:前端
一、視頻需通過加密在進行發佈,先建立一個編碼模板(配置以下):java
二、進入 媒資管理 進行視頻上傳,選擇 自定義轉碼模板組:ajax
二:視頻的播放後端
一、後端代碼api
1 # 獲取tokon; 2 def course_token(request): 3 # video:是視頻文件的完整連接 4 file = request.GET.get('video') 5 # 過時時間; 6 expiration_time = int(time.time()) + 2 * 60 * 60 7 8 USER_ID = settings.BAIDU_CLOUD_USER_ID 9 USER_KEY = settings.BAIDU_CLOUD_USER_KEY 10 11 # file=http://hemvpc6ui1kef2g0dd2.exp.bcevod.com/mda-igjsr8g7z7zqwnav/mda-igjsr8g7z7zqwnav.m3u8 12 # 先獲取擴展名;再經過‘/’分割取最後一個值,並將擴展名替換爲空字符串;獲得‘ID’; 13 extension = os.path.splitext(file)[1] 14 media_id = file.split('/')[-1].replace(extension, '') 15 16 # 將‘USER_KEY’進行編碼;後面的‘hmac.new()’只能接受bytes類型; 17 # unicode->bytes=unicode.encode('utf-8')bytes 18 key = USER_KEY.encode('utf-8') 19 message = '/{0}/{1}'.format(media_id, expiration_time).encode('utf-8') 20 21 # signature:生成簽名; 22 # disgestmod:指定加密方式; 23 signature = hmac.new(key, message, digestmod=hashlib.sha256).hexdigest() 24 token = '{0}_{1}_{2}'.format(signature, USER_ID, expiration_time) 25 return restful.result(data={'token': token})
(url 映射地址:cms/course_token)安全
二、前端代碼微信
1 <!-- HTML中需引用的文件 --> 2 <script src="{% static 'videojs/video.min.js' %}"></script> 3 <script src="{% static 'videojs/videojs-contrib-hls.min.js' %}"></script> 4 <script src="{% static 'videojs/videojs-contrib-quality-levels.min.js' %}"></script> 5 6 <!-- 加載播放器 --> 7 <script type="text/javascript" src="https://cdn.bdstatic.com/jwplayer/latest/cyberplayer.js"></script> 8 9 <!-- 初始化播放器的 js 文件 --> 10 <script src="{% static 'js/cms/course_detail.min.js' %}"></script>
1 <div class="video-group"> 2 <span id="video-info" hidden data-video-url="{{ course.video_url }}" 3 data-cover-url="{{ course.cover_url }}"></span> 4 <!-- 用來存儲播放器 --> 5 <div id="playercontainer"></div> 6 </div>
1 function CourseDetail() { 2 3 } 4 5 // 視頻播放; 6 CourseDetail.prototype.initPlayer = function () { 7 var videoInfoSpan = $('#video-info'); 8 var video_url = videoInfoSpan.attr('data-video-url'); 9 var cover_url = videoInfoSpan.attr('data-cover-url'); 10 var player = cyberplayer("playercontainer").setup({ 11 width: '100%', 12 height: '100%', 13 file: video_url, // 視頻連接; 14 image: cover_url, // 封面圖連接; 15 autostart: false, // 是否自動播放; 16 stretching: 'uniform', // 擴大; 17 repeat: false, // 是否重複; 18 volume: 100, // 音量; 19 controls: true, // 底部控制欄; 20 primary: "flash", // 使用flash; 21 tokenEncrypt: true, // 採用token加密; 22 ak: '86dc62e1dbd4455080ab1cfc5e587b17', // AccessKey; 23 }); 24 25 // 視頻播放前的事件; 26 player.on('beforePlay', function (e) { 27 if (!/m3u8/.test(e.file)) { 28 return; 29 } 30 xfzajax.get({ 31 'url': '/course/course_token/', 32 'data': { 33 'video': video_url 34 }, 35 'success': function (result) { 36 if (result['code'] === 200) { 37 var token = result['data']['token']; 38 // 將‘tokon’設置入‘player’中,使播放器獲取視頻密碼; 39 player.setToken(e.file, token); 40 } else { 41 alert('token錯誤!') 42 } 43 }, 44 'fail': function (error) { 45 console.log(error) 46 } 47 }) 48 }) 49 }; 50 51 CourseDetail.prototype.run = function () { 52 this.initPlayer(); 53 }; 54 55 $(function () { 56 var course = new CourseDetail(); 57 course.run() 58 });
3、購買視頻及播放;restful
一、使用 PaysApi
二、代碼
1 # 支付頁面模板; 2 class CourseOrder(models.Model): 3 uid = ShortUUIDField(primary_key=True) 4 course = models.ForeignKey("Course",on_delete=models.DO_NOTHING) 5 buyer = models.ForeignKey("xfzauth.User",on_delete=models.DO_NOTHING) 6 amount = models.FloatField(default=0) 7 pub_time = models.DateTimeField(auto_now_add=True) 8 # 支付渠道:1:表明支付寶支付;2:表明微信支付; 9 istype = models.SmallIntegerField(default=1) 10 # 支付狀態:1:表明未支付;2:表明支付成功; 11 status = models.SmallIntegerField(default=1)
1 <!-- 部分相關HTML模板 --> 2 <form action="https://pay.bbbapi.com/" method="post" id="pay-form"> 3 <input type="hidden" name="uid" value="49dc532695baa99e16e01bc0"> 4 <input type="hidden" name="price" value="{{ course.price }}"> 5 <input type="hidden" name="notify_url" value="{{ notify_url }}"> 6 <input type="hidden" name="return_url" value="{{ return_url }}"> 7 <input type="hidden" name="orderid" value="{{ order.pk }}"> 8 <input type="hidden" name="orderuid" value="{{ request.user.pk }}"> 9 <input type="hidden" name="goodsname" value="{{ course.title }}"> 10 <input type="hidden" name="key" value=""> 11 <!-- 微信與支付寶支付 --> 12 <div class="pay-way"> 13 <label for="istype-wx" class="label"> 14 <input id="istype-wx" type="radio" name="istype" value="2"> 15 <span class="wx-btn fk-btn"> 16 <img src="http://nos.netease.com/test-edu-image/1BD9F69D6760CE1508D2269864AA54F8.png" alt=""> 17 </span> 18 </label> 19 <label for="istype-zfb" class="label"> 20 <input id="istype-zfb" type="radio" name="istype" value="1" checked> 21 <span class="zfb-btn fk-btn"></span> 22 </label> 23 <div style="clear: both;"></div> 24 </div> 25 <div class="submit-group"> 26 <input type="submit" value="去支付" id="submit-btn" class="submit-btn"> 27 </div> 28 </form>
1 from utils import restful 2 from apps.xfzauth.decorators import xfz_login_required 3 from hashlib import md5 4 5 # 獲取「key」;用戶登陸的狀況下; 6 @xfz_login_required 7 def course_order_key(request): 8 goodsname = request.POST.get("goodsname") # 商品名稱; 9 istype = request.POST.get("istype") # 支付渠道; 10 notify_url = request.POST.get("notify_url") # 通知回調網址; 11 orderid = request.POST.get("orderid") # 商戶自定義訂單號; 12 orderuid = str(request.user.pk) # 商戶自定義客戶號; 13 price = request.POST.get("price") # 價格; 14 return_url = request.POST.get("return_url") # 跳轉網址; 15 16 # 在官網我的帳戶設置中的「API接口信息」中獲取「token」與「uid」的值: 17 token = 'e6110f92abcb11040ba153967847b7a6' 18 uid = '49dc532695baa99e16e01bc0' 19 20 # 祕鑰「key」的拼接順序:goodsname + istype + notify_url + orderid + orderuid + price + return_url + token + uid 21 key = md5((goodsname + istype + notify_url + orderid + orderuid + price + return_url + token + uid).encode( 22 "utf-8")).hexdigest() 23 return restful.result(data={"key": key})
1 # 購買課程; 2 @xfz_login_required 3 def course_order(request, course_id): 4 course = Course.objects.get(pk=course_id) 5 order = CourseOrder.objects.create(course=course, buyer=request.user, status=1, amount=course.price) 6 context = { 7 'course': course, 8 'order': order, 9 # 跳轉連接:/course/notify_url/; 10 'notify_url': request.build_absolute_uri(reverse('course:notify_view')), 11 'return_url': request.build_absolute_uri(reverse('course:course_detail', kwargs={"course_id": course.pk})) 12 } 13 return render(request, 'course/course_order.html', context=context) 14 15 # 使用訂單ID更新支付狀態;(關閉CSRF保護) 16 @csrf_exempt 17 def notify_view(request): 18 orderid = request.POST.get('orderid') 19 CourseOrder.objects.filter(pk=orderid).update(status=2) 20 return restful.ok()
1 function CourseOrder() { 2 3 } 4 5 CourseOrder.prototype.run = function () { 6 this.OrderEvent(); 7 }; 8 9 // 訂購事件;(發起付款接口時需處理的數據) 10 CourseOrder.prototype.OrderEvent = (function () { 11 var submitBtn = $("#submit-btn"); 12 submitBtn.click(function (event) { 13 event.preventDefault(); 14 var goodsname = $("input[name='goodsname']").val(); 15 var istype = $("input[name='istype']:checked").val(); 16 var notify_url = $("input[name='notify_url']").val(); 17 var orderid = $("input[name='orderid']").val(); 18 var price = $("input[name='price']").val(); 19 var return_url = $("input[name='return_url']").val(); 20 xfzajax.post({ 21 'url': '/course/course_order_key/', 22 'data': { 23 'goodsname': goodsname, 24 'istype': istype, 25 'notify_url': notify_url, 26 'orderid': orderid, 27 'price': price, 28 'return_url': return_url 29 }, 30 'success': function (result) { 31 if(result['code'] === 200){ 32 var key = result['data']['key']; 33 var keyInput = $("input[name='key']"); 34 keyInput.val(key); 35 $("#pay-form").submit(); 36 } 37 } 38 }); 39 }); 40 }); 41 42 $(function () { 43 var order = new CourseOrder(); 44 order.run(); 45 });