摘要:上傳結(jié)構(gòu)與網(wǎng)宿云要求上傳結(jié)構(gòu)的不同上圖是翻自網(wǎng)宿云的文檔的分片上傳流程。鑒于網(wǎng)宿云的上傳一片一塊在邏輯上沒(méi)毛病,我們同樣能一塊一塊完成上傳這里注意請(qǐng)仔細(xì)看網(wǎng)宿云或七牛云分片上傳的文檔,了解如何分片上傳。
webuploader踩坑
webuploader是百度f(wàn)ex團(tuán)隊(duì)開(kāi)發(fā)的一個(gè)十分便捷的上傳插件,但是我們?cè)趯?shí)際生產(chǎn)中,會(huì)發(fā)現(xiàn)使用它與我們的需求有各種各樣的出入。最近做上傳功能,踩了不少坑,現(xiàn)在來(lái)記錄一下。如果我的文章中有任何不妥或者不對(duì)的地方,歡迎指正。
webuploader上傳結(jié)構(gòu)與網(wǎng)宿云要求上傳結(jié)構(gòu)的不同上圖是翻自網(wǎng)宿云的文檔的分片上傳流程。
通過(guò)該圖,我們可知網(wǎng)宿云組織上傳文件形式是
{文件[塊1(分片1,分片2,分片3,…),塊2,塊3,…]}
而webuploader對(duì)文件分片的形式如下
{文件[塊1(分片1),塊2(分片1),塊3(分片1),…]}
即一塊即是一片。鑒于網(wǎng)宿云的上傳一片一塊在邏輯上沒(méi)毛病,我們同樣能一塊一塊完成上傳
這里注意,請(qǐng)仔細(xì)看網(wǎng)宿云或七牛云分片上傳的文檔,了解如何分片上傳。其中一個(gè)很重要的概念是塊,片上下文,即ctx,請(qǐng)前往查看
webuploader上傳流程上與需求不符合的原因我們先來(lái)看webuploader一個(gè)文件上傳流程中,觸發(fā)的鉤子和事件
一個(gè)文件的上傳只觸發(fā)三個(gè)實(shí)際使用的鉤子
1. before-send-file 上傳文件前 2. before-send 上傳塊前 3. after-send-file 上傳文件結(jié)束
觸發(fā)多個(gè)事件
1. uploadStart 開(kāi)始上傳前 2. uploadAccept 驗(yàn)證上傳是否合法的事件,取ctx只能在這一步進(jìn)行,比較悲慘 3. uploadBeforeSend 上傳文件前,對(duì)應(yīng)before-send-file 4. uploadProgress 文件上傳進(jìn)度事件 5. uploadSkip 跳過(guò)當(dāng)前文件上傳事件,當(dāng)出現(xiàn)該事件,uploader內(nèi)部標(biāo)記該文件已經(jīng)上傳成功 6. stopUpload 暫停當(dāng)前文件上傳時(shí)觸發(fā) 7. startUpload 恢復(fù)上傳當(dāng)前文件觸發(fā),或開(kāi)始上傳也會(huì)觸發(fā) 8. uploadSuccess 文件上傳成功觸發(fā) 9. uploadError 文件上傳失敗觸發(fā)
通過(guò)比對(duì)網(wǎng)宿云的分片上傳流程,我們會(huì)發(fā)現(xiàn)他遠(yuǎn)遠(yuǎn)不滿(mǎn)足我們當(dāng)下需求,缺少上傳分片前的鉤子,缺少上傳分片后的鉤子,這是不同的分片姿勢(shì)決定的,目前來(lái)說(shuō)除非我們自己修改widgets/upload模塊,要不沒(méi)什么好的方式解決他
所以下面是修改該模塊的內(nèi)容
// 負(fù)責(zé)將文件切片。 function CuteFile( file, chunkSize ) { ... // 七牛云,網(wǎng)宿云規(guī)定的最大的塊的大小,chunkSize不能大于它 var blockSize = 4 * 1024 * 1024 while ( index < chunks ) { len = Math.min( chunkSize, total - start ); let block = { file: file, start: start, end: chunkSize ? (start + len) : total, total: total, chunks: chunks, chunk: index, cuted: api } // 增加塊id block.blockIndex = Math.floor(block.start / blockSize); // 增加塊內(nèi)片偏移量標(biāo)識(shí) block.offset = block.start % blockSize; // 增加塊內(nèi)最后一片標(biāo)識(shí)(網(wǎng)宿云要求在組合文件的時(shí)候,需要用每塊最后一片上傳成功的ctx作為參數(shù)來(lái)組合文件) block.lastChunk = block.end % blockSize === 0 || block.end === total; if (block.start % blockSize === 0) { // 增加塊頭標(biāo)識(shí) block.mkblk = true; // 計(jì)算總塊數(shù) let blocks = Math.ceil( total / opts.blockSize ); // 增加塊大小標(biāo)識(shí) block.size = (block.blockIndex + 1) === blocks ? (total - block.start) : blockSize; } pending.push(block); index++; start += len; } file.blocks = pending.concat(); file.remaning = pending.length; return api; }
這樣改過(guò)后有一個(gè)毛病,那就是由于片上傳是順序上傳,片上傳是無(wú)法并發(fā)的~這樣改的結(jié)果就是,一個(gè)文件只能順序上傳所有片了。。~本修改只是一個(gè)示例,如果真的要完全支持塊并發(fā),片順序上傳,必須要修改block的結(jié)構(gòu),讓block存儲(chǔ)該塊中所有片內(nèi)容。其結(jié)構(gòu)應(yīng)該是
block: { ... file: 父節(jié)點(diǎn)的引用 cutes: [ 片1, 片2, 片3 ], percents: x, remaning: cutes.length }
除此之外,把實(shí)施上傳的主體變更為片,并實(shí)現(xiàn)或觸發(fā)一些支持分片上傳的自定義事件,這樣就可以以塊為單位,并發(fā)上傳,塊中片順序上傳了。
上傳過(guò)程中,鉤子執(zhí)行的方式和修改上傳配置所帶來(lái)的困擾通過(guò)網(wǎng)上大量的例子,如下:
uploader.register({ "before-send-file": "bsf", "before-send": "bbs", "after-send-file": "afs" }, { "bsf": function () { ... }, "bbs": function (block) { var server = ""; var D = webUploader.Deferred() if (block.chunk === 1) { uploader.options.server = "xxxx" } else { uploader.options.server = "xxxxx" } setTimeout(function () { D.resolve() }, 200) return D.promise() }, "afs": function () { ... } })
從例子看,似乎webuploader只有一個(gè)通用的options來(lái)配置服務(wù)器地址,formData, headers信息等,由于before-send-file, before-send, after-send-file三個(gè)鉤子是異步執(zhí)行的,所以在并發(fā)上傳時(shí),修改分片上傳或mkblk操作所需的服務(wù)配置可能會(huì)給我們帶來(lái)困擾。按照這個(gè)思路,一個(gè)解決方案是實(shí)現(xiàn)一個(gè)uploadTaskManager,使用worker來(lái)進(jìn)行多實(shí)例并發(fā)上傳操作。
然而近期,通過(guò)讀webuploader/widgets/upload.js的源代碼,我們發(fā)現(xiàn)以下內(nèi)容:
_doSend: function( block ) { var me = this, owner = me.owner, // 可喜可賀 opts = $.extend({}, me.options, block.options), file = block.file, tr = new Transport( opts ), data = $.extend({}, opts.formData ), headers = $.extend({}, opts.headers ), requestAccept, ret; ...
可喜可賀,我們完全可以通過(guò)直接給block增加options來(lái)保證before-send鉤子執(zhí)行時(shí)不擾亂整體options配置
// appendWidget不用管,是我添加用于追加注冊(cè)一個(gè)掛件的方法。 // 由于register方法是在webuploader實(shí)例化的時(shí)候才將注冊(cè)的掛件掛載上,所以才有了這個(gè)方法 this.$uploader.appendWidget({ "before-send-file": "bsf", "before-send": "bbs", "after-send-file": "afs", "name": "progress" }, { bsf: (file) => { // 這個(gè)也不用管,是我為vue增加的插件,每次響應(yīng)get操作都返回一個(gè)webuploader.Deferred() let deferred = this.$deferred // 為webuploader增加的sha1hash計(jì)算方法 this.$uploader.sha1File(file) .progress((e) => { // console.log(file.name, e) }) .then((sha1Hash) => { file.sha1Hash = sha1Hash api.path.upload({ name: file.name, pid: file.pid, hash: file.sha1Hash }) .then((res) => { let data = res.body if (data.msg === "file already exists") { this.$uploader.skipFile(file) } else { file.token = data.token file.server = data.url } deferred.resolve() }) }) return deferred.promise() }, bbs: (block) => { let deferred = this.$deferred if (!block.options) { let file = block.file // 直接設(shè)置options來(lái)達(dá)到修改server,headers配置的目的 block.options = { headers: { "Content-Type": "application/octet-stream", "Authorization": file.token, "UploadBatch": file.source.uid } } // webuploader切出的block上沒(méi)有mkblk, blockIndex, size, offset屬性等,這是我為了支持分片上傳做的修改,請(qǐng)注意 if (block.mkblk) { block.options.server = file.server + "/mkblk/" + block.size + "/" + block.blockIndex } else { // 尋找當(dāng)前片在整個(gè)塊中的偏移 block.options.server = file.server + "/bput/" + file.ctxs[block.chunk - 1] + "/" + block.offset } } deferred.resolve() return deferred.promise() }, afs: (file) => { let deferred = this.$deferred if (file.skipped) { deferred.resolve() } else { let server = file.server + "/mkfile/" + file.size this.$http.post(server, file.mkblkctxs.join(","), { headers: { Authorization: file.token, "Content-Type": "text/plain", UploadBatch: file.source.uid } }) .then(res => { if (res.body.code) { deferred.reject(res.body.message) } else { deferred.resolve() } }) } return deferred.promise() }, "name": "progress" })關(guān)于webuploader如何和vue組合的探索
這里用html5無(wú)依賴(lài)版本進(jìn)行說(shuō)明
1.html5版本沒(méi)有提供md5File的具體實(shí)現(xiàn),而是以鉤子的形式給你了,如果真的需要聚合md5計(jì)算方法,可以按照全量版本里的模塊注冊(cè)形式,依次引入md5計(jì)算輔助庫(kù),引入全量包里的lib/md5, runtime/html5/md5, widgets/md5三個(gè)模塊,并在preset模塊中引入widgets/md5, runtime/html5/md5兩個(gè)模塊,完成模塊組合。如果不需要在內(nèi)部聚合,可以直接使用register注冊(cè)一個(gè)匿名掛件,并把md5-file這個(gè)命令鉤子所對(duì)應(yīng)的函數(shù)實(shí)現(xiàn)即可。 2.無(wú)依賴(lài)版本的內(nèi)建jquery還不完全,這導(dǎo)致了無(wú)依賴(lài)版本無(wú)法運(yùn)行,請(qǐng)自行為dollar-builtin模塊增加$.param, $.inArray兩個(gè)方法,并將weuploader中用到了$.map方法的地方改為$.each(內(nèi)建的jquery不支持$.map) 3.刪除所有與dom相關(guān)的依賴(lài),只保留無(wú)dom操作相關(guān)的純邏輯模塊(其實(shí)不刪除也可以,只要不配置dom相關(guān)掛件即可) 4.將webuploader實(shí)現(xiàn)為vue的插件,可以直接為Vue.prototype添加一個(gè)uploader的實(shí)例
以下是一個(gè)內(nèi)聚實(shí)現(xiàn)七牛云qeTag hash的代碼,由于是臨時(shí)測(cè)試修改,沒(méi)有在意語(yǔ)法和模塊引入,見(jiàn)諒。
修改uploader模塊,為webuploader添加sha1File方法的命令
// 批量添加純命令式方法。 $.each({ upload: "start-upload", stop: "stop-upload", getFile: "get-file", getFiles: "get-files", addFile: "add-file", addFiles: "add-file", sort: "sort-files", removeFile: "remove-file", cancelFile: "cancel-file", skipFile: "skip-file", retry: "retry", isInProgress: "is-in-progress", makeThumb: "make-thumb", md5File: "md5-file", sha1File: "sha1-file", // 這里添加~ getDimension: "get-dimension", addButton: "add-btn", predictRuntimeType: "predict-runtime-type", refresh: "refresh", disable: "disable", enable: "enable", reset: "reset" }, function( fn, command ) { Uploader.prototype[ fn ] = function() { return this.request( command, arguments ); }; });
加入一個(gè)sha1的依賴(lài),這里我使用的是js-sha1
實(shí)現(xiàn)/widgets/sha1,實(shí)現(xiàn)sha1File接口
/** * @fileOverview sha1計(jì)算 */ import Base from "../base" import Uploader from "../uploader" import Sha1 from "../lib/sha1" import Blob from "../lib/blob" export default Uploader.register({ name: "sha1", /** * 計(jì)算文件 sha1_hash 值,返回一個(gè) promise 對(duì)象,可以監(jiān)聽(tīng) progress 進(jìn)度。 * * * @method sha1File * @grammar sha1File( file[, start[, end]] ) => promise * @for Uploader * @example * * uploader.on( "fileQueued", function( file ) { * var $li = ...; * * uploader.sha1File( file ) * * // 及時(shí)顯示進(jìn)度 * .progress(function(percentage) { * console.log("Percentage:", percentage); * }) * * // 完成 * .then(function(val) { * console.log("sha1 result:", val); * }); * * }); */ sha1File: function( file, start, end ) { var sha1 = new Sha1(), deferred = Base.Deferred(), blob = (file instanceof Blob) ? file : this.request( "get-file", file ).source; sha1.on( "progress load", function( e ) { e = e || {}; deferred.notify( e.total ? e.loaded / e.total : 1 ); }); sha1.on( "complete", function() { deferred.resolve( sha1.getResult() ); }); sha1.on( "error", function( reason ) { deferred.reject( reason ); }); if ( arguments.length > 1 ) { start = start || 0; end = end || 0; start < 0 && (start = blob.size + start); end < 0 && (end = blob.size + end); end = Math.min( end, blob.size ); blob = blob.slice( start, end ); } sha1.loadFromBlob( blob ); return deferred.promise(); } });
實(shí)現(xiàn)/lib/sha1,連接運(yùn)行時(shí)sha1庫(kù)的封裝
/** * @fileOverview sha1 */ import RuntimeClient from "../runtime/client" import Mediator from "../mediator" function Sha1() { RuntimeClient.call( this, "Sha1" ); } // 讓 Sha1 具備事件功能。 Mediator.installTo( Sha1.prototype ); Sha1.prototype.loadFromBlob = function( blob ) { var me = this; if ( me.getRuid() ) { me.disconnectRuntime(); } // 連接到blob歸屬的同一個(gè)runtime. me.connectRuntime( blob.ruid, function() { me.exec("init"); me.exec( "loadFromBlob", blob ); }); }; Sha1.prototype.getResult = function() { return this.exec("getResult"); }; export default Sha1;
創(chuàng)建一個(gè)運(yùn)行時(shí)庫(kù)/runtime/html5/sha1,這里使用了Crypto-JS v2.5.1進(jìn)行輔助計(jì)算
/** * @fileOverview Transport flash實(shí)現(xiàn) */ import Html5Runtime from "./runtime" import Sha1 from "@/plugins/sha1" import Uploader from "../../uploader" import Crypto from "@/libs/Crypto" export default Html5Runtime.register( "Sha1", { init: function() { // do nothing. }, loadFromBlob: function( file ) { var blob = file.getSource(), chunkSize = 4 * 1024 * 1024, chunks = Math.ceil( blob.size / chunkSize ), chunk = 0, owner = this.owner, me = this, blobSlice = blob.mozSlice || blob.webkitSlice || blob.slice, loadNext, fr; var hashs = [], ret = ""; fr = new FileReader(); loadNext = function() { var start, end; start = chunk * chunkSize; end = Math.min( start + chunkSize, blob.size ); fr.onload = function( e ) { // var block = Tool.Crypto.util.bytesToWords( new Uint8Array(e.target.result)); var sha1 = Sha1.create(); var hash = sha1.update(e.target.result).digest(); hashs = hashs.concat(hash); if (end === file.size) { var perfex = 0x16; if (chunks > 1) { perfex = 0x96 sha1 = Sha1.create(); hash = sha1.update(hashs).digest() hashs = hash } hashs.unshift(perfex) ret = Crypto.util.bytesToBase64(hashs); } owner.trigger( "progress", { total: file.size, loaded: end }); }; fr.onloadend = function() { fr.onloadend = fr.onload = null; if ( ++chunk < chunks ) { setTimeout( loadNext, 1 ); } else { setTimeout(function(){ owner.trigger("load"); // 導(dǎo)出的是urlsafe的base64 me.result = ret.replace(///g,"_").replace(/+/g,"-"); loadNext = file = blob = hashs = null; owner.trigger("complete"); }, 50 ); } }; fr.readAsArrayBuffer( blobSlice.call( blob, start, end ) ); }; loadNext(); }, getResult: function() { return this.result; } });
為preset/html5only掛載依賴(lài)
/** * @fileOverview 只有html5實(shí)現(xiàn)的文件版本。 */ import Base from "../base" import "../widgets/widget" import "../widgets/queue" import "../widgets/runtime" import "../widgets/upload" import "../widgets/validator" import "../widgets/md5" import "../widgets/sha1" import "../runtime/html5/blob" import "../runtime/html5/transport" import "../runtime/html5/md5" import "../runtime/html5/sha1" export default Base;
如何使用?和md5File使用姿勢(shì)一模一樣
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/88811.html
摘要:簡(jiǎn)介是由團(tuán)隊(duì)開(kāi)發(fā)的一個(gè)簡(jiǎn)單的以為主,為輔的現(xiàn)代文件上傳組件。采用大文件分片并發(fā)上傳,極大的提高了文件上傳效率。另外分片傳輸能夠更加實(shí)時(shí)的跟蹤上傳進(jìn)度。選擇文件的按鈕。 簡(jiǎn)介:WebUploader是由Baidu WebFE(FEX)團(tuán)隊(duì)開(kāi)發(fā)的一個(gè)簡(jiǎn)單的以HTML5為主,F(xiàn)LASH為輔的現(xiàn)代文件上傳組件。在現(xiàn)代的瀏覽器里面能充分發(fā)揮HTML5的優(yōu)勢(shì),同時(shí)又不摒棄主流IE瀏覽器,沿用原來(lái)的...
摘要:簡(jiǎn)介是由團(tuán)隊(duì)開(kāi)發(fā)的一個(gè)簡(jiǎn)單的以為主,為輔的現(xiàn)代文件上傳組件。采用大文件分片并發(fā)上傳,極大的提高了文件上傳效率。另外分片傳輸能夠更加實(shí)時(shí)的跟蹤上傳進(jìn)度。選擇文件的按鈕。 簡(jiǎn)介:WebUploader是由Baidu WebFE(FEX)團(tuán)隊(duì)開(kāi)發(fā)的一個(gè)簡(jiǎn)單的以HTML5為主,F(xiàn)LASH為輔的現(xiàn)代文件上傳組件。在現(xiàn)代的瀏覽器里面能充分發(fā)揮HTML5的優(yōu)勢(shì),同時(shí)又不摒棄主流IE瀏覽器,沿用原來(lái)的...
摘要:簡(jiǎn)介是由團(tuán)隊(duì)開(kāi)發(fā)的一個(gè)簡(jiǎn)單的以為主,為輔的現(xiàn)代文件上傳組件。采用大文件分片并發(fā)上傳,極大的提高了文件上傳效率。另外分片傳輸能夠更加實(shí)時(shí)的跟蹤上傳進(jìn)度。選擇文件的按鈕。 簡(jiǎn)介:WebUploader是由Baidu WebFE(FEX)團(tuán)隊(duì)開(kāi)發(fā)的一個(gè)簡(jiǎn)單的以HTML5為主,F(xiàn)LASH為輔的現(xiàn)代文件上傳組件。在現(xiàn)代的瀏覽器里面能充分發(fā)揮HTML5的優(yōu)勢(shì),同時(shí)又不摒棄主流IE瀏覽器,沿用原來(lái)的...
摘要:否則強(qiáng)制轉(zhuǎn)換成指定的類(lèi)型。是否要分片處理大文件上傳還有其他配置項(xiàng)上傳事件選擇需要上傳的文件后,文件就會(huì)加入文件隊(duì)列,并觸發(fā)事件上傳進(jìn)度回調(diào)事件,在文件上傳中,多次調(diào)用此事件當(dāng)文件上傳成功時(shí)觸發(fā)當(dāng)文件上傳出錯(cuò)時(shí)觸發(fā)。 WebUploader簡(jiǎn)述 具有兩套運(yùn)行時(shí)支持:HTML5與FLASH 分片、并發(fā) 預(yù)覽、壓縮 多途徑添加文件 MD5驗(yàn)證 引入文件 雖然官方?jīng)]說(shuō)必須要引入JQuery...
閱讀 3247·2021-11-22 12:07
閱讀 1887·2021-10-12 10:11
閱讀 1051·2019-08-30 15:44
閱讀 2951·2019-08-30 12:45
閱讀 2214·2019-08-29 16:41
閱讀 1645·2019-08-29 16:35
閱讀 2637·2019-08-29 12:57
閱讀 1158·2019-08-26 13:51