You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					502 lines
				
				16 KiB
			
		
		
			
		
	
	
					502 lines
				
				16 KiB
			| 
								 
											3 years ago
										 
									 | 
							
								"use strict";
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								var support = require("./support");
							 | 
						||
| 
								 | 
							
								var base64 = require("./base64");
							 | 
						||
| 
								 | 
							
								var nodejsUtils = require("./nodejsUtils");
							 | 
						||
| 
								 | 
							
								var external = require("./external");
							 | 
						||
| 
								 | 
							
								require("setimmediate");
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/**
							 | 
						||
| 
								 | 
							
								 * Convert a string that pass as a "binary string": it should represent a byte
							 | 
						||
| 
								 | 
							
								 * array but may have > 255 char codes. Be sure to take only the first byte
							 | 
						||
| 
								 | 
							
								 * and returns the byte array.
							 | 
						||
| 
								 | 
							
								 * @param {String} str the string to transform.
							 | 
						||
| 
								 | 
							
								 * @return {Array|Uint8Array} the string in a binary format.
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								function string2binary(str) {
							 | 
						||
| 
								 | 
							
								    var result = null;
							 | 
						||
| 
								 | 
							
								    if (support.uint8array) {
							 | 
						||
| 
								 | 
							
								        result = new Uint8Array(str.length);
							 | 
						||
| 
								 | 
							
								    } else {
							 | 
						||
| 
								 | 
							
								        result = new Array(str.length);
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    return stringToArrayLike(str, result);
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/**
							 | 
						||
| 
								 | 
							
								 * Create a new blob with the given content and the given type.
							 | 
						||
| 
								 | 
							
								 * @param {String|ArrayBuffer} part the content to put in the blob. DO NOT use
							 | 
						||
| 
								 | 
							
								 * an Uint8Array because the stock browser of android 4 won't accept it (it
							 | 
						||
| 
								 | 
							
								 * will be silently converted to a string, "[object Uint8Array]").
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Use only ONE part to build the blob to avoid a memory leak in IE11 / Edge:
							 | 
						||
| 
								 | 
							
								 * when a large amount of Array is used to create the Blob, the amount of
							 | 
						||
| 
								 | 
							
								 * memory consumed is nearly 100 times the original data amount.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * @param {String} type the mime type of the blob.
							 | 
						||
| 
								 | 
							
								 * @return {Blob} the created blob.
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								exports.newBlob = function(part, type) {
							 | 
						||
| 
								 | 
							
								    exports.checkSupport("blob");
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    try {
							 | 
						||
| 
								 | 
							
								        // Blob constructor
							 | 
						||
| 
								 | 
							
								        return new Blob([part], {
							 | 
						||
| 
								 | 
							
								            type: type
							 | 
						||
| 
								 | 
							
								        });
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    catch (e) {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        try {
							 | 
						||
| 
								 | 
							
								            // deprecated, browser only, old way
							 | 
						||
| 
								 | 
							
								            var Builder = self.BlobBuilder || self.WebKitBlobBuilder || self.MozBlobBuilder || self.MSBlobBuilder;
							 | 
						||
| 
								 | 
							
								            var builder = new Builder();
							 | 
						||
| 
								 | 
							
								            builder.append(part);
							 | 
						||
| 
								 | 
							
								            return builder.getBlob(type);
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								        catch (e) {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            // well, fuck ?!
							 | 
						||
| 
								 | 
							
								            throw new Error("Bug : can't construct the Blob.");
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								/**
							 | 
						||
| 
								 | 
							
								 * The identity function.
							 | 
						||
| 
								 | 
							
								 * @param {Object} input the input.
							 | 
						||
| 
								 | 
							
								 * @return {Object} the same input.
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								function identity(input) {
							 | 
						||
| 
								 | 
							
								    return input;
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/**
							 | 
						||
| 
								 | 
							
								 * Fill in an array with a string.
							 | 
						||
| 
								 | 
							
								 * @param {String} str the string to use.
							 | 
						||
| 
								 | 
							
								 * @param {Array|ArrayBuffer|Uint8Array|Buffer} array the array to fill in (will be mutated).
							 | 
						||
| 
								 | 
							
								 * @return {Array|ArrayBuffer|Uint8Array|Buffer} the updated array.
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								function stringToArrayLike(str, array) {
							 | 
						||
| 
								 | 
							
								    for (var i = 0; i < str.length; ++i) {
							 | 
						||
| 
								 | 
							
								        array[i] = str.charCodeAt(i) & 0xFF;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    return array;
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/**
							 | 
						||
| 
								 | 
							
								 * An helper for the function arrayLikeToString.
							 | 
						||
| 
								 | 
							
								 * This contains static information and functions that
							 | 
						||
| 
								 | 
							
								 * can be optimized by the browser JIT compiler.
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								var arrayToStringHelper = {
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * Transform an array of int into a string, chunk by chunk.
							 | 
						||
| 
								 | 
							
								     * See the performances notes on arrayLikeToString.
							 | 
						||
| 
								 | 
							
								     * @param {Array|ArrayBuffer|Uint8Array|Buffer} array the array to transform.
							 | 
						||
| 
								 | 
							
								     * @param {String} type the type of the array.
							 | 
						||
| 
								 | 
							
								     * @param {Integer} chunk the chunk size.
							 | 
						||
| 
								 | 
							
								     * @return {String} the resulting string.
							 | 
						||
| 
								 | 
							
								     * @throws Error if the chunk is too big for the stack.
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    stringifyByChunk: function(array, type, chunk) {
							 | 
						||
| 
								 | 
							
								        var result = [], k = 0, len = array.length;
							 | 
						||
| 
								 | 
							
								        // shortcut
							 | 
						||
| 
								 | 
							
								        if (len <= chunk) {
							 | 
						||
| 
								 | 
							
								            return String.fromCharCode.apply(null, array);
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								        while (k < len) {
							 | 
						||
| 
								 | 
							
								            if (type === "array" || type === "nodebuffer") {
							 | 
						||
| 
								 | 
							
								                result.push(String.fromCharCode.apply(null, array.slice(k, Math.min(k + chunk, len))));
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								            else {
							 | 
						||
| 
								 | 
							
								                result.push(String.fromCharCode.apply(null, array.subarray(k, Math.min(k + chunk, len))));
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								            k += chunk;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								        return result.join("");
							 | 
						||
| 
								 | 
							
								    },
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * Call String.fromCharCode on every item in the array.
							 | 
						||
| 
								 | 
							
								     * This is the naive implementation, which generate A LOT of intermediate string.
							 | 
						||
| 
								 | 
							
								     * This should be used when everything else fail.
							 | 
						||
| 
								 | 
							
								     * @param {Array|ArrayBuffer|Uint8Array|Buffer} array the array to transform.
							 | 
						||
| 
								 | 
							
								     * @return {String} the result.
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    stringifyByChar: function(array){
							 | 
						||
| 
								 | 
							
								        var resultStr = "";
							 | 
						||
| 
								 | 
							
								        for(var i = 0; i < array.length; i++) {
							 | 
						||
| 
								 | 
							
								            resultStr += String.fromCharCode(array[i]);
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								        return resultStr;
							 | 
						||
| 
								 | 
							
								    },
							 | 
						||
| 
								 | 
							
								    applyCanBeUsed : {
							 | 
						||
| 
								 | 
							
								        /**
							 | 
						||
| 
								 | 
							
								         * true if the browser accepts to use String.fromCharCode on Uint8Array
							 | 
						||
| 
								 | 
							
								         */
							 | 
						||
