成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專(zhuān)欄INFORMATION COLUMN

webuploader與網(wǎng)宿云踩坑

Yi_Zhi_Yu / 1462人閱讀

摘要:上傳結(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

相關(guān)文章

  • 在Vue項(xiàng)目中使用WebUploader實(shí)現(xiàn)文件上傳

    摘要:簡(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)的...

    mindwind 評(píng)論0 收藏0
  • 在Vue項(xiàng)目中使用WebUploader實(shí)現(xiàn)文件上傳

    摘要:簡(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)的...

    endiat 評(píng)論0 收藏0
  • 在Vue項(xiàng)目中使用WebUploader實(shí)現(xiàn)文件上傳

    摘要:簡(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)的...

    funnyZhang 評(píng)論0 收藏0
  • WebUploader上傳插件使用說(shuō)明

    摘要:否則強(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...

    xbynet 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<