| 
								 | 
							
								        uint8array : (function () {
							 | 
						||
| 
								 | 
							
								            try {
							 | 
						||
| 
								 | 
							
								                return support.uint8array && String.fromCharCode.apply(null, new Uint8Array(1)).length === 1;
							 | 
						||
| 
								 | 
							
								            } catch (e) {
							 | 
						||
| 
								 | 
							
								                return false;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        })(),
							 | 
						||
| 
								 | 
							
								        /**
							 | 
						||
| 
								 | 
							
								         * true if the browser accepts to use String.fromCharCode on nodejs Buffer.
							 | 
						||
| 
								 | 
							
								         */
							 | 
						||
| 
								 | 
							
								        nodebuffer : (function () {
							 | 
						||
| 
								 | 
							
								            try {
							 | 
						||
| 
								 | 
							
								                return support.nodebuffer && String.fromCharCode.apply(null, nodejsUtils.allocBuffer(1)).length === 1;
							 | 
						||
| 
								 | 
							
								            } catch (e) {
							 | 
						||
| 
								 | 
							
								                return false;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        })()
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/**
							 | 
						||
| 
								 | 
							
								 * Transform an array-like object to a string.
							 | 
						||
| 
								 | 
							
								 * @param {Array|ArrayBuffer|Uint8Array|Buffer} array the array to transform.
							 | 
						||
| 
								 | 
							
								 * @return {String} the result.
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								function arrayLikeToString(array) {
							 | 
						||
| 
								 | 
							
								    // Performances notes :
							 | 
						||
| 
								 | 
							
								    // --------------------
							 | 
						||
| 
								 | 
							
								    // String.fromCharCode.apply(null, array) is the fastest, see
							 | 
						||
| 
								 | 
							
								    // see http://jsperf.com/converting-a-uint8array-to-a-string/2
							 | 
						||
| 
								 | 
							
								    // but the stack is limited (and we can get huge arrays !).
							 | 
						||
| 
								 | 
							
								    //
							 | 
						||
| 
								 | 
							
								    // result += String.fromCharCode(array[i]); generate too many strings !
							 | 
						||
| 
								 | 
							
								    //
							 | 
						||
| 
								 | 
							
								    // This code is inspired by http://jsperf.com/arraybuffer-to-string-apply-performance/2
							 | 
						||
| 
								 | 
							
								    // TODO : we now have workers that split the work. Do we still need that ?
							 | 
						||
| 
								 | 
							
								    var chunk = 65536,
							 | 
						||
| 
								 | 
							
								        type = exports.getTypeOf(array),
							 | 
						||
| 
								 | 
							
								        canUseApply = true;
							 | 
						||
| 
								 | 
							
								    if (type === "uint8array") {
							 | 
						||
| 
								 | 
							
								        canUseApply = arrayToStringHelper.applyCanBeUsed.uint8array;
							 | 
						||
| 
								 | 
							
								    } else if (type === "nodebuffer") {
							 | 
						||
| 
								 | 
							
								        canUseApply = arrayToStringHelper.applyCanBeUsed.nodebuffer;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    if (canUseApply) {
							 | 
						||
| 
								 | 
							
								        while (chunk > 1) {
							 | 
						||
| 
								 | 
							
								            try {
							 | 
						||
| 
								 | 
							
								                return arrayToStringHelper.stringifyByChunk(array, type, chunk);
							 | 
						||
| 
								 | 
							
								            } catch (e) {
							 | 
						||
| 
								 | 
							
								                chunk = Math.floor(chunk / 2);
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    // no apply or chunk error : slow and painful algorithm
							 | 
						||
| 
								 | 
							
								    // default browser on android 4.*
							 | 
						||
| 
								 | 
							
								    return arrayToStringHelper.stringifyByChar(array);
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								exports.applyFromCharCode = arrayLikeToString;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/**
							 | 
						||
| 
								 | 
							
								 * Copy the data from an array-like to an other array-like.
							 | 
						||
| 
								 | 
							
								 * @param {Array|ArrayBuffer|Uint8Array|Buffer} arrayFrom the origin array.
							 | 
						||
| 
								 | 
							
								 * @param {Array|ArrayBuffer|Uint8Array|Buffer} arrayTo the destination array which will be mutated.
							 | 
						||
| 
								 | 
							
								 * @return {Array|ArrayBuffer|Uint8Array|Buffer} the updated destination array.
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								function arrayLikeToArrayLike(arrayFrom, arrayTo) {
							 | 
						||
| 
								 | 
							
								    for (var i = 0; i < arrayFrom.length; i++) {
							 | 
						||
| 
								 | 
							
								        arrayTo[i] = arrayFrom[i];
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    return arrayTo;
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								// a matrix containing functions to transform everything into everything.
							 | 
						||
| 
								 | 
							
								var transform = {};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								// string to ?
							 | 
						||
| 
								 | 
							
								transform["string"] = {
							 | 
						||
| 
								 | 
							
								    "string": identity,
							 | 
						||
| 
								 | 
							
								    "array": function(input) {
							 | 
						||
| 
								 | 
							
								        return stringToArrayLike(input, new Array(input.length));
							 | 
						||
| 
								 | 
							
								    },
							 | 
						||
| 
								 | 
							
								    "arraybuffer": function(input) {
							 | 
						||
| 
								 | 
							
								        return transform["string"]["uint8array"](input).buffer;
							 | 
						||
| 
								 | 
							
								    },
							 | 
						||
| 
								 | 
							
								    "uint8array": function(input) {
							 | 
						||
| 
								 | 
							
								        return stringToArrayLike(input, new Uint8Array(input.length));
							 | 
						||
| 
								 | 
							
								    },
							 | 
						||
| 
								 | 
							
								    "nodebuffer": function(input) {
							 | 
						||
| 
								 | 
							
								        return stringToArrayLike(input, nodejsUtils.allocBuffer(input.length));
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								// array to ?
							 | 
						||
| 
								 | 
							
								transform["array"] = {
							 | 
						||
| 
								 | 
							
								    "string": arrayLikeToString,
							 | 
						||
| 
								 | 
							
								    "array": identity,
							 | 
						||
| 
								 | 
							
								    "arraybuffer": function(input) {
							 | 
						||
| 
								 | 
							
								        return (new Uint8Array(input)).buffer;
							 | 
						||
| 
								 | 
							
								    },
							 | 
						||
| 
								 | 
							
								    "uint8array": function(input) {
							 | 
						||
| 
								 | 
							
								        return new Uint8Array(input);
							 | 
						||
| 
								 | 
							
								    },
							 | 
						||
| 
								 | 
							
								    "nodebuffer": function(input) {
							 | 
						||
| 
								 | 
							
								        return nodejsUtils.newBufferFrom(input);
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								// arraybuffer to ?
							 | 
						||
| 
								 | 
							
								transform["arraybuffer"] = {
							 | 
						||
| 
								 | 
							
								    "string": function(input) {
							 | 
						||
| 
								 | 
							
								        return arrayLikeToString(new Uint8Array(input));
							 | 
						||
| 
								 | 
							
								    },
							 | 
						||
| 
								 | 
							
								    "array": function(input) {
							 | 
						||
| 
								 | 
							
								        return arrayLikeToArrayLike(new Uint8Array(input), new Array(input.byteLength));
							 | 
						||
| 
								 | 
							
								    },
							 | 
						||
| 
								 | 
							
								    "arraybuffer": identity,
							 | 
						||
| 
								 | 
							
								    "uint8array": function(input) {
							 | 
						||
| 
								 | 
							
								        return new Uint8Array(input);
							 | 
						||
| 
								 | 
							
								    },
							 | 
						||
| 
								 | 
							
								    "nodebuffer": function(input) {
							 | 
						||
| 
								 | 
							
								        return nodejsUtils.newBufferFrom(new Uint8Array(input));
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								// uint8array to ?
							 | 
						||
| 
								 | 
							
								transform["uint8array"] = {
							 | 
						||
| 
								 | 
							
								    "string": arrayLikeToString,
							 | 
						||
| 
								 | 
							
								    "array": function(input) {
							 | 
						||
| 
								 | 
							
								        return arrayLikeToArrayLike(input, new Array(input.length));
							 | 
						||
| 
								 | 
							
								    },
							 | 
						||
| 
								 | 
							
								    "arraybuffer": function(input) {
							 | 
						||
| 
								 | 
							
								        return input.buffer;
							 | 
						||
| 
								 | 
							
								    },
							 | 
						||
| 
								 | 
							
								    "uint8array": identity,
							 | 
						||
| 
								 | 
							
								    "nodebuffer": function(input) {
							 | 
						||
| 
								 | 
							
								        return nodejsUtils.newBufferFrom(input);
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								// nodebuffer to ?
							 | 
						||
| 
								 | 
							
								transform["nodebuffer"] = {
							 | 
						||
| 
								 | 
							
								    "string": arrayLikeToString,
							 | 
						||
| 
								 | 
							
								    "array": function(input) {
							 | 
						||
| 
								 | 
							
								        return arrayLikeToArrayLike(input, new Array(input.length));
							 | 
						||
| 
								 | 
							
								    },
							 | 
						||
| 
								 | 
							
								    "arraybuffer": function(input) {
							 | 
						||
| 
								 | 
							
								        return transform["nodebuffer"]["uint8array"](input).buffer;
							 | 
						||
| 
								 | 
							
								    },
							 | 
						||
| 
								 | 
							
								    "uint8array": function(input) {
							 | 
						||
| 
								 | 
							
								        return arrayLikeToArrayLike(input, new Uint8Array(input.length));
							 | 
						||
| 
								 | 
							
								    },
							 | 
						||
| 
								 | 
							
								    "nodebuffer": identity
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/**
							 | 
						||
| 
								 | 
							
								 * Transform an input into any type.
							 | 
						||
| 
								 | 
							
								 * The supported output type are : string, array, uint8array, arraybuffer, nodebuffer.
							 | 
						||
| 
								 | 
							
								 * If no output type is specified, the unmodified input will be returned.
							 | 
						||
| 
								 | 
							
								 * @param {String} outputType the output type.
							 | 
						||
| 
								 | 
							
								 * @param {String|Array|ArrayBuffer|Uint8Array|Buffer} input the input to convert.
							 | 
						||
| 
								 | 
							
								 * @throws {Error} an Error if the browser doesn't support the requested output type.
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								exports.transformTo = function(outputType, input) {
							 | 
						||
| 
								 | 
							
								    if (!input) {
							 | 
						||
| 
								 | 
							
								        // undefined, null, etc
							 | 
						||
| 
								 | 
							
								        // an empty string won't harm.
							 | 
						||
| 
								 | 
							
								        input = "";
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    if (!outputType) {
							 | 
						||
| 
								 | 
							
								        return input;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    exports.checkSupport(outputType);
							 | 
						||
| 
								 | 
							
								    var inputType = exports.getTypeOf(input);
							 | 
						||
| 
								 | 
							
								    var result = transform[inputType][outputType](input);
							 | 
						||
| 
								 | 
							
								    return result;
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/**
							 | 
						||
| 
								 | 
							
								 * Resolve all relative path components, "." and "..", in a path. If these relative components
							 | 
						||
| 
								 | 
							
								 * traverse above the root then the resulting path will only contain the final path component.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * All empty components, e.g. "//", are removed.
							 | 
						||
| 
								 | 
							
								 * @param {string} path A path with / or \ separators
							 | 
						||
| 
								 | 
							
								 * @returns {string} The path with all relative path components resolved.
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								exports.resolve = function(path) {
							 | 
						||
| 
								 | 
							
								    var parts = path.split("/");
							 | 
						||
| 
								 | 
							
								    var result = [];
							 | 
						||
| 
								 | 
							
								    for (var index = 0; index < parts.length; index++) {
							 | 
						||
| 
								 | 
							
								        var part = parts[index];
							 | 
						||
| 
								 | 
							
								        // Allow the first and last component to be empty for trailing slashes.
							 | 
						||
| 
								 | 
							
								        if (part === "." || (part === "" && index !== 0 && index !== parts.length - 1)) {
							 | 
						||
| 
								 | 
							
								            continue;
							 | 
						||
| 
								 | 
							
								        } else if (part === "..") {
							 | 
						||
| 
								 | 
							
								            result.pop();
							 | 
						||
| 
								 | 
							
								        } else {
							 | 
						||
| 
								 | 
							
								            result.push(part);
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    return result.join("/");
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/**
							 | 
						||
| 
								 | 
							
								 * Return the type of the input.
							 | 
						||
| 
								 | 
							
								 * The type will be in a format valid for JSZip.utils.transformTo : string, array, uint8array, arraybuffer.
							 | 
						||
| 
								 | 
							
								 * @param {Object} input the input to identify.
							 | 
						||
| 
								 | 
							
								 * @return {String} the (lowercase) type of the input.
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								exports.getTypeOf = function(input) {
							 | 
						||
| 
								 | 
							
								    if (typeof input === "string") {
							 | 
						||
| 
								 | 
							
								        return "string";
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    if (Object.prototype.toString.call(input) === "[object Array]") {
							 | 
						||
| 
								 | 
							
								        return "array";
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    if (support.nodebuffer && nodejsUtils.isBuffer(input)) {
							 | 
						||
| 
								 | 
							
								        return "nodebuffer";
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    if (support.uint8array && input instanceof Uint8Array) {
							 | 
						||
| 
								 | 
							
								        return "uint8array";
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    if (support.arraybuffer && input instanceof ArrayBuffer) {
							 | 
						||
| 
								 | 
							
								        return "arraybuffer";
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/**
							 | 
						||
| 
								 | 
							
								 * Throw an exception if the type is not supported.
							 | 
						||
| 
								 | 
							
								 * @param {String} type the type to check.
							 | 
						||
| 
								 | 
							
								 * @throws {Error} an Error if the browser doesn't support the requested type.
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								exports.checkSupport = function(type) {
							 | 
						||
| 
								 | 
							
								    var supported = support[type.toLowerCase()];
							 | 
						||
| 
								 | 
							
								    if (!supported) {
							 | 
						||
| 
								 | 
							
								        throw new Error(type + " is not supported by this platform");
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								exports.MAX_VALUE_16BITS = 65535;
							 | 
						||
| 
								 | 
							
								exports.MAX_VALUE_32BITS = -1; // well, "\xFF\xFF\xFF\xFF\xFF\xFF\xFF\xFF" is parsed as -1
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/**
							 | 
						||
| 
								 | 
							
								 * Prettify a string read as binary.
							 | 
						||
| 
								 | 
							
								 * @param {string} str the string to prettify.
							 | 
						||
| 
								 | 
							
								 * @return {string} a pretty string.
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								exports.pretty = function(str) {
							 | 
						||
| 
								 | 
							
								    var res = "",
							 | 
						||
| 
								 | 
							
								        code, i;
							 | 
						||
| 
								 | 
							
								    for (i = 0; i < (str || "").length; i++) {
							 | 
						||
| 
								 | 
							
								        code = str.charCodeAt(i);
							 | 
						||
| 
								 | 
							
								        res += "\\x" + (code < 16 ? "0" : "") + code.toString(16).toUpperCase();
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    return res;
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/**
							 | 
						||
| 
								 | 
							
								 * Defer the call of a function.
							 | 
						||
| 
								 | 
							
								 * @param {Function} callback the function to call asynchronously.
							 | 
						||
| 
								 | 
							
								 * @param {Array} args the arguments to give to the callback.
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								exports.delay = function(callback, args, self) {
							 | 
						||
| 
								 | 
							
								    setImmediate(function () {
							 | 
						||
| 
								 | 
							
								        callback.apply(self || null, args || []);
							 | 
						||
| 
								 | 
							
								    });
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/**
							 | 
						||
| 
								 | 
							
								 * Extends a prototype with an other, without calling a constructor with
							 | 
						||
| 
								 | 
							
								 * side effects. Inspired by nodejs' `utils.inherits`
							 | 
						||
| 
								 | 
							
								 * @param {Function} ctor the constructor to augment
							 | 
						||
| 
								 | 
							
								 * @param {Function} superCtor the parent constructor to use
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								exports.inherits = function (ctor, superCtor) {
							 | 
						||
| 
								 | 
							
								    var Obj = function() {};
							 | 
						||
| 
								 | 
							
								    Obj.prototype = superCtor.prototype;
							 | 
						||
| 
								 | 
							
								    ctor.prototype = new Obj();
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/**
							 | 
						||
| 
								 | 
							
								 * Merge the objects passed as parameters into a new one.
							 | 
						||
| 
								 | 
							
								 * @private
							 | 
						||
| 
								 | 
							
								 * @param {...Object} var_args All objects to merge.
							 | 
						||
| 
								 | 
							
								 * @return {Object} a new object with the data of the others.
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								exports.extend = function() {
							 | 
						||
| 
								 | 
							
								    var result = {}, i, attr;
							 | 
						||
| 
								 | 
							
								    for (i = 0; i < arguments.length; i++) { // arguments is not enumerable in some browsers
							 | 
						||
| 
								 | 
							
								        for (attr in arguments[i]) {
							 | 
						||
| 
								 | 
							
								            if (Object.prototype.hasOwnProperty.call(arguments[i], attr) && typeof result[attr] === "undefined") {
							 | 
						||
| 
								 | 
							
								                result[attr] = arguments[i][attr];
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    return result;
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/**
							 | 
						||
| 
								 | 
							
								 * Transform arbitrary content into a Promise.
							 | 
						||
| 
								 | 
							
								 * @param {String} name a name for the content being processed.
							 | 
						||
| 
								 | 
							
								 * @param {Object} inputData the content to process.
							 | 
						||
| 
								 | 
							
								 * @param {Boolean} isBinary true if the content is not an unicode string
							 | 
						||
| 
								 | 
							
								 * @param {Boolean} isOptimizedBinaryString true if the string content only has one byte per character.
							 | 
						||
| 
								 | 
							
								 * @param {Boolean} isBase64 true if the string content is encoded with base64.
							 | 
						||
| 
								 | 
							
								 * @return {Promise} a promise in a format usable by JSZip.
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								exports.prepareContent = function(name, inputData, isBinary, isOptimizedBinaryString, isBase64) {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    // if inputData is already a promise, this flatten it.
							 | 
						||
| 
								 | 
							
								    var promise = external.Promise.resolve(inputData).then(function(data) {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        var isBlob = support.blob && (data instanceof Blob || ["[object File]", "[object Blob]"].indexOf(Object.prototype.toString.call(data)) !== -1);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if (isBlob && typeof FileReader !== "undefined") {
							 | 
						||
| 
								 | 
							
								            return new external.Promise(function (resolve, reject) {
							 | 
						||
| 
								 | 
							
								                var reader = new FileReader();
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                reader.onload = function(e) {
							 | 
						||
| 
								 | 
							
								                    resolve(e.target.result);
							 | 
						||
| 
								 | 
							
								                };
							 | 
						||
| 
								 | 
							
								                reader.onerror = function(e) {
							 | 
						||
| 
								 | 
							
								                    reject(e.target.error);
							 | 
						||
| 
								 | 
							
								                };
							 | 
						||
| 
								 | 
							
								                reader.readAsArrayBuffer(data);
							 | 
						||
| 
								 | 
							
								            });
							 | 
						||
| 
								 | 
							
								        } else {
							 | 
						||
| 
								 | 
							
								            return data;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								    });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    return promise.then(function(data) {
							 | 
						||
| 
								 | 
							
								        var dataType = exports.getTypeOf(data);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if (!dataType) {
							 | 
						||
| 
								 | 
							
								            return external.Promise.reject(
							 | 
						||
| 
								 | 
							
								                new Error("Can't read the data of '" + name + "'. Is it " +
							 | 
						||
| 
								 | 
							
								                          "in a supported JavaScript type (String, Blob, ArrayBuffer, etc) ?")
							 | 
						||
| 
								 | 
							
								            );
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								        // special case : it's way easier to work with Uint8Array than with ArrayBuffer
							 | 
						||
| 
								 | 
							
								        if (dataType === "arraybuffer") {
							 | 
						||
| 
								 | 
							
								            data = exports.transformTo("uint8array", data);
							 | 
						||
| 
								 | 
							
								        } else if (dataType === "string") {
							 | 
						||
| 
								 | 
							
								            if (isBase64) {
							 | 
						||
| 
								 | 
							
								                data = base64.decode(data);
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								            else if (isBinary) {
							 | 
						||
| 
								 | 
							
								                // optimizedBinaryString === true means that the file has already been filtered with a 0xFF mask
							 | 
						||
| 
								 | 
							
								                if (isOptimizedBinaryString !== true) {
							 | 
						||
| 
								 | 
							
								                    // this is a string, not in a base64 format.
							 | 
						||
| 
								 | 
							
								                    // Be sure that this is a correct "binary string"
							 | 
						||
| 
								 | 
							
								                    data = string2binary(data);
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								        return data;
							 | 
						||
| 
								 | 
							
								    });
							 | 
						||
| 
								 | 
							
								};
							 |