mirror of
https://github.com/CTCaer/RetroArch.git
synced 2025-01-13 22:31:57 +00:00
11847 lines
440 KiB
JavaScript
11847 lines
440 KiB
JavaScript
|
(function() {
|
|||
|
/**
|
|||
|
* This file installs all of the polyfills that BrowserFS requires.
|
|||
|
*/
|
|||
|
// IE < 9 does not define this function.
|
|||
|
if (!Date.now) {
|
|||
|
Date.now = function now() {
|
|||
|
return new Date().getTime();
|
|||
|
};
|
|||
|
}
|
|||
|
|
|||
|
// IE < 9 does not define this function.
|
|||
|
if (!Array.isArray) {
|
|||
|
Array.isArray = function (vArg) {
|
|||
|
return Object.prototype.toString.call(vArg) === "[object Array]";
|
|||
|
};
|
|||
|
}
|
|||
|
|
|||
|
// IE < 9 does not define this function.
|
|||
|
// From https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Object/keys
|
|||
|
if (!Object.keys) {
|
|||
|
Object.keys = (function () {
|
|||
|
|
|||
|
var hasOwnProperty = Object.prototype.hasOwnProperty, hasDontEnumBug = !({ toString: null }).propertyIsEnumerable('toString'), dontEnums = [
|
|||
|
'toString',
|
|||
|
'toLocaleString',
|
|||
|
'valueOf',
|
|||
|
'hasOwnProperty',
|
|||
|
'isPrototypeOf',
|
|||
|
'propertyIsEnumerable',
|
|||
|
'constructor'
|
|||
|
], dontEnumsLength = dontEnums.length;
|
|||
|
|
|||
|
return function (obj) {
|
|||
|
if (typeof obj !== 'object' && (typeof obj !== 'function' || obj === null)) {
|
|||
|
throw new TypeError('Object.keys called on non-object');
|
|||
|
}
|
|||
|
|
|||
|
var result = [], prop, i;
|
|||
|
|
|||
|
for (prop in obj) {
|
|||
|
if (hasOwnProperty.call(obj, prop)) {
|
|||
|
result.push(prop);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
if (hasDontEnumBug) {
|
|||
|
for (i = 0; i < dontEnumsLength; i++) {
|
|||
|
if (hasOwnProperty.call(obj, dontEnums[i])) {
|
|||
|
result.push(dontEnums[i]);
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
return result;
|
|||
|
};
|
|||
|
}());
|
|||
|
}
|
|||
|
|
|||
|
// IE substr does not support negative indices
|
|||
|
if ('ab'.substr(-1) !== 'b') {
|
|||
|
String.prototype.substr = function (substr) {
|
|||
|
return function (start, length) {
|
|||
|
// did we get a negative start, calculate how much it is from the
|
|||
|
// beginning of the string
|
|||
|
if (start < 0)
|
|||
|
start = this.length + start;
|
|||
|
|
|||
|
// call the original function
|
|||
|
return substr.call(this, start, length);
|
|||
|
};
|
|||
|
}(String.prototype.substr);
|
|||
|
}
|
|||
|
|
|||
|
// IE < 9 does not support forEach
|
|||
|
if (!Array.prototype.forEach) {
|
|||
|
Array.prototype.forEach = function (fn, scope) {
|
|||
|
for (var i = 0; i < this.length; ++i) {
|
|||
|
if (i in this) {
|
|||
|
fn.call(scope, this[i], i, this);
|
|||
|
}
|
|||
|
}
|
|||
|
};
|
|||
|
}
|
|||
|
|
|||
|
// Only IE10 has setImmediate.
|
|||
|
// @todo: Determine viability of switching to the 'proper' polyfill for this.
|
|||
|
if (typeof setImmediate === 'undefined') {
|
|||
|
// XXX avoid importing the global module.
|
|||
|
var gScope = typeof window !== 'undefined' ? window : typeof self !== 'undefined' ? self : global;
|
|||
|
var timeouts = [];
|
|||
|
var messageName = "zero-timeout-message";
|
|||
|
var canUsePostMessage = function () {
|
|||
|
if (typeof gScope.importScripts !== 'undefined' || !gScope.postMessage) {
|
|||
|
return false;
|
|||
|
}
|
|||
|
var postMessageIsAsync = true;
|
|||
|
var oldOnMessage = gScope.onmessage;
|
|||
|
gScope.onmessage = function () {
|
|||
|
postMessageIsAsync = false;
|
|||
|
};
|
|||
|
gScope.postMessage('', '*');
|
|||
|
gScope.onmessage = oldOnMessage;
|
|||
|
return postMessageIsAsync;
|
|||
|
};
|
|||
|
if (canUsePostMessage()) {
|
|||
|
gScope.setImmediate = function (fn) {
|
|||
|
timeouts.push(fn);
|
|||
|
gScope.postMessage(messageName, "*");
|
|||
|
};
|
|||
|
var handleMessage = function (event) {
|
|||
|
if (event.source === self && event.data === messageName) {
|
|||
|
if (event.stopPropagation) {
|
|||
|
event.stopPropagation();
|
|||
|
} else {
|
|||
|
event.cancelBubble = true;
|
|||
|
}
|
|||
|
if (timeouts.length > 0) {
|
|||
|
var fn = timeouts.shift();
|
|||
|
return fn();
|
|||
|
}
|
|||
|
}
|
|||
|
};
|
|||
|
if (gScope.addEventListener) {
|
|||
|
gScope.addEventListener('message', handleMessage, true);
|
|||
|
} else {
|
|||
|
gScope.attachEvent('onmessage', handleMessage);
|
|||
|
}
|
|||
|
} else if (gScope.MessageChannel) {
|
|||
|
// WebWorker MessageChannel
|
|||
|
var channel = new gScope.MessageChannel();
|
|||
|
channel.port1.onmessage = function (event) {
|
|||
|
if (timeouts.length > 0) {
|
|||
|
return timeouts.shift()();
|
|||
|
}
|
|||
|
};
|
|||
|
gScope.setImmediate = function (fn) {
|
|||
|
timeouts.push(fn);
|
|||
|
channel.port2.postMessage('');
|
|||
|
};
|
|||
|
} else {
|
|||
|
gScope.setImmediate = function (fn) {
|
|||
|
return setTimeout(fn, 0);
|
|||
|
var scriptEl = window.document.createElement("script");
|
|||
|
scriptEl.onreadystatechange = function () {
|
|||
|
fn();
|
|||
|
scriptEl.onreadystatechange = null;
|
|||
|
scriptEl.parentNode.removeChild(scriptEl);
|
|||
|
return scriptEl = null;
|
|||
|
};
|
|||
|
gScope.document.documentElement.appendChild(scriptEl);
|
|||
|
};
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
// IE<9 does not define indexOf.
|
|||
|
// From: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/indexOf
|
|||
|
if (!Array.prototype.indexOf) {
|
|||
|
Array.prototype.indexOf = function (searchElement, fromIndex) {
|
|||
|
if (typeof fromIndex === "undefined") { fromIndex = 0; }
|
|||
|
if (!this) {
|
|||
|
throw new TypeError();
|
|||
|
}
|
|||
|
|
|||
|
var length = this.length;
|
|||
|
if (length === 0 || pivot >= length) {
|
|||
|
return -1;
|
|||
|
}
|
|||
|
|
|||
|
var pivot = fromIndex;
|
|||
|
if (pivot < 0) {
|
|||
|
pivot = length + pivot;
|
|||
|
}
|
|||
|
|
|||
|
for (var i = pivot; i < length; i++) {
|
|||
|
if (this[i] === searchElement) {
|
|||
|
return i;
|
|||
|
}
|
|||
|
}
|
|||
|
return -1;
|
|||
|
};
|
|||
|
}
|
|||
|
|
|||
|
// IE<9 does not support forEach
|
|||
|
// From: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/forEach
|
|||
|
if (!Array.prototype.forEach) {
|
|||
|
Array.prototype.forEach = function (fn, scope) {
|
|||
|
var i, len;
|
|||
|
for (i = 0, len = this.length; i < len; ++i) {
|
|||
|
if (i in this) {
|
|||
|
fn.call(scope, this[i], i, this);
|
|||
|
}
|
|||
|
}
|
|||
|
};
|
|||
|
}
|
|||
|
|
|||
|
// IE<9 does not support map
|
|||
|
// From: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/map
|
|||
|
if (!Array.prototype.map) {
|
|||
|
Array.prototype.map = function (callback, thisArg) {
|
|||
|
var T, A, k;
|
|||
|
if (this == null) {
|
|||
|
throw new TypeError(" this is null or not defined");
|
|||
|
}
|
|||
|
|
|||
|
// 1. Let O be the result of calling ToObject passing the |this| value as the argument.
|
|||
|
var O = Object(this);
|
|||
|
|
|||
|
// 2. Let lenValue be the result of calling the Get internal method of O with the argument "length".
|
|||
|
// 3. Let len be ToUint32(lenValue).
|
|||
|
var len = O.length >>> 0;
|
|||
|
|
|||
|
// 4. If IsCallable(callback) is false, throw a TypeError exception.
|
|||
|
// See: http://es5.github.com/#x9.11
|
|||
|
if (typeof callback !== "function") {
|
|||
|
throw new TypeError(callback + " is not a function");
|
|||
|
}
|
|||
|
|
|||
|
// 5. If thisArg was supplied, let T be thisArg; else let T be undefined.
|
|||
|
if (thisArg) {
|
|||
|
T = thisArg;
|
|||
|
}
|
|||
|
|
|||
|
// 6. Let A be a new array created as if by the expression new Array(len) where Array is
|
|||
|
// the standard built-in constructor with that name and len is the value of len.
|
|||
|
A = new Array(len);
|
|||
|
|
|||
|
// 7. Let k be 0
|
|||
|
k = 0;
|
|||
|
|
|||
|
while (k < len) {
|
|||
|
var kValue, mappedValue;
|
|||
|
|
|||
|
// a. Let Pk be ToString(k).
|
|||
|
// This is implicit for LHS operands of the in operator
|
|||
|
// b. Let kPresent be the result of calling the HasProperty internal method of O with argument Pk.
|
|||
|
// This step can be combined with c
|
|||
|
// c. If kPresent is true, then
|
|||
|
if (k in O) {
|
|||
|
// i. Let kValue be the result of calling the Get internal method of O with argument Pk.
|
|||
|
kValue = O[k];
|
|||
|
|
|||
|
// ii. Let mappedValue be the result of calling the Call internal method of callback
|
|||
|
// with T as the this value and argument list containing kValue, k, and O.
|
|||
|
mappedValue = callback.call(T, kValue, k, O);
|
|||
|
|
|||
|
// iii. Call the DefineOwnProperty internal method of A with arguments
|
|||
|
// Pk, Property Descriptor {Value: mappedValue, : true, Enumerable: true, Configurable: true},
|
|||
|
// and false.
|
|||
|
// In browsers that support Object.defineProperty, use the following:
|
|||
|
// Object.defineProperty(A, Pk, { value: mappedValue, writable: true, enumerable: true, configurable: true });
|
|||
|
// For best browser support, use the following:
|
|||
|
A[k] = mappedValue;
|
|||
|
}
|
|||
|
|
|||
|
// d. Increase k by 1.
|
|||
|
k++;
|
|||
|
}
|
|||
|
|
|||
|
// 9. return A
|
|||
|
return A;
|
|||
|
};
|
|||
|
}
|
|||
|
|
|||
|
/**
|
|||
|
* IE9 and below only: Injects a VBScript function that converts the
|
|||
|
* 'responseBody' attribute of an XMLHttpRequest into a bytestring.
|
|||
|
* From: http://miskun.com/javascript/internet-explorer-and-binary-files-data-access/#comment-17
|
|||
|
*
|
|||
|
* This must be performed *before* the page finishes loading, otherwise
|
|||
|
* document.write will refresh the page. :(
|
|||
|
*
|
|||
|
* This is harmless to inject into non-IE browsers.
|
|||
|
*/
|
|||
|
if (typeof document !== 'undefined' && window['chrome'] === undefined) {
|
|||
|
document.write("<!-- IEBinaryToArray_ByteStr -->\r\n" + "<script type='text/vbscript'>\r\n" + "Function IEBinaryToArray_ByteStr(Binary)\r\n" + " IEBinaryToArray_ByteStr = CStr(Binary)\r\n" + "End Function\r\n" + "Function IEBinaryToArray_ByteStr_Last(Binary)\r\n" + " Dim lastIndex\r\n" + " lastIndex = LenB(Binary)\r\n" + " if lastIndex mod 2 Then\r\n" + " IEBinaryToArray_ByteStr_Last = Chr( AscB( MidB( Binary, lastIndex, 1 ) ) )\r\n" + " Else\r\n" + " IEBinaryToArray_ByteStr_Last = " + '""' + "\r\n" + " End If\r\n" + "End Function\r\n" + "</script>\r\n");
|
|||
|
}
|
|||
|
//# sourceMappingURL=polyfills.js.map
|
|||
|
|
|||
|
|
|||
|
|
|||
|
/**
|
|||
|
* @license almond 0.3.0 Copyright (c) 2011-2014, The Dojo Foundation All Rights Reserved.
|
|||
|
* Available via the MIT or new BSD license.
|
|||
|
* see: http://github.com/jrburke/almond for details
|
|||
|
*/
|
|||
|
//Going sloppy to avoid 'use strict' string cost, but strict practices should
|
|||
|
//be followed.
|
|||
|
/*jslint sloppy: true */
|
|||
|
/*global setTimeout: false */
|
|||
|
|
|||
|
var requirejs, require, define;
|
|||
|
(function (undef) {
|
|||
|
var main, req, makeMap, handlers,
|
|||
|
defined = {},
|
|||
|
waiting = {},
|
|||
|
config = {},
|
|||
|
defining = {},
|
|||
|
hasOwn = Object.prototype.hasOwnProperty,
|
|||
|
aps = [].slice,
|
|||
|
jsSuffixRegExp = /\.js$/;
|
|||
|
|
|||
|
function hasProp(obj, prop) {
|
|||
|
return hasOwn.call(obj, prop);
|
|||
|
}
|
|||
|
|
|||
|
/**
|
|||
|
* Given a relative module name, like ./something, normalize it to
|
|||
|
* a real name that can be mapped to a path.
|
|||
|
* @param {String} name the relative name
|
|||
|
* @param {String} baseName a real name that the name arg is relative
|
|||
|
* to.
|
|||
|
* @returns {String} normalized name
|
|||
|
*/
|
|||
|
function normalize(name, baseName) {
|
|||
|
var nameParts, nameSegment, mapValue, foundMap, lastIndex,
|
|||
|
foundI, foundStarMap, starI, i, j, part,
|
|||
|
baseParts = baseName && baseName.split("/"),
|
|||
|
map = config.map,
|
|||
|
starMap = (map && map['*']) || {};
|
|||
|
|
|||
|
//Adjust any relative paths.
|
|||
|
if (name && name.charAt(0) === ".") {
|
|||
|
//If have a base name, try to normalize against it,
|
|||
|
//otherwise, assume it is a top-level require that will
|
|||
|
//be relative to baseUrl in the end.
|
|||
|
if (baseName) {
|
|||
|
//Convert baseName to array, and lop off the last part,
|
|||
|
//so that . matches that "directory" and not name of the baseName's
|
|||
|
//module. For instance, baseName of "one/two/three", maps to
|
|||
|
//"one/two/three.js", but we want the directory, "one/two" for
|
|||
|
//this normalization.
|
|||
|
baseParts = baseParts.slice(0, baseParts.length - 1);
|
|||
|
name = name.split('/');
|
|||
|
lastIndex = name.length - 1;
|
|||
|
|
|||
|
// Node .js allowance:
|
|||
|
if (config.nodeIdCompat && jsSuffixRegExp.test(name[lastIndex])) {
|
|||
|
name[lastIndex] = name[lastIndex].replace(jsSuffixRegExp, '');
|
|||
|
}
|
|||
|
|
|||
|
name = baseParts.concat(name);
|
|||
|
|
|||
|
//start trimDots
|
|||
|
for (i = 0; i < name.length; i += 1) {
|
|||
|
part = name[i];
|
|||
|
if (part === ".") {
|
|||
|
name.splice(i, 1);
|
|||
|
i -= 1;
|
|||
|
} else if (part === "..") {
|
|||
|
if (i === 1 && (name[2] === '..' || name[0] === '..')) {
|
|||
|
//End of the line. Keep at least one non-dot
|
|||
|
//path segment at the front so it can be mapped
|
|||
|
//correctly to disk. Otherwise, there is likely
|
|||
|
//no path mapping for a path starting with '..'.
|
|||
|
//This can still fail, but catches the most reasonable
|
|||
|
//uses of ..
|
|||
|
break;
|
|||
|
} else if (i > 0) {
|
|||
|
name.splice(i - 1, 2);
|
|||
|
i -= 2;
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
//end trimDots
|
|||
|
|
|||
|
name = name.join("/");
|
|||
|
} else if (name.indexOf('./') === 0) {
|
|||
|
// No baseName, so this is ID is resolved relative
|
|||
|
// to baseUrl, pull off the leading dot.
|
|||
|
name = name.substring(2);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
//Apply map config if available.
|
|||
|
if ((baseParts || starMap) && map) {
|
|||
|
nameParts = name.split('/');
|
|||
|
|
|||
|
for (i = nameParts.length; i > 0; i -= 1) {
|
|||
|
nameSegment = nameParts.slice(0, i).join("/");
|
|||
|
|
|||
|
if (baseParts) {
|
|||
|
//Find the longest baseName segment match in the config.
|
|||
|
//So, do joins on the biggest to smallest lengths of baseParts.
|
|||
|
for (j = baseParts.length; j > 0; j -= 1) {
|
|||
|
mapValue = map[baseParts.slice(0, j).join('/')];
|
|||
|
|
|||
|
//baseName segment has config, find if it has one for
|
|||
|
//this name.
|
|||
|
if (mapValue) {
|
|||
|
mapValue = mapValue[nameSegment];
|
|||
|
if (mapValue) {
|
|||
|
//Match, update name to the new value.
|
|||
|
foundMap = mapValue;
|
|||
|
foundI = i;
|
|||
|
break;
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
if (foundMap) {
|
|||
|
break;
|
|||
|
}
|
|||
|
|
|||
|
//Check for a star map match, but just hold on to it,
|
|||
|
//if there is a shorter segment match later in a matching
|
|||
|
//config, then favor over this star map.
|
|||
|
if (!foundStarMap && starMap && starMap[nameSegment]) {
|
|||
|
foundStarMap = starMap[nameSegment];
|
|||
|
starI = i;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
if (!foundMap && foundStarMap) {
|
|||
|
foundMap = foundStarMap;
|
|||
|
foundI = starI;
|
|||
|
}
|
|||
|
|
|||
|
if (foundMap) {
|
|||
|
nameParts.splice(0, foundI, foundMap);
|
|||
|
name = nameParts.join('/');
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
return name;
|
|||
|
}
|
|||
|
|
|||
|
function makeRequire(relName, forceSync) {
|
|||
|
return function () {
|
|||
|
//A version of a require function that passes a moduleName
|
|||
|
//value for items that may need to
|
|||
|
//look up paths relative to the moduleName
|
|||
|
var args = aps.call(arguments, 0);
|
|||
|
|
|||
|
//If first arg is not require('string'), and there is only
|
|||
|
//one arg, it is the array form without a callback. Insert
|
|||
|
//a null so that the following concat is correct.
|
|||
|
if (typeof args[0] !== 'string' && args.length === 1) {
|
|||
|
args.push(null);
|
|||
|
}
|
|||
|
return req.apply(undef, args.concat([relName, forceSync]));
|
|||
|
};
|
|||
|
}
|
|||
|
|
|||
|
function makeNormalize(relName) {
|
|||
|
return function (name) {
|
|||
|
return normalize(name, relName);
|
|||
|
};
|
|||
|
}
|
|||
|
|
|||
|
function makeLoad(depName) {
|
|||
|
return function (value) {
|
|||
|
defined[depName] = value;
|
|||
|
};
|
|||
|
}
|
|||
|
|
|||
|
function callDep(name) {
|
|||
|
if (hasProp(waiting, name)) {
|
|||
|
var args = waiting[name];
|
|||
|
delete waiting[name];
|
|||
|
defining[name] = true;
|
|||
|
main.apply(undef, args);
|
|||
|
}
|
|||
|
|
|||
|
if (!hasProp(defined, name) && !hasProp(defining, name)) {
|
|||
|
throw new Error('No ' + name);
|
|||
|
}
|
|||
|
return defined[name];
|
|||
|
}
|
|||
|
|
|||
|
//Turns a plugin!resource to [plugin, resource]
|
|||
|
//with the plugin being undefined if the name
|
|||
|
//did not have a plugin prefix.
|
|||
|
function splitPrefix(name) {
|
|||
|
var prefix,
|
|||
|
index = name ? name.indexOf('!') : -1;
|
|||
|
if (index > -1) {
|
|||
|
prefix = name.substring(0, index);
|
|||
|
name = name.substring(index + 1, name.length);
|
|||
|
}
|
|||
|
return [prefix, name];
|
|||
|
}
|
|||
|
|
|||
|
/**
|
|||
|
* Makes a name map, normalizing the name, and using a plugin
|
|||
|
* for normalization if necessary. Grabs a ref to plugin
|
|||
|
* too, as an optimization.
|
|||
|
*/
|
|||
|
makeMap = function (name, relName) {
|
|||
|
var plugin,
|
|||
|
parts = splitPrefix(name),
|
|||
|
prefix = parts[0];
|
|||
|
|
|||
|
name = parts[1];
|
|||
|
|
|||
|
if (prefix) {
|
|||
|
prefix = normalize(prefix, relName);
|
|||
|
plugin = callDep(prefix);
|
|||
|
}
|
|||
|
|
|||
|
//Normalize according
|
|||
|
if (prefix) {
|
|||
|
if (plugin && plugin.normalize) {
|
|||
|
name = plugin.normalize(name, makeNormalize(relName));
|
|||
|
} else {
|
|||
|
name = normalize(name, relName);
|
|||
|
}
|
|||
|
} else {
|
|||
|
name = normalize(name, relName);
|
|||
|
parts = splitPrefix(name);
|
|||
|
prefix = parts[0];
|
|||
|
name = parts[1];
|
|||
|
if (prefix) {
|
|||
|
plugin = callDep(prefix);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
//Using ridiculous property names for space reasons
|
|||
|
return {
|
|||
|
f: prefix ? prefix + '!' + name : name, //fullName
|
|||
|
n: name,
|
|||
|
pr: prefix,
|
|||
|
p: plugin
|
|||
|
};
|
|||
|
};
|
|||
|
|
|||
|
function makeConfig(name) {
|
|||
|
return function () {
|
|||
|
return (config && config.config && config.config[name]) || {};
|
|||
|
};
|
|||
|
}
|
|||
|
|
|||
|
handlers = {
|
|||
|
require: function (name) {
|
|||
|
return makeRequire(name);
|
|||
|
},
|
|||
|
exports: function (name) {
|
|||
|
var e = defined[name];
|
|||
|
if (typeof e !== 'undefined') {
|
|||
|
return e;
|
|||
|
} else {
|
|||
|
return (defined[name] = {});
|
|||
|
}
|
|||
|
},
|
|||
|
module: function (name) {
|
|||
|
return {
|
|||
|
id: name,
|
|||
|
uri: '',
|
|||
|
exports: defined[name],
|
|||
|
config: makeConfig(name)
|
|||
|
};
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
main = function (name, deps, callback, relName) {
|
|||
|
var cjsModule, depName, ret, map, i,
|
|||
|
args = [],
|
|||
|
callbackType = typeof callback,
|
|||
|
usingExports;
|
|||
|
|
|||
|
//Use name if no relName
|
|||
|
relName = relName || name;
|
|||
|
|
|||
|
//Call the callback to define the module, if necessary.
|
|||
|
if (callbackType === 'undefined' || callbackType === 'function') {
|
|||
|
//Pull out the defined dependencies and pass the ordered
|
|||
|
//values to the callback.
|
|||
|
//Default to [require, exports, module] if no deps
|
|||
|
deps = !deps.length && callback.length ? ['require', 'exports', 'module'] : deps;
|
|||
|
for (i = 0; i < deps.length; i += 1) {
|
|||
|
map = makeMap(deps[i], relName);
|
|||
|
depName = map.f;
|
|||
|
|
|||
|
//Fast path CommonJS standard dependencies.
|
|||
|
if (depName === "require") {
|
|||
|
args[i] = handlers.require(name);
|
|||
|
} else if (depName === "exports") {
|
|||
|
//CommonJS module spec 1.1
|
|||
|
args[i] = handlers.exports(name);
|
|||
|
usingExports = true;
|
|||
|
} else if (depName === "module") {
|
|||
|
//CommonJS module spec 1.1
|
|||
|
cjsModule = args[i] = handlers.module(name);
|
|||
|
} else if (hasProp(defined, depName) ||
|
|||
|
hasProp(waiting, depName) ||
|
|||
|
hasProp(defining, depName)) {
|
|||
|
args[i] = callDep(depName);
|
|||
|
} else if (map.p) {
|
|||
|
map.p.load(map.n, makeRequire(relName, true), makeLoad(depName), {});
|
|||
|
args[i] = defined[depName];
|
|||
|
} else {
|
|||
|
throw new Error(name + ' missing ' + depName);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
ret = callback ? callback.apply(defined[name], args) : undefined;
|
|||
|
|
|||
|
if (name) {
|
|||
|
//If setting exports via "module" is in play,
|
|||
|
//favor that over return value and exports. After that,
|
|||
|
//favor a non-undefined return value over exports use.
|
|||
|
if (cjsModule && cjsModule.exports !== undef &&
|
|||
|
cjsModule.exports !== defined[name]) {
|
|||
|
defined[name] = cjsModule.exports;
|
|||
|
} else if (ret !== undef || !usingExports) {
|
|||
|
//Use the return value from the function.
|
|||
|
defined[name] = ret;
|
|||
|
}
|
|||
|
}
|
|||
|
} else if (name) {
|
|||
|
//May just be an object definition for the module. Only
|
|||
|
//worry about defining if have a module name.
|
|||
|
defined[name] = callback;
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
requirejs = require = req = function (deps, callback, relName, forceSync, alt) {
|
|||
|
if (typeof deps === "string") {
|
|||
|
if (handlers[deps]) {
|
|||
|
//callback in this case is really relName
|
|||
|
return handlers[deps](callback);
|
|||
|
}
|
|||
|
//Just return the module wanted. In this scenario, the
|
|||
|
//deps arg is the module name, and second arg (if passed)
|
|||
|
//is just the relName.
|
|||
|
//Normalize module name, if it contains . or ..
|
|||
|
return callDep(makeMap(deps, callback).f);
|
|||
|
} else if (!deps.splice) {
|
|||
|
//deps is a config object, not an array.
|
|||
|
config = deps;
|
|||
|
if (config.deps) {
|
|||
|
req(config.deps, config.callback);
|
|||
|
}
|
|||
|
if (!callback) {
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
if (callback.splice) {
|
|||
|
//callback is an array, which means it is a dependency list.
|
|||
|
//Adjust args if there are dependencies
|
|||
|
deps = callback;
|
|||
|
callback = relName;
|
|||
|
relName = null;
|
|||
|
} else {
|
|||
|
deps = undef;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
//Support require(['a'])
|
|||
|
callback = callback || function () {};
|
|||
|
|
|||
|
//If relName is a function, it is an errback handler,
|
|||
|
//so remove it.
|
|||
|
if (typeof relName === 'function') {
|
|||
|
relName = forceSync;
|
|||
|
forceSync = alt;
|
|||
|
}
|
|||
|
|
|||
|
//Simulate async callback;
|
|||
|
if (forceSync) {
|
|||
|
main(undef, deps, callback, relName);
|
|||
|
} else {
|
|||
|
//Using a non-zero value because of concern for what old browsers
|
|||
|
//do, and latest browsers "upgrade" to 4 if lower value is used:
|
|||
|
//http://www.whatwg.org/specs/web-apps/current-work/multipage/timers.html#dom-windowtimers-settimeout:
|
|||
|
//If want a value immediately, use require('id') instead -- something
|
|||
|
//that works in almond on the global level, but not guaranteed and
|
|||
|
//unlikely to work in other AMD implementations.
|
|||
|
setTimeout(function () {
|
|||
|
main(undef, deps, callback, relName);
|
|||
|
}, 4);
|
|||
|
}
|
|||
|
|
|||
|
return req;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Just drops the config on the floor, but returns req in case
|
|||
|
* the config return value is used.
|
|||
|
*/
|
|||
|
req.config = function (cfg) {
|
|||
|
return req(cfg);
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Expose module registry for debugging and tooling
|
|||
|
*/
|
|||
|
requirejs._defined = defined;
|
|||
|
|
|||
|
define = function (name, deps, callback) {
|
|||
|
|
|||
|
//This module may not have dependencies
|
|||
|
if (!deps.splice) {
|
|||
|
//deps is not an array, so probably means
|
|||
|
//an object literal or factory function for
|
|||
|
//the value. Adjust args.
|
|||
|
callback = deps;
|
|||
|
deps = [];
|
|||
|
}
|
|||
|
|
|||
|
if (!hasProp(defined, name) && !hasProp(waiting, name)) {
|
|||
|
waiting[name] = [name, deps, callback];
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
define.amd = {
|
|||
|
jQuery: true
|
|||
|
};
|
|||
|
}());
|
|||
|
|
|||
|
define("../../vendor/almond/almond", function(){});
|
|||
|
|
|||
|
/**
|
|||
|
* @module core/api_error
|
|||
|
*/
|
|||
|
define('core/api_error',["require", "exports"], function(require, exports) {
|
|||
|
/**
|
|||
|
* Standard libc error codes. Add more to this enum and ErrorStrings as they are
|
|||
|
* needed.
|
|||
|
* @url http://www.gnu.org/software/libc/manual/html_node/Error-Codes.html
|
|||
|
*/
|
|||
|
(function (ErrorCode) {
|
|||
|
ErrorCode[ErrorCode["EPERM"] = 0] = "EPERM";
|
|||
|
ErrorCode[ErrorCode["ENOENT"] = 1] = "ENOENT";
|
|||
|
ErrorCode[ErrorCode["EIO"] = 2] = "EIO";
|
|||
|
ErrorCode[ErrorCode["EBADF"] = 3] = "EBADF";
|
|||
|
ErrorCode[ErrorCode["EACCES"] = 4] = "EACCES";
|
|||
|
ErrorCode[ErrorCode["EBUSY"] = 5] = "EBUSY";
|
|||
|
ErrorCode[ErrorCode["EEXIST"] = 6] = "EEXIST";
|
|||
|
ErrorCode[ErrorCode["ENOTDIR"] = 7] = "ENOTDIR";
|
|||
|
ErrorCode[ErrorCode["EISDIR"] = 8] = "EISDIR";
|
|||
|
ErrorCode[ErrorCode["EINVAL"] = 9] = "EINVAL";
|
|||
|
ErrorCode[ErrorCode["EFBIG"] = 10] = "EFBIG";
|
|||
|
ErrorCode[ErrorCode["ENOSPC"] = 11] = "ENOSPC";
|
|||
|
ErrorCode[ErrorCode["EROFS"] = 12] = "EROFS";
|
|||
|
ErrorCode[ErrorCode["ENOTEMPTY"] = 13] = "ENOTEMPTY";
|
|||
|
ErrorCode[ErrorCode["ENOTSUP"] = 14] = "ENOTSUP";
|
|||
|
})(exports.ErrorCode || (exports.ErrorCode = {}));
|
|||
|
var ErrorCode = exports.ErrorCode;
|
|||
|
|
|||
|
/**
|
|||
|
* Strings associated with each error code.
|
|||
|
*/
|
|||
|
var ErrorStrings = {};
|
|||
|
ErrorStrings[0 /* EPERM */] = 'Operation not permitted.';
|
|||
|
ErrorStrings[1 /* ENOENT */] = 'No such file or directory.';
|
|||
|
ErrorStrings[2 /* EIO */] = 'Input/output error.';
|
|||
|
ErrorStrings[3 /* EBADF */] = 'Bad file descriptor.';
|
|||
|
ErrorStrings[4 /* EACCES */] = 'Permission denied.';
|
|||
|
ErrorStrings[5 /* EBUSY */] = 'Resource busy or locked.';
|
|||
|
ErrorStrings[6 /* EEXIST */] = 'File exists.';
|
|||
|
ErrorStrings[7 /* ENOTDIR */] = 'File is not a directory.';
|
|||
|
ErrorStrings[8 /* EISDIR */] = 'File is a directory.';
|
|||
|
ErrorStrings[9 /* EINVAL */] = 'Invalid argument.';
|
|||
|
ErrorStrings[10 /* EFBIG */] = 'File is too big.';
|
|||
|
ErrorStrings[11 /* ENOSPC */] = 'No space left on disk.';
|
|||
|
ErrorStrings[12 /* EROFS */] = 'Cannot modify a read-only file system.';
|
|||
|
ErrorStrings[13 /* ENOTEMPTY */] = 'Directory is not empty.';
|
|||
|
ErrorStrings[14 /* ENOTSUP */] = 'Operation is not supported.';
|
|||
|
|
|||
|
/**
|
|||
|
* Represents a BrowserFS error. Passed back to applications after a failed
|
|||
|
* call to the BrowserFS API.
|
|||
|
*/
|
|||
|
var ApiError = (function () {
|
|||
|
/**
|
|||
|
* Represents a BrowserFS error. Passed back to applications after a failed
|
|||
|
* call to the BrowserFS API.
|
|||
|
*
|
|||
|
* Error codes mirror those returned by regular Unix file operations, which is
|
|||
|
* what Node returns.
|
|||
|
* @constructor ApiError
|
|||
|
* @param type The type of the error.
|
|||
|
* @param [message] A descriptive error message.
|
|||
|
*/
|
|||
|
function ApiError(type, message) {
|
|||
|
this.type = type;
|
|||
|
this.code = ErrorCode[type];
|
|||
|
if (message != null) {
|
|||
|
this.message = message;
|
|||
|
} else {
|
|||
|
this.message = ErrorStrings[type];
|
|||
|
}
|
|||
|
}
|
|||
|
/**
|
|||
|
* @return A friendly error message.
|
|||
|
*/
|
|||
|
ApiError.prototype.toString = function () {
|
|||
|
return this.code + ": " + ErrorStrings[this.type] + " " + this.message;
|
|||
|
};
|
|||
|
|
|||
|
ApiError.FileError = function (code, p) {
|
|||
|
return new ApiError(code, p + ": " + ErrorStrings[code]);
|
|||
|
};
|
|||
|
ApiError.ENOENT = function (path) {
|
|||
|
return this.FileError(1 /* ENOENT */, path);
|
|||
|
};
|
|||
|
|
|||
|
ApiError.EEXIST = function (path) {
|
|||
|
return this.FileError(6 /* EEXIST */, path);
|
|||
|
};
|
|||
|
|
|||
|
ApiError.EISDIR = function (path) {
|
|||
|
return this.FileError(8 /* EISDIR */, path);
|
|||
|
};
|
|||
|
|
|||
|
ApiError.ENOTDIR = function (path) {
|
|||
|
return this.FileError(7 /* ENOTDIR */, path);
|
|||
|
};
|
|||
|
|
|||
|
ApiError.EPERM = function (path) {
|
|||
|
return this.FileError(0 /* EPERM */, path);
|
|||
|
};
|
|||
|
return ApiError;
|
|||
|
})();
|
|||
|
exports.ApiError = ApiError;
|
|||
|
});
|
|||
|
//# sourceMappingURL=api_error.js.map
|
|||
|
;
|
|||
|
define('core/buffer_core',["require", "exports", './api_error'], function(require, exports, api_error) {
|
|||
|
var FLOAT_POS_INFINITY = Math.pow(2, 128);
|
|||
|
var FLOAT_NEG_INFINITY = -1 * FLOAT_POS_INFINITY;
|
|||
|
var FLOAT_POS_INFINITY_AS_INT = 0x7F800000;
|
|||
|
var FLOAT_NEG_INFINITY_AS_INT = -8388608;
|
|||
|
var FLOAT_NaN_AS_INT = 0x7fc00000;
|
|||
|
|
|||
|
|
|||
|
|
|||
|
/**
|
|||
|
* Contains common definitions for most of the BufferCore classes.
|
|||
|
* Subclasses only need to implement write/readUInt8 for full functionality.
|
|||
|
*/
|
|||
|
var BufferCoreCommon = (function () {
|
|||
|
function BufferCoreCommon() {
|
|||
|
}
|
|||
|
BufferCoreCommon.prototype.getLength = function () {
|
|||
|
throw new api_error.ApiError(14 /* ENOTSUP */, 'BufferCore implementations should implement getLength.');
|
|||
|
};
|
|||
|
BufferCoreCommon.prototype.writeInt8 = function (i, data) {
|
|||
|
// Pack the sign bit as the highest bit.
|
|||
|
// Note that we keep the highest bit in the value byte as the sign bit if it
|
|||
|
// exists.
|
|||
|
this.writeUInt8(i, (data & 0xFF) | ((data & 0x80000000) >>> 24));
|
|||
|
};
|
|||
|
BufferCoreCommon.prototype.writeInt16LE = function (i, data) {
|
|||
|
this.writeUInt8(i, data & 0xFF);
|
|||
|
|
|||
|
// Pack the sign bit as the highest bit.
|
|||
|
// Note that we keep the highest bit in the value byte as the sign bit if it
|
|||
|
// exists.
|
|||
|
this.writeUInt8(i + 1, ((data >>> 8) & 0xFF) | ((data & 0x80000000) >>> 24));
|
|||
|
};
|
|||
|
BufferCoreCommon.prototype.writeInt16BE = function (i, data) {
|
|||
|
this.writeUInt8(i + 1, data & 0xFF);
|
|||
|
|
|||
|
// Pack the sign bit as the highest bit.
|
|||
|
// Note that we keep the highest bit in the value byte as the sign bit if it
|
|||
|
// exists.
|
|||
|
this.writeUInt8(i, ((data >>> 8) & 0xFF) | ((data & 0x80000000) >>> 24));
|
|||
|
};
|
|||
|
BufferCoreCommon.prototype.writeInt32LE = function (i, data) {
|
|||
|
this.writeUInt8(i, data & 0xFF);
|
|||
|
this.writeUInt8(i + 1, (data >>> 8) & 0xFF);
|
|||
|
this.writeUInt8(i + 2, (data >>> 16) & 0xFF);
|
|||
|
this.writeUInt8(i + 3, (data >>> 24) & 0xFF);
|
|||
|
};
|
|||
|
BufferCoreCommon.prototype.writeInt32BE = function (i, data) {
|
|||
|
this.writeUInt8(i + 3, data & 0xFF);
|
|||
|
this.writeUInt8(i + 2, (data >>> 8) & 0xFF);
|
|||
|
this.writeUInt8(i + 1, (data >>> 16) & 0xFF);
|
|||
|
this.writeUInt8(i, (data >>> 24) & 0xFF);
|
|||
|
};
|
|||
|
BufferCoreCommon.prototype.writeUInt8 = function (i, data) {
|
|||
|
throw new api_error.ApiError(14 /* ENOTSUP */, 'BufferCore implementations should implement writeUInt8.');
|
|||
|
};
|
|||
|
BufferCoreCommon.prototype.writeUInt16LE = function (i, data) {
|
|||
|
this.writeUInt8(i, data & 0xFF);
|
|||
|
this.writeUInt8(i + 1, (data >> 8) & 0xFF);
|
|||
|
};
|
|||
|
BufferCoreCommon.prototype.writeUInt16BE = function (i, data) {
|
|||
|
this.writeUInt8(i + 1, data & 0xFF);
|
|||
|
this.writeUInt8(i, (data >> 8) & 0xFF);
|
|||
|
};
|
|||
|
BufferCoreCommon.prototype.writeUInt32LE = function (i, data) {
|
|||
|
this.writeInt32LE(i, data | 0);
|
|||
|
};
|
|||
|
BufferCoreCommon.prototype.writeUInt32BE = function (i, data) {
|
|||
|
this.writeInt32BE(i, data | 0);
|
|||
|
};
|
|||
|
BufferCoreCommon.prototype.writeFloatLE = function (i, data) {
|
|||
|
this.writeInt32LE(i, this.float2intbits(data));
|
|||
|
};
|
|||
|
BufferCoreCommon.prototype.writeFloatBE = function (i, data) {
|
|||
|
this.writeInt32BE(i, this.float2intbits(data));
|
|||
|
};
|
|||
|
BufferCoreCommon.prototype.writeDoubleLE = function (i, data) {
|
|||
|
var doubleBits = this.double2longbits(data);
|
|||
|
this.writeInt32LE(i, doubleBits[0]);
|
|||
|
this.writeInt32LE(i + 4, doubleBits[1]);
|
|||
|
};
|
|||
|
BufferCoreCommon.prototype.writeDoubleBE = function (i, data) {
|
|||
|
var doubleBits = this.double2longbits(data);
|
|||
|
this.writeInt32BE(i + 4, doubleBits[0]);
|
|||
|
this.writeInt32BE(i, doubleBits[1]);
|
|||
|
};
|
|||
|
BufferCoreCommon.prototype.readInt8 = function (i) {
|
|||
|
var val = this.readUInt8(i);
|
|||
|
if (val & 0x80) {
|
|||
|
// Sign bit is set, so perform sign extension.
|
|||
|
return val | 0xFFFFFF80;
|
|||
|
} else {
|
|||
|
return val;
|
|||
|
}
|
|||
|
};
|
|||
|
BufferCoreCommon.prototype.readInt16LE = function (i) {
|
|||
|
var val = this.readUInt16LE(i);
|
|||
|
if (val & 0x8000) {
|
|||
|
// Sign bit is set, so perform sign extension.
|
|||
|
return val | 0xFFFF8000;
|
|||
|
} else {
|
|||
|
return val;
|
|||
|
}
|
|||
|
};
|
|||
|
BufferCoreCommon.prototype.readInt16BE = function (i) {
|
|||
|
var val = this.readUInt16BE(i);
|
|||
|
if (val & 0x8000) {
|
|||
|
// Sign bit is set, so perform sign extension.
|
|||
|
return val | 0xFFFF8000;
|
|||
|
} else {
|
|||
|
return val;
|
|||
|
}
|
|||
|
};
|
|||
|
BufferCoreCommon.prototype.readInt32LE = function (i) {
|
|||
|
return this.readUInt32LE(i) | 0;
|
|||
|
};
|
|||
|
BufferCoreCommon.prototype.readInt32BE = function (i) {
|
|||
|
return this.readUInt32BE(i) | 0;
|
|||
|
};
|
|||
|
BufferCoreCommon.prototype.readUInt8 = function (i) {
|
|||
|
throw new api_error.ApiError(14 /* ENOTSUP */, 'BufferCore implementations should implement readUInt8.');
|
|||
|
};
|
|||
|
BufferCoreCommon.prototype.readUInt16LE = function (i) {
|
|||
|
return (this.readUInt8(i + 1) << 8) | this.readUInt8(i);
|
|||
|
};
|
|||
|
BufferCoreCommon.prototype.readUInt16BE = function (i) {
|
|||
|
return (this.readUInt8(i) << 8) | this.readUInt8(i + 1);
|
|||
|
};
|
|||
|
BufferCoreCommon.prototype.readUInt32LE = function (i) {
|
|||
|
return ((this.readUInt8(i + 3) << 24) | (this.readUInt8(i + 2) << 16) | (this.readUInt8(i + 1) << 8) | this.readUInt8(i)) >>> 0;
|
|||
|
};
|
|||
|
BufferCoreCommon.prototype.readUInt32BE = function (i) {
|
|||
|
return ((this.readUInt8(i) << 24) | (this.readUInt8(i + 1) << 16) | (this.readUInt8(i + 2) << 8) | this.readUInt8(i + 3)) >>> 0;
|
|||
|
};
|
|||
|
BufferCoreCommon.prototype.readFloatLE = function (i) {
|
|||
|
return this.intbits2float(this.readInt32LE(i));
|
|||
|
};
|
|||
|
BufferCoreCommon.prototype.readFloatBE = function (i) {
|
|||
|
return this.intbits2float(this.readInt32BE(i));
|
|||
|
};
|
|||
|
BufferCoreCommon.prototype.readDoubleLE = function (i) {
|
|||
|
return this.longbits2double(this.readInt32LE(i + 4), this.readInt32LE(i));
|
|||
|
};
|
|||
|
BufferCoreCommon.prototype.readDoubleBE = function (i) {
|
|||
|
return this.longbits2double(this.readInt32BE(i), this.readInt32BE(i + 4));
|
|||
|
};
|
|||
|
BufferCoreCommon.prototype.copy = function (start, end) {
|
|||
|
throw new api_error.ApiError(14 /* ENOTSUP */, 'BufferCore implementations should implement copy.');
|
|||
|
};
|
|||
|
BufferCoreCommon.prototype.fill = function (value, start, end) {
|
|||
|
for (var i = start; i < end; i++) {
|
|||
|
this.writeUInt8(i, value);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
BufferCoreCommon.prototype.float2intbits = function (f_val) {
|
|||
|
var exp, f_view, i_view, sig, sign;
|
|||
|
|
|||
|
// Special cases!
|
|||
|
if (f_val === 0) {
|
|||
|
return 0;
|
|||
|
}
|
|||
|
|
|||
|
// We map the infinities to JavaScript infinities. Map them back.
|
|||
|
if (f_val === Number.POSITIVE_INFINITY) {
|
|||
|
return FLOAT_POS_INFINITY_AS_INT;
|
|||
|
}
|
|||
|
if (f_val === Number.NEGATIVE_INFINITY) {
|
|||
|
return FLOAT_NEG_INFINITY_AS_INT;
|
|||
|
}
|
|||
|
|
|||
|
// Convert JavaScript NaN to Float NaN value.
|
|||
|
if (isNaN(f_val)) {
|
|||
|
return FLOAT_NaN_AS_INT;
|
|||
|
}
|
|||
|
|
|||
|
// We have more bits of precision than a float, so below we round to
|
|||
|
// the nearest significand. This appears to be what the x86
|
|||
|
// Java does for normal floating point operations.
|
|||
|
sign = f_val < 0 ? 1 : 0;
|
|||
|
f_val = Math.abs(f_val);
|
|||
|
|
|||
|
// Subnormal zone!
|
|||
|
// (−1)^signbits×2^−126×0.significandbits
|
|||
|
// Largest subnormal magnitude:
|
|||
|
// 0000 0000 0111 1111 1111 1111 1111 1111
|
|||
|
// Smallest subnormal magnitude:
|
|||
|
// 0000 0000 0000 0000 0000 0000 0000 0001
|
|||
|
if (f_val <= 1.1754942106924411e-38 && f_val >= 1.4012984643248170e-45) {
|
|||
|
exp = 0;
|
|||
|
sig = Math.round((f_val / Math.pow(2, -126)) * Math.pow(2, 23));
|
|||
|
return (sign << 31) | (exp << 23) | sig;
|
|||
|
} else {
|
|||
|
// Regular FP numbers
|
|||
|
exp = Math.floor(Math.log(f_val) / Math.LN2);
|
|||
|
sig = Math.round((f_val / Math.pow(2, exp) - 1) * Math.pow(2, 23));
|
|||
|
return (sign << 31) | ((exp + 127) << 23) | sig;
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
BufferCoreCommon.prototype.double2longbits = function (d_val) {
|
|||
|
var d_view, exp, high_bits, i_view, sig, sign;
|
|||
|
|
|||
|
// Special cases
|
|||
|
if (d_val === 0) {
|
|||
|
return [0, 0];
|
|||
|
}
|
|||
|
if (d_val === Number.POSITIVE_INFINITY) {
|
|||
|
// High bits: 0111 1111 1111 0000 0000 0000 0000 0000
|
|||
|
// Low bits: 0000 0000 0000 0000 0000 0000 0000 0000
|
|||
|
return [0, 2146435072];
|
|||
|
} else if (d_val === Number.NEGATIVE_INFINITY) {
|
|||
|
// High bits: 1111 1111 1111 0000 0000 0000 0000 0000
|
|||
|
// Low bits: 0000 0000 0000 0000 0000 0000 0000 0000
|
|||
|
return [0, -1048576];
|
|||
|
} else if (isNaN(d_val)) {
|
|||
|
// High bits: 0111 1111 1111 1000 0000 0000 0000 0000
|
|||
|
// Low bits: 0000 0000 0000 0000 0000 0000 0000 0000
|
|||
|
return [0, 2146959360];
|
|||
|
}
|
|||
|
sign = d_val < 0 ? 1 << 31 : 0;
|
|||
|
d_val = Math.abs(d_val);
|
|||
|
|
|||
|
// Check if it is a subnormal number.
|
|||
|
// (-1)s × 0.f × 2-1022
|
|||
|
// Largest subnormal magnitude:
|
|||
|
// 0000 0000 0000 1111 1111 1111 1111 1111
|
|||
|
// 1111 1111 1111 1111 1111 1111 1111 1111
|
|||
|
// Smallest subnormal magnitude:
|
|||
|
// 0000 0000 0000 0000 0000 0000 0000 0000
|
|||
|
// 0000 0000 0000 0000 0000 0000 0000 0001
|
|||
|
if (d_val <= 2.2250738585072010e-308 && d_val >= 5.0000000000000000e-324) {
|
|||
|
exp = 0;
|
|||
|
sig = (d_val / Math.pow(2, -1022)) * Math.pow(2, 52);
|
|||
|
} else {
|
|||
|
exp = Math.floor(Math.log(d_val) / Math.LN2);
|
|||
|
|
|||
|
// If d_val is close to a power of two, there's a chance that exp
|
|||
|
// will be 1 greater than it should due to loss of accuracy in the
|
|||
|
// log result.
|
|||
|
if (d_val < Math.pow(2, exp)) {
|
|||
|
exp = exp - 1;
|
|||
|
}
|
|||
|
sig = (d_val / Math.pow(2, exp) - 1) * Math.pow(2, 52);
|
|||
|
exp = (exp + 1023) << 20;
|
|||
|
}
|
|||
|
|
|||
|
// Simulate >> 32
|
|||
|
high_bits = ((sig * Math.pow(2, -32)) | 0) | sign | exp;
|
|||
|
return [sig & 0xFFFF, high_bits];
|
|||
|
};
|
|||
|
|
|||
|
BufferCoreCommon.prototype.intbits2float = function (int32) {
|
|||
|
// Map +/- infinity to JavaScript equivalents
|
|||
|
if (int32 === FLOAT_POS_INFINITY_AS_INT) {
|
|||
|
return Number.POSITIVE_INFINITY;
|
|||
|
} else if (int32 === FLOAT_NEG_INFINITY_AS_INT) {
|
|||
|
return Number.NEGATIVE_INFINITY;
|
|||
|
}
|
|||
|
var sign = (int32 & 0x80000000) >>> 31;
|
|||
|
var exponent = (int32 & 0x7F800000) >>> 23;
|
|||
|
var significand = int32 & 0x007FFFFF;
|
|||
|
var value;
|
|||
|
if (exponent === 0) {
|
|||
|
value = Math.pow(-1, sign) * significand * Math.pow(2, -149);
|
|||
|
} else {
|
|||
|
value = Math.pow(-1, sign) * (1 + significand * Math.pow(2, -23)) * Math.pow(2, exponent - 127);
|
|||
|
}
|
|||
|
|
|||
|
// NaN check
|
|||
|
if (value < FLOAT_NEG_INFINITY || value > FLOAT_POS_INFINITY) {
|
|||
|
value = NaN;
|
|||
|
}
|
|||
|
return value;
|
|||
|
};
|
|||
|
|
|||
|
BufferCoreCommon.prototype.longbits2double = function (uint32_a, uint32_b) {
|
|||
|
var sign = (uint32_a & 0x80000000) >>> 31;
|
|||
|
var exponent = (uint32_a & 0x7FF00000) >>> 20;
|
|||
|
var significand = ((uint32_a & 0x000FFFFF) * Math.pow(2, 32)) + uint32_b;
|
|||
|
|
|||
|
// Special values!
|
|||
|
if (exponent === 0 && significand === 0) {
|
|||
|
return 0;
|
|||
|
}
|
|||
|
if (exponent === 2047) {
|
|||
|
if (significand === 0) {
|
|||
|
if (sign === 1) {
|
|||
|
return Number.NEGATIVE_INFINITY;
|
|||
|
}
|
|||
|
return Number.POSITIVE_INFINITY;
|
|||
|
} else {
|
|||
|
return NaN;
|
|||
|
}
|
|||
|
}
|
|||
|
if (exponent === 0)
|
|||
|
return Math.pow(-1, sign) * significand * Math.pow(2, -1074);
|
|||
|
return Math.pow(-1, sign) * (1 + significand * Math.pow(2, -52)) * Math.pow(2, exponent - 1023);
|
|||
|
};
|
|||
|
return BufferCoreCommon;
|
|||
|
})();
|
|||
|
exports.BufferCoreCommon = BufferCoreCommon;
|
|||
|
});
|
|||
|
//# sourceMappingURL=buffer_core.js.map
|
|||
|
;
|
|||
|
var __extends = this.__extends || function (d, b) {
|
|||
|
for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p];
|
|||
|
function __() { this.constructor = d; }
|
|||
|
__.prototype = b.prototype;
|
|||
|
d.prototype = new __();
|
|||
|
};
|
|||
|
define('core/buffer_core_array',["require", "exports", './buffer_core'], function(require, exports, buffer_core) {
|
|||
|
// Used to clear segments of an array index.
|
|||
|
var clearMasks = [0xFFFFFF00, 0xFFFF00FF, 0xFF00FFFF, 0x00FFFFFF];
|
|||
|
|
|||
|
/**
|
|||
|
* Implementation of BufferCore that is backed by an array of 32-bit ints.
|
|||
|
* Data is stored little endian.
|
|||
|
* Example: Bytes 0 through 3 are present in the first int:
|
|||
|
* BYTE 3 BYTE 2 BYTE 1 BYTE 0
|
|||
|
* 0000 0000 | 0000 0000 | 0000 0000 | 0000 0000
|
|||
|
*/
|
|||
|
var BufferCoreArray = (function (_super) {
|
|||
|
__extends(BufferCoreArray, _super);
|
|||
|
function BufferCoreArray(length) {
|
|||
|
_super.call(this);
|
|||
|
this.length = length;
|
|||
|
this.buff = new Array(Math.ceil(length / 4));
|
|||
|
|
|||
|
// Zero-fill the array.
|
|||
|
var bufflen = this.buff.length;
|
|||
|
for (var i = 0; i < bufflen; i++) {
|
|||
|
this.buff[i] = 0;
|
|||
|
}
|
|||
|
}
|
|||
|
BufferCoreArray.isAvailable = function () {
|
|||
|
return true;
|
|||
|
};
|
|||
|
|
|||
|
BufferCoreArray.prototype.getLength = function () {
|
|||
|
return this.length;
|
|||
|
};
|
|||
|
BufferCoreArray.prototype.writeUInt8 = function (i, data) {
|
|||
|
data &= 0xFF;
|
|||
|
|
|||
|
// Which int? (Equivalent to (i/4)|0)
|
|||
|
var arrIdx = i >> 2;
|
|||
|
|
|||
|
// Which offset? (Equivalent to i - arrIdx*4)
|
|||
|
var intIdx = i & 3;
|
|||
|
this.buff[arrIdx] = this.buff[arrIdx] & clearMasks[intIdx];
|
|||
|
this.buff[arrIdx] = this.buff[arrIdx] | (data << (intIdx << 3));
|
|||
|
};
|
|||
|
BufferCoreArray.prototype.readUInt8 = function (i) {
|
|||
|
// Which int?
|
|||
|
var arrIdx = i >> 2;
|
|||
|
|
|||
|
// Which offset?
|
|||
|
var intIdx = i & 3;
|
|||
|
|
|||
|
// Bring the data we want into the lowest 8 bits, and truncate.
|
|||
|
return (this.buff[arrIdx] >> (intIdx << 3)) & 0xFF;
|
|||
|
};
|
|||
|
BufferCoreArray.prototype.copy = function (start, end) {
|
|||
|
// Stupid unoptimized copy. Later, we could do optimizations when aligned.
|
|||
|
var newBC = new BufferCoreArray(end - start);
|
|||
|
for (var i = start; i < end; i++) {
|
|||
|
newBC.writeUInt8(i - start, this.readUInt8(i));
|
|||
|
}
|
|||
|
return newBC;
|
|||
|
};
|
|||
|
return BufferCoreArray;
|
|||
|
})(buffer_core.BufferCoreCommon);
|
|||
|
exports.BufferCoreArray = BufferCoreArray;
|
|||
|
|
|||
|
// Type-check the class.
|
|||
|
var _ = BufferCoreArray;
|
|||
|
});
|
|||
|
//# sourceMappingURL=buffer_core_array.js.map
|
|||
|
;
|
|||
|
var __extends = this.__extends || function (d, b) {
|
|||
|
for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p];
|
|||
|
function __() { this.constructor = d; }
|
|||
|
__.prototype = b.prototype;
|
|||
|
d.prototype = new __();
|
|||
|
};
|
|||
|
define('core/buffer_core_arraybuffer',["require", "exports", './buffer_core'], function(require, exports, buffer_core) {
|
|||
|
/**
|
|||
|
* Represents data using an ArrayBuffer.
|
|||
|
*/
|
|||
|
var BufferCoreArrayBuffer = (function (_super) {
|
|||
|
__extends(BufferCoreArrayBuffer, _super);
|
|||
|
function BufferCoreArrayBuffer(arg1) {
|
|||
|
_super.call(this);
|
|||
|
if (typeof arg1 === 'number') {
|
|||
|
this.buff = new DataView(new ArrayBuffer(arg1));
|
|||
|
} else if (arg1 instanceof DataView) {
|
|||
|
this.buff = arg1;
|
|||
|
} else {
|
|||
|
this.buff = new DataView(arg1);
|
|||
|
}
|
|||
|
this.length = this.buff.byteLength;
|
|||
|
}
|
|||
|
BufferCoreArrayBuffer.isAvailable = function () {
|
|||
|
return typeof DataView !== 'undefined';
|
|||
|
};
|
|||
|
|
|||
|
BufferCoreArrayBuffer.prototype.getLength = function () {
|
|||
|
return this.length;
|
|||
|
};
|
|||
|
BufferCoreArrayBuffer.prototype.writeInt8 = function (i, data) {
|
|||
|
this.buff.setInt8(i, data);
|
|||
|
};
|
|||
|
BufferCoreArrayBuffer.prototype.writeInt16LE = function (i, data) {
|
|||
|
this.buff.setInt16(i, data, true);
|
|||
|
};
|
|||
|
BufferCoreArrayBuffer.prototype.writeInt16BE = function (i, data) {
|
|||
|
this.buff.setInt16(i, data, false);
|
|||
|
};
|
|||
|
BufferCoreArrayBuffer.prototype.writeInt32LE = function (i, data) {
|
|||
|
this.buff.setInt32(i, data, true);
|
|||
|
};
|
|||
|
BufferCoreArrayBuffer.prototype.writeInt32BE = function (i, data) {
|
|||
|
this.buff.setInt32(i, data, false);
|
|||
|
};
|
|||
|
BufferCoreArrayBuffer.prototype.writeUInt8 = function (i, data) {
|
|||
|
this.buff.setUint8(i, data);
|
|||
|
};
|
|||
|
BufferCoreArrayBuffer.prototype.writeUInt16LE = function (i, data) {
|
|||
|
this.buff.setUint16(i, data, true);
|
|||
|
};
|
|||
|
BufferCoreArrayBuffer.prototype.writeUInt16BE = function (i, data) {
|
|||
|
this.buff.setUint16(i, data, false);
|
|||
|
};
|
|||
|
BufferCoreArrayBuffer.prototype.writeUInt32LE = function (i, data) {
|
|||
|
this.buff.setUint32(i, data, true);
|
|||
|
};
|
|||
|
BufferCoreArrayBuffer.prototype.writeUInt32BE = function (i, data) {
|
|||
|
this.buff.setUint32(i, data, false);
|
|||
|
};
|
|||
|
BufferCoreArrayBuffer.prototype.writeFloatLE = function (i, data) {
|
|||
|
this.buff.setFloat32(i, data, true);
|
|||
|
};
|
|||
|
BufferCoreArrayBuffer.prototype.writeFloatBE = function (i, data) {
|
|||
|
this.buff.setFloat32(i, data, false);
|
|||
|
};
|
|||
|
BufferCoreArrayBuffer.prototype.writeDoubleLE = function (i, data) {
|
|||
|
this.buff.setFloat64(i, data, true);
|
|||
|
};
|
|||
|
BufferCoreArrayBuffer.prototype.writeDoubleBE = function (i, data) {
|
|||
|
this.buff.setFloat64(i, data, false);
|
|||
|
};
|
|||
|
BufferCoreArrayBuffer.prototype.readInt8 = function (i) {
|
|||
|
return this.buff.getInt8(i);
|
|||
|
};
|
|||
|
BufferCoreArrayBuffer.prototype.readInt16LE = function (i) {
|
|||
|
return this.buff.getInt16(i, true);
|
|||
|
};
|
|||
|
BufferCoreArrayBuffer.prototype.readInt16BE = function (i) {
|
|||
|
return this.buff.getInt16(i, false);
|
|||
|
};
|
|||
|
BufferCoreArrayBuffer.prototype.readInt32LE = function (i) {
|
|||
|
return this.buff.getInt32(i, true);
|
|||
|
};
|
|||
|
BufferCoreArrayBuffer.prototype.readInt32BE = function (i) {
|
|||
|
return this.buff.getInt32(i, false);
|
|||
|
};
|
|||
|
BufferCoreArrayBuffer.prototype.readUInt8 = function (i) {
|
|||
|
return this.buff.getUint8(i);
|
|||
|
};
|
|||
|
BufferCoreArrayBuffer.prototype.readUInt16LE = function (i) {
|
|||
|
return this.buff.getUint16(i, true);
|
|||
|
};
|
|||
|
BufferCoreArrayBuffer.prototype.readUInt16BE = function (i) {
|
|||
|
return this.buff.getUint16(i, false);
|
|||
|
};
|
|||
|
BufferCoreArrayBuffer.prototype.readUInt32LE = function (i) {
|
|||
|
return this.buff.getUint32(i, true);
|
|||
|
};
|
|||
|
BufferCoreArrayBuffer.prototype.readUInt32BE = function (i) {
|
|||
|
return this.buff.getUint32(i, false);
|
|||
|
};
|
|||
|
BufferCoreArrayBuffer.prototype.readFloatLE = function (i) {
|
|||
|
return this.buff.getFloat32(i, true);
|
|||
|
};
|
|||
|
BufferCoreArrayBuffer.prototype.readFloatBE = function (i) {
|
|||
|
return this.buff.getFloat32(i, false);
|
|||
|
};
|
|||
|
BufferCoreArrayBuffer.prototype.readDoubleLE = function (i) {
|
|||
|
return this.buff.getFloat64(i, true);
|
|||
|
};
|
|||
|
BufferCoreArrayBuffer.prototype.readDoubleBE = function (i) {
|
|||
|
return this.buff.getFloat64(i, false);
|
|||
|
};
|
|||
|
BufferCoreArrayBuffer.prototype.copy = function (start, end) {
|
|||
|
var aBuff = this.buff.buffer;
|
|||
|
var newBuff;
|
|||
|
|
|||
|
// Some ArrayBuffer implementations (IE10) do not have 'slice'.
|
|||
|
// XXX: Type hacks - the typings don't have slice either.
|
|||
|
if (ArrayBuffer.prototype.slice) {
|
|||
|
// ArrayBuffer.slice is copying; exactly what we want.
|
|||
|
newBuff = aBuff.slice(start, end);
|
|||
|
} else {
|
|||
|
var len = end - start;
|
|||
|
newBuff = new ArrayBuffer(len);
|
|||
|
|
|||
|
// Copy the old contents in.
|
|||
|
var newUintArray = new Uint8Array(newBuff);
|
|||
|
var oldUintArray = new Uint8Array(aBuff);
|
|||
|
newUintArray.set(oldUintArray.subarray(start, end));
|
|||
|
}
|
|||
|
return new BufferCoreArrayBuffer(newBuff);
|
|||
|
};
|
|||
|
BufferCoreArrayBuffer.prototype.fill = function (value, start, end) {
|
|||
|
// Value must be a byte wide.
|
|||
|
value = value & 0xFF;
|
|||
|
var i;
|
|||
|
var len = end - start;
|
|||
|
var intBytes = (((len) / 4) | 0) * 4;
|
|||
|
|
|||
|
// Optimization: Write 4 bytes at a time.
|
|||
|
// TODO: Could we copy 8 bytes at a time using Float64, or could we
|
|||
|
// lose precision?
|
|||
|
var intVal = (value << 24) | (value << 16) | (value << 8) | value;
|
|||
|
for (i = 0; i < intBytes; i += 4) {
|
|||
|
this.writeInt32LE(i + start, intVal);
|
|||
|
}
|
|||
|
for (i = intBytes; i < len; i++) {
|
|||
|
this.writeUInt8(i + start, value);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Custom method for this buffer core. Get the backing object.
|
|||
|
*/
|
|||
|
BufferCoreArrayBuffer.prototype.getDataView = function () {
|
|||
|
return this.buff;
|
|||
|
};
|
|||
|
return BufferCoreArrayBuffer;
|
|||
|
})(buffer_core.BufferCoreCommon);
|
|||
|
exports.BufferCoreArrayBuffer = BufferCoreArrayBuffer;
|
|||
|
|
|||
|
// Type-check the class.
|
|||
|
var _ = BufferCoreArrayBuffer;
|
|||
|
});
|
|||
|
//# sourceMappingURL=buffer_core_arraybuffer.js.map
|
|||
|
;
|
|||
|
var __extends = this.__extends || function (d, b) {
|
|||
|
for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p];
|
|||
|
function __() { this.constructor = d; }
|
|||
|
__.prototype = b.prototype;
|
|||
|
d.prototype = new __();
|
|||
|
};
|
|||
|
define('core/buffer_core_imagedata',["require", "exports", './buffer_core'], function(require, exports, buffer_core) {
|
|||
|
|
|||
|
|
|||
|
/**
|
|||
|
* Implementation of BufferCore that is backed by an ImageData object.
|
|||
|
* Useful in browsers with HTML5 canvas support, but no TypedArray support
|
|||
|
* (IE9).
|
|||
|
*/
|
|||
|
var BufferCoreImageData = (function (_super) {
|
|||
|
__extends(BufferCoreImageData, _super);
|
|||
|
function BufferCoreImageData(length) {
|
|||
|
_super.call(this);
|
|||
|
this.length = length;
|
|||
|
this.buff = BufferCoreImageData.getCanvasPixelArray(length);
|
|||
|
}
|
|||
|
/**
|
|||
|
* Constructs a CanvasPixelArray that represents the given amount of bytes.
|
|||
|
*/
|
|||
|
BufferCoreImageData.getCanvasPixelArray = function (bytes) {
|
|||
|
var ctx = BufferCoreImageData.imageDataFactory;
|
|||
|
|
|||
|
// Lazily initialize, otherwise every browser (even those that will never
|
|||
|
// use this code) will create a canvas on script load.
|
|||
|
if (ctx === undefined) {
|
|||
|
BufferCoreImageData.imageDataFactory = ctx = document.createElement('canvas').getContext('2d');
|
|||
|
}
|
|||
|
|
|||
|
// You cannot create image data with size 0, so up it to size 1.
|
|||
|
if (bytes === 0)
|
|||
|
bytes = 1;
|
|||
|
return ctx.createImageData(Math.ceil(bytes / 4), 1).data;
|
|||
|
};
|
|||
|
BufferCoreImageData.isAvailable = function () {
|
|||
|
// Modern browsers have removed this deprecated API, so it is not always around.
|
|||
|
return typeof CanvasPixelArray !== 'undefined';
|
|||
|
};
|
|||
|
|
|||
|
BufferCoreImageData.prototype.getLength = function () {
|
|||
|
return this.length;
|
|||
|
};
|
|||
|
BufferCoreImageData.prototype.writeUInt8 = function (i, data) {
|
|||
|
this.buff[i] = data;
|
|||
|
};
|
|||
|
BufferCoreImageData.prototype.readUInt8 = function (i) {
|
|||
|
return this.buff[i];
|
|||
|
};
|
|||
|
BufferCoreImageData.prototype.copy = function (start, end) {
|
|||
|
// AFAIK, there's no efficient way to clone ImageData.
|
|||
|
var newBC = new BufferCoreImageData(end - start);
|
|||
|
for (var i = start; i < end; i++) {
|
|||
|
newBC.writeUInt8(i - start, this.buff[i]);
|
|||
|
}
|
|||
|
return newBC;
|
|||
|
};
|
|||
|
return BufferCoreImageData;
|
|||
|
})(buffer_core.BufferCoreCommon);
|
|||
|
exports.BufferCoreImageData = BufferCoreImageData;
|
|||
|
|
|||
|
// Type-check the class.
|
|||
|
var _ = BufferCoreImageData;
|
|||
|
});
|
|||
|
//# sourceMappingURL=buffer_core_imagedata.js.map
|
|||
|
;
|
|||
|
define('core/string_util',["require", "exports"], function(require, exports) {
|
|||
|
|
|||
|
|
|||
|
/**
|
|||
|
* Find the 'utility' object for the given string encoding. Throws an exception
|
|||
|
* if the encoding is invalid.
|
|||
|
* @param [String] encoding a string encoding
|
|||
|
* @return [BrowserFS.StringUtil.*] The StringUtil object for the given encoding
|
|||
|
*/
|
|||
|
function FindUtil(encoding) {
|
|||
|
encoding = (function () {
|
|||
|
switch (typeof encoding) {
|
|||
|
case 'object':
|
|||
|
return "" + encoding;
|
|||
|
case 'string':
|
|||
|
return encoding;
|
|||
|
default:
|
|||
|
throw new Error('Invalid encoding argument specified');
|
|||
|
}
|
|||
|
})();
|
|||
|
encoding = encoding.toLowerCase();
|
|||
|
|
|||
|
switch (encoding) {
|
|||
|
case 'utf8':
|
|||
|
case 'utf-8':
|
|||
|
return UTF8;
|
|||
|
case 'ascii':
|
|||
|
return ASCII;
|
|||
|
case 'binary':
|
|||
|
return BINARY;
|
|||
|
case 'ucs2':
|
|||
|
case 'ucs-2':
|
|||
|
case 'utf16le':
|
|||
|
case 'utf-16le':
|
|||
|
return UCS2;
|
|||
|
case 'hex':
|
|||
|
return HEX;
|
|||
|
case 'base64':
|
|||
|
return BASE64;
|
|||
|
|
|||
|
case 'binary_string':
|
|||
|
return BINSTR;
|
|||
|
case 'binary_string_ie':
|
|||
|
return BINSTRIE;
|
|||
|
case 'extended_ascii':
|
|||
|
return ExtendedASCII;
|
|||
|
|
|||
|
default:
|
|||
|
throw new Error("Unknown encoding: " + encoding);
|
|||
|
}
|
|||
|
}
|
|||
|
exports.FindUtil = FindUtil;
|
|||
|
|
|||
|
/**
|
|||
|
* String utility functions for UTF-8. Note that some UTF-8 strings *cannot* be
|
|||
|
* expressed in terms of JavaScript UTF-16 strings.
|
|||
|
* @see http://en.wikipedia.org/wiki/UTF-8
|
|||
|
*/
|
|||
|
var UTF8 = (function () {
|
|||
|
function UTF8() {
|
|||
|
}
|
|||
|
UTF8.str2byte = function (str, buf) {
|
|||
|
var length = buf.length;
|
|||
|
var i = 0;
|
|||
|
var j = 0;
|
|||
|
var maxJ = length;
|
|||
|
var rv = [];
|
|||
|
var numChars = 0;
|
|||
|
while (i < str.length && j < maxJ) {
|
|||
|
var code = str.charCodeAt(i++);
|
|||
|
var next = str.charCodeAt(i);
|
|||
|
if (0xD800 <= code && code <= 0xDBFF && 0xDC00 <= next && next <= 0xDFFF) {
|
|||
|
// 4 bytes: Surrogate pairs! UTF-16 fun time.
|
|||
|
if (j + 3 >= maxJ) {
|
|||
|
break;
|
|||
|
} else {
|
|||
|
numChars++;
|
|||
|
}
|
|||
|
|
|||
|
// First pair: 10 bits of data, with an implicitly set 11th bit
|
|||
|
// Second pair: 10 bits of data
|
|||
|
var codePoint = (((code & 0x3FF) | 0x400) << 10) | (next & 0x3FF);
|
|||
|
|
|||
|
// Highest 3 bits in first byte
|
|||
|
buf.writeUInt8((codePoint >> 18) | 0xF0, j++);
|
|||
|
|
|||
|
// Rest are all 6 bits
|
|||
|
buf.writeUInt8(((codePoint >> 12) & 0x3F) | 0x80, j++);
|
|||
|
buf.writeUInt8(((codePoint >> 6) & 0x3F) | 0x80, j++);
|
|||
|
buf.writeUInt8((codePoint & 0x3F) | 0x80, j++);
|
|||
|
i++;
|
|||
|
} else if (code < 0x80) {
|
|||
|
// One byte
|
|||
|
buf.writeUInt8(code, j++);
|
|||
|
numChars++;
|
|||
|
} else if (code < 0x800) {
|
|||
|
// Two bytes
|
|||
|
if (j + 1 >= maxJ) {
|
|||
|
break;
|
|||
|
} else {
|
|||
|
numChars++;
|
|||
|
}
|
|||
|
|
|||
|
// Highest 5 bits in first byte
|
|||
|
buf.writeUInt8((code >> 6) | 0xC0, j++);
|
|||
|
|
|||
|
// Lower 6 bits in second byte
|
|||
|
buf.writeUInt8((code & 0x3F) | 0x80, j++);
|
|||
|
} else if (code < 0x10000) {
|
|||
|
// Three bytes
|
|||
|
if (j + 2 >= maxJ) {
|
|||
|
break;
|
|||
|
} else {
|
|||
|
numChars++;
|
|||
|
}
|
|||
|
|
|||
|
// Highest 4 bits in first byte
|
|||
|
buf.writeUInt8((code >> 12) | 0xE0, j++);
|
|||
|
|
|||
|
// Middle 6 bits in second byte
|
|||
|
buf.writeUInt8(((code >> 6) & 0x3F) | 0x80, j++);
|
|||
|
|
|||
|
// Lowest 6 bits in third byte
|
|||
|
buf.writeUInt8((code & 0x3F) | 0x80, j++);
|
|||
|
}
|
|||
|
}
|
|||
|
return j;
|
|||
|
};
|
|||
|
|
|||
|
UTF8.byte2str = function (buff) {
|
|||
|
var chars = [];
|
|||
|
var i = 0;
|
|||
|
while (i < buff.length) {
|
|||
|
var code = buff.readUInt8(i++);
|
|||
|
if (code < 0x80) {
|
|||
|
chars.push(String.fromCharCode(code));
|
|||
|
} else if (code < 0xC0) {
|
|||
|
throw new Error('Found incomplete part of character in string.');
|
|||
|
} else if (code < 0xE0) {
|
|||
|
// 2 bytes: 5 and 6 bits
|
|||
|
chars.push(String.fromCharCode(((code & 0x1F) << 6) | (buff.readUInt8(i++) & 0x3F)));
|
|||
|
} else if (code < 0xF0) {
|
|||
|
// 3 bytes: 4, 6, and 6 bits
|
|||
|
chars.push(String.fromCharCode(((code & 0xF) << 12) | ((buff.readUInt8(i++) & 0x3F) << 6) | (buff.readUInt8(i++) & 0x3F)));
|
|||
|
} else if (code < 0xF8) {
|
|||
|
// 4 bytes: 3, 6, 6, 6 bits; surrogate pairs time!
|
|||
|
// First 11 bits; remove 11th bit as per UTF-16 standard
|
|||
|
var byte3 = buff.readUInt8(i + 2);
|
|||
|
chars.push(String.fromCharCode(((((code & 0x7) << 8) | ((buff.readUInt8(i++) & 0x3F) << 2) | ((buff.readUInt8(i++) & 0x3F) >> 4)) & 0x3FF) | 0xD800));
|
|||
|
|
|||
|
// Final 10 bits
|
|||
|
chars.push(String.fromCharCode((((byte3 & 0xF) << 6) | (buff.readUInt8(i++) & 0x3F)) | 0xDC00));
|
|||
|
} else {
|
|||
|
throw new Error('Unable to represent UTF-8 string as UTF-16 JavaScript string.');
|
|||
|
}
|
|||
|
}
|
|||
|
return chars.join('');
|
|||
|
};
|
|||
|
|
|||
|
UTF8.byteLength = function (str) {
|
|||
|
// Matches only the 10.. bytes that are non-initial characters in a
|
|||
|
// multi-byte sequence.
|
|||
|
// @todo This may be slower than iterating through the string in some cases.
|
|||
|
var m = encodeURIComponent(str).match(/%[89ABab]/g);
|
|||
|
return str.length + (m ? m.length : 0);
|
|||
|
};
|
|||
|
return UTF8;
|
|||
|
})();
|
|||
|
exports.UTF8 = UTF8;
|
|||
|
|
|||
|
/**
|
|||
|
* String utility functions for 8-bit ASCII. Like Node, we mask the high bits of
|
|||
|
* characters in JavaScript UTF-16 strings.
|
|||
|
* @see http://en.wikipedia.org/wiki/ASCII
|
|||
|
*/
|
|||
|
var ASCII = (function () {
|
|||
|
function ASCII() {
|
|||
|
}
|
|||
|
ASCII.str2byte = function (str, buf) {
|
|||
|
var length = str.length > buf.length ? buf.length : str.length;
|
|||
|
for (var i = 0; i < length; i++) {
|
|||
|
buf.writeUInt8(str.charCodeAt(i) % 256, i);
|
|||
|
}
|
|||
|
return length;
|
|||
|
};
|
|||
|
|
|||
|
ASCII.byte2str = function (buff) {
|
|||
|
var chars = new Array(buff.length);
|
|||
|
for (var i = 0; i < buff.length; i++) {
|
|||
|
chars[i] = String.fromCharCode(buff.readUInt8(i) & 0x7F);
|
|||
|
}
|
|||
|
return chars.join('');
|
|||
|
};
|
|||
|
|
|||
|
ASCII.byteLength = function (str) {
|
|||
|
return str.length;
|
|||
|
};
|
|||
|
return ASCII;
|
|||
|
})();
|
|||
|
exports.ASCII = ASCII;
|
|||
|
|
|||
|
/**
|
|||
|
* (Nonstandard) String utility function for 8-bit ASCII with the extended
|
|||
|
* character set. Unlike the ASCII above, we do not mask the high bits.
|
|||
|
* @see http://en.wikipedia.org/wiki/Extended_ASCII
|
|||
|
*/
|
|||
|
var ExtendedASCII = (function () {
|
|||
|
function ExtendedASCII() {
|
|||
|
}
|
|||
|
ExtendedASCII.str2byte = function (str, buf) {
|
|||
|
var length = str.length > buf.length ? buf.length : str.length;
|
|||
|
for (var i = 0; i < length; i++) {
|
|||
|
var charCode = str.charCodeAt(i);
|
|||
|
if (charCode > 0x7F) {
|
|||
|
// Check if extended ASCII.
|
|||
|
var charIdx = ExtendedASCII.extendedChars.indexOf(str.charAt(i));
|
|||
|
if (charIdx > -1) {
|
|||
|
charCode = charIdx + 0x80;
|
|||
|
}
|
|||
|
// Otherwise, keep it as-is.
|
|||
|
}
|
|||
|
buf.writeUInt8(charCode, i);
|
|||
|
}
|
|||
|
return length;
|
|||
|
};
|
|||
|
|
|||
|
ExtendedASCII.byte2str = function (buff) {
|
|||
|
var chars = new Array(buff.length);
|
|||
|
for (var i = 0; i < buff.length; i++) {
|
|||
|
var charCode = buff.readUInt8(i);
|
|||
|
if (charCode > 0x7F) {
|
|||
|
chars[i] = ExtendedASCII.extendedChars[charCode - 128];
|
|||
|
} else {
|
|||
|
chars[i] = String.fromCharCode(charCode);
|
|||
|
}
|
|||
|
}
|
|||
|
return chars.join('');
|
|||
|
};
|
|||
|
|
|||
|
ExtendedASCII.byteLength = function (str) {
|
|||
|
return str.length;
|
|||
|
};
|
|||
|
ExtendedASCII.extendedChars = [
|
|||
|
'\u00C7', '\u00FC', '\u00E9', '\u00E2', '\u00E4',
|
|||
|
'\u00E0', '\u00E5', '\u00E7', '\u00EA', '\u00EB', '\u00E8', '\u00EF',
|
|||
|
'\u00EE', '\u00EC', '\u00C4', '\u00C5', '\u00C9', '\u00E6', '\u00C6',
|
|||
|
'\u00F4', '\u00F6', '\u00F2', '\u00FB', '\u00F9', '\u00FF', '\u00D6',
|
|||
|
'\u00DC', '\u00F8', '\u00A3', '\u00D8', '\u00D7', '\u0192', '\u00E1',
|
|||
|
'\u00ED', '\u00F3', '\u00FA', '\u00F1', '\u00D1', '\u00AA', '\u00BA',
|
|||
|
'\u00BF', '\u00AE', '\u00AC', '\u00BD', '\u00BC', '\u00A1', '\u00AB',
|
|||
|
'\u00BB', '_', '_', '_', '\u00A6', '\u00A6', '\u00C1', '\u00C2', '\u00C0',
|
|||
|
'\u00A9', '\u00A6', '\u00A6', '+', '+', '\u00A2', '\u00A5', '+', '+', '-',
|
|||
|
'-', '+', '-', '+', '\u00E3', '\u00C3', '+', '+', '-', '-', '\u00A6', '-',
|
|||
|
'+', '\u00A4', '\u00F0', '\u00D0', '\u00CA', '\u00CB', '\u00C8', 'i',
|
|||
|
'\u00CD', '\u00CE', '\u00CF', '+', '+', '_', '_', '\u00A6', '\u00CC', '_',
|
|||
|
'\u00D3', '\u00DF', '\u00D4', '\u00D2', '\u00F5', '\u00D5', '\u00B5',
|
|||
|
'\u00FE', '\u00DE', '\u00DA', '\u00DB', '\u00D9', '\u00FD', '\u00DD',
|
|||
|
'\u00AF', '\u00B4', '\u00AD', '\u00B1', '_', '\u00BE', '\u00B6', '\u00A7',
|
|||
|
'\u00F7', '\u00B8', '\u00B0', '\u00A8', '\u00B7', '\u00B9', '\u00B3',
|
|||
|
'\u00B2', '_', ' '];
|
|||
|
return ExtendedASCII;
|
|||
|
})();
|
|||
|
exports.ExtendedASCII = ExtendedASCII;
|
|||
|
|
|||
|
/**
|
|||
|
* String utility functions for Node's BINARY strings, which represent a single
|
|||
|
* byte per character.
|
|||
|
*/
|
|||
|
var BINARY = (function () {
|
|||
|
function BINARY() {
|
|||
|
}
|
|||
|
BINARY.str2byte = function (str, buf) {
|
|||
|
var length = str.length > buf.length ? buf.length : str.length;
|
|||
|
for (var i = 0; i < length; i++) {
|
|||
|
buf.writeUInt8(str.charCodeAt(i) & 0xFF, i);
|
|||
|
}
|
|||
|
return length;
|
|||
|
};
|
|||
|
|
|||
|
BINARY.byte2str = function (buff) {
|
|||
|
var chars = new Array(buff.length);
|
|||
|
for (var i = 0; i < buff.length; i++) {
|
|||
|
chars[i] = String.fromCharCode(buff.readUInt8(i) & 0xFF);
|
|||
|
}
|
|||
|
return chars.join('');
|
|||
|
};
|
|||
|
|
|||
|
BINARY.byteLength = function (str) {
|
|||
|
return str.length;
|
|||
|
};
|
|||
|
return BINARY;
|
|||
|
})();
|
|||
|
exports.BINARY = BINARY;
|
|||
|
|
|||
|
/**
|
|||
|
* Contains string utility functions for base-64 encoding.
|
|||
|
*
|
|||
|
* Adapted from the StackOverflow comment linked below.
|
|||
|
* @see http://stackoverflow.com/questions/246801/how-can-you-encode-to-base64-using-javascript#246813
|
|||
|
* @see http://en.wikipedia.org/wiki/Base64
|
|||
|
* @todo Bake in support for btoa() and atob() if available.
|
|||
|
*/
|
|||
|
var BASE64 = (function () {
|
|||
|
function BASE64() {
|
|||
|
}
|
|||
|
BASE64.byte2str = function (buff) {
|
|||
|
var output = '';
|
|||
|
var i = 0;
|
|||
|
while (i < buff.length) {
|
|||
|
var chr1 = buff.readUInt8(i++);
|
|||
|
var chr2 = i < buff.length ? buff.readUInt8(i++) : NaN;
|
|||
|
var chr3 = i < buff.length ? buff.readUInt8(i++) : NaN;
|
|||
|
var enc1 = chr1 >> 2;
|
|||
|
var enc2 = ((chr1 & 3) << 4) | (chr2 >> 4);
|
|||
|
var enc3 = ((chr2 & 15) << 2) | (chr3 >> 6);
|
|||
|
var enc4 = chr3 & 63;
|
|||
|
if (isNaN(chr2)) {
|
|||
|
enc3 = enc4 = 64;
|
|||
|
} else if (isNaN(chr3)) {
|
|||
|
enc4 = 64;
|
|||
|
}
|
|||
|
output = output + BASE64.num2b64[enc1] + BASE64.num2b64[enc2] + BASE64.num2b64[enc3] + BASE64.num2b64[enc4];
|
|||
|
}
|
|||
|
return output;
|
|||
|
};
|
|||
|
|
|||
|
BASE64.str2byte = function (str, buf) {
|
|||
|
var length = buf.length;
|
|||
|
var output = '';
|
|||
|
var i = 0;
|
|||
|
str = str.replace(/[^A-Za-z0-9\+\/\=\-\_]/g, '');
|
|||
|
var j = 0;
|
|||
|
while (i < str.length) {
|
|||
|
var enc1 = BASE64.b642num[str.charAt(i++)];
|
|||
|
var enc2 = BASE64.b642num[str.charAt(i++)];
|
|||
|
var enc3 = BASE64.b642num[str.charAt(i++)];
|
|||
|
var enc4 = BASE64.b642num[str.charAt(i++)];
|
|||
|
var chr1 = (enc1 << 2) | (enc2 >> 4);
|
|||
|
var chr2 = ((enc2 & 15) << 4) | (enc3 >> 2);
|
|||
|
var chr3 = ((enc3 & 3) << 6) | enc4;
|
|||
|
buf.writeUInt8(chr1, j++);
|
|||
|
if (j === length) {
|
|||
|
break;
|
|||
|
}
|
|||
|
if (enc3 !== 64) {
|
|||
|
output += buf.writeUInt8(chr2, j++);
|
|||
|
}
|
|||
|
if (j === length) {
|
|||
|
break;
|
|||
|
}
|
|||
|
if (enc4 !== 64) {
|
|||
|
output += buf.writeUInt8(chr3, j++);
|
|||
|
}
|
|||
|
if (j === length) {
|
|||
|
break;
|
|||
|
}
|
|||
|
}
|
|||
|
return j;
|
|||
|
};
|
|||
|
|
|||
|
BASE64.byteLength = function (str) {
|
|||
|
return Math.floor(((str.replace(/[^A-Za-z0-9\+\/\-\_]/g, '')).length * 6) / 8);
|
|||
|
};
|
|||
|
BASE64.b64chars = ['A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z', 'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 'x', 'y', 'z', '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '+', '/', '='];
|
|||
|
BASE64.num2b64 = (function () {
|
|||
|
var obj = new Array(BASE64.b64chars.length);
|
|||
|
for (var idx = 0; idx < BASE64.b64chars.length; idx++) {
|
|||
|
var i = BASE64.b64chars[idx];
|
|||
|
obj[idx] = i;
|
|||
|
}
|
|||
|
return obj;
|
|||
|
})();
|
|||
|
|
|||
|
BASE64.b642num = (function () {
|
|||
|
var obj = {};
|
|||
|
for (var idx = 0; idx < BASE64.b64chars.length; idx++) {
|
|||
|
var i = BASE64.b64chars[idx];
|
|||
|
obj[i] = idx;
|
|||
|
}
|
|||
|
obj['-'] = 62;
|
|||
|
obj['_'] = 63;
|
|||
|
return obj;
|
|||
|
})();
|
|||
|
return BASE64;
|
|||
|
})();
|
|||
|
exports.BASE64 = BASE64;
|
|||
|
|
|||
|
/**
|
|||
|
* String utility functions for the UCS-2 encoding. Note that our UCS-2 handling
|
|||
|
* is identical to our UTF-16 handling.
|
|||
|
*
|
|||
|
* Note: UCS-2 handling is identical to UTF-16.
|
|||
|
* @see http://en.wikipedia.org/wiki/UCS2
|
|||
|
*/
|
|||
|
var UCS2 = (function () {
|
|||
|
function UCS2() {
|
|||
|
}
|
|||
|
UCS2.str2byte = function (str, buf) {
|
|||
|
var len = str.length;
|
|||
|
|
|||
|
// Clip length to longest string of valid characters that can fit in the
|
|||
|
// byte range.
|
|||
|
if (len * 2 > buf.length) {
|
|||
|
len = buf.length % 2 === 1 ? (buf.length - 1) / 2 : buf.length / 2;
|
|||
|
}
|
|||
|
for (var i = 0; i < len; i++) {
|
|||
|
buf.writeUInt16LE(str.charCodeAt(i), i * 2);
|
|||
|
}
|
|||
|
return len * 2;
|
|||
|
};
|
|||
|
|
|||
|
UCS2.byte2str = function (buff) {
|
|||
|
if (buff.length % 2 !== 0) {
|
|||
|
throw new Error('Invalid UCS2 byte array.');
|
|||
|
}
|
|||
|
var chars = new Array(buff.length / 2);
|
|||
|
for (var i = 0; i < buff.length; i += 2) {
|
|||
|
chars[i / 2] = String.fromCharCode(buff.readUInt8(i) | (buff.readUInt8(i + 1) << 8));
|
|||
|
}
|
|||
|
return chars.join('');
|
|||
|
};
|
|||
|
|
|||
|
UCS2.byteLength = function (str) {
|
|||
|
return str.length * 2;
|
|||
|
};
|
|||
|
return UCS2;
|
|||
|
})();
|
|||
|
exports.UCS2 = UCS2;
|
|||
|
|
|||
|
/**
|
|||
|
* Contains string utility functions for hex encoding.
|
|||
|
* @see http://en.wikipedia.org/wiki/Hexadecimal
|
|||
|
*/
|
|||
|
var HEX = (function () {
|
|||
|
function HEX() {
|
|||
|
}
|
|||
|
HEX.str2byte = function (str, buf) {
|
|||
|
if (str.length % 2 === 1) {
|
|||
|
throw new Error('Invalid hex string');
|
|||
|
}
|
|||
|
|
|||
|
// Each character is 1 byte encoded as two hex characters; so 1 byte becomes
|
|||
|
// 2 bytes.
|
|||
|
var numBytes = str.length >> 1;
|
|||
|
if (numBytes > buf.length) {
|
|||
|
numBytes = buf.length;
|
|||
|
}
|
|||
|
for (var i = 0; i < numBytes; i++) {
|
|||
|
var char1 = this.hex2num[str.charAt(i << 1)];
|
|||
|
var char2 = this.hex2num[str.charAt((i << 1) + 1)];
|
|||
|
buf.writeUInt8((char1 << 4) | char2, i);
|
|||
|
}
|
|||
|
return numBytes;
|
|||
|
};
|
|||
|
|
|||
|
HEX.byte2str = function (buff) {
|
|||
|
var len = buff.length;
|
|||
|
var chars = new Array(len << 1);
|
|||
|
var j = 0;
|
|||
|
for (var i = 0; i < len; i++) {
|
|||
|
var hex2 = buff.readUInt8(i) & 0xF;
|
|||
|
var hex1 = buff.readUInt8(i) >> 4;
|
|||
|
chars[j++] = this.num2hex[hex1];
|
|||
|
chars[j++] = this.num2hex[hex2];
|
|||
|
}
|
|||
|
return chars.join('');
|
|||
|
};
|
|||
|
|
|||
|
HEX.byteLength = function (str) {
|
|||
|
// Assuming a valid string.
|
|||
|
return str.length >> 1;
|
|||
|
};
|
|||
|
HEX.HEXCHARS = '0123456789abcdef';
|
|||
|
|
|||
|
HEX.num2hex = (function () {
|
|||
|
var obj = new Array(HEX.HEXCHARS.length);
|
|||
|
for (var idx = 0; idx < HEX.HEXCHARS.length; idx++) {
|
|||
|
var i = HEX.HEXCHARS[idx];
|
|||
|
obj[idx] = i;
|
|||
|
}
|
|||
|
return obj;
|
|||
|
})();
|
|||
|
|
|||
|
HEX.hex2num = (function () {
|
|||
|
var idx, i;
|
|||
|
var obj = {};
|
|||
|
for (idx = 0; idx < HEX.HEXCHARS.length; idx++) {
|
|||
|
i = HEX.HEXCHARS[idx];
|
|||
|
obj[i] = idx;
|
|||
|
}
|
|||
|
var capitals = 'ABCDEF';
|
|||
|
for (idx = 0; idx < capitals.length; idx++) {
|
|||
|
i = capitals[idx];
|
|||
|
obj[i] = idx + 10;
|
|||
|
}
|
|||
|
return obj;
|
|||
|
})();
|
|||
|
return HEX;
|
|||
|
})();
|
|||
|
exports.HEX = HEX;
|
|||
|
|
|||
|
/**
|
|||
|
* Contains string utility functions for binary string encoding. This is where we
|
|||
|
* pack arbitrary binary data as a UTF-16 string.
|
|||
|
*
|
|||
|
* Each character in the string is two bytes. The first character in the string
|
|||
|
* is special: The first byte specifies if the binary data is of odd byte length.
|
|||
|
* If it is, then it is a 1 and the second byte is the first byte of data; if
|
|||
|
* not, it is a 0 and the second byte is 0.
|
|||
|
*
|
|||
|
* Everything is little endian.
|
|||
|
*/
|
|||
|
var BINSTR = (function () {
|
|||
|
function BINSTR() {
|
|||
|
}
|
|||
|
BINSTR.str2byte = function (str, buf) {
|
|||
|
// Special case: Empty string
|
|||
|
if (str.length === 0) {
|
|||
|
return 0;
|
|||
|
}
|
|||
|
var numBytes = BINSTR.byteLength(str);
|
|||
|
if (numBytes > buf.length) {
|
|||
|
numBytes = buf.length;
|
|||
|
}
|
|||
|
var j = 0;
|
|||
|
var startByte = 0;
|
|||
|
var endByte = startByte + numBytes;
|
|||
|
|
|||
|
// Handle first character separately
|
|||
|
var firstChar = str.charCodeAt(j++);
|
|||
|
if (firstChar !== 0) {
|
|||
|
buf.writeUInt8(firstChar & 0xFF, 0);
|
|||
|
startByte = 1;
|
|||
|
}
|
|||
|
for (var i = startByte; i < endByte; i += 2) {
|
|||
|
var chr = str.charCodeAt(j++);
|
|||
|
if (endByte - i === 1) {
|
|||
|
// Write first byte of character
|
|||
|
buf.writeUInt8(chr >> 8, i);
|
|||
|
}
|
|||
|
if (endByte - i >= 2) {
|
|||
|
// Write both bytes in character
|
|||
|
buf.writeUInt16BE(chr, i);
|
|||
|
}
|
|||
|
}
|
|||
|
return numBytes;
|
|||
|
};
|
|||
|
|
|||
|
BINSTR.byte2str = function (buff) {
|
|||
|
var len = buff.length;
|
|||
|
|
|||
|
// Special case: Empty string
|
|||
|
if (len === 0) {
|
|||
|
return '';
|
|||
|
}
|
|||
|
var chars = new Array((len >> 1) + 1);
|
|||
|
var j = 0;
|
|||
|
for (var i = 0; i < chars.length; i++) {
|
|||
|
if (i === 0) {
|
|||
|
if (len % 2 === 1) {
|
|||
|
chars[i] = String.fromCharCode((1 << 8) | buff.readUInt8(j++));
|
|||
|
} else {
|
|||
|
chars[i] = String.fromCharCode(0);
|
|||
|
}
|
|||
|
} else {
|
|||
|
chars[i] = String.fromCharCode((buff.readUInt8(j++) << 8) | buff.readUInt8(j++));
|
|||
|
}
|
|||
|
}
|
|||
|
return chars.join('');
|
|||
|
};
|
|||
|
|
|||
|
BINSTR.byteLength = function (str) {
|
|||
|
if (str.length === 0) {
|
|||
|
// Special case: Empty string.
|
|||
|
return 0;
|
|||
|
}
|
|||
|
var firstChar = str.charCodeAt(0);
|
|||
|
var bytelen = (str.length - 1) << 1;
|
|||
|
if (firstChar !== 0) {
|
|||
|
bytelen++;
|
|||
|
}
|
|||
|
return bytelen;
|
|||
|
};
|
|||
|
return BINSTR;
|
|||
|
})();
|
|||
|
exports.BINSTR = BINSTR;
|
|||
|
|
|||
|
/**
|
|||
|
* IE/older FF version of binary string. One byte per character, offset by 0x20.
|
|||
|
*/
|
|||
|
var BINSTRIE = (function () {
|
|||
|
function BINSTRIE() {
|
|||
|
}
|
|||
|
BINSTRIE.str2byte = function (str, buf) {
|
|||
|
var length = str.length > buf.length ? buf.length : str.length;
|
|||
|
for (var i = 0; i < length; i++) {
|
|||
|
buf.writeUInt8(str.charCodeAt(i) - 0x20, i);
|
|||
|
}
|
|||
|
return length;
|
|||
|
};
|
|||
|
|
|||
|
BINSTRIE.byte2str = function (buff) {
|
|||
|
var chars = new Array(buff.length);
|
|||
|
for (var i = 0; i < buff.length; i++) {
|
|||
|
chars[i] = String.fromCharCode(buff.readUInt8(i) + 0x20);
|
|||
|
}
|
|||
|
return chars.join('');
|
|||
|
};
|
|||
|
|
|||
|
BINSTRIE.byteLength = function (str) {
|
|||
|
return str.length;
|
|||
|
};
|
|||
|
return BINSTRIE;
|
|||
|
})();
|
|||
|
exports.BINSTRIE = BINSTRIE;
|
|||
|
});
|
|||
|
//# sourceMappingURL=string_util.js.map
|
|||
|
;
|
|||
|
define('core/buffer',["require", "exports", './buffer_core', './buffer_core_array', './buffer_core_arraybuffer', './buffer_core_imagedata', './string_util'], function(require, exports, buffer_core, buffer_core_array, buffer_core_arraybuffer, buffer_core_imagedata, string_util) {
|
|||
|
// BC implementations earlier in the array are preferred.
|
|||
|
var BufferCorePreferences = [
|
|||
|
buffer_core_arraybuffer.BufferCoreArrayBuffer,
|
|||
|
buffer_core_imagedata.BufferCoreImageData,
|
|||
|
buffer_core_array.BufferCoreArray
|
|||
|
];
|
|||
|
|
|||
|
var PreferredBufferCore = (function () {
|
|||
|
var i, bci;
|
|||
|
for (i = 0; i < BufferCorePreferences.length; i++) {
|
|||
|
bci = BufferCorePreferences[i];
|
|||
|
if (bci.isAvailable())
|
|||
|
return bci;
|
|||
|
}
|
|||
|
|
|||
|
throw new Error("This browser does not support any available BufferCore implementations.");
|
|||
|
})();
|
|||
|
|
|||
|
|
|||
|
|
|||
|
|
|||
|
|
|||
|
/**
|
|||
|
* Emulates Node's Buffer API. Wraps a BufferCore object that is responsible
|
|||
|
* for actually writing/reading data from some data representation in memory.
|
|||
|
*/
|
|||
|
var Buffer = (function () {
|
|||
|
function Buffer(arg1, arg2, arg3) {
|
|||
|
if (typeof arg2 === "undefined") { arg2 = 'utf8'; }
|
|||
|
this.offset = 0;
|
|||
|
var i;
|
|||
|
|
|||
|
// Node apparently allows you to construct buffers w/o 'new'.
|
|||
|
if (!(this instanceof Buffer)) {
|
|||
|
return new Buffer(arg1, arg2);
|
|||
|
}
|
|||
|
|
|||
|
if (arg1 instanceof buffer_core.BufferCoreCommon) {
|
|||
|
// constructor (data: buffer_core.BufferCore, start?: number, end?: number)
|
|||
|
this.data = arg1;
|
|||
|
var start = typeof arg2 === 'number' ? arg2 : 0;
|
|||
|
var end = typeof arg3 === 'number' ? arg3 : this.data.getLength();
|
|||
|
this.offset = start;
|
|||
|
this.length = end - start;
|
|||
|
} else if (typeof arg1 === 'number') {
|
|||
|
// constructor (size: number);
|
|||
|
if (arg1 !== (arg1 >>> 0)) {
|
|||
|
throw new TypeError('Buffer size must be a uint32.');
|
|||
|
}
|
|||
|
this.length = arg1;
|
|||
|
this.data = new PreferredBufferCore(arg1);
|
|||
|
} else if (typeof DataView !== 'undefined' && arg1 instanceof DataView) {
|
|||
|
// constructor (data: DataView);
|
|||
|
this.data = new buffer_core_arraybuffer.BufferCoreArrayBuffer(arg1);
|
|||
|
this.length = arg1.byteLength;
|
|||
|
} else if (typeof ArrayBuffer !== 'undefined' && arg1 instanceof ArrayBuffer) {
|
|||
|
// constructor (data: ArrayBuffer);
|
|||
|
this.data = new buffer_core_arraybuffer.BufferCoreArrayBuffer(arg1);
|
|||
|
this.length = arg1.byteLength;
|
|||
|
} else if (arg1 instanceof Buffer) {
|
|||
|
// constructor (data: Buffer);
|
|||
|
var argBuff = arg1;
|
|||
|
this.data = new PreferredBufferCore(arg1.length);
|
|||
|
this.length = arg1.length;
|
|||
|
argBuff.copy(this);
|
|||
|
} else if (Array.isArray(arg1) || (arg1 != null && typeof arg1 === 'object' && typeof arg1[0] === 'number')) {
|
|||
|
// constructor (data: number[]);
|
|||
|
this.data = new PreferredBufferCore(arg1.length);
|
|||
|
for (i = 0; i < arg1.length; i++) {
|
|||
|
this.data.writeUInt8(i, arg1[i]);
|
|||
|
}
|
|||
|
this.length = arg1.length;
|
|||
|
} else if (typeof arg1 === 'string') {
|
|||
|
// constructor (data: string, encoding?: string);
|
|||
|
this.length = Buffer.byteLength(arg1, arg2);
|
|||
|
this.data = new PreferredBufferCore(this.length);
|
|||
|
this.write(arg1, 0, this.length, arg2);
|
|||
|
} else {
|
|||
|
throw new Error("Invalid argument to Buffer constructor: " + arg1);
|
|||
|
}
|
|||
|
}
|
|||
|
Buffer.prototype.getBufferCore = function () {
|
|||
|
return this.data;
|
|||
|
};
|
|||
|
|
|||
|
Buffer.prototype.getOffset = function () {
|
|||
|
return this.offset;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* **NONSTANDARD**: Set the octet at index. Emulates NodeJS buffer's index
|
|||
|
* operation. Octet can be signed or unsigned.
|
|||
|
* @param {number} index - the index to set the value at
|
|||
|
* @param {number} value - the value to set at the given index
|
|||
|
*/
|
|||
|
Buffer.prototype.set = function (index, value) {
|
|||
|
// In Node, the following happens:
|
|||
|
// buffer[0] = -1;
|
|||
|
// buffer[0]; // 255
|
|||
|
if (value < 0) {
|
|||
|
return this.writeInt8(value, index);
|
|||
|
} else {
|
|||
|
return this.writeUInt8(value, index);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* **NONSTANDARD**: Get the octet at index.
|
|||
|
* @param {number} index - index to fetch the value at
|
|||
|
* @return {number} the value at the given index
|
|||
|
*/
|
|||
|
Buffer.prototype.get = function (index) {
|
|||
|
return this.readUInt8(index);
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Writes string to the buffer at offset using the given encoding.
|
|||
|
* If buffer did not contain enough space to fit the entire string, it will
|
|||
|
* write a partial amount of the string.
|
|||
|
* @param {string} str - Data to be written to buffer
|
|||
|
* @param {number} [offset=0] - Offset in the buffer to write to
|
|||
|
* @param {number} [length=this.length] - Number of bytes to write
|
|||
|
* @param {string} [encoding=utf8] - Character encoding
|
|||
|
* @return {number} Number of octets written.
|
|||
|
*/
|
|||
|
Buffer.prototype.write = function (str, offset, length, encoding) {
|
|||
|
if (typeof offset === "undefined") { offset = 0; }
|
|||
|
if (typeof length === "undefined") { length = this.length; }
|
|||
|
if (typeof encoding === "undefined") { encoding = 'utf8'; }
|
|||
|
// I hate Node's optional arguments.
|
|||
|
if (typeof offset === 'string') {
|
|||
|
// 'str' and 'encoding' specified
|
|||
|
encoding = "" + offset;
|
|||
|
offset = 0;
|
|||
|
length = this.length;
|
|||
|
} else if (typeof length === 'string') {
|
|||
|
// 'str', 'offset', and 'encoding' specified
|
|||
|
encoding = "" + length;
|
|||
|
length = this.length;
|
|||
|
}
|
|||
|
|
|||
|
// Don't waste our time if the offset is beyond the buffer length
|
|||
|
if (offset >= this.length) {
|
|||
|
return 0;
|
|||
|
}
|
|||
|
var strUtil = string_util.FindUtil(encoding);
|
|||
|
|
|||
|
// Are we trying to write past the buffer?
|
|||
|
length = length + offset > this.length ? this.length - offset : length;
|
|||
|
offset += this.offset;
|
|||
|
return strUtil.str2byte(str, offset === 0 && length === this.length ? this : new Buffer(this.data, offset, length + offset));
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Decodes a portion of the Buffer into a String.
|
|||
|
* @param {string} encoding - Character encoding to decode to
|
|||
|
* @param {number} [start=0] - Start position in the buffer
|
|||
|
* @param {number} [end=this.length] - Ending position in the buffer
|
|||
|
* @return {string} A string from buffer data encoded with encoding, beginning
|
|||
|
* at start, and ending at end.
|
|||
|
*/
|
|||
|
Buffer.prototype.toString = function (encoding, start, end) {
|
|||
|
if (typeof encoding === "undefined") { encoding = 'utf8'; }
|
|||
|
if (typeof start === "undefined") { start = 0; }
|
|||
|
if (typeof end === "undefined") { end = this.length; }
|
|||
|
if (!(start <= end)) {
|
|||
|
throw new Error("Invalid start/end positions: " + start + " - " + end);
|
|||
|
}
|
|||
|
if (start === end) {
|
|||
|
return '';
|
|||
|
}
|
|||
|
if (end > this.length) {
|
|||
|
end = this.length;
|
|||
|
}
|
|||
|
var strUtil = string_util.FindUtil(encoding);
|
|||
|
|
|||
|
// Get the string representation of the given slice. Create a new buffer
|
|||
|
// if need be.
|
|||
|
return strUtil.byte2str(start === 0 && end === this.length ? this : new Buffer(this.data, start + this.offset, end + this.offset));
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Returns a JSON-representation of the Buffer instance, which is identical to
|
|||
|
* the output for JSON Arrays. JSON.stringify implicitly calls this function
|
|||
|
* when stringifying a Buffer instance.
|
|||
|
* @return {object} An object that can be used for JSON stringification.
|
|||
|
*/
|
|||
|
Buffer.prototype.toJSON = function () {
|
|||
|
// Construct a byte array for the JSON 'data'.
|
|||
|
var len = this.length;
|
|||
|
var byteArr = new Array(len);
|
|||
|
for (var i = 0; i < len; i++) {
|
|||
|
byteArr[i] = this.readUInt8(i);
|
|||
|
}
|
|||
|
return {
|
|||
|
type: 'Buffer',
|
|||
|
data: byteArr
|
|||
|
};
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Does copy between buffers. The source and target regions can be overlapped.
|
|||
|
* All values passed that are undefined/NaN or are out of bounds are set equal
|
|||
|
* to their respective defaults.
|
|||
|
* @param {Buffer} target - Buffer to copy into
|
|||
|
* @param {number} [targetStart=0] - Index to start copying to in the targetBuffer
|
|||
|
* @param {number} [sourceStart=0] - Index in this buffer to start copying from
|
|||
|
* @param {number} [sourceEnd=this.length] - Index in this buffer stop copying at
|
|||
|
* @return {number} The number of bytes copied into the target buffer.
|
|||
|
*/
|
|||
|
Buffer.prototype.copy = function (target, targetStart, sourceStart, sourceEnd) {
|
|||
|
if (typeof targetStart === "undefined") { targetStart = 0; }
|
|||
|
if (typeof sourceStart === "undefined") { sourceStart = 0; }
|
|||
|
if (typeof sourceEnd === "undefined") { sourceEnd = this.length; }
|
|||
|
// The Node code is weird. It sets some out-of-bounds args to their defaults
|
|||
|
// and throws exceptions for others (sourceEnd).
|
|||
|
targetStart = targetStart < 0 ? 0 : targetStart;
|
|||
|
sourceStart = sourceStart < 0 ? 0 : sourceStart;
|
|||
|
|
|||
|
// Need to sanity check all of the input. Node has really odd rules regarding
|
|||
|
// when to apply default arguments. I decided to copy Node's logic.
|
|||
|
if (sourceEnd < sourceStart) {
|
|||
|
throw new RangeError('sourceEnd < sourceStart');
|
|||
|
}
|
|||
|
if (sourceEnd === sourceStart) {
|
|||
|
return 0;
|
|||
|
}
|
|||
|
if (targetStart >= target.length) {
|
|||
|
throw new RangeError('targetStart out of bounds');
|
|||
|
}
|
|||
|
if (sourceStart >= this.length) {
|
|||
|
throw new RangeError('sourceStart out of bounds');
|
|||
|
}
|
|||
|
if (sourceEnd > this.length) {
|
|||
|
throw new RangeError('sourceEnd out of bounds');
|
|||
|
}
|
|||
|
var bytesCopied = Math.min(sourceEnd - sourceStart, target.length - targetStart, this.length - sourceStart);
|
|||
|
|
|||
|
for (var i = 0; i < bytesCopied; i++) {
|
|||
|
target.writeUInt8(this.readUInt8(sourceStart + i), targetStart + i);
|
|||
|
}
|
|||
|
return bytesCopied;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Returns a slice of this buffer.
|
|||
|
* @param {number} [start=0] - Index to start slicing from
|
|||
|
* @param {number} [end=this.length] - Index to stop slicing at
|
|||
|
* @return {Buffer} A new buffer which references the same
|
|||
|
* memory as the old, but offset and cropped by the start (defaults to 0) and
|
|||
|
* end (defaults to buffer.length) indexes. Negative indexes start from the end
|
|||
|
* of the buffer.
|
|||
|
*/
|
|||
|
Buffer.prototype.slice = function (start, end) {
|
|||
|
if (typeof start === "undefined") { start = 0; }
|
|||
|
if (typeof end === "undefined") { end = this.length; }
|
|||
|
// Translate negative indices to positive ones.
|
|||
|
if (start < 0) {
|
|||
|
start += this.length;
|
|||
|
if (start < 0) {
|
|||
|
start = 0;
|
|||
|
}
|
|||
|
}
|
|||
|
if (end < 0) {
|
|||
|
end += this.length;
|
|||
|
if (end < 0) {
|
|||
|
end = 0;
|
|||
|
}
|
|||
|
}
|
|||
|
if (end > this.length) {
|
|||
|
end = this.length;
|
|||
|
}
|
|||
|
if (start > end) {
|
|||
|
start = end;
|
|||
|
}
|
|||
|
|
|||
|
// Sanity check.
|
|||
|
if (start < 0 || end < 0 || start >= this.length || end > this.length) {
|
|||
|
throw new Error("Invalid slice indices.");
|
|||
|
}
|
|||
|
|
|||
|
// Create a new buffer backed by the same BufferCore.
|
|||
|
return new Buffer(this.data, start + this.offset, end + this.offset);
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* [NONSTANDARD] A copy-based version of Buffer.slice.
|
|||
|
*/
|
|||
|
Buffer.prototype.sliceCopy = function (start, end) {
|
|||
|
if (typeof start === "undefined") { start = 0; }
|
|||
|
if (typeof end === "undefined") { end = this.length; }
|
|||
|
// Translate negative indices to positive ones.
|
|||
|
if (start < 0) {
|
|||
|
start += this.length;
|
|||
|
if (start < 0) {
|
|||
|
start = 0;
|
|||
|
}
|
|||
|
}
|
|||
|
if (end < 0) {
|
|||
|
end += this.length;
|
|||
|
if (end < 0) {
|
|||
|
end = 0;
|
|||
|
}
|
|||
|
}
|
|||
|
if (end > this.length) {
|
|||
|
end = this.length;
|
|||
|
}
|
|||
|
if (start > end) {
|
|||
|
start = end;
|
|||
|
}
|
|||
|
|
|||
|
// Sanity check.
|
|||
|
if (start < 0 || end < 0 || start >= this.length || end > this.length) {
|
|||
|
throw new Error("Invalid slice indices.");
|
|||
|
}
|
|||
|
|
|||
|
// Copy the BufferCore.
|
|||
|
return new Buffer(this.data.copy(start + this.offset, end + this.offset));
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Fills the buffer with the specified value. If the offset and end are not
|
|||
|
* given it will fill the entire buffer.
|
|||
|
* @param {(string|number)} value - The value to fill the buffer with
|
|||
|
* @param {number} [offset=0]
|
|||
|
* @param {number} [end=this.length]
|
|||
|
*/
|
|||
|
Buffer.prototype.fill = function (value, offset, end) {
|
|||
|
if (typeof offset === "undefined") { offset = 0; }
|
|||
|
if (typeof end === "undefined") { end = this.length; }
|
|||
|
var i;
|
|||
|
var valType = typeof value;
|
|||
|
switch (valType) {
|
|||
|
case "string":
|
|||
|
// Trim to a byte.
|
|||
|
value = value.charCodeAt(0) & 0xFF;
|
|||
|
break;
|
|||
|
case "number":
|
|||
|
break;
|
|||
|
default:
|
|||
|
throw new Error('Invalid argument to fill.');
|
|||
|
}
|
|||
|
offset += this.offset;
|
|||
|
end += this.offset;
|
|||
|
this.data.fill(value, offset, end);
|
|||
|
};
|
|||
|
|
|||
|
Buffer.prototype.readUInt8 = function (offset, noAssert) {
|
|||
|
if (typeof noAssert === "undefined") { noAssert = false; }
|
|||
|
offset += this.offset;
|
|||
|
return this.data.readUInt8(offset);
|
|||
|
};
|
|||
|
|
|||
|
Buffer.prototype.readUInt16LE = function (offset, noAssert) {
|
|||
|
if (typeof noAssert === "undefined") { noAssert = false; }
|
|||
|
offset += this.offset;
|
|||
|
return this.data.readUInt16LE(offset);
|
|||
|
};
|
|||
|
|
|||
|
Buffer.prototype.readUInt16BE = function (offset, noAssert) {
|
|||
|
if (typeof noAssert === "undefined") { noAssert = false; }
|
|||
|
offset += this.offset;
|
|||
|
return this.data.readUInt16BE(offset);
|
|||
|
};
|
|||
|
|
|||
|
Buffer.prototype.readUInt32LE = function (offset, noAssert) {
|
|||
|
if (typeof noAssert === "undefined") { noAssert = false; }
|
|||
|
offset += this.offset;
|
|||
|
return this.data.readUInt32LE(offset);
|
|||
|
};
|
|||
|
|
|||
|
Buffer.prototype.readUInt32BE = function (offset, noAssert) {
|
|||
|
if (typeof noAssert === "undefined") { noAssert = false; }
|
|||
|
offset += this.offset;
|
|||
|
return this.data.readUInt32BE(offset);
|
|||
|
};
|
|||
|
|
|||
|
Buffer.prototype.readInt8 = function (offset, noAssert) {
|
|||
|
if (typeof noAssert === "undefined") { noAssert = false; }
|
|||
|
offset += this.offset;
|
|||
|
return this.data.readInt8(offset);
|
|||
|
};
|
|||
|
|
|||
|
Buffer.prototype.readInt16LE = function (offset, noAssert) {
|
|||
|
if (typeof noAssert === "undefined") { noAssert = false; }
|
|||
|
offset += this.offset;
|
|||
|
return this.data.readInt16LE(offset);
|
|||
|
};
|
|||
|
|
|||
|
Buffer.prototype.readInt16BE = function (offset, noAssert) {
|
|||
|
if (typeof noAssert === "undefined") { noAssert = false; }
|
|||
|
offset += this.offset;
|
|||
|
return this.data.readInt16BE(offset);
|
|||
|
};
|
|||
|
|
|||
|
Buffer.prototype.readInt32LE = function (offset, noAssert) {
|
|||
|
if (typeof noAssert === "undefined") { noAssert = false; }
|
|||
|
offset += this.offset;
|
|||
|
return this.data.readInt32LE(offset);
|
|||
|
};
|
|||
|
|
|||
|
Buffer.prototype.readInt32BE = function (offset, noAssert) {
|
|||
|
if (typeof noAssert === "undefined") { noAssert = false; }
|
|||
|
offset += this.offset;
|
|||
|
return this.data.readInt32BE(offset);
|
|||
|
};
|
|||
|
|
|||
|
Buffer.prototype.readFloatLE = function (offset, noAssert) {
|
|||
|
if (typeof noAssert === "undefined") { noAssert = false; }
|
|||
|
offset += this.offset;
|
|||
|
return this.data.readFloatLE(offset);
|
|||
|
};
|
|||
|
|
|||
|
Buffer.prototype.readFloatBE = function (offset, noAssert) {
|
|||
|
if (typeof noAssert === "undefined") { noAssert = false; }
|
|||
|
offset += this.offset;
|
|||
|
return this.data.readFloatBE(offset);
|
|||
|
};
|
|||
|
|
|||
|
Buffer.prototype.readDoubleLE = function (offset, noAssert) {
|
|||
|
if (typeof noAssert === "undefined") { noAssert = false; }
|
|||
|
offset += this.offset;
|
|||
|
return this.data.readDoubleLE(offset);
|
|||
|
};
|
|||
|
|
|||
|
Buffer.prototype.readDoubleBE = function (offset, noAssert) {
|
|||
|
if (typeof noAssert === "undefined") { noAssert = false; }
|
|||
|
offset += this.offset;
|
|||
|
return this.data.readDoubleBE(offset);
|
|||
|
};
|
|||
|
|
|||
|
Buffer.prototype.writeUInt8 = function (value, offset, noAssert) {
|
|||
|
if (typeof noAssert === "undefined") { noAssert = false; }
|
|||
|
offset += this.offset;
|
|||
|
this.data.writeUInt8(offset, value);
|
|||
|
};
|
|||
|
|
|||
|
Buffer.prototype.writeUInt16LE = function (value, offset, noAssert) {
|
|||
|
if (typeof noAssert === "undefined") { noAssert = false; }
|
|||
|
offset += this.offset;
|
|||
|
this.data.writeUInt16LE(offset, value);
|
|||
|
};
|
|||
|
|
|||
|
Buffer.prototype.writeUInt16BE = function (value, offset, noAssert) {
|
|||
|
if (typeof noAssert === "undefined") { noAssert = false; }
|
|||
|
offset += this.offset;
|
|||
|
this.data.writeUInt16BE(offset, value);
|
|||
|
};
|
|||
|
|
|||
|
Buffer.prototype.writeUInt32LE = function (value, offset, noAssert) {
|
|||
|
if (typeof noAssert === "undefined") { noAssert = false; }
|
|||
|
offset += this.offset;
|
|||
|
this.data.writeUInt32LE(offset, value);
|
|||
|
};
|
|||
|
|
|||
|
Buffer.prototype.writeUInt32BE = function (value, offset, noAssert) {
|
|||
|
if (typeof noAssert === "undefined") { noAssert = false; }
|
|||
|
offset += this.offset;
|
|||
|
this.data.writeUInt32BE(offset, value);
|
|||
|
};
|
|||
|
|
|||
|
Buffer.prototype.writeInt8 = function (value, offset, noAssert) {
|
|||
|
if (typeof noAssert === "undefined") { noAssert = false; }
|
|||
|
offset += this.offset;
|
|||
|
this.data.writeInt8(offset, value);
|
|||
|
};
|
|||
|
|
|||
|
Buffer.prototype.writeInt16LE = function (value, offset, noAssert) {
|
|||
|
if (typeof noAssert === "undefined") { noAssert = false; }
|
|||
|
offset += this.offset;
|
|||
|
this.data.writeInt16LE(offset, value);
|
|||
|
};
|
|||
|
|
|||
|
Buffer.prototype.writeInt16BE = function (value, offset, noAssert) {
|
|||
|
if (typeof noAssert === "undefined") { noAssert = false; }
|
|||
|
offset += this.offset;
|
|||
|
this.data.writeInt16BE(offset, value);
|
|||
|
};
|
|||
|
|
|||
|
Buffer.prototype.writeInt32LE = function (value, offset, noAssert) {
|
|||
|
if (typeof noAssert === "undefined") { noAssert = false; }
|
|||
|
offset += this.offset;
|
|||
|
this.data.writeInt32LE(offset, value);
|
|||
|
};
|
|||
|
|
|||
|
Buffer.prototype.writeInt32BE = function (value, offset, noAssert) {
|
|||
|
if (typeof noAssert === "undefined") { noAssert = false; }
|
|||
|
offset += this.offset;
|
|||
|
this.data.writeInt32BE(offset, value);
|
|||
|
};
|
|||
|
|
|||
|
Buffer.prototype.writeFloatLE = function (value, offset, noAssert) {
|
|||
|
if (typeof noAssert === "undefined") { noAssert = false; }
|
|||
|
offset += this.offset;
|
|||
|
this.data.writeFloatLE(offset, value);
|
|||
|
};
|
|||
|
|
|||
|
Buffer.prototype.writeFloatBE = function (value, offset, noAssert) {
|
|||
|
if (typeof noAssert === "undefined") { noAssert = false; }
|
|||
|
offset += this.offset;
|
|||
|
this.data.writeFloatBE(offset, value);
|
|||
|
};
|
|||
|
|
|||
|
Buffer.prototype.writeDoubleLE = function (value, offset, noAssert) {
|
|||
|
if (typeof noAssert === "undefined") { noAssert = false; }
|
|||
|
offset += this.offset;
|
|||
|
this.data.writeDoubleLE(offset, value);
|
|||
|
};
|
|||
|
|
|||
|
Buffer.prototype.writeDoubleBE = function (value, offset, noAssert) {
|
|||
|
if (typeof noAssert === "undefined") { noAssert = false; }
|
|||
|
offset += this.offset;
|
|||
|
this.data.writeDoubleBE(offset, value);
|
|||
|
};
|
|||
|
|
|||
|
///**************************STATIC METHODS********************************///
|
|||
|
/**
|
|||
|
* Checks if enc is a valid string encoding type.
|
|||
|
* @param {string} enc - Name of a string encoding type.
|
|||
|
* @return {boolean} Whether or not enc is a valid encoding type.
|
|||
|
*/
|
|||
|
Buffer.isEncoding = function (enc) {
|
|||
|
try {
|
|||
|
string_util.FindUtil(enc);
|
|||
|
} catch (e) {
|
|||
|
return false;
|
|||
|
}
|
|||
|
return true;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Tests if obj is a Buffer.
|
|||
|
* @param {object} obj - An arbitrary object
|
|||
|
* @return {boolean} True if this object is a Buffer.
|
|||
|
*/
|
|||
|
Buffer.isBuffer = function (obj) {
|
|||
|
return obj instanceof Buffer;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Gives the actual byte length of a string. This is not the same as
|
|||
|
* String.prototype.length since that returns the number of characters in a
|
|||
|
* string.
|
|||
|
* @param {string} str - The string to get the byte length of
|
|||
|
* @param {string} [encoding=utf8] - Character encoding of the string
|
|||
|
* @return {number} The number of bytes in the string
|
|||
|
*/
|
|||
|
Buffer.byteLength = function (str, encoding) {
|
|||
|
if (typeof encoding === "undefined") { encoding = 'utf8'; }
|
|||
|
var strUtil = string_util.FindUtil(encoding);
|
|||
|
return strUtil.byteLength(str);
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Returns a buffer which is the result of concatenating all the buffers in the
|
|||
|
* list together.
|
|||
|
* If the list has no items, or if the totalLength is 0, then it returns a
|
|||
|
* zero-length buffer.
|
|||
|
* If the list has exactly one item, then the first item of the list is
|
|||
|
* returned.
|
|||
|
* If the list has more than one item, then a new Buffer is created.
|
|||
|
* If totalLength is not provided, it is read from the buffers in the list.
|
|||
|
* However, this adds an additional loop to the function, so it is faster to
|
|||
|
* provide the length explicitly.
|
|||
|
* @param {Buffer[]} list - List of Buffer objects to concat
|
|||
|
* @param {number} [totalLength] - Total length of the buffers when concatenated
|
|||
|
* @return {Buffer}
|
|||
|
*/
|
|||
|
Buffer.concat = function (list, totalLength) {
|
|||
|
var item;
|
|||
|
if (list.length === 0 || totalLength === 0) {
|
|||
|
return new Buffer(0);
|
|||
|
} else if (list.length === 1) {
|
|||
|
return list[0];
|
|||
|
} else {
|
|||
|
if (totalLength == null) {
|
|||
|
// Calculate totalLength
|
|||
|
totalLength = 0;
|
|||
|
for (var i = 0; i < list.length; i++) {
|
|||
|
item = list[i];
|
|||
|
totalLength += item.length;
|
|||
|
}
|
|||
|
}
|
|||
|
var buf = new Buffer(totalLength);
|
|||
|
var curPos = 0;
|
|||
|
for (var j = 0; j < list.length; j++) {
|
|||
|
item = list[j];
|
|||
|
curPos += item.copy(buf, curPos);
|
|||
|
}
|
|||
|
return buf;
|
|||
|
}
|
|||
|
};
|
|||
|
return Buffer;
|
|||
|
})();
|
|||
|
exports.Buffer = Buffer;
|
|||
|
|
|||
|
// Type-check the class.
|
|||
|
var _ = Buffer;
|
|||
|
});
|
|||
|
//# sourceMappingURL=buffer.js.map
|
|||
|
;
|
|||
|
define('core/file_flag',["require", "exports", './api_error'], function(require, exports, api_error) {
|
|||
|
/**
|
|||
|
* @class
|
|||
|
*/
|
|||
|
(function (ActionType) {
|
|||
|
// Indicates that the code should not do anything.
|
|||
|
ActionType[ActionType["NOP"] = 0] = "NOP";
|
|||
|
|
|||
|
// Indicates that the code should throw an exception.
|
|||
|
ActionType[ActionType["THROW_EXCEPTION"] = 1] = "THROW_EXCEPTION";
|
|||
|
|
|||
|
// Indicates that the code should truncate the file, but only if it is a file.
|
|||
|
ActionType[ActionType["TRUNCATE_FILE"] = 2] = "TRUNCATE_FILE";
|
|||
|
|
|||
|
// Indicates that the code should create the file.
|
|||
|
ActionType[ActionType["CREATE_FILE"] = 3] = "CREATE_FILE";
|
|||
|
})(exports.ActionType || (exports.ActionType = {}));
|
|||
|
var ActionType = exports.ActionType;
|
|||
|
|
|||
|
/**
|
|||
|
* Represents one of the following file flags. A convenience object.
|
|||
|
*
|
|||
|
* * `'r'` - Open file for reading. An exception occurs if the file does not exist.
|
|||
|
* * `'r+'` - Open file for reading and writing. An exception occurs if the file does not exist.
|
|||
|
* * `'rs'` - Open file for reading in synchronous mode. Instructs the filesystem to not cache writes.
|
|||
|
* * `'rs+'` - Open file for reading and writing, and opens the file in synchronous mode.
|
|||
|
* * `'w'` - Open file for writing. The file is created (if it does not exist) or truncated (if it exists).
|
|||
|
* * `'wx'` - Like 'w' but opens the file in exclusive mode.
|
|||
|
* * `'w+'` - Open file for reading and writing. The file is created (if it does not exist) or truncated (if it exists).
|
|||
|
* * `'wx+'` - Like 'w+' but opens the file in exclusive mode.
|
|||
|
* * `'a'` - Open file for appending. The file is created if it does not exist.
|
|||
|
* * `'ax'` - Like 'a' but opens the file in exclusive mode.
|
|||
|
* * `'a+'` - Open file for reading and appending. The file is created if it does not exist.
|
|||
|
* * `'ax+'` - Like 'a+' but opens the file in exclusive mode.
|
|||
|
*
|
|||
|
* Exclusive mode ensures that the file path is newly created.
|
|||
|
* @class
|
|||
|
*/
|
|||
|
var FileFlag = (function () {
|
|||
|
/**
|
|||
|
* This should never be called directly.
|
|||
|
* @param [String] modeStr The string representing the mode
|
|||
|
* @throw [BrowserFS.ApiError] when the mode string is invalid
|
|||
|
*/
|
|||
|
function FileFlag(flagStr) {
|
|||
|
this.flagStr = flagStr;
|
|||
|
if (FileFlag.validFlagStrs.indexOf(flagStr) < 0) {
|
|||
|
throw new api_error.ApiError(9 /* EINVAL */, "Invalid flag: " + flagStr);
|
|||
|
}
|
|||
|
}
|
|||
|
/**
|
|||
|
* Get an object representing the given file mode.
|
|||
|
* @param [String] modeStr The string representing the mode
|
|||
|
* @return [BrowserFS.FileMode] The FileMode object representing the mode
|
|||
|
* @throw [BrowserFS.ApiError] when the mode string is invalid
|
|||
|
*/
|
|||
|
FileFlag.getFileFlag = function (flagStr) {
|
|||
|
// Check cache first.
|
|||
|
if (FileFlag.flagCache.hasOwnProperty(flagStr)) {
|
|||
|
return FileFlag.flagCache[flagStr];
|
|||
|
}
|
|||
|
return FileFlag.flagCache[flagStr] = new FileFlag(flagStr);
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Returns true if the file is readable.
|
|||
|
* @return [Boolean]
|
|||
|
*/
|
|||
|
FileFlag.prototype.isReadable = function () {
|
|||
|
return this.flagStr.indexOf('r') !== -1 || this.flagStr.indexOf('+') !== -1;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Returns true if the file is writeable.
|
|||
|
* @return [Boolean]
|
|||
|
*/
|
|||
|
FileFlag.prototype.isWriteable = function () {
|
|||
|
return this.flagStr.indexOf('w') !== -1 || this.flagStr.indexOf('a') !== -1 || this.flagStr.indexOf('+') !== -1;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Returns true if the file mode should truncate.
|
|||
|
* @return [Boolean]
|
|||
|
*/
|
|||
|
FileFlag.prototype.isTruncating = function () {
|
|||
|
return this.flagStr.indexOf('w') !== -1;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Returns true if the file is appendable.
|
|||
|
* @return [Boolean]
|
|||
|
*/
|
|||
|
FileFlag.prototype.isAppendable = function () {
|
|||
|
return this.flagStr.indexOf('a') !== -1;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Returns true if the file is open in synchronous mode.
|
|||
|
* @return [Boolean]
|
|||
|
*/
|
|||
|
FileFlag.prototype.isSynchronous = function () {
|
|||
|
return this.flagStr.indexOf('s') !== -1;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Returns true if the file is open in exclusive mode.
|
|||
|
* @return [Boolean]
|
|||
|
*/
|
|||
|
FileFlag.prototype.isExclusive = function () {
|
|||
|
return this.flagStr.indexOf('x') !== -1;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Returns one of the static fields on this object that indicates the
|
|||
|
* appropriate response to the path existing.
|
|||
|
* @return [Number]
|
|||
|
*/
|
|||
|
FileFlag.prototype.pathExistsAction = function () {
|
|||
|
if (this.isExclusive()) {
|
|||
|
return 1 /* THROW_EXCEPTION */;
|
|||
|
} else if (this.isTruncating()) {
|
|||
|
return 2 /* TRUNCATE_FILE */;
|
|||
|
} else {
|
|||
|
return 0 /* NOP */;
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Returns one of the static fields on this object that indicates the
|
|||
|
* appropriate response to the path not existing.
|
|||
|
* @return [Number]
|
|||
|
*/
|
|||
|
FileFlag.prototype.pathNotExistsAction = function () {
|
|||
|
if ((this.isWriteable() || this.isAppendable()) && this.flagStr !== 'r+') {
|
|||
|
return 3 /* CREATE_FILE */;
|
|||
|
} else {
|
|||
|
return 1 /* THROW_EXCEPTION */;
|
|||
|
}
|
|||
|
};
|
|||
|
FileFlag.flagCache = {};
|
|||
|
|
|||
|
FileFlag.validFlagStrs = ['r', 'r+', 'rs', 'rs+', 'w', 'wx', 'w+', 'wx+', 'a', 'ax', 'a+', 'ax+'];
|
|||
|
return FileFlag;
|
|||
|
})();
|
|||
|
exports.FileFlag = FileFlag;
|
|||
|
});
|
|||
|
//# sourceMappingURL=file_flag.js.map
|
|||
|
;
|
|||
|
var __extends = this.__extends || function (d, b) {
|
|||
|
for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p];
|
|||
|
function __() { this.constructor = d; }
|
|||
|
__.prototype = b.prototype;
|
|||
|
d.prototype = new __();
|
|||
|
};
|
|||
|
define('core/node_eventemitter',["require", "exports", './buffer', './api_error'], function(require, exports, buffer, api_error) {
|
|||
|
var Buffer = buffer.Buffer;
|
|||
|
var ApiError = api_error.ApiError;
|
|||
|
var ErrorCode = api_error.ErrorCode;
|
|||
|
|
|||
|
/**
|
|||
|
* Internal class. Represents a buffered event.
|
|||
|
*/
|
|||
|
var BufferedEvent = (function () {
|
|||
|
function BufferedEvent(data, encoding, cb) {
|
|||
|
this.data = data;
|
|||
|
this.encoding = encoding;
|
|||
|
this.cb = cb;
|
|||
|
this.size = typeof (data) !== 'string' ? data.length : Buffer.byteLength(data, encoding != null ? encoding : undefined);
|
|||
|
|
|||
|
// If data is a buffer, we need to copy it.
|
|||
|
if (typeof (this.data) !== 'string') {
|
|||
|
this.data = this.data.sliceCopy();
|
|||
|
}
|
|||
|
}
|
|||
|
BufferedEvent.prototype.getData = function (encoding) {
|
|||
|
if (encoding == null) {
|
|||
|
if (typeof (this.data) === 'string') {
|
|||
|
return new Buffer(this.data, this.encoding != null ? this.encoding : undefined);
|
|||
|
} else {
|
|||
|
return this.data;
|
|||
|
}
|
|||
|
} else {
|
|||
|
if (typeof (this.data) === 'string') {
|
|||
|
if (encoding === this.encoding) {
|
|||
|
return this.data;
|
|||
|
} else {
|
|||
|
return (new Buffer(this.data, this.encoding != null ? this.encoding : undefined)).toString(encoding);
|
|||
|
}
|
|||
|
} else {
|
|||
|
return this.data.toString(encoding);
|
|||
|
}
|
|||
|
}
|
|||
|
};
|
|||
|
return BufferedEvent;
|
|||
|
})();
|
|||
|
|
|||
|
/**
|
|||
|
* Provides an abstract implementation of the EventEmitter interface.
|
|||
|
*/
|
|||
|
var AbstractEventEmitter = (function () {
|
|||
|
function AbstractEventEmitter() {
|
|||
|
this._listeners = {};
|
|||
|
this.maxListeners = 10;
|
|||
|
}
|
|||
|
/**
|
|||
|
* Adds a listener for the particular event.
|
|||
|
*/
|
|||
|
AbstractEventEmitter.prototype.addListener = function (event, listener) {
|
|||
|
if (typeof (this._listeners[event]) === 'undefined') {
|
|||
|
this._listeners[event] = [];
|
|||
|
}
|
|||
|
if (this._listeners[event].push(listener) > this.maxListeners) {
|
|||
|
process.stdout.write("Warning: Event " + event + " has more than " + this.maxListeners + " listeners.\n");
|
|||
|
}
|
|||
|
this.emit('newListener', event, listener);
|
|||
|
return this;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Adds a listener for the particular event.
|
|||
|
*/
|
|||
|
AbstractEventEmitter.prototype.on = function (event, listener) {
|
|||
|
return this.addListener(event, listener);
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Adds a listener for the particular event that fires only once.
|
|||
|
*/
|
|||
|
AbstractEventEmitter.prototype.once = function (event, listener) {
|
|||
|
// Create a new callback that will only fire once.
|
|||
|
var fired = false, newListener = function () {
|
|||
|
this.removeListener(event, newListener);
|
|||
|
|
|||
|
if (!fired) {
|
|||
|
fired = true;
|
|||
|
listener.apply(this, arguments);
|
|||
|
}
|
|||
|
};
|
|||
|
return this.addListener(event, newListener);
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Emits the 'removeListener' event for the specified listeners.
|
|||
|
*/
|
|||
|
AbstractEventEmitter.prototype._emitRemoveListener = function (event, listeners) {
|
|||
|
var i;
|
|||
|
|
|||
|
// Only emit the event if someone is listening.
|
|||
|
if (this._listeners['removeListener'] && this._listeners['removeListener'].length > 0) {
|
|||
|
for (i = 0; i < listeners.length; i++) {
|
|||
|
this.emit('removeListener', event, listeners[i]);
|
|||
|
}
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Removes the particular listener for the given event.
|
|||
|
*/
|
|||
|
AbstractEventEmitter.prototype.removeListener = function (event, listener) {
|
|||
|
var listeners = this._listeners[event];
|
|||
|
if (typeof (listeners) !== 'undefined') {
|
|||
|
// Remove listener, if present.
|
|||
|
var idx = listeners.indexOf(listener);
|
|||
|
if (idx > -1) {
|
|||
|
listeners.splice(idx, 1);
|
|||
|
}
|
|||
|
}
|
|||
|
this.emit('removeListener', event, listener);
|
|||
|
return this;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Removes all listeners, or those of the specified event.
|
|||
|
*/
|
|||
|
AbstractEventEmitter.prototype.removeAllListeners = function (event) {
|
|||
|
var removed, keys, i;
|
|||
|
if (typeof (event) !== 'undefined') {
|
|||
|
removed = this._listeners[event];
|
|||
|
|
|||
|
// Clear one event.
|
|||
|
this._listeners[event] = [];
|
|||
|
this._emitRemoveListener(event, removed);
|
|||
|
} else {
|
|||
|
// Clear all events.
|
|||
|
keys = Object.keys(this._listeners);
|
|||
|
for (i = 0; i < keys.length; i++) {
|
|||
|
this.removeAllListeners(keys[i]);
|
|||
|
}
|
|||
|
}
|
|||
|
return this;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* EventEmitters print a warning when an event has greater than this specified
|
|||
|
* number of listeners.
|
|||
|
*/
|
|||
|
AbstractEventEmitter.prototype.setMaxListeners = function (n) {
|
|||
|
this.maxListeners = n;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Returns the listeners for the given event.
|
|||
|
*/
|
|||
|
AbstractEventEmitter.prototype.listeners = function (event) {
|
|||
|
if (typeof (this._listeners[event]) === 'undefined') {
|
|||
|
this._listeners[event] = [];
|
|||
|
}
|
|||
|
|
|||
|
// Return a *copy* of our internal structure.
|
|||
|
return this._listeners[event].slice(0);
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Emits the specified event to all listeners of the particular event.
|
|||
|
*/
|
|||
|
AbstractEventEmitter.prototype.emit = function (event) {
|
|||
|
var args = [];
|
|||
|
for (var _i = 0; _i < (arguments.length - 1); _i++) {
|
|||
|
args[_i] = arguments[_i + 1];
|
|||
|
}
|
|||
|
var listeners = this._listeners[event], rv = false;
|
|||
|
if (typeof (listeners) !== 'undefined') {
|
|||
|
var i;
|
|||
|
for (i = 0; i < listeners.length; i++) {
|
|||
|
rv = true;
|
|||
|
listeners[i].apply(this, args);
|
|||
|
}
|
|||
|
}
|
|||
|
return rv;
|
|||
|
};
|
|||
|
return AbstractEventEmitter;
|
|||
|
})();
|
|||
|
exports.AbstractEventEmitter = AbstractEventEmitter;
|
|||
|
|
|||
|
/**
|
|||
|
* Provides an abstract implementation of the WritableStream and ReadableStream
|
|||
|
* interfaces.
|
|||
|
* @todo: Check readable/writable status.
|
|||
|
*/
|
|||
|
var AbstractDuplexStream = (function (_super) {
|
|||
|
__extends(AbstractDuplexStream, _super);
|
|||
|
/**
|
|||
|
* Abstract stream implementation that can be configured to be readable and/or
|
|||
|
* writable.
|
|||
|
*/
|
|||
|
function AbstractDuplexStream(writable, readable) {
|
|||
|
_super.call(this);
|
|||
|
this.writable = writable;
|
|||
|
this.readable = readable;
|
|||
|
/**
|
|||
|
* How should the data output be encoded? 'null' means 'Buffer'.
|
|||
|
*/
|
|||
|
this.encoding = null;
|
|||
|
/**
|
|||
|
* Is this stream currently flowing (resumed) or non-flowing (paused)?
|
|||
|
*/
|
|||
|
this.flowing = false;
|
|||
|
/**
|
|||
|
* Event buffer. Simply queues up all write requests.
|
|||
|
*/
|
|||
|
this.buffer = [];
|
|||
|
/**
|
|||
|
* Once set, the stream is closed. Emitted once 'buffer' is empty.
|
|||
|
*/
|
|||
|
this.endEvent = null;
|
|||
|
/**
|
|||
|
* Has the stream ended?
|
|||
|
*/
|
|||
|
this.ended = false;
|
|||
|
/**
|
|||
|
* The last time we checked, was the buffer empty?
|
|||
|
* We emit 'readable' events when this transitions from 'true' -> 'false'.
|
|||
|
*/
|
|||
|
this.drained = true;
|
|||
|
}
|
|||
|
/**
|
|||
|
* Adds a listener for the particular event.
|
|||
|
* Implemented here so that we can capture data EventListeners, which trigger
|
|||
|
* us to 'resume'.
|
|||
|
*/
|
|||
|
AbstractDuplexStream.prototype.addListener = function (event, listener) {
|
|||
|
var rv = _super.prototype.addListener.call(this, event, listener), _this = this;
|
|||
|
if (event === 'data' && !this.flowing) {
|
|||
|
this.resume();
|
|||
|
} else if (event === 'readable' && this.buffer.length > 0) {
|
|||
|
setTimeout(function () {
|
|||
|
_this.emit('readable');
|
|||
|
}, 0);
|
|||
|
}
|
|||
|
return rv;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Helper function for 'write' and 'end' functions.
|
|||
|
*/
|
|||
|
AbstractDuplexStream.prototype._processArgs = function (data, arg2, arg3) {
|
|||
|
if (typeof (arg2) === 'string') {
|
|||
|
// data, encoding, cb?
|
|||
|
return new BufferedEvent(data, arg2, arg3);
|
|||
|
} else {
|
|||
|
// data, cb?
|
|||
|
return new BufferedEvent(data, null, arg2);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* If flowing, this will process pending events.
|
|||
|
*/
|
|||
|
AbstractDuplexStream.prototype._processEvents = function () {
|
|||
|
var drained = this.buffer.length === 0;
|
|||
|
if (this.drained !== drained) {
|
|||
|
if (this.drained) {
|
|||
|
// Went from drained to not drained. New stuff is available.
|
|||
|
// @todo: Is this event relevant in flowing mode?
|
|||
|
this.emit('readable');
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
if (this.flowing && this.buffer.length !== 0) {
|
|||
|
this.emit('data', this.read());
|
|||
|
}
|
|||
|
|
|||
|
// Are we drained? Check.
|
|||
|
this.drained = this.buffer.length === 0;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Emits the given buffered event.
|
|||
|
*/
|
|||
|
AbstractDuplexStream.prototype.emitEvent = function (type, event) {
|
|||
|
this.emit(type, event.getData(this.encoding));
|
|||
|
if (event.cb) {
|
|||
|
event.cb();
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
AbstractDuplexStream.prototype.write = function (data, arg2, arg3) {
|
|||
|
if (this.ended) {
|
|||
|
throw new ApiError(0 /* EPERM */, 'Cannot write to an ended stream.');
|
|||
|
}
|
|||
|
var event = this._processArgs(data, arg2, arg3);
|
|||
|
this._push(event);
|
|||
|
return this.flowing;
|
|||
|
};
|
|||
|
|
|||
|
AbstractDuplexStream.prototype.end = function (data, arg2, arg3) {
|
|||
|
if (this.ended) {
|
|||
|
throw new ApiError(0 /* EPERM */, 'Stream is already closed.');
|
|||
|
}
|
|||
|
var event = this._processArgs(data, arg2, arg3);
|
|||
|
this.ended = true;
|
|||
|
this.endEvent = event;
|
|||
|
this._processEvents();
|
|||
|
};
|
|||
|
|
|||
|
/**** Readable Interface ****/
|
|||
|
/**
|
|||
|
* Read a given number of bytes from the buffer. Should only be called in
|
|||
|
* non-flowing mode.
|
|||
|
* If we do not have `size` bytes available, return null.
|
|||
|
*/
|
|||
|
AbstractDuplexStream.prototype.read = function (size) {
|
|||
|
var events = [], eventsCbs = [], lastCb, eventsSize = 0, event, buff, trueSize, i = 0, sizeUnspecified = typeof (size) !== 'number';
|
|||
|
|
|||
|
// I do this so I do not need to specialize the loop below.
|
|||
|
if (sizeUnspecified)
|
|||
|
size = 4294967295;
|
|||
|
|
|||
|
for (i = 0; i < this.buffer.length && eventsSize < size; i++) {
|
|||
|
event = this.buffer[i];
|
|||
|
events.push(event.getData());
|
|||
|
if (event.cb) {
|
|||
|
eventsCbs.push(event.cb);
|
|||
|
}
|
|||
|
eventsSize += event.size;
|
|||
|
lastCb = event.cb;
|
|||
|
}
|
|||
|
|
|||
|
if (!sizeUnspecified && eventsSize < size) {
|
|||
|
// For some reason, the Node stream API specifies that we either return
|
|||
|
// 'size' bytes of data, or nothing at all.
|
|||
|
return null;
|
|||
|
}
|
|||
|
|
|||
|
// Remove all of the events we are processing from the buffer.
|
|||
|
this.buffer = this.buffer.slice(events.length);
|
|||
|
|
|||
|
// The 'true size' of the final event we're going to send out.
|
|||
|
trueSize = eventsSize > size ? size : eventsSize;
|
|||
|
|
|||
|
// Concat at all of the events into one buffer.
|
|||
|
buff = Buffer.concat(events);
|
|||
|
if (eventsSize > size) {
|
|||
|
// If last event had a cb, ignore it -- we trigger it when that *entire*
|
|||
|
// write finishes.
|
|||
|
if (lastCb)
|
|||
|
eventsCbs.pop();
|
|||
|
|
|||
|
// Make a new event for the remaining data.
|
|||
|
this._push(new BufferedEvent(buff.slice(size), null, lastCb));
|
|||
|
}
|
|||
|
|
|||
|
// Schedule the relevant cbs to fire *after* we've returned these values.
|
|||
|
if (eventsCbs.length > 0) {
|
|||
|
setTimeout(function () {
|
|||
|
var i;
|
|||
|
for (i = 0; i < eventsCbs.length; i++) {
|
|||
|
eventsCbs[i]();
|
|||
|
}
|
|||
|
}, 0);
|
|||
|
}
|
|||
|
|
|||
|
// If we're at the end of the buffer and an endEvent is specified, schedule
|
|||
|
// the event to fire.
|
|||
|
if (this.ended && this.buffer.length === 0 && this.endEvent !== null) {
|
|||
|
var endEvent = this.endEvent, _this = this;
|
|||
|
|
|||
|
// Erase it so we don't accidentally trigger it again.
|
|||
|
this.endEvent = null;
|
|||
|
setTimeout(function () {
|
|||
|
_this.emitEvent('end', endEvent);
|
|||
|
}, 0);
|
|||
|
}
|
|||
|
|
|||
|
// Return in correct encoding.
|
|||
|
if (events.length === 0) {
|
|||
|
// Buffer was empty. We're supposed to return 'null', as opposed to an
|
|||
|
// empty buffer or string.
|
|||
|
// [BFS] Emit a '_read' event to signal that maybe the write-end of this
|
|||
|
// should push some data into the pipe.
|
|||
|
this.emit('_read');
|
|||
|
return null;
|
|||
|
} else if (this.encoding === null) {
|
|||
|
return buff.slice(0, trueSize);
|
|||
|
} else {
|
|||
|
return buff.toString(this.encoding, 0, trueSize);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Set the encoding for the 'data' event.
|
|||
|
*/
|
|||
|
AbstractDuplexStream.prototype.setEncoding = function (encoding) {
|
|||
|
this.encoding = encoding;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Pause the stream.
|
|||
|
*/
|
|||
|
AbstractDuplexStream.prototype.pause = function () {
|
|||
|
this.flowing = false;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Resume the stream.
|
|||
|
*/
|
|||
|
AbstractDuplexStream.prototype.resume = function () {
|
|||
|
this.flowing = true;
|
|||
|
|
|||
|
// Process any buffered writes.
|
|||
|
this._processEvents();
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Pipe a readable stream into a writable stream. Currently unimplemented.
|
|||
|
*/
|
|||
|
AbstractDuplexStream.prototype.pipe = function (destination, options) {
|
|||
|
throw new ApiError(0 /* EPERM */, "Unimplemented.");
|
|||
|
};
|
|||
|
AbstractDuplexStream.prototype.unpipe = function (destination) {
|
|||
|
};
|
|||
|
|
|||
|
AbstractDuplexStream.prototype.unshift = function (chunk) {
|
|||
|
if (this.ended) {
|
|||
|
throw new ApiError(0 /* EPERM */, "Stream has ended.");
|
|||
|
}
|
|||
|
this.buffer.unshift(new BufferedEvent(chunk, this.encoding));
|
|||
|
this._processEvents();
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* 'Push' the given piece of data to the back of the buffer.
|
|||
|
* Returns true if the event was sent out, false if buffered.
|
|||
|
*/
|
|||
|
AbstractDuplexStream.prototype._push = function (event) {
|
|||
|
this.buffer.push(event);
|
|||
|
this._processEvents();
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Enables backwards-compatibility with older versions of Node and their
|
|||
|
* stream interface. Unimplemented.
|
|||
|
*/
|
|||
|
AbstractDuplexStream.prototype.wrap = function (stream) {
|
|||
|
throw new ApiError(0 /* EPERM */, "Unimplemented.");
|
|||
|
};
|
|||
|
return AbstractDuplexStream;
|
|||
|
})(AbstractEventEmitter);
|
|||
|
exports.AbstractDuplexStream = AbstractDuplexStream;
|
|||
|
});
|
|||
|
//# sourceMappingURL=node_eventemitter.js.map
|
|||
|
;
|
|||
|
var __extends = this.__extends || function (d, b) {
|
|||
|
for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p];
|
|||
|
function __() { this.constructor = d; }
|
|||
|
__.prototype = b.prototype;
|
|||
|
d.prototype = new __();
|
|||
|
};
|
|||
|
define('core/node_process',["require", "exports", './node_eventemitter'], function(require, exports, eventemitter) {
|
|||
|
var path = null;
|
|||
|
|
|||
|
var TTY = (function (_super) {
|
|||
|
__extends(TTY, _super);
|
|||
|
function TTY() {
|
|||
|
_super.call(this, true, true);
|
|||
|
this.isRaw = false;
|
|||
|
this.columns = 80;
|
|||
|
this.rows = 120;
|
|||
|
this.isTTY = true;
|
|||
|
}
|
|||
|
/**
|
|||
|
* Set read mode to 'true' to enable raw mode.
|
|||
|
*/
|
|||
|
TTY.prototype.setReadMode = function (mode) {
|
|||
|
if (this.isRaw !== mode) {
|
|||
|
this.isRaw = mode;
|
|||
|
|
|||
|
// [BFS] TTY implementations can use this to change their event emitting
|
|||
|
// patterns.
|
|||
|
this.emit('modeChange');
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* [BFS] Update the number of columns available on the terminal.
|
|||
|
*/
|
|||
|
TTY.prototype.changeColumns = function (columns) {
|
|||
|
if (columns !== this.columns) {
|
|||
|
this.columns = columns;
|
|||
|
|
|||
|
// Resize event.
|
|||
|
this.emit('resize');
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* [BFS] Update the number of rows available on the terminal.
|
|||
|
*/
|
|||
|
TTY.prototype.changeRows = function (rows) {
|
|||
|
if (rows !== this.rows) {
|
|||
|
this.rows = rows;
|
|||
|
|
|||
|
// Resize event.
|
|||
|
this.emit('resize');
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Returns 'true' if the given object is a TTY.
|
|||
|
*/
|
|||
|
TTY.isatty = function (fd) {
|
|||
|
return fd instanceof TTY;
|
|||
|
};
|
|||
|
return TTY;
|
|||
|
})(eventemitter.AbstractDuplexStream);
|
|||
|
exports.TTY = TTY;
|
|||
|
|
|||
|
/**
|
|||
|
* Partial implementation of Node's `process` module.
|
|||
|
* We implement the portions that are relevant for the filesystem.
|
|||
|
* @see http://nodejs.org/api/process.html
|
|||
|
* @class
|
|||
|
*/
|
|||
|
var Process = (function () {
|
|||
|
function Process() {
|
|||
|
this.startTime = Date.now();
|
|||
|
this._cwd = '/';
|
|||
|
/**
|
|||
|
* Returns what platform you are running on.
|
|||
|
* @return [String]
|
|||
|
*/
|
|||
|
this.platform = 'browser';
|
|||
|
this.argv = [];
|
|||
|
this.stdout = new TTY();
|
|||
|
this.stderr = new TTY();
|
|||
|
this.stdin = new TTY();
|
|||
|
}
|
|||
|
/**
|
|||
|
* Changes the current working directory.
|
|||
|
*
|
|||
|
* **Note**: BrowserFS does not validate that the directory actually exists.
|
|||
|
*
|
|||
|
* @example Usage example
|
|||
|
* console.log('Starting directory: ' + process.cwd());
|
|||
|
* process.chdir('/tmp');
|
|||
|
* console.log('New directory: ' + process.cwd());
|
|||
|
* @param [String] dir The directory to change to.
|
|||
|
*/
|
|||
|
Process.prototype.chdir = function (dir) {
|
|||
|
// XXX: Circular dependency hack.
|
|||
|
if (path === null) {
|
|||
|
path = require('./node_path').path;
|
|||
|
}
|
|||
|
this._cwd = path.resolve(dir);
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Returns the current working directory.
|
|||
|
* @example Usage example
|
|||
|
* console.log('Current directory: ' + process.cwd());
|
|||
|
* @return [String] The current working directory.
|
|||
|
*/
|
|||
|
Process.prototype.cwd = function () {
|
|||
|
return this._cwd;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Number of seconds BrowserFS has been running.
|
|||
|
* @return [Number]
|
|||
|
*/
|
|||
|
Process.prototype.uptime = function () {
|
|||
|
return ((Date.now() - this.startTime) / 1000) | 0;
|
|||
|
};
|
|||
|
return Process;
|
|||
|
})();
|
|||
|
exports.Process = Process;
|
|||
|
|
|||
|
// process is a singleton.
|
|||
|
exports.process = new Process();
|
|||
|
});
|
|||
|
//# sourceMappingURL=node_process.js.map
|
|||
|
;
|
|||
|
define('core/node_path',["require", "exports", './node_process'], function(require, exports, node_process) {
|
|||
|
var process = node_process.process;
|
|||
|
|
|||
|
/**
|
|||
|
* Emulates Node's `path` module. This module contains utilities for handling and
|
|||
|
* transforming file paths. **All** of these methods perform only string
|
|||
|
* transformations. The file system is not consulted to check whether paths are
|
|||
|
* valid.
|
|||
|
* @see http://nodejs.org/api/path.html
|
|||
|
* @class
|
|||
|
*/
|
|||
|
var path = (function () {
|
|||
|
function path() {
|
|||
|
}
|
|||
|
/**
|
|||
|
* Normalize a string path, taking care of '..' and '.' parts.
|
|||
|
*
|
|||
|
* When multiple slashes are found, they're replaced by a single one; when the path contains a trailing slash, it is preserved. On Windows backslashes are used.
|
|||
|
* @example Usage example
|
|||
|
* path.normalize('/foo/bar//baz/asdf/quux/..')
|
|||
|
* // returns
|
|||
|
* '/foo/bar/baz/asdf'
|
|||
|
* @param [String] p The path to normalize.
|
|||
|
* @return [String]
|
|||
|
*/
|
|||
|
path.normalize = function (p) {
|
|||
|
// Special case: '' -> '.'
|
|||
|
if (p === '') {
|
|||
|
p = '.';
|
|||
|
}
|
|||
|
|
|||
|
// It's very important to know if the path is relative or not, since it
|
|||
|
// changes how we process .. and reconstruct the split string.
|
|||
|
var absolute = p.charAt(0) === path.sep;
|
|||
|
|
|||
|
// Remove repeated //s
|
|||
|
p = path._removeDuplicateSeps(p);
|
|||
|
|
|||
|
// Try to remove as many '../' as possible, and remove '.' completely.
|
|||
|
var components = p.split(path.sep);
|
|||
|
var goodComponents = [];
|
|||
|
for (var idx = 0; idx < components.length; idx++) {
|
|||
|
var c = components[idx];
|
|||
|
if (c === '.') {
|
|||
|
continue;
|
|||
|
} else if (c === '..' && (absolute || (!absolute && goodComponents.length > 0 && goodComponents[0] !== '..'))) {
|
|||
|
// In the absolute case: Path is relative to root, so we may pop even if
|
|||
|
// goodComponents is empty (e.g. /../ => /)
|
|||
|
// In the relative case: We're getting rid of a directory that preceded
|
|||
|
// it (e.g. /foo/../bar -> /bar)
|
|||
|
goodComponents.pop();
|
|||
|
} else {
|
|||
|
goodComponents.push(c);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
// Add in '.' when it's a relative path with no other nonempty components.
|
|||
|
// Possible results: '.' and './' (input: [''] or [])
|
|||
|
// @todo Can probably simplify this logic.
|
|||
|
if (!absolute && goodComponents.length < 2) {
|
|||
|
switch (goodComponents.length) {
|
|||
|
case 1:
|
|||
|
if (goodComponents[0] === '') {
|
|||
|
goodComponents.unshift('.');
|
|||
|
}
|
|||
|
break;
|
|||
|
default:
|
|||
|
goodComponents.push('.');
|
|||
|
}
|
|||
|
}
|
|||
|
p = goodComponents.join(path.sep);
|
|||
|
if (absolute && p.charAt(0) !== path.sep) {
|
|||
|
p = path.sep + p;
|
|||
|
}
|
|||
|
return p;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Join all arguments together and normalize the resulting path.
|
|||
|
*
|
|||
|
* Arguments must be strings.
|
|||
|
* @example Usage
|
|||
|
* path.join('/foo', 'bar', 'baz/asdf', 'quux', '..')
|
|||
|
* // returns
|
|||
|
* '/foo/bar/baz/asdf'
|
|||
|
*
|
|||
|
* path.join('foo', {}, 'bar')
|
|||
|
* // throws exception
|
|||
|
* TypeError: Arguments to path.join must be strings
|
|||
|
* @param [String,...] paths Each component of the path
|
|||
|
* @return [String]
|
|||
|
*/
|
|||
|
path.join = function () {
|
|||
|
var paths = [];
|
|||
|
for (var _i = 0; _i < (arguments.length - 0); _i++) {
|
|||
|
paths[_i] = arguments[_i + 0];
|
|||
|
}
|
|||
|
// Required: Prune any non-strings from the path. I also prune empty segments
|
|||
|
// so we can do a simple join of the array.
|
|||
|
var processed = [];
|
|||
|
for (var i = 0; i < paths.length; i++) {
|
|||
|
var segment = paths[i];
|
|||
|
if (typeof segment !== 'string') {
|
|||
|
throw new TypeError("Invalid argument type to path.join: " + (typeof segment));
|
|||
|
} else if (segment !== '') {
|
|||
|
processed.push(segment);
|
|||
|
}
|
|||
|
}
|
|||
|
return path.normalize(processed.join(path.sep));
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Resolves to to an absolute path.
|
|||
|
*
|
|||
|
* If to isn't already absolute from arguments are prepended in right to left
|
|||
|
* order, until an absolute path is found. If after using all from paths still
|
|||
|
* no absolute path is found, the current working directory is used as well.
|
|||
|
* The resulting path is normalized, and trailing slashes are removed unless
|
|||
|
* the path gets resolved to the root directory. Non-string arguments are
|
|||
|
* ignored.
|
|||
|
*
|
|||
|
* Another way to think of it is as a sequence of cd commands in a shell.
|
|||
|
*
|
|||
|
* path.resolve('foo/bar', '/tmp/file/', '..', 'a/../subfile')
|
|||
|
*
|
|||
|
* Is similar to:
|
|||
|
*
|
|||
|
* cd foo/bar
|
|||
|
* cd /tmp/file/
|
|||
|
* cd ..
|
|||
|
* cd a/../subfile
|
|||
|
* pwd
|
|||
|
*
|
|||
|
* The difference is that the different paths don't need to exist and may also
|
|||
|
* be files.
|
|||
|
* @example Usage example
|
|||
|
* path.resolve('/foo/bar', './baz')
|
|||
|
* // returns
|
|||
|
* '/foo/bar/baz'
|
|||
|
*
|
|||
|
* path.resolve('/foo/bar', '/tmp/file/')
|
|||
|
* // returns
|
|||
|
* '/tmp/file'
|
|||
|
*
|
|||
|
* path.resolve('wwwroot', 'static_files/png/', '../gif/image.gif')
|
|||
|
* // if currently in /home/myself/node, it returns
|
|||
|
* '/home/myself/node/wwwroot/static_files/gif/image.gif'
|
|||
|
* @param [String,...] paths
|
|||
|
* @return [String]
|
|||
|
*/
|
|||
|
path.resolve = function () {
|
|||
|
var paths = [];
|
|||
|
for (var _i = 0; _i < (arguments.length - 0); _i++) {
|
|||
|
paths[_i] = arguments[_i + 0];
|
|||
|
}
|
|||
|
// Monitor for invalid paths, throw out empty paths, and look for the *last*
|
|||
|
// absolute path that we see.
|
|||
|
var processed = [];
|
|||
|
for (var i = 0; i < paths.length; i++) {
|
|||
|
var p = paths[i];
|
|||
|
if (typeof p !== 'string') {
|
|||
|
throw new TypeError("Invalid argument type to path.join: " + (typeof p));
|
|||
|
} else if (p !== '') {
|
|||
|
// Remove anything that has occurred before this absolute path, as it
|
|||
|
// doesn't matter.
|
|||
|
if (p.charAt(0) === path.sep) {
|
|||
|
processed = [];
|
|||
|
}
|
|||
|
processed.push(p);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
// Special: Remove trailing slash unless it's the root
|
|||
|
var resolved = path.normalize(processed.join(path.sep));
|
|||
|
if (resolved.length > 1 && resolved.charAt(resolved.length - 1) === path.sep) {
|
|||
|
return resolved.substr(0, resolved.length - 1);
|
|||
|
}
|
|||
|
|
|||
|
// Special: If it doesn't start with '/', it's relative and we need to append
|
|||
|
// the current directory.
|
|||
|
if (resolved.charAt(0) !== path.sep) {
|
|||
|
// Remove ./, since we're going to append the current directory.
|
|||
|
if (resolved.charAt(0) === '.' && (resolved.length === 1 || resolved.charAt(1) === path.sep)) {
|
|||
|
resolved = resolved.length === 1 ? '' : resolved.substr(2);
|
|||
|
}
|
|||
|
|
|||
|
// Append the current directory, which *must* be an absolute path.
|
|||
|
var cwd = process.cwd();
|
|||
|
if (resolved !== '') {
|
|||
|
// cwd will never end in a /... unless it's the root.
|
|||
|
resolved = this.normalize(cwd + (cwd !== '/' ? path.sep : '') + resolved);
|
|||
|
} else {
|
|||
|
resolved = cwd;
|
|||
|
}
|
|||
|
}
|
|||
|
return resolved;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Solve the relative path from from to to.
|
|||
|
*
|
|||
|
* At times we have two absolute paths, and we need to derive the relative path
|
|||
|
* from one to the other. This is actually the reverse transform of
|
|||
|
* path.resolve, which means we see that:
|
|||
|
*
|
|||
|
* path.resolve(from, path.relative(from, to)) == path.resolve(to)
|
|||
|
*
|
|||
|
* @example Usage example
|
|||
|
* path.relative('C:\\orandea\\test\\aaa', 'C:\\orandea\\impl\\bbb')
|
|||
|
* // returns
|
|||
|
* '..\\..\\impl\\bbb'
|
|||
|
*
|
|||
|
* path.relative('/data/orandea/test/aaa', '/data/orandea/impl/bbb')
|
|||
|
* // returns
|
|||
|
* '../../impl/bbb'
|
|||
|
* @param [String] from
|
|||
|
* @param [String] to
|
|||
|
* @return [String]
|
|||
|
*/
|
|||
|
path.relative = function (from, to) {
|
|||
|
var i;
|
|||
|
|
|||
|
// Alright. Let's resolve these two to absolute paths and remove any
|
|||
|
// weirdness.
|
|||
|
from = path.resolve(from);
|
|||
|
to = path.resolve(to);
|
|||
|
var fromSegs = from.split(path.sep);
|
|||
|
var toSegs = to.split(path.sep);
|
|||
|
|
|||
|
// Remove the first segment on both, as it's '' (both are absolute paths)
|
|||
|
toSegs.shift();
|
|||
|
fromSegs.shift();
|
|||
|
|
|||
|
// There are two segments to this path:
|
|||
|
// * Going *up* the directory hierarchy with '..'
|
|||
|
// * Going *down* the directory hierarchy with foo/baz/bat.
|
|||
|
var upCount = 0;
|
|||
|
var downSegs = [];
|
|||
|
|
|||
|
for (i = 0; i < fromSegs.length; i++) {
|
|||
|
var seg = fromSegs[i];
|
|||
|
if (seg === toSegs[i]) {
|
|||
|
continue;
|
|||
|
}
|
|||
|
|
|||
|
// The rest of 'from', including the current element, indicates how many
|
|||
|
// directories we need to go up.
|
|||
|
upCount = fromSegs.length - i;
|
|||
|
break;
|
|||
|
}
|
|||
|
|
|||
|
// The rest of 'to' indicates where we need to change to. We place this
|
|||
|
// outside of the loop, as toSegs.length may be greater than fromSegs.length.
|
|||
|
downSegs = toSegs.slice(i);
|
|||
|
|
|||
|
// Special case: If 'from' is '/'
|
|||
|
if (fromSegs.length === 1 && fromSegs[0] === '') {
|
|||
|
upCount = 0;
|
|||
|
}
|
|||
|
|
|||
|
// upCount can't be greater than the number of fromSegs
|
|||
|
// (cd .. from / is still /)
|
|||
|
if (upCount > fromSegs.length) {
|
|||
|
upCount = fromSegs.length;
|
|||
|
}
|
|||
|
|
|||
|
// Create the final string!
|
|||
|
var rv = '';
|
|||
|
for (i = 0; i < upCount; i++) {
|
|||
|
rv += '../';
|
|||
|
}
|
|||
|
rv += downSegs.join(path.sep);
|
|||
|
|
|||
|
// Special case: Remove trailing '/'. Happens if it's all up and no down.
|
|||
|
if (rv.length > 1 && rv.charAt(rv.length - 1) === path.sep) {
|
|||
|
rv = rv.substr(0, rv.length - 1);
|
|||
|
}
|
|||
|
return rv;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Return the directory name of a path. Similar to the Unix `dirname` command.
|
|||
|
*
|
|||
|
* Note that BrowserFS does not validate if the path is actually a valid
|
|||
|
* directory.
|
|||
|
* @example Usage example
|
|||
|
* path.dirname('/foo/bar/baz/asdf/quux')
|
|||
|
* // returns
|
|||
|
* '/foo/bar/baz/asdf'
|
|||
|
* @param [String] p The path to get the directory name of.
|
|||
|
* @return [String]
|
|||
|
*/
|
|||
|
path.dirname = function (p) {
|
|||
|
// We get rid of //, but we don't modify anything else (e.g. any extraneous .
|
|||
|
// and ../ are kept intact)
|
|||
|
p = path._removeDuplicateSeps(p);
|
|||
|
var absolute = p.charAt(0) === path.sep;
|
|||
|
var sections = p.split(path.sep);
|
|||
|
|
|||
|
// Do 1 if it's /foo/bar, 2 if it's /foo/bar/
|
|||
|
if (sections.pop() === '' && sections.length > 0) {
|
|||
|
sections.pop();
|
|||
|
}
|
|||
|
if (sections.length > 1) {
|
|||
|
return sections.join(path.sep);
|
|||
|
} else if (absolute) {
|
|||
|
return path.sep;
|
|||
|
} else {
|
|||
|
return '.';
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Return the last portion of a path. Similar to the Unix basename command.
|
|||
|
* @example Usage example
|
|||
|
* path.basename('/foo/bar/baz/asdf/quux.html')
|
|||
|
* // returns
|
|||
|
* 'quux.html'
|
|||
|
*
|
|||
|
* path.basename('/foo/bar/baz/asdf/quux.html', '.html')
|
|||
|
* // returns
|
|||
|
* 'quux'
|
|||
|
* @param [String] p
|
|||
|
* @param [String?] ext
|
|||
|
* @return [String]
|
|||
|
*/
|
|||
|
path.basename = function (p, ext) {
|
|||
|
if (typeof ext === "undefined") { ext = ""; }
|
|||
|
// Special case: Normalize will modify this to '.'
|
|||
|
if (p === '') {
|
|||
|
return p;
|
|||
|
}
|
|||
|
|
|||
|
// Normalize the string first to remove any weirdness.
|
|||
|
p = path.normalize(p);
|
|||
|
|
|||
|
// Get the last part of the string.
|
|||
|
var sections = p.split(path.sep);
|
|||
|
var lastPart = sections[sections.length - 1];
|
|||
|
|
|||
|
// Special case: If it's empty, then we have a string like so: foo/
|
|||
|
// Meaning, 'foo' is guaranteed to be a directory.
|
|||
|
if (lastPart === '' && sections.length > 1) {
|
|||
|
return sections[sections.length - 2];
|
|||
|
}
|
|||
|
|
|||
|
// Remove the extension, if need be.
|
|||
|
if (ext.length > 0) {
|
|||
|
var lastPartExt = lastPart.substr(lastPart.length - ext.length);
|
|||
|
if (lastPartExt === ext) {
|
|||
|
return lastPart.substr(0, lastPart.length - ext.length);
|
|||
|
}
|
|||
|
}
|
|||
|
return lastPart;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Return the extension of the path, from the last '.' to end of string in the
|
|||
|
* last portion of the path. If there is no '.' in the last portion of the path
|
|||
|
* or the first character of it is '.', then it returns an empty string.
|
|||
|
* @example Usage example
|
|||
|
* path.extname('index.html')
|
|||
|
* // returns
|
|||
|
* '.html'
|
|||
|
*
|
|||
|
* path.extname('index.')
|
|||
|
* // returns
|
|||
|
* '.'
|
|||
|
*
|
|||
|
* path.extname('index')
|
|||
|
* // returns
|
|||
|
* ''
|
|||
|
* @param [String] p
|
|||
|
* @return [String]
|
|||
|
*/
|
|||
|
path.extname = function (p) {
|
|||
|
p = path.normalize(p);
|
|||
|
var sections = p.split(path.sep);
|
|||
|
p = sections.pop();
|
|||
|
|
|||
|
// Special case: foo/file.ext/ should return '.ext'
|
|||
|
if (p === '' && sections.length > 0) {
|
|||
|
p = sections.pop();
|
|||
|
}
|
|||
|
if (p === '..') {
|
|||
|
return '';
|
|||
|
}
|
|||
|
var i = p.lastIndexOf('.');
|
|||
|
if (i === -1 || i === 0) {
|
|||
|
return '';
|
|||
|
}
|
|||
|
return p.substr(i);
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Checks if the given path is an absolute path.
|
|||
|
*
|
|||
|
* Despite not being documented, this is a tested part of Node's path API.
|
|||
|
* @param [String] p
|
|||
|
* @return [Boolean] True if the path appears to be an absolute path.
|
|||
|
*/
|
|||
|
path.isAbsolute = function (p) {
|
|||
|
return p.length > 0 && p.charAt(0) === path.sep;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Unknown. Undocumented.
|
|||
|
*/
|
|||
|
path._makeLong = function (p) {
|
|||
|
return p;
|
|||
|
};
|
|||
|
|
|||
|
path._removeDuplicateSeps = function (p) {
|
|||
|
p = p.replace(this._replaceRegex, this.sep);
|
|||
|
return p;
|
|||
|
};
|
|||
|
path.sep = '/';
|
|||
|
|
|||
|
path._replaceRegex = new RegExp("//+", 'g');
|
|||
|
|
|||
|
path.delimiter = ':';
|
|||
|
return path;
|
|||
|
})();
|
|||
|
exports.path = path;
|
|||
|
});
|
|||
|
//# sourceMappingURL=node_path.js.map
|
|||
|
;
|
|||
|
define('core/node_fs',["require", "exports", './api_error', './file_flag', './buffer', './node_path'], function(require, exports, api_error, file_flag, buffer, node_path) {
|
|||
|
var ApiError = api_error.ApiError;
|
|||
|
var ErrorCode = api_error.ErrorCode;
|
|||
|
var FileFlag = file_flag.FileFlag;
|
|||
|
var Buffer = buffer.Buffer;
|
|||
|
var path = node_path.path;
|
|||
|
|
|||
|
/**
|
|||
|
* Wraps a callback with a setImmediate call.
|
|||
|
* @param [Function] cb The callback to wrap.
|
|||
|
* @param [Number] numArgs The number of arguments that the callback takes.
|
|||
|
* @return [Function] The wrapped callback.
|
|||
|
*/
|
|||
|
function wrapCb(cb, numArgs) {
|
|||
|
if (typeof cb !== 'function') {
|
|||
|
throw new ApiError(9 /* EINVAL */, 'Callback must be a function.');
|
|||
|
}
|
|||
|
|
|||
|
// @todo This is used for unit testing. Maybe we should inject this logic
|
|||
|
// dynamically rather than bundle it in 'production' code.
|
|||
|
if (typeof __numWaiting === 'undefined') {
|
|||
|
__numWaiting = 0;
|
|||
|
}
|
|||
|
__numWaiting++;
|
|||
|
|
|||
|
switch (numArgs) {
|
|||
|
case 1:
|
|||
|
return function (arg1) {
|
|||
|
setImmediate(function () {
|
|||
|
__numWaiting--;
|
|||
|
return cb(arg1);
|
|||
|
});
|
|||
|
};
|
|||
|
case 2:
|
|||
|
return function (arg1, arg2) {
|
|||
|
setImmediate(function () {
|
|||
|
__numWaiting--;
|
|||
|
return cb(arg1, arg2);
|
|||
|
});
|
|||
|
};
|
|||
|
case 3:
|
|||
|
return function (arg1, arg2, arg3) {
|
|||
|
setImmediate(function () {
|
|||
|
__numWaiting--;
|
|||
|
return cb(arg1, arg2, arg3);
|
|||
|
});
|
|||
|
};
|
|||
|
default:
|
|||
|
throw new Error('Invalid invocation of wrapCb.');
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
/**
|
|||
|
* Checks if the fd is valid.
|
|||
|
* @param [BrowserFS.File] fd A file descriptor (in BrowserFS, it's a File object)
|
|||
|
* @return [Boolean, BrowserFS.ApiError] Returns `true` if the FD is OK,
|
|||
|
* otherwise returns an ApiError.
|
|||
|
*/
|
|||
|
function checkFd(fd) {
|
|||
|
if (typeof fd['write'] !== 'function') {
|
|||
|
throw new ApiError(3 /* EBADF */, 'Invalid file descriptor.');
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
function normalizeMode(mode, def) {
|
|||
|
switch (typeof mode) {
|
|||
|
case 'number':
|
|||
|
// (path, flag, mode, cb?)
|
|||
|
return mode;
|
|||
|
case 'string':
|
|||
|
// (path, flag, modeString, cb?)
|
|||
|
var trueMode = parseInt(mode, 8);
|
|||
|
if (trueMode !== NaN) {
|
|||
|
return trueMode;
|
|||
|
}
|
|||
|
|
|||
|
default:
|
|||
|
return def;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
function normalizePath(p) {
|
|||
|
// Node doesn't allow null characters in paths.
|
|||
|
if (p.indexOf('\u0000') >= 0) {
|
|||
|
throw new ApiError(9 /* EINVAL */, 'Path must be a string without null bytes.');
|
|||
|
} else if (p === '') {
|
|||
|
throw new ApiError(9 /* EINVAL */, 'Path must not be empty.');
|
|||
|
}
|
|||
|
return path.resolve(p);
|
|||
|
}
|
|||
|
|
|||
|
function normalizeOptions(options, defEnc, defFlag, defMode) {
|
|||
|
switch (typeof options) {
|
|||
|
case 'object':
|
|||
|
return {
|
|||
|
encoding: typeof options['encoding'] !== 'undefined' ? options['encoding'] : defEnc,
|
|||
|
flag: typeof options['flag'] !== 'undefined' ? options['flag'] : defFlag,
|
|||
|
mode: normalizeMode(options['mode'], defMode)
|
|||
|
};
|
|||
|
case 'string':
|
|||
|
return {
|
|||
|
encoding: options,
|
|||
|
flag: defFlag,
|
|||
|
mode: defMode
|
|||
|
};
|
|||
|
default:
|
|||
|
return {
|
|||
|
encoding: defEnc,
|
|||
|
flag: defFlag,
|
|||
|
mode: defMode
|
|||
|
};
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
// The default callback is a NOP.
|
|||
|
function nopCb() {
|
|||
|
}
|
|||
|
;
|
|||
|
|
|||
|
/**
|
|||
|
* The node frontend to all filesystems.
|
|||
|
* This layer handles:
|
|||
|
*
|
|||
|
* * Sanity checking inputs.
|
|||
|
* * Normalizing paths.
|
|||
|
* * Resetting stack depth for asynchronous operations which may not go through
|
|||
|
* the browser by wrapping all input callbacks using `setImmediate`.
|
|||
|
* * Performing the requested operation through the filesystem or the file
|
|||
|
* descriptor, as appropriate.
|
|||
|
* * Handling optional arguments and setting default arguments.
|
|||
|
* @see http://nodejs.org/api/fs.html
|
|||
|
* @class
|
|||
|
*/
|
|||
|
var fs = (function () {
|
|||
|
function fs() {
|
|||
|
}
|
|||
|
fs._initialize = function (rootFS) {
|
|||
|
if (!rootFS.constructor.isAvailable()) {
|
|||
|
throw new ApiError(9 /* EINVAL */, 'Tried to instantiate BrowserFS with an unavailable file system.');
|
|||
|
}
|
|||
|
return fs.root = rootFS;
|
|||
|
};
|
|||
|
|
|||
|
fs._toUnixTimestamp = function (time) {
|
|||
|
if (typeof time === 'number') {
|
|||
|
return time;
|
|||
|
} else if (time instanceof Date) {
|
|||
|
return time.getTime() / 1000;
|
|||
|
}
|
|||
|
throw new Error("Cannot parse time: " + time);
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* **NONSTANDARD**: Grab the FileSystem instance that backs this API.
|
|||
|
* @return [BrowserFS.FileSystem | null] Returns null if the file system has
|
|||
|
* not been initialized.
|
|||
|
*/
|
|||
|
fs.getRootFS = function () {
|
|||
|
if (fs.root) {
|
|||
|
return fs.root;
|
|||
|
} else {
|
|||
|
return null;
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
// FILE OR DIRECTORY METHODS
|
|||
|
/**
|
|||
|
* Asynchronous rename. No arguments other than a possible exception are given
|
|||
|
* to the completion callback.
|
|||
|
* @param [String] oldPath
|
|||
|
* @param [String] newPath
|
|||
|
* @param [Function(BrowserFS.ApiError)] callback
|
|||
|
*/
|
|||
|
fs.rename = function (oldPath, newPath, cb) {
|
|||
|
if (typeof cb === "undefined") { cb = nopCb; }
|
|||
|
var newCb = wrapCb(cb, 1);
|
|||
|
try {
|
|||
|
fs.root.rename(normalizePath(oldPath), normalizePath(newPath), newCb);
|
|||
|
} catch (e) {
|
|||
|
newCb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Synchronous rename.
|
|||
|
* @param [String] oldPath
|
|||
|
* @param [String] newPath
|
|||
|
*/
|
|||
|
fs.renameSync = function (oldPath, newPath) {
|
|||
|
fs.root.renameSync(normalizePath(oldPath), normalizePath(newPath));
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Test whether or not the given path exists by checking with the file system.
|
|||
|
* Then call the callback argument with either true or false.
|
|||
|
* @example Sample invocation
|
|||
|
* fs.exists('/etc/passwd', function (exists) {
|
|||
|
* util.debug(exists ? "it's there" : "no passwd!");
|
|||
|
* });
|
|||
|
* @param [String] path
|
|||
|
* @param [Function(Boolean)] callback
|
|||
|
*/
|
|||
|
fs.exists = function (path, cb) {
|
|||
|
if (typeof cb === "undefined") { cb = nopCb; }
|
|||
|
var newCb = wrapCb(cb, 1);
|
|||
|
try {
|
|||
|
return fs.root.exists(normalizePath(path), newCb);
|
|||
|
} catch (e) {
|
|||
|
// Doesn't return an error. If something bad happens, we assume it just
|
|||
|
// doesn't exist.
|
|||
|
return newCb(false);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Test whether or not the given path exists by checking with the file system.
|
|||
|
* @param [String] path
|
|||
|
* @return [boolean]
|
|||
|
*/
|
|||
|
fs.existsSync = function (path) {
|
|||
|
try {
|
|||
|
return fs.root.existsSync(normalizePath(path));
|
|||
|
} catch (e) {
|
|||
|
// Doesn't return an error. If something bad happens, we assume it just
|
|||
|
// doesn't exist.
|
|||
|
return false;
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Asynchronous `stat`.
|
|||
|
* @param [String] path
|
|||
|
* @param [Function(BrowserFS.ApiError, BrowserFS.node.fs.Stats)] callback
|
|||
|
*/
|
|||
|
fs.stat = function (path, cb) {
|
|||
|
if (typeof cb === "undefined") { cb = nopCb; }
|
|||
|
var newCb = wrapCb(cb, 2);
|
|||
|
try {
|
|||
|
return fs.root.stat(normalizePath(path), false, newCb);
|
|||
|
} catch (e) {
|
|||
|
return newCb(e, null);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Synchronous `stat`.
|
|||
|
* @param [String] path
|
|||
|
* @return [BrowserFS.node.fs.Stats]
|
|||
|
*/
|
|||
|
fs.statSync = function (path) {
|
|||
|
return fs.root.statSync(normalizePath(path), false);
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Asynchronous `lstat`.
|
|||
|
* `lstat()` is identical to `stat()`, except that if path is a symbolic link,
|
|||
|
* then the link itself is stat-ed, not the file that it refers to.
|
|||
|
* @param [String] path
|
|||
|
* @param [Function(BrowserFS.ApiError, BrowserFS.node.fs.Stats)] callback
|
|||
|
*/
|
|||
|
fs.lstat = function (path, cb) {
|
|||
|
if (typeof cb === "undefined") { cb = nopCb; }
|
|||
|
var newCb = wrapCb(cb, 2);
|
|||
|
try {
|
|||
|
return fs.root.stat(normalizePath(path), true, newCb);
|
|||
|
} catch (e) {
|
|||
|
return newCb(e, null);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Synchronous `lstat`.
|
|||
|
* `lstat()` is identical to `stat()`, except that if path is a symbolic link,
|
|||
|
* then the link itself is stat-ed, not the file that it refers to.
|
|||
|
* @param [String] path
|
|||
|
* @return [BrowserFS.node.fs.Stats]
|
|||
|
*/
|
|||
|
fs.lstatSync = function (path) {
|
|||
|
return fs.root.statSync(normalizePath(path), true);
|
|||
|
};
|
|||
|
|
|||
|
fs.truncate = function (path, arg2, cb) {
|
|||
|
if (typeof arg2 === "undefined") { arg2 = 0; }
|
|||
|
if (typeof cb === "undefined") { cb = nopCb; }
|
|||
|
var len = 0;
|
|||
|
if (typeof arg2 === 'function') {
|
|||
|
cb = arg2;
|
|||
|
} else if (typeof arg2 === 'number') {
|
|||
|
len = arg2;
|
|||
|
}
|
|||
|
|
|||
|
var newCb = wrapCb(cb, 1);
|
|||
|
try {
|
|||
|
if (len < 0) {
|
|||
|
throw new ApiError(9 /* EINVAL */);
|
|||
|
}
|
|||
|
return fs.root.truncate(normalizePath(path), len, newCb);
|
|||
|
} catch (e) {
|
|||
|
return newCb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Synchronous `truncate`.
|
|||
|
* @param [String] path
|
|||
|
* @param [Number] len
|
|||
|
*/
|
|||
|
fs.truncateSync = function (path, len) {
|
|||
|
if (typeof len === "undefined") { len = 0; }
|
|||
|
if (len < 0) {
|
|||
|
throw new ApiError(9 /* EINVAL */);
|
|||
|
}
|
|||
|
return fs.root.truncateSync(normalizePath(path), len);
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Asynchronous `unlink`.
|
|||
|
* @param [String] path
|
|||
|
* @param [Function(BrowserFS.ApiError)] callback
|
|||
|
*/
|
|||
|
fs.unlink = function (path, cb) {
|
|||
|
if (typeof cb === "undefined") { cb = nopCb; }
|
|||
|
var newCb = wrapCb(cb, 1);
|
|||
|
try {
|
|||
|
return fs.root.unlink(normalizePath(path), newCb);
|
|||
|
} catch (e) {
|
|||
|
return newCb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Synchronous `unlink`.
|
|||
|
* @param [String] path
|
|||
|
*/
|
|||
|
fs.unlinkSync = function (path) {
|
|||
|
return fs.root.unlinkSync(normalizePath(path));
|
|||
|
};
|
|||
|
|
|||
|
fs.open = function (path, flag, arg2, cb) {
|
|||
|
if (typeof cb === "undefined") { cb = nopCb; }
|
|||
|
var mode = normalizeMode(arg2, 0x1a4);
|
|||
|
cb = typeof arg2 === 'function' ? arg2 : cb;
|
|||
|
var newCb = wrapCb(cb, 2);
|
|||
|
try {
|
|||
|
return fs.root.open(normalizePath(path), FileFlag.getFileFlag(flag), mode, newCb);
|
|||
|
} catch (e) {
|
|||
|
return newCb(e, null);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
fs.openSync = function (path, flag, mode) {
|
|||
|
if (typeof mode === "undefined") { mode = 0x1a4; }
|
|||
|
return fs.root.openSync(normalizePath(path), FileFlag.getFileFlag(flag), mode);
|
|||
|
};
|
|||
|
|
|||
|
fs.readFile = function (filename, arg2, cb) {
|
|||
|
if (typeof arg2 === "undefined") { arg2 = {}; }
|
|||
|
if (typeof cb === "undefined") { cb = nopCb; }
|
|||
|
var options = normalizeOptions(arg2, null, 'r', null);
|
|||
|
cb = typeof arg2 === 'function' ? arg2 : cb;
|
|||
|
var newCb = wrapCb(cb, 2);
|
|||
|
try {
|
|||
|
var flag = FileFlag.getFileFlag(options['flag']);
|
|||
|
if (!flag.isReadable()) {
|
|||
|
return newCb(new ApiError(9 /* EINVAL */, 'Flag passed to readFile must allow for reading.'));
|
|||
|
}
|
|||
|
return fs.root.readFile(normalizePath(filename), options.encoding, flag, newCb);
|
|||
|
} catch (e) {
|
|||
|
return newCb(e, null);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
fs.readFileSync = function (filename, arg2) {
|
|||
|
if (typeof arg2 === "undefined") { arg2 = {}; }
|
|||
|
var options = normalizeOptions(arg2, null, 'r', null);
|
|||
|
var flag = FileFlag.getFileFlag(options.flag);
|
|||
|
if (!flag.isReadable()) {
|
|||
|
throw new ApiError(9 /* EINVAL */, 'Flag passed to readFile must allow for reading.');
|
|||
|
}
|
|||
|
return fs.root.readFileSync(normalizePath(filename), options.encoding, flag);
|
|||
|
};
|
|||
|
|
|||
|
fs.writeFile = function (filename, data, arg3, cb) {
|
|||
|
if (typeof arg3 === "undefined") { arg3 = {}; }
|
|||
|
if (typeof cb === "undefined") { cb = nopCb; }
|
|||
|
var options = normalizeOptions(arg3, 'utf8', 'w', 0x1a4);
|
|||
|
cb = typeof arg3 === 'function' ? arg3 : cb;
|
|||
|
var newCb = wrapCb(cb, 1);
|
|||
|
try {
|
|||
|
var flag = FileFlag.getFileFlag(options.flag);
|
|||
|
if (!flag.isWriteable()) {
|
|||
|
return newCb(new ApiError(9 /* EINVAL */, 'Flag passed to writeFile must allow for writing.'));
|
|||
|
}
|
|||
|
return fs.root.writeFile(normalizePath(filename), data, options.encoding, flag, options.mode, newCb);
|
|||
|
} catch (e) {
|
|||
|
return newCb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
fs.writeFileSync = function (filename, data, arg3) {
|
|||
|
var options = normalizeOptions(arg3, 'utf8', 'w', 0x1a4);
|
|||
|
var flag = FileFlag.getFileFlag(options.flag);
|
|||
|
if (!flag.isWriteable()) {
|
|||
|
throw new ApiError(9 /* EINVAL */, 'Flag passed to writeFile must allow for writing.');
|
|||
|
}
|
|||
|
return fs.root.writeFileSync(normalizePath(filename), data, options.encoding, flag, options.mode);
|
|||
|
};
|
|||
|
|
|||
|
fs.appendFile = function (filename, data, arg3, cb) {
|
|||
|
if (typeof cb === "undefined") { cb = nopCb; }
|
|||
|
var options = normalizeOptions(arg3, 'utf8', 'a', 0x1a4);
|
|||
|
cb = typeof arg3 === 'function' ? arg3 : cb;
|
|||
|
var newCb = wrapCb(cb, 1);
|
|||
|
try {
|
|||
|
var flag = FileFlag.getFileFlag(options.flag);
|
|||
|
if (!flag.isAppendable()) {
|
|||
|
return newCb(new ApiError(9 /* EINVAL */, 'Flag passed to appendFile must allow for appending.'));
|
|||
|
}
|
|||
|
fs.root.appendFile(normalizePath(filename), data, options.encoding, flag, options.mode, newCb);
|
|||
|
} catch (e) {
|
|||
|
newCb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
fs.appendFileSync = function (filename, data, arg3) {
|
|||
|
var options = normalizeOptions(arg3, 'utf8', 'a', 0x1a4);
|
|||
|
var flag = FileFlag.getFileFlag(options.flag);
|
|||
|
if (!flag.isAppendable()) {
|
|||
|
throw new ApiError(9 /* EINVAL */, 'Flag passed to appendFile must allow for appending.');
|
|||
|
}
|
|||
|
return fs.root.appendFileSync(normalizePath(filename), data, options.encoding, flag, options.mode);
|
|||
|
};
|
|||
|
|
|||
|
// FILE DESCRIPTOR METHODS
|
|||
|
/**
|
|||
|
* Asynchronous `fstat`.
|
|||
|
* `fstat()` is identical to `stat()`, except that the file to be stat-ed is
|
|||
|
* specified by the file descriptor `fd`.
|
|||
|
* @param [BrowserFS.File] fd
|
|||
|
* @param [Function(BrowserFS.ApiError, BrowserFS.node.fs.Stats)] callback
|
|||
|
*/
|
|||
|
fs.fstat = function (fd, cb) {
|
|||
|
if (typeof cb === "undefined") { cb = nopCb; }
|
|||
|
var newCb = wrapCb(cb, 2);
|
|||
|
try {
|
|||
|
checkFd(fd);
|
|||
|
fd.stat(newCb);
|
|||
|
} catch (e) {
|
|||
|
newCb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Synchronous `fstat`.
|
|||
|
* `fstat()` is identical to `stat()`, except that the file to be stat-ed is
|
|||
|
* specified by the file descriptor `fd`.
|
|||
|
* @param [BrowserFS.File] fd
|
|||
|
* @return [BrowserFS.node.fs.Stats]
|
|||
|
*/
|
|||
|
fs.fstatSync = function (fd) {
|
|||
|
checkFd(fd);
|
|||
|
return fd.statSync();
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Asynchronous close.
|
|||
|
* @param [BrowserFS.File] fd
|
|||
|
* @param [Function(BrowserFS.ApiError)] callback
|
|||
|
*/
|
|||
|
fs.close = function (fd, cb) {
|
|||
|
if (typeof cb === "undefined") { cb = nopCb; }
|
|||
|
var newCb = wrapCb(cb, 1);
|
|||
|
try {
|
|||
|
checkFd(fd);
|
|||
|
fd.close(newCb);
|
|||
|
} catch (e) {
|
|||
|
newCb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Synchronous close.
|
|||
|
* @param [BrowserFS.File] fd
|
|||
|
*/
|
|||
|
fs.closeSync = function (fd) {
|
|||
|
checkFd(fd);
|
|||
|
return fd.closeSync();
|
|||
|
};
|
|||
|
|
|||
|
fs.ftruncate = function (fd, arg2, cb) {
|
|||
|
if (typeof cb === "undefined") { cb = nopCb; }
|
|||
|
var length = typeof arg2 === 'number' ? arg2 : 0;
|
|||
|
cb = typeof arg2 === 'function' ? arg2 : cb;
|
|||
|
var newCb = wrapCb(cb, 1);
|
|||
|
try {
|
|||
|
checkFd(fd);
|
|||
|
if (length < 0) {
|
|||
|
throw new ApiError(9 /* EINVAL */);
|
|||
|
}
|
|||
|
fd.truncate(length, newCb);
|
|||
|
} catch (e) {
|
|||
|
newCb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Synchronous ftruncate.
|
|||
|
* @param [BrowserFS.File] fd
|
|||
|
* @param [Number] len
|
|||
|
*/
|
|||
|
fs.ftruncateSync = function (fd, len) {
|
|||
|
if (typeof len === "undefined") { len = 0; }
|
|||
|
checkFd(fd);
|
|||
|
return fd.truncateSync(len);
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Asynchronous fsync.
|
|||
|
* @param [BrowserFS.File] fd
|
|||
|
* @param [Function(BrowserFS.ApiError)] callback
|
|||
|
*/
|
|||
|
fs.fsync = function (fd, cb) {
|
|||
|
if (typeof cb === "undefined") { cb = nopCb; }
|
|||
|
var newCb = wrapCb(cb, 1);
|
|||
|
try {
|
|||
|
checkFd(fd);
|
|||
|
fd.sync(newCb);
|
|||
|
} catch (e) {
|
|||
|
newCb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Synchronous fsync.
|
|||
|
* @param [BrowserFS.File] fd
|
|||
|
*/
|
|||
|
fs.fsyncSync = function (fd) {
|
|||
|
checkFd(fd);
|
|||
|
return fd.syncSync();
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Asynchronous fdatasync.
|
|||
|
* @param [BrowserFS.File] fd
|
|||
|
* @param [Function(BrowserFS.ApiError)] callback
|
|||
|
*/
|
|||
|
fs.fdatasync = function (fd, cb) {
|
|||
|
if (typeof cb === "undefined") { cb = nopCb; }
|
|||
|
var newCb = wrapCb(cb, 1);
|
|||
|
try {
|
|||
|
checkFd(fd);
|
|||
|
fd.datasync(newCb);
|
|||
|
} catch (e) {
|
|||
|
newCb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Synchronous fdatasync.
|
|||
|
* @param [BrowserFS.File] fd
|
|||
|
*/
|
|||
|
fs.fdatasyncSync = function (fd) {
|
|||
|
checkFd(fd);
|
|||
|
fd.datasyncSync();
|
|||
|
};
|
|||
|
|
|||
|
fs.write = function (fd, arg2, arg3, arg4, arg5, cb) {
|
|||
|
if (typeof cb === "undefined") { cb = nopCb; }
|
|||
|
var buffer, offset, length, position = null;
|
|||
|
if (typeof arg2 === 'string') {
|
|||
|
// Signature 1: (fd, string, [position?, [encoding?]], cb?)
|
|||
|
var encoding = 'utf8';
|
|||
|
switch (typeof arg3) {
|
|||
|
case 'function':
|
|||
|
// (fd, string, cb)
|
|||
|
cb = arg3;
|
|||
|
break;
|
|||
|
case 'number':
|
|||
|
// (fd, string, position, encoding?, cb?)
|
|||
|
position = arg3;
|
|||
|
encoding = typeof arg4 === 'string' ? arg4 : 'utf8';
|
|||
|
cb = typeof arg5 === 'function' ? arg5 : cb;
|
|||
|
break;
|
|||
|
default:
|
|||
|
// ...try to find the callback and get out of here!
|
|||
|
cb = typeof arg4 === 'function' ? arg4 : typeof arg5 === 'function' ? arg5 : cb;
|
|||
|
return cb(new ApiError(9 /* EINVAL */, 'Invalid arguments.'));
|
|||
|
}
|
|||
|
buffer = new Buffer(arg2, encoding);
|
|||
|
offset = 0;
|
|||
|
length = buffer.length;
|
|||
|
} else {
|
|||
|
// Signature 2: (fd, buffer, offset, length, position?, cb?)
|
|||
|
buffer = arg2;
|
|||
|
offset = arg3;
|
|||
|
length = arg4;
|
|||
|
position = typeof arg5 === 'number' ? arg5 : null;
|
|||
|
cb = typeof arg5 === 'function' ? arg5 : cb;
|
|||
|
}
|
|||
|
|
|||
|
var newCb = wrapCb(cb, 3);
|
|||
|
try {
|
|||
|
checkFd(fd);
|
|||
|
if (position == null) {
|
|||
|
position = fd.getPos();
|
|||
|
}
|
|||
|
fd.write(buffer, offset, length, position, newCb);
|
|||
|
} catch (e) {
|
|||
|
newCb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
fs.writeSync = function (fd, arg2, arg3, arg4, arg5) {
|
|||
|
var buffer, offset = 0, length, position;
|
|||
|
if (typeof arg2 === 'string') {
|
|||
|
// Signature 1: (fd, string, [position?, [encoding?]])
|
|||
|
position = typeof arg3 === 'number' ? arg3 : null;
|
|||
|
var encoding = typeof arg4 === 'string' ? arg4 : 'utf8';
|
|||
|
offset = 0;
|
|||
|
buffer = new Buffer(arg2, encoding);
|
|||
|
length = buffer.length;
|
|||
|
} else {
|
|||
|
// Signature 2: (fd, buffer, offset, length, position?)
|
|||
|
buffer = arg2;
|
|||
|
offset = arg3;
|
|||
|
length = arg4;
|
|||
|
position = typeof arg5 === 'number' ? arg5 : null;
|
|||
|
}
|
|||
|
|
|||
|
checkFd(fd);
|
|||
|
if (position == null) {
|
|||
|
position = fd.getPos();
|
|||
|
}
|
|||
|
return fd.writeSync(buffer, offset, length, position);
|
|||
|
};
|
|||
|
|
|||
|
fs.read = function (fd, arg2, arg3, arg4, arg5, cb) {
|
|||
|
if (typeof cb === "undefined") { cb = nopCb; }
|
|||
|
var position, offset, length, buffer, newCb;
|
|||
|
if (typeof arg2 === 'number') {
|
|||
|
// legacy interface
|
|||
|
// (fd, length, position, encoding, callback)
|
|||
|
length = arg2;
|
|||
|
position = arg3;
|
|||
|
var encoding = arg4;
|
|||
|
cb = typeof arg5 === 'function' ? arg5 : cb;
|
|||
|
offset = 0;
|
|||
|
buffer = new Buffer(length);
|
|||
|
|
|||
|
// XXX: Inefficient.
|
|||
|
// Wrap the cb so we shelter upper layers of the API from these
|
|||
|
// shenanigans.
|
|||
|
newCb = wrapCb((function (err, bytesRead, buf) {
|
|||
|
if (err) {
|
|||
|
return cb(err);
|
|||
|
}
|
|||
|
cb(err, buf.toString(encoding), bytesRead);
|
|||
|
}), 3);
|
|||
|
} else {
|
|||
|
buffer = arg2;
|
|||
|
offset = arg3;
|
|||
|
length = arg4;
|
|||
|
position = arg5;
|
|||
|
newCb = wrapCb(cb, 3);
|
|||
|
}
|
|||
|
|
|||
|
try {
|
|||
|
checkFd(fd);
|
|||
|
if (position == null) {
|
|||
|
position = fd.getPos();
|
|||
|
}
|
|||
|
fd.read(buffer, offset, length, position, newCb);
|
|||
|
} catch (e) {
|
|||
|
newCb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
fs.readSync = function (fd, arg2, arg3, arg4, arg5) {
|
|||
|
var shenanigans = false;
|
|||
|
var buffer, offset, length, position;
|
|||
|
if (typeof arg2 === 'number') {
|
|||
|
length = arg2;
|
|||
|
position = arg3;
|
|||
|
var encoding = arg4;
|
|||
|
offset = 0;
|
|||
|
buffer = new Buffer(length);
|
|||
|
shenanigans = true;
|
|||
|
} else {
|
|||
|
buffer = arg2;
|
|||
|
offset = arg3;
|
|||
|
length = arg4;
|
|||
|
position = arg5;
|
|||
|
}
|
|||
|
checkFd(fd);
|
|||
|
if (position == null) {
|
|||
|
position = fd.getPos();
|
|||
|
}
|
|||
|
|
|||
|
var rv = fd.readSync(buffer, offset, length, position);
|
|||
|
if (!shenanigans) {
|
|||
|
return rv;
|
|||
|
} else {
|
|||
|
return [buffer.toString(encoding), rv];
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Asynchronous `fchown`.
|
|||
|
* @param [BrowserFS.File] fd
|
|||
|
* @param [Number] uid
|
|||
|
* @param [Number] gid
|
|||
|
* @param [Function(BrowserFS.ApiError)] callback
|
|||
|
*/
|
|||
|
fs.fchown = function (fd, uid, gid, callback) {
|
|||
|
if (typeof callback === "undefined") { callback = nopCb; }
|
|||
|
var newCb = wrapCb(callback, 1);
|
|||
|
try {
|
|||
|
checkFd(fd);
|
|||
|
fd.chown(uid, gid, newCb);
|
|||
|
} catch (e) {
|
|||
|
newCb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Synchronous `fchown`.
|
|||
|
* @param [BrowserFS.File] fd
|
|||
|
* @param [Number] uid
|
|||
|
* @param [Number] gid
|
|||
|
*/
|
|||
|
fs.fchownSync = function (fd, uid, gid) {
|
|||
|
checkFd(fd);
|
|||
|
return fd.chownSync(uid, gid);
|
|||
|
};
|
|||
|
|
|||
|
fs.fchmod = function (fd, mode, cb) {
|
|||
|
if (typeof cb === "undefined") { cb = nopCb; }
|
|||
|
var newCb = wrapCb(cb, 1);
|
|||
|
try {
|
|||
|
mode = typeof mode === 'string' ? parseInt(mode, 8) : mode;
|
|||
|
checkFd(fd);
|
|||
|
fd.chmod(mode, newCb);
|
|||
|
} catch (e) {
|
|||
|
newCb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
fs.fchmodSync = function (fd, mode) {
|
|||
|
mode = typeof mode === 'string' ? parseInt(mode, 8) : mode;
|
|||
|
checkFd(fd);
|
|||
|
return fd.chmodSync(mode);
|
|||
|
};
|
|||
|
|
|||
|
fs.futimes = function (fd, atime, mtime, cb) {
|
|||
|
if (typeof cb === "undefined") { cb = nopCb; }
|
|||
|
var newCb = wrapCb(cb, 1);
|
|||
|
try {
|
|||
|
checkFd(fd);
|
|||
|
if (typeof atime === 'number') {
|
|||
|
atime = new Date(atime * 1000);
|
|||
|
}
|
|||
|
if (typeof mtime === 'number') {
|
|||
|
mtime = new Date(mtime * 1000);
|
|||
|
}
|
|||
|
fd.utimes(atime, mtime, newCb);
|
|||
|
} catch (e) {
|
|||
|
newCb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
fs.futimesSync = function (fd, atime, mtime) {
|
|||
|
checkFd(fd);
|
|||
|
if (typeof atime === 'number') {
|
|||
|
atime = new Date(atime * 1000);
|
|||
|
}
|
|||
|
if (typeof mtime === 'number') {
|
|||
|
mtime = new Date(mtime * 1000);
|
|||
|
}
|
|||
|
return fd.utimesSync(atime, mtime);
|
|||
|
};
|
|||
|
|
|||
|
// DIRECTORY-ONLY METHODS
|
|||
|
/**
|
|||
|
* Asynchronous `rmdir`.
|
|||
|
* @param [String] path
|
|||
|
* @param [Function(BrowserFS.ApiError)] callback
|
|||
|
*/
|
|||
|
fs.rmdir = function (path, cb) {
|
|||
|
if (typeof cb === "undefined") { cb = nopCb; }
|
|||
|
var newCb = wrapCb(cb, 1);
|
|||
|
try {
|
|||
|
path = normalizePath(path);
|
|||
|
fs.root.rmdir(path, newCb);
|
|||
|
} catch (e) {
|
|||
|
newCb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Synchronous `rmdir`.
|
|||
|
* @param [String] path
|
|||
|
*/
|
|||
|
fs.rmdirSync = function (path) {
|
|||
|
path = normalizePath(path);
|
|||
|
return fs.root.rmdirSync(path);
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Asynchronous `mkdir`.
|
|||
|
* @param [String] path
|
|||
|
* @param [Number?] mode defaults to `0777`
|
|||
|
* @param [Function(BrowserFS.ApiError)] callback
|
|||
|
*/
|
|||
|
fs.mkdir = function (path, mode, cb) {
|
|||
|
if (typeof cb === "undefined") { cb = nopCb; }
|
|||
|
if (typeof mode === 'function') {
|
|||
|
cb = mode;
|
|||
|
mode = 0x1ff;
|
|||
|
}
|
|||
|
var newCb = wrapCb(cb, 1);
|
|||
|
try {
|
|||
|
path = normalizePath(path);
|
|||
|
fs.root.mkdir(path, mode, newCb);
|
|||
|
} catch (e) {
|
|||
|
newCb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
fs.mkdirSync = function (path, mode) {
|
|||
|
if (typeof mode === "undefined") { mode = 0x1ff; }
|
|||
|
mode = typeof mode === 'string' ? parseInt(mode, 8) : mode;
|
|||
|
path = normalizePath(path);
|
|||
|
return fs.root.mkdirSync(path, mode);
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Asynchronous `readdir`. Reads the contents of a directory.
|
|||
|
* The callback gets two arguments `(err, files)` where `files` is an array of
|
|||
|
* the names of the files in the directory excluding `'.'` and `'..'`.
|
|||
|
* @param [String] path
|
|||
|
* @param [Function(BrowserFS.ApiError, String[])] callback
|
|||
|
*/
|
|||
|
fs.readdir = function (path, cb) {
|
|||
|
if (typeof cb === "undefined") { cb = nopCb; }
|
|||
|
var newCb = wrapCb(cb, 2);
|
|||
|
try {
|
|||
|
path = normalizePath(path);
|
|||
|
fs.root.readdir(path, newCb);
|
|||
|
} catch (e) {
|
|||
|
newCb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Synchronous `readdir`. Reads the contents of a directory.
|
|||
|
* @param [String] path
|
|||
|
* @return [String[]]
|
|||
|
*/
|
|||
|
fs.readdirSync = function (path) {
|
|||
|
path = normalizePath(path);
|
|||
|
return fs.root.readdirSync(path);
|
|||
|
};
|
|||
|
|
|||
|
// SYMLINK METHODS
|
|||
|
/**
|
|||
|
* Asynchronous `link`.
|
|||
|
* @param [String] srcpath
|
|||
|
* @param [String] dstpath
|
|||
|
* @param [Function(BrowserFS.ApiError)] callback
|
|||
|
*/
|
|||
|
fs.link = function (srcpath, dstpath, cb) {
|
|||
|
if (typeof cb === "undefined") { cb = nopCb; }
|
|||
|
var newCb = wrapCb(cb, 1);
|
|||
|
try {
|
|||
|
srcpath = normalizePath(srcpath);
|
|||
|
dstpath = normalizePath(dstpath);
|
|||
|
fs.root.link(srcpath, dstpath, newCb);
|
|||
|
} catch (e) {
|
|||
|
newCb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Synchronous `link`.
|
|||
|
* @param [String] srcpath
|
|||
|
* @param [String] dstpath
|
|||
|
*/
|
|||
|
fs.linkSync = function (srcpath, dstpath) {
|
|||
|
srcpath = normalizePath(srcpath);
|
|||
|
dstpath = normalizePath(dstpath);
|
|||
|
return fs.root.linkSync(srcpath, dstpath);
|
|||
|
};
|
|||
|
|
|||
|
fs.symlink = function (srcpath, dstpath, arg3, cb) {
|
|||
|
if (typeof cb === "undefined") { cb = nopCb; }
|
|||
|
var type = typeof arg3 === 'string' ? arg3 : 'file';
|
|||
|
cb = typeof arg3 === 'function' ? arg3 : cb;
|
|||
|
var newCb = wrapCb(cb, 1);
|
|||
|
try {
|
|||
|
if (type !== 'file' && type !== 'dir') {
|
|||
|
return newCb(new ApiError(9 /* EINVAL */, "Invalid type: " + type));
|
|||
|
}
|
|||
|
srcpath = normalizePath(srcpath);
|
|||
|
dstpath = normalizePath(dstpath);
|
|||
|
fs.root.symlink(srcpath, dstpath, type, newCb);
|
|||
|
} catch (e) {
|
|||
|
newCb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Synchronous `symlink`.
|
|||
|
* @param [String] srcpath
|
|||
|
* @param [String] dstpath
|
|||
|
* @param [String?] type can be either `'dir'` or `'file'` (default is `'file'`)
|
|||
|
*/
|
|||
|
fs.symlinkSync = function (srcpath, dstpath, type) {
|
|||
|
if (type == null) {
|
|||
|
type = 'file';
|
|||
|
} else if (type !== 'file' && type !== 'dir') {
|
|||
|
throw new ApiError(9 /* EINVAL */, "Invalid type: " + type);
|
|||
|
}
|
|||
|
srcpath = normalizePath(srcpath);
|
|||
|
dstpath = normalizePath(dstpath);
|
|||
|
return fs.root.symlinkSync(srcpath, dstpath, type);
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Asynchronous readlink.
|
|||
|
* @param [String] path
|
|||
|
* @param [Function(BrowserFS.ApiError, String)] callback
|
|||
|
*/
|
|||
|
fs.readlink = function (path, cb) {
|
|||
|
if (typeof cb === "undefined") { cb = nopCb; }
|
|||
|
var newCb = wrapCb(cb, 2);
|
|||
|
try {
|
|||
|
path = normalizePath(path);
|
|||
|
fs.root.readlink(path, newCb);
|
|||
|
} catch (e) {
|
|||
|
newCb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Synchronous readlink.
|
|||
|
* @param [String] path
|
|||
|
* @return [String]
|
|||
|
*/
|
|||
|
fs.readlinkSync = function (path) {
|
|||
|
path = normalizePath(path);
|
|||
|
return fs.root.readlinkSync(path);
|
|||
|
};
|
|||
|
|
|||
|
// PROPERTY OPERATIONS
|
|||
|
/**
|
|||
|
* Asynchronous `chown`.
|
|||
|
* @param [String] path
|
|||
|
* @param [Number] uid
|
|||
|
* @param [Number] gid
|
|||
|
* @param [Function(BrowserFS.ApiError)] callback
|
|||
|
*/
|
|||
|
fs.chown = function (path, uid, gid, cb) {
|
|||
|
if (typeof cb === "undefined") { cb = nopCb; }
|
|||
|
var newCb = wrapCb(cb, 1);
|
|||
|
try {
|
|||
|
path = normalizePath(path);
|
|||
|
fs.root.chown(path, false, uid, gid, newCb);
|
|||
|
} catch (e) {
|
|||
|
newCb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Synchronous `chown`.
|
|||
|
* @param [String] path
|
|||
|
* @param [Number] uid
|
|||
|
* @param [Number] gid
|
|||
|
*/
|
|||
|
fs.chownSync = function (path, uid, gid) {
|
|||
|
path = normalizePath(path);
|
|||
|
fs.root.chownSync(path, false, uid, gid);
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Asynchronous `lchown`.
|
|||
|
* @param [String] path
|
|||
|
* @param [Number] uid
|
|||
|
* @param [Number] gid
|
|||
|
* @param [Function(BrowserFS.ApiError)] callback
|
|||
|
*/
|
|||
|
fs.lchown = function (path, uid, gid, cb) {
|
|||
|
if (typeof cb === "undefined") { cb = nopCb; }
|
|||
|
var newCb = wrapCb(cb, 1);
|
|||
|
try {
|
|||
|
path = normalizePath(path);
|
|||
|
fs.root.chown(path, true, uid, gid, newCb);
|
|||
|
} catch (e) {
|
|||
|
newCb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Synchronous `lchown`.
|
|||
|
* @param [String] path
|
|||
|
* @param [Number] uid
|
|||
|
* @param [Number] gid
|
|||
|
*/
|
|||
|
fs.lchownSync = function (path, uid, gid) {
|
|||
|
path = normalizePath(path);
|
|||
|
return fs.root.chownSync(path, true, uid, gid);
|
|||
|
};
|
|||
|
|
|||
|
fs.chmod = function (path, mode, cb) {
|
|||
|
if (typeof cb === "undefined") { cb = nopCb; }
|
|||
|
var newCb = wrapCb(cb, 1);
|
|||
|
try {
|
|||
|
mode = typeof mode === 'string' ? parseInt(mode, 8) : mode;
|
|||
|
path = normalizePath(path);
|
|||
|
fs.root.chmod(path, false, mode, newCb);
|
|||
|
} catch (e) {
|
|||
|
newCb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
fs.chmodSync = function (path, mode) {
|
|||
|
mode = typeof mode === 'string' ? parseInt(mode, 8) : mode;
|
|||
|
path = normalizePath(path);
|
|||
|
return fs.root.chmodSync(path, false, mode);
|
|||
|
};
|
|||
|
|
|||
|
fs.lchmod = function (path, mode, cb) {
|
|||
|
if (typeof cb === "undefined") { cb = nopCb; }
|
|||
|
var newCb = wrapCb(cb, 1);
|
|||
|
try {
|
|||
|
mode = typeof mode === 'string' ? parseInt(mode, 8) : mode;
|
|||
|
path = normalizePath(path);
|
|||
|
fs.root.chmod(path, true, mode, newCb);
|
|||
|
} catch (e) {
|
|||
|
newCb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
fs.lchmodSync = function (path, mode) {
|
|||
|
path = normalizePath(path);
|
|||
|
mode = typeof mode === 'string' ? parseInt(mode, 8) : mode;
|
|||
|
return fs.root.chmodSync(path, true, mode);
|
|||
|
};
|
|||
|
|
|||
|
fs.utimes = function (path, atime, mtime, cb) {
|
|||
|
if (typeof cb === "undefined") { cb = nopCb; }
|
|||
|
var newCb = wrapCb(cb, 1);
|
|||
|
try {
|
|||
|
path = normalizePath(path);
|
|||
|
if (typeof atime === 'number') {
|
|||
|
atime = new Date(atime * 1000);
|
|||
|
}
|
|||
|
if (typeof mtime === 'number') {
|
|||
|
mtime = new Date(mtime * 1000);
|
|||
|
}
|
|||
|
fs.root.utimes(path, atime, mtime, newCb);
|
|||
|
} catch (e) {
|
|||
|
newCb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
fs.utimesSync = function (path, atime, mtime) {
|
|||
|
path = normalizePath(path);
|
|||
|
if (typeof atime === 'number') {
|
|||
|
atime = new Date(atime * 1000);
|
|||
|
}
|
|||
|
if (typeof mtime === 'number') {
|
|||
|
mtime = new Date(mtime * 1000);
|
|||
|
}
|
|||
|
return fs.root.utimesSync(path, atime, mtime);
|
|||
|
};
|
|||
|
|
|||
|
fs.realpath = function (path, arg2, cb) {
|
|||
|
if (typeof cb === "undefined") { cb = nopCb; }
|
|||
|
var cache = typeof arg2 === 'object' ? arg2 : {};
|
|||
|
cb = typeof arg2 === 'function' ? arg2 : nopCb;
|
|||
|
var newCb = wrapCb(cb, 2);
|
|||
|
try {
|
|||
|
path = normalizePath(path);
|
|||
|
fs.root.realpath(path, cache, newCb);
|
|||
|
} catch (e) {
|
|||
|
newCb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Synchronous `realpath`.
|
|||
|
* @param [String] path
|
|||
|
* @param [Object?] cache An object literal of mapped paths that can be used to
|
|||
|
* force a specific path resolution or avoid additional `fs.stat` calls for
|
|||
|
* known real paths.
|
|||
|
* @return [String]
|
|||
|
*/
|
|||
|
fs.realpathSync = function (path, cache) {
|
|||
|
if (typeof cache === "undefined") { cache = {}; }
|
|||
|
path = normalizePath(path);
|
|||
|
return fs.root.realpathSync(path, cache);
|
|||
|
};
|
|||
|
fs.root = null;
|
|||
|
return fs;
|
|||
|
})();
|
|||
|
exports.fs = fs;
|
|||
|
});
|
|||
|
//# sourceMappingURL=node_fs.js.map
|
|||
|
;
|
|||
|
define('core/browserfs',["require", "exports", './buffer', './node_fs', './node_path', './node_process'], function(require, exports, buffer, node_fs, node_path, node_process) {
|
|||
|
/**
|
|||
|
* Installs BrowserFS onto the given object.
|
|||
|
* We recommend that you run install with the 'window' object to make things
|
|||
|
* global, as in Node.
|
|||
|
*
|
|||
|
* Properties installed:
|
|||
|
*
|
|||
|
* * Buffer
|
|||
|
* * process
|
|||
|
* * require (we monkey-patch it)
|
|||
|
*
|
|||
|
* This allows you to write code as if you were running inside Node.
|
|||
|
* @param {object} obj - The object to install things onto (e.g. window)
|
|||
|
*/
|
|||
|
function install(obj) {
|
|||
|
obj.Buffer = buffer.Buffer;
|
|||
|
obj.process = node_process.process;
|
|||
|
var oldRequire = obj.require != null ? obj.require : null;
|
|||
|
|
|||
|
// Monkey-patch require for Node-style code.
|
|||
|
obj.require = function (arg) {
|
|||
|
var rv = exports.BFSRequire(arg);
|
|||
|
if (rv == null) {
|
|||
|
return oldRequire.apply(null, Array.prototype.slice.call(arguments, 0));
|
|||
|
} else {
|
|||
|
return rv;
|
|||
|
}
|
|||
|
};
|
|||
|
}
|
|||
|
exports.install = install;
|
|||
|
|
|||
|
exports.FileSystem = {};
|
|||
|
function registerFileSystem(name, fs) {
|
|||
|
exports.FileSystem[name] = fs;
|
|||
|
}
|
|||
|
exports.registerFileSystem = registerFileSystem;
|
|||
|
|
|||
|
function BFSRequire(module) {
|
|||
|
switch (module) {
|
|||
|
case 'fs':
|
|||
|
return node_fs.fs;
|
|||
|
case 'path':
|
|||
|
return node_path.path;
|
|||
|
case 'buffer':
|
|||
|
// The 'buffer' module has 'Buffer' as a property.
|
|||
|
return buffer;
|
|||
|
case 'process':
|
|||
|
return node_process.process;
|
|||
|
default:
|
|||
|
return exports.FileSystem[module];
|
|||
|
}
|
|||
|
}
|
|||
|
exports.BFSRequire = BFSRequire;
|
|||
|
|
|||
|
/**
|
|||
|
* You must call this function with a properly-instantiated root file system
|
|||
|
* before using any file system API method.
|
|||
|
* @param {BrowserFS.FileSystem} rootFS - The root filesystem to use for the
|
|||
|
* entire BrowserFS file system.
|
|||
|
*/
|
|||
|
function initialize(rootfs) {
|
|||
|
return node_fs.fs._initialize(rootfs);
|
|||
|
}
|
|||
|
exports.initialize = initialize;
|
|||
|
});
|
|||
|
//# sourceMappingURL=browserfs.js.map
|
|||
|
;
|
|||
|
define('generic/emscripten_fs',["require", "exports", '../core/browserfs', '../core/node_fs', '../core/buffer', '../core/buffer_core_arraybuffer'], function(require, exports, BrowserFS, node_fs, buffer, buffer_core_arraybuffer) {
|
|||
|
var Buffer = buffer.Buffer;
|
|||
|
var BufferCoreArrayBuffer = buffer_core_arraybuffer.BufferCoreArrayBuffer;
|
|||
|
var fs = node_fs.fs;
|
|||
|
|
|||
|
var BFSEmscriptenStreamOps = (function () {
|
|||
|
function BFSEmscriptenStreamOps(fs) {
|
|||
|
this.fs = fs;
|
|||
|
}
|
|||
|
BFSEmscriptenStreamOps.prototype.open = function (stream) {
|
|||
|
var path = this.fs.realPath(stream.node);
|
|||
|
try {
|
|||
|
if (FS.isFile(stream.node.mode)) {
|
|||
|
stream.nfd = fs.openSync(path, this.fs.flagsToPermissionString(stream.flags));
|
|||
|
}
|
|||
|
} catch (e) {
|
|||
|
if (!e.code)
|
|||
|
throw e;
|
|||
|
throw new FS.ErrnoError(ERRNO_CODES[e.code]);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
BFSEmscriptenStreamOps.prototype.close = function (stream) {
|
|||
|
try {
|
|||
|
if (FS.isFile(stream.node.mode) && stream.nfd) {
|
|||
|
fs.closeSync(stream.nfd);
|
|||
|
}
|
|||
|
} catch (e) {
|
|||
|
if (!e.code)
|
|||
|
throw e;
|
|||
|
throw new FS.ErrnoError(ERRNO_CODES[e.code]);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
BFSEmscriptenStreamOps.prototype.read = function (stream, buffer, offset, length, position) {
|
|||
|
// Avoid copying overhead by reading directly into buffer.
|
|||
|
var bcore = new BufferCoreArrayBuffer(buffer.buffer);
|
|||
|
var nbuffer = new Buffer(bcore, buffer.byteOffset + offset, buffer.byteOffset + offset + length);
|
|||
|
var res;
|
|||
|
try {
|
|||
|
res = fs.readSync(stream.nfd, nbuffer, 0, length, position);
|
|||
|
} catch (e) {
|
|||
|
throw new FS.ErrnoError(ERRNO_CODES[e.code]);
|
|||
|
}
|
|||
|
|
|||
|
// No copying needed, since we wrote directly into UintArray.
|
|||
|
return res;
|
|||
|
};
|
|||
|
|
|||
|
BFSEmscriptenStreamOps.prototype.write = function (stream, buffer, offset, length, position) {
|
|||
|
// Avoid copying overhead; plug the buffer directly into a BufferCore.
|
|||
|
var bcore = new BufferCoreArrayBuffer(buffer.buffer);
|
|||
|
var nbuffer = new Buffer(bcore, buffer.byteOffset + offset, buffer.byteOffset + offset + length);
|
|||
|
var res;
|
|||
|
try {
|
|||
|
res = fs.writeSync(stream.nfd, nbuffer, 0, length, position);
|
|||
|
} catch (e) {
|
|||
|
throw new FS.ErrnoError(ERRNO_CODES[e.code]);
|
|||
|
}
|
|||
|
return res;
|
|||
|
};
|
|||
|
|
|||
|
BFSEmscriptenStreamOps.prototype.llseek = function (stream, offset, whence) {
|
|||
|
var position = offset;
|
|||
|
if (whence === 1) {
|
|||
|
position += stream.position;
|
|||
|
} else if (whence === 2) {
|
|||
|
if (FS.isFile(stream.node.mode)) {
|
|||
|
try {
|
|||
|
var stat = fs.fstatSync(stream.nfd);
|
|||
|
position += stat.size;
|
|||
|
} catch (e) {
|
|||
|
throw new FS.ErrnoError(ERRNO_CODES[e.code]);
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
if (position < 0) {
|
|||
|
throw new FS.ErrnoError(ERRNO_CODES.EINVAL);
|
|||
|
}
|
|||
|
|
|||
|
stream.position = position;
|
|||
|
return position;
|
|||
|
};
|
|||
|
return BFSEmscriptenStreamOps;
|
|||
|
})();
|
|||
|
|
|||
|
var BFSEmscriptenNodeOps = (function () {
|
|||
|
function BFSEmscriptenNodeOps(fs) {
|
|||
|
this.fs = fs;
|
|||
|
}
|
|||
|
BFSEmscriptenNodeOps.prototype.getattr = function (node) {
|
|||
|
var path = this.fs.realPath(node);
|
|||
|
var stat;
|
|||
|
try {
|
|||
|
stat = fs.lstatSync(path);
|
|||
|
} catch (e) {
|
|||
|
if (!e.code)
|
|||
|
throw e;
|
|||
|
throw new FS.ErrnoError(ERRNO_CODES[e.code]);
|
|||
|
}
|
|||
|
return {
|
|||
|
dev: stat.dev,
|
|||
|
ino: stat.ino,
|
|||
|
mode: stat.mode,
|
|||
|
nlink: stat.nlink,
|
|||
|
uid: stat.uid,
|
|||
|
gid: stat.gid,
|
|||
|
rdev: stat.rdev,
|
|||
|
size: stat.size,
|
|||
|
atime: stat.atime,
|
|||
|
mtime: stat.mtime,
|
|||
|
ctime: stat.ctime,
|
|||
|
blksize: stat.blksize,
|
|||
|
blocks: stat.blocks
|
|||
|
};
|
|||
|
};
|
|||
|
|
|||
|
BFSEmscriptenNodeOps.prototype.setattr = function (node, attr) {
|
|||
|
var path = this.fs.realPath(node);
|
|||
|
try {
|
|||
|
if (attr.mode !== undefined) {
|
|||
|
fs.chmodSync(path, attr.mode);
|
|||
|
|
|||
|
// update the common node structure mode as well
|
|||
|
node.mode = attr.mode;
|
|||
|
}
|
|||
|
if (attr.timestamp !== undefined) {
|
|||
|
var date = new Date(attr.timestamp);
|
|||
|
fs.utimesSync(path, date, date);
|
|||
|
}
|
|||
|
if (attr.size !== undefined) {
|
|||
|
fs.truncateSync(path, attr.size);
|
|||
|
}
|
|||
|
} catch (e) {
|
|||
|
if (!e.code)
|
|||
|
throw e;
|
|||
|
if (e.code === "ENOTSUP") {
|
|||
|
// Ignore not supported errors. Emscripten does utimesSync when it
|
|||
|
// writes files, but never really requires the value to be set.
|
|||
|
return;
|
|||
|
}
|
|||
|
throw new FS.ErrnoError(ERRNO_CODES[e.code]);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
BFSEmscriptenNodeOps.prototype.lookup = function (parent, name) {
|
|||
|
var path = PATH.join2(this.fs.realPath(parent), name);
|
|||
|
var mode = this.fs.getMode(path);
|
|||
|
return this.fs.createNode(parent, name, mode);
|
|||
|
};
|
|||
|
|
|||
|
BFSEmscriptenNodeOps.prototype.mknod = function (parent, name, mode, dev) {
|
|||
|
var node = this.fs.createNode(parent, name, mode, dev);
|
|||
|
|
|||
|
// create the backing node for this in the fs root as well
|
|||
|
var path = this.fs.realPath(node);
|
|||
|
try {
|
|||
|
if (FS.isDir(node.mode)) {
|
|||
|
fs.mkdirSync(path, node.mode);
|
|||
|
} else {
|
|||
|
fs.writeFileSync(path, '', { mode: node.mode });
|
|||
|
}
|
|||
|
} catch (e) {
|
|||
|
if (!e.code)
|
|||
|
throw e;
|
|||
|
throw new FS.ErrnoError(ERRNO_CODES[e.code]);
|
|||
|
}
|
|||
|
return node;
|
|||
|
};
|
|||
|
|
|||
|
BFSEmscriptenNodeOps.prototype.rename = function (oldNode, newDir, newName) {
|
|||
|
var oldPath = this.fs.realPath(oldNode);
|
|||
|
var newPath = PATH.join2(this.fs.realPath(newDir), newName);
|
|||
|
try {
|
|||
|
fs.renameSync(oldPath, newPath);
|
|||
|
} catch (e) {
|
|||
|
if (!e.code)
|
|||
|
throw e;
|
|||
|
throw new FS.ErrnoError(ERRNO_CODES[e.code]);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
BFSEmscriptenNodeOps.prototype.unlink = function (parent, name) {
|
|||
|
var path = PATH.join2(this.fs.realPath(parent), name);
|
|||
|
try {
|
|||
|
fs.unlinkSync(path);
|
|||
|
} catch (e) {
|
|||
|
if (!e.code)
|
|||
|
throw e;
|
|||
|
throw new FS.ErrnoError(ERRNO_CODES[e.code]);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
BFSEmscriptenNodeOps.prototype.rmdir = function (parent, name) {
|
|||
|
var path = PATH.join2(this.fs.realPath(parent), name);
|
|||
|
try {
|
|||
|
fs.rmdirSync(path);
|
|||
|
} catch (e) {
|
|||
|
if (!e.code)
|
|||
|
throw e;
|
|||
|
throw new FS.ErrnoError(ERRNO_CODES[e.code]);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
BFSEmscriptenNodeOps.prototype.readdir = function (node) {
|
|||
|
var path = this.fs.realPath(node);
|
|||
|
try {
|
|||
|
return fs.readdirSync(path);
|
|||
|
} catch (e) {
|
|||
|
if (!e.code)
|
|||
|
throw e;
|
|||
|
throw new FS.ErrnoError(ERRNO_CODES[e.code]);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
BFSEmscriptenNodeOps.prototype.symlink = function (parent, newName, oldPath) {
|
|||
|
var newPath = PATH.join2(this.fs.realPath(parent), newName);
|
|||
|
try {
|
|||
|
fs.symlinkSync(oldPath, newPath);
|
|||
|
} catch (e) {
|
|||
|
if (!e.code)
|
|||
|
throw e;
|
|||
|
throw new FS.ErrnoError(ERRNO_CODES[e.code]);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
BFSEmscriptenNodeOps.prototype.readlink = function (node) {
|
|||
|
var path = this.fs.realPath(node);
|
|||
|
try {
|
|||
|
return fs.readlinkSync(path);
|
|||
|
} catch (e) {
|
|||
|
if (!e.code)
|
|||
|
throw e;
|
|||
|
throw new FS.ErrnoError(ERRNO_CODES[e.code]);
|
|||
|
}
|
|||
|
};
|
|||
|
return BFSEmscriptenNodeOps;
|
|||
|
})();
|
|||
|
|
|||
|
var BFSEmscriptenFS = (function () {
|
|||
|
function BFSEmscriptenFS() {
|
|||
|
// This maps the integer permission modes from http://linux.die.net/man/3/open
|
|||
|
// to node.js-specific file open permission strings at http://nodejs.org/api/fs.html#fs_fs_open_path_flags_mode_callback
|
|||
|
this.flagsToPermissionStringMap = {
|
|||
|
0: 'r',
|
|||
|
1: 'r+',
|
|||
|
2: 'r+',
|
|||
|
64: 'r',
|
|||
|
65: 'r+',
|
|||
|
66: 'r+',
|
|||
|
129: 'rx+',
|
|||
|
193: 'rx+',
|
|||
|
514: 'w+',
|
|||
|
577: 'w',
|
|||
|
578: 'w+',
|
|||
|
705: 'wx',
|
|||
|
706: 'wx+',
|
|||
|
1024: 'a',
|
|||
|
1025: 'a',
|
|||
|
1026: 'a+',
|
|||
|
1089: 'a',
|
|||
|
1090: 'a+',
|
|||
|
1153: 'ax',
|
|||
|
1154: 'ax+',
|
|||
|
1217: 'ax',
|
|||
|
1218: 'ax+',
|
|||
|
4096: 'rs',
|
|||
|
4098: 'rs+'
|
|||
|
};
|
|||
|
this.node_ops = new BFSEmscriptenNodeOps(this);
|
|||
|
this.stream_ops = new BFSEmscriptenStreamOps(this);
|
|||
|
if (typeof BrowserFS === 'undefined') {
|
|||
|
throw new Error("BrowserFS is not loaded. Please load it before this library.");
|
|||
|
}
|
|||
|
}
|
|||
|
BFSEmscriptenFS.prototype.mount = function (mount) {
|
|||
|
return this.createNode(null, '/', this.getMode(mount.opts.root), 0);
|
|||
|
};
|
|||
|
|
|||
|
BFSEmscriptenFS.prototype.createNode = function (parent, name, mode, dev) {
|
|||
|
if (!FS.isDir(mode) && !FS.isFile(mode) && !FS.isLink(mode)) {
|
|||
|
throw new FS.ErrnoError(ERRNO_CODES.EINVAL);
|
|||
|
}
|
|||
|
var node = FS.createNode(parent, name, mode);
|
|||
|
node.node_ops = this.node_ops;
|
|||
|
node.stream_ops = this.stream_ops;
|
|||
|
return node;
|
|||
|
};
|
|||
|
|
|||
|
BFSEmscriptenFS.prototype.getMode = function (path) {
|
|||
|
var stat;
|
|||
|
try {
|
|||
|
stat = fs.lstatSync(path);
|
|||
|
} catch (e) {
|
|||
|
if (!e.code)
|
|||
|
throw e;
|
|||
|
throw new FS.ErrnoError(ERRNO_CODES[e.code]);
|
|||
|
}
|
|||
|
return stat.mode;
|
|||
|
};
|
|||
|
|
|||
|
BFSEmscriptenFS.prototype.realPath = function (node) {
|
|||
|
var parts = [];
|
|||
|
while (node.parent !== node) {
|
|||
|
parts.push(node.name);
|
|||
|
node = node.parent;
|
|||
|
}
|
|||
|
parts.push(node.mount.opts.root);
|
|||
|
parts.reverse();
|
|||
|
return PATH.join.apply(null, parts);
|
|||
|
};
|
|||
|
|
|||
|
BFSEmscriptenFS.prototype.flagsToPermissionString = function (flags) {
|
|||
|
if (flags in this.flagsToPermissionStringMap) {
|
|||
|
return this.flagsToPermissionStringMap[flags];
|
|||
|
} else {
|
|||
|
return flags;
|
|||
|
}
|
|||
|
};
|
|||
|
return BFSEmscriptenFS;
|
|||
|
})();
|
|||
|
exports.BFSEmscriptenFS = BFSEmscriptenFS;
|
|||
|
|
|||
|
// Make it available on the global BrowserFS object.
|
|||
|
BrowserFS['EmscriptenFS'] = BFSEmscriptenFS;
|
|||
|
});
|
|||
|
//# sourceMappingURL=emscripten_fs.js.map
|
|||
|
;
|
|||
|
var __extends = this.__extends || function (d, b) {
|
|||
|
for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p];
|
|||
|
function __() { this.constructor = d; }
|
|||
|
__.prototype = b.prototype;
|
|||
|
d.prototype = new __();
|
|||
|
};
|
|||
|
define('core/file_system',["require", "exports", './api_error', './file_flag', './node_path', './buffer'], function(require, exports, api_error, file_flag, node_path, buffer) {
|
|||
|
var ApiError = api_error.ApiError;
|
|||
|
var ErrorCode = api_error.ErrorCode;
|
|||
|
var path = node_path.path;
|
|||
|
var Buffer = buffer.Buffer;
|
|||
|
var ActionType = file_flag.ActionType;
|
|||
|
|
|||
|
|
|||
|
|
|||
|
|
|||
|
|
|||
|
/**
|
|||
|
* Basic filesystem class. Most filesystems should extend this class, as it
|
|||
|
* provides default implementations for a handful of methods.
|
|||
|
*/
|
|||
|
var BaseFileSystem = (function () {
|
|||
|
function BaseFileSystem() {
|
|||
|
}
|
|||
|
BaseFileSystem.prototype.supportsLinks = function () {
|
|||
|
return false;
|
|||
|
};
|
|||
|
BaseFileSystem.prototype.diskSpace = function (p, cb) {
|
|||
|
cb(0, 0);
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Opens the file at path p with the given flag. The file must exist.
|
|||
|
* @param p The path to open.
|
|||
|
* @param flag The flag to use when opening the file.
|
|||
|
*/
|
|||
|
BaseFileSystem.prototype.openFile = function (p, flag, cb) {
|
|||
|
throw new ApiError(14 /* ENOTSUP */);
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Create the file at path p with the given mode. Then, open it with the given
|
|||
|
* flag.
|
|||
|
*/
|
|||
|
BaseFileSystem.prototype.createFile = function (p, flag, mode, cb) {
|
|||
|
throw new ApiError(14 /* ENOTSUP */);
|
|||
|
};
|
|||
|
BaseFileSystem.prototype.open = function (p, flag, mode, cb) {
|
|||
|
var _this = this;
|
|||
|
var must_be_file = function (e, stats) {
|
|||
|
if (e) {
|
|||
|
switch (flag.pathNotExistsAction()) {
|
|||
|
case 3 /* CREATE_FILE */:
|
|||
|
// Ensure parent exists.
|
|||
|
return _this.stat(path.dirname(p), false, function (e, parentStats) {
|
|||
|
if (e) {
|
|||
|
cb(e);
|
|||
|
} else if (!parentStats.isDirectory()) {
|
|||
|
cb(new ApiError(7 /* ENOTDIR */, path.dirname(p) + " is not a directory."));
|
|||
|
} else {
|
|||
|
_this.createFile(p, flag, mode, cb);
|
|||
|
}
|
|||
|
});
|
|||
|
case 1 /* THROW_EXCEPTION */:
|
|||
|
return cb(new ApiError(1 /* ENOENT */, "" + p + " doesn't exist."));
|
|||
|
default:
|
|||
|
return cb(new ApiError(9 /* EINVAL */, 'Invalid FileFlag object.'));
|
|||
|
}
|
|||
|
} else {
|
|||
|
// File exists.
|
|||
|
if (stats.isDirectory()) {
|
|||
|
return cb(new ApiError(8 /* EISDIR */, p + " is a directory."));
|
|||
|
}
|
|||
|
switch (flag.pathExistsAction()) {
|
|||
|
case 1 /* THROW_EXCEPTION */:
|
|||
|
return cb(new ApiError(6 /* EEXIST */, p + " already exists."));
|
|||
|
case 2 /* TRUNCATE_FILE */:
|
|||
|
// NOTE: In a previous implementation, we deleted the file and
|
|||
|
// re-created it. However, this created a race condition if another
|
|||
|
// asynchronous request was trying to read the file, as the file
|
|||
|
// would not exist for a small period of time.
|
|||
|
return _this.openFile(p, flag, function (e, fd) {
|
|||
|
if (e) {
|
|||
|
cb(e);
|
|||
|
} else {
|
|||
|
fd.truncate(0, function () {
|
|||
|
fd.sync(function () {
|
|||
|
cb(null, fd);
|
|||
|
});
|
|||
|
});
|
|||
|
}
|
|||
|
});
|
|||
|
case 0 /* NOP */:
|
|||
|
return _this.openFile(p, flag, cb);
|
|||
|
default:
|
|||
|
return cb(new ApiError(9 /* EINVAL */, 'Invalid FileFlag object.'));
|
|||
|
}
|
|||
|
}
|
|||
|
};
|
|||
|
this.stat(p, false, must_be_file);
|
|||
|
};
|
|||
|
BaseFileSystem.prototype.rename = function (oldPath, newPath, cb) {
|
|||
|
cb(new ApiError(14 /* ENOTSUP */));
|
|||
|
};
|
|||
|
BaseFileSystem.prototype.renameSync = function (oldPath, newPath) {
|
|||
|
throw new ApiError(14 /* ENOTSUP */);
|
|||
|
};
|
|||
|
BaseFileSystem.prototype.stat = function (p, isLstat, cb) {
|
|||
|
cb(new ApiError(14 /* ENOTSUP */));
|
|||
|
};
|
|||
|
BaseFileSystem.prototype.statSync = function (p, isLstat) {
|
|||
|
throw new ApiError(14 /* ENOTSUP */);
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Opens the file at path p with the given flag. The file must exist.
|
|||
|
* @param p The path to open.
|
|||
|
* @param flag The flag to use when opening the file.
|
|||
|
* @return A File object corresponding to the opened file.
|
|||
|
*/
|
|||
|
BaseFileSystem.prototype.openFileSync = function (p, flag) {
|
|||
|
throw new ApiError(14 /* ENOTSUP */);
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Create the file at path p with the given mode. Then, open it with the given
|
|||
|
* flag.
|
|||
|
*/
|
|||
|
BaseFileSystem.prototype.createFileSync = function (p, flag, mode) {
|
|||
|
throw new ApiError(14 /* ENOTSUP */);
|
|||
|
};
|
|||
|
BaseFileSystem.prototype.openSync = function (p, flag, mode) {
|
|||
|
// Check if the path exists, and is a file.
|
|||
|
var stats;
|
|||
|
try {
|
|||
|
stats = this.statSync(p, false);
|
|||
|
} catch (e) {
|
|||
|
switch (flag.pathNotExistsAction()) {
|
|||
|
case 3 /* CREATE_FILE */:
|
|||
|
// Ensure parent exists.
|
|||
|
var parentStats = this.statSync(path.dirname(p), false);
|
|||
|
if (!parentStats.isDirectory()) {
|
|||
|
throw new ApiError(7 /* ENOTDIR */, path.dirname(p) + " is not a directory.");
|
|||
|
}
|
|||
|
return this.createFileSync(p, flag, mode);
|
|||
|
case 1 /* THROW_EXCEPTION */:
|
|||
|
throw new ApiError(1 /* ENOENT */, "" + p + " doesn't exist.");
|
|||
|
default:
|
|||
|
throw new ApiError(9 /* EINVAL */, 'Invalid FileFlag object.');
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
// File exists.
|
|||
|
if (stats.isDirectory()) {
|
|||
|
throw new ApiError(8 /* EISDIR */, p + " is a directory.");
|
|||
|
}
|
|||
|
switch (flag.pathExistsAction()) {
|
|||
|
case 1 /* THROW_EXCEPTION */:
|
|||
|
throw new ApiError(6 /* EEXIST */, p + " already exists.");
|
|||
|
case 2 /* TRUNCATE_FILE */:
|
|||
|
// Delete file.
|
|||
|
this.unlinkSync(p);
|
|||
|
|
|||
|
// Create file. Use the same mode as the old file.
|
|||
|
// Node itself modifies the ctime when this occurs, so this action
|
|||
|
// will preserve that behavior if the underlying file system
|
|||
|
// supports those properties.
|
|||
|
return this.createFileSync(p, flag, stats.mode);
|
|||
|
case 0 /* NOP */:
|
|||
|
return this.openFileSync(p, flag);
|
|||
|
default:
|
|||
|
throw new ApiError(9 /* EINVAL */, 'Invalid FileFlag object.');
|
|||
|
}
|
|||
|
};
|
|||
|
BaseFileSystem.prototype.unlink = function (p, cb) {
|
|||
|
cb(new ApiError(14 /* ENOTSUP */));
|
|||
|
};
|
|||
|
BaseFileSystem.prototype.unlinkSync = function (p) {
|
|||
|
throw new ApiError(14 /* ENOTSUP */);
|
|||
|
};
|
|||
|
BaseFileSystem.prototype.rmdir = function (p, cb) {
|
|||
|
cb(new ApiError(14 /* ENOTSUP */));
|
|||
|
};
|
|||
|
BaseFileSystem.prototype.rmdirSync = function (p) {
|
|||
|
throw new ApiError(14 /* ENOTSUP */);
|
|||
|
};
|
|||
|
BaseFileSystem.prototype.mkdir = function (p, mode, cb) {
|
|||
|
cb(new ApiError(14 /* ENOTSUP */));
|
|||
|
};
|
|||
|
BaseFileSystem.prototype.mkdirSync = function (p, mode) {
|
|||
|
throw new ApiError(14 /* ENOTSUP */);
|
|||
|
};
|
|||
|
BaseFileSystem.prototype.readdir = function (p, cb) {
|
|||
|
cb(new ApiError(14 /* ENOTSUP */));
|
|||
|
};
|
|||
|
BaseFileSystem.prototype.readdirSync = function (p) {
|
|||
|
throw new ApiError(14 /* ENOTSUP */);
|
|||
|
};
|
|||
|
BaseFileSystem.prototype.exists = function (p, cb) {
|
|||
|
this.stat(p, null, function (err) {
|
|||
|
cb(err == null);
|
|||
|
});
|
|||
|
};
|
|||
|
BaseFileSystem.prototype.existsSync = function (p) {
|
|||
|
try {
|
|||
|
this.statSync(p, true);
|
|||
|
return true;
|
|||
|
} catch (e) {
|
|||
|
return false;
|
|||
|
}
|
|||
|
};
|
|||
|
BaseFileSystem.prototype.realpath = function (p, cache, cb) {
|
|||
|
if (this.supportsLinks()) {
|
|||
|
// The path could contain symlinks. Split up the path,
|
|||
|
// resolve any symlinks, return the resolved string.
|
|||
|
var splitPath = p.split(path.sep);
|
|||
|
|
|||
|
for (var i = 0; i < splitPath.length; i++) {
|
|||
|
var addPaths = splitPath.slice(0, i + 1);
|
|||
|
splitPath[i] = path.join.apply(null, addPaths);
|
|||
|
}
|
|||
|
} else {
|
|||
|
// No symlinks. We just need to verify that it exists.
|
|||
|
this.exists(p, function (doesExist) {
|
|||
|
if (doesExist) {
|
|||
|
cb(null, p);
|
|||
|
} else {
|
|||
|
cb(new ApiError(1 /* ENOENT */, "File " + p + " not found."));
|
|||
|
}
|
|||
|
});
|
|||
|
}
|
|||
|
};
|
|||
|
BaseFileSystem.prototype.realpathSync = function (p, cache) {
|
|||
|
if (this.supportsLinks()) {
|
|||
|
// The path could contain symlinks. Split up the path,
|
|||
|
// resolve any symlinks, return the resolved string.
|
|||
|
var splitPath = p.split(path.sep);
|
|||
|
|
|||
|
for (var i = 0; i < splitPath.length; i++) {
|
|||
|
var addPaths = splitPath.slice(0, i + 1);
|
|||
|
splitPath[i] = path.join.apply(null, addPaths);
|
|||
|
}
|
|||
|
} else {
|
|||
|
// No symlinks. We just need to verify that it exists.
|
|||
|
if (this.existsSync(p)) {
|
|||
|
return p;
|
|||
|
} else {
|
|||
|
throw new ApiError(1 /* ENOENT */, "File " + p + " not found.");
|
|||
|
}
|
|||
|
}
|
|||
|
};
|
|||
|
BaseFileSystem.prototype.truncate = function (p, len, cb) {
|
|||
|
this.open(p, file_flag.FileFlag.getFileFlag('r+'), 0x1a4, (function (er, fd) {
|
|||
|
if (er) {
|
|||
|
return cb(er);
|
|||
|
}
|
|||
|
fd.truncate(len, (function (er) {
|
|||
|
fd.close((function (er2) {
|
|||
|
cb(er || er2);
|
|||
|
}));
|
|||
|
}));
|
|||
|
}));
|
|||
|
};
|
|||
|
BaseFileSystem.prototype.truncateSync = function (p, len) {
|
|||
|
var fd = this.openSync(p, file_flag.FileFlag.getFileFlag('r+'), 0x1a4);
|
|||
|
|
|||
|
try {
|
|||
|
fd.truncateSync(len);
|
|||
|
} catch (e) {
|
|||
|
throw e;
|
|||
|
} finally {
|
|||
|
fd.closeSync();
|
|||
|
}
|
|||
|
};
|
|||
|
BaseFileSystem.prototype.readFile = function (fname, encoding, flag, cb) {
|
|||
|
// Wrap cb in file closing code.
|
|||
|
var oldCb = cb;
|
|||
|
|
|||
|
// Get file.
|
|||
|
this.open(fname, flag, 0x1a4, function (err, fd) {
|
|||
|
if (err) {
|
|||
|
return cb(err);
|
|||
|
}
|
|||
|
cb = function (err, arg) {
|
|||
|
fd.close(function (err2) {
|
|||
|
if (err == null) {
|
|||
|
err = err2;
|
|||
|
}
|
|||
|
return oldCb(err, arg);
|
|||
|
});
|
|||
|
};
|
|||
|
fd.stat(function (err, stat) {
|
|||
|
if (err != null) {
|
|||
|
return cb(err);
|
|||
|
}
|
|||
|
|
|||
|
// Allocate buffer.
|
|||
|
var buf = new Buffer(stat.size);
|
|||
|
fd.read(buf, 0, stat.size, 0, function (err) {
|
|||
|
if (err != null) {
|
|||
|
return cb(err);
|
|||
|
} else if (encoding === null) {
|
|||
|
return cb(err, buf);
|
|||
|
}
|
|||
|
try {
|
|||
|
cb(null, buf.toString(encoding));
|
|||
|
} catch (e) {
|
|||
|
cb(e);
|
|||
|
}
|
|||
|
});
|
|||
|
});
|
|||
|
});
|
|||
|
};
|
|||
|
BaseFileSystem.prototype.readFileSync = function (fname, encoding, flag) {
|
|||
|
// Get file.
|
|||
|
var fd = this.openSync(fname, flag, 0x1a4);
|
|||
|
try {
|
|||
|
var stat = fd.statSync();
|
|||
|
|
|||
|
// Allocate buffer.
|
|||
|
var buf = new Buffer(stat.size);
|
|||
|
fd.readSync(buf, 0, stat.size, 0);
|
|||
|
fd.closeSync();
|
|||
|
if (encoding === null) {
|
|||
|
return buf;
|
|||
|
}
|
|||
|
return buf.toString(encoding);
|
|||
|
} finally {
|
|||
|
fd.closeSync();
|
|||
|
}
|
|||
|
};
|
|||
|
BaseFileSystem.prototype.writeFile = function (fname, data, encoding, flag, mode, cb) {
|
|||
|
// Wrap cb in file closing code.
|
|||
|
var oldCb = cb;
|
|||
|
|
|||
|
// Get file.
|
|||
|
this.open(fname, flag, 0x1a4, function (err, fd) {
|
|||
|
if (err != null) {
|
|||
|
return cb(err);
|
|||
|
}
|
|||
|
cb = function (err) {
|
|||
|
fd.close(function (err2) {
|
|||
|
oldCb(err != null ? err : err2);
|
|||
|
});
|
|||
|
};
|
|||
|
|
|||
|
try {
|
|||
|
if (typeof data === 'string') {
|
|||
|
data = new Buffer(data, encoding);
|
|||
|
}
|
|||
|
} catch (e) {
|
|||
|
return cb(e);
|
|||
|
}
|
|||
|
|
|||
|
// Write into file.
|
|||
|
fd.write(data, 0, data.length, 0, cb);
|
|||
|
});
|
|||
|
};
|
|||
|
BaseFileSystem.prototype.writeFileSync = function (fname, data, encoding, flag, mode) {
|
|||
|
// Get file.
|
|||
|
var fd = this.openSync(fname, flag, mode);
|
|||
|
try {
|
|||
|
if (typeof data === 'string') {
|
|||
|
data = new Buffer(data, encoding);
|
|||
|
}
|
|||
|
|
|||
|
// Write into file.
|
|||
|
fd.writeSync(data, 0, data.length, 0);
|
|||
|
} finally {
|
|||
|
fd.closeSync();
|
|||
|
}
|
|||
|
};
|
|||
|
BaseFileSystem.prototype.appendFile = function (fname, data, encoding, flag, mode, cb) {
|
|||
|
// Wrap cb in file closing code.
|
|||
|
var oldCb = cb;
|
|||
|
this.open(fname, flag, mode, function (err, fd) {
|
|||
|
if (err != null) {
|
|||
|
return cb(err);
|
|||
|
}
|
|||
|
cb = function (err) {
|
|||
|
fd.close(function (err2) {
|
|||
|
oldCb(err != null ? err : err2);
|
|||
|
});
|
|||
|
};
|
|||
|
if (typeof data === 'string') {
|
|||
|
data = new Buffer(data, encoding);
|
|||
|
}
|
|||
|
fd.write(data, 0, data.length, null, cb);
|
|||
|
});
|
|||
|
};
|
|||
|
BaseFileSystem.prototype.appendFileSync = function (fname, data, encoding, flag, mode) {
|
|||
|
var fd = this.openSync(fname, flag, mode);
|
|||
|
try {
|
|||
|
if (typeof data === 'string') {
|
|||
|
data = new Buffer(data, encoding);
|
|||
|
}
|
|||
|
fd.writeSync(data, 0, data.length, null);
|
|||
|
} finally {
|
|||
|
fd.closeSync();
|
|||
|
}
|
|||
|
};
|
|||
|
BaseFileSystem.prototype.chmod = function (p, isLchmod, mode, cb) {
|
|||
|
cb(new ApiError(14 /* ENOTSUP */));
|
|||
|
};
|
|||
|
BaseFileSystem.prototype.chmodSync = function (p, isLchmod, mode) {
|
|||
|
throw new ApiError(14 /* ENOTSUP */);
|
|||
|
};
|
|||
|
BaseFileSystem.prototype.chown = function (p, isLchown, uid, gid, cb) {
|
|||
|
cb(new ApiError(14 /* ENOTSUP */));
|
|||
|
};
|
|||
|
BaseFileSystem.prototype.chownSync = function (p, isLchown, uid, gid) {
|
|||
|
throw new ApiError(14 /* ENOTSUP */);
|
|||
|
};
|
|||
|
BaseFileSystem.prototype.utimes = function (p, atime, mtime, cb) {
|
|||
|
cb(new ApiError(14 /* ENOTSUP */));
|
|||
|
};
|
|||
|
BaseFileSystem.prototype.utimesSync = function (p, atime, mtime) {
|
|||
|
throw new ApiError(14 /* ENOTSUP */);
|
|||
|
};
|
|||
|
BaseFileSystem.prototype.link = function (srcpath, dstpath, cb) {
|
|||
|
cb(new ApiError(14 /* ENOTSUP */));
|
|||
|
};
|
|||
|
BaseFileSystem.prototype.linkSync = function (srcpath, dstpath) {
|
|||
|
throw new ApiError(14 /* ENOTSUP */);
|
|||
|
};
|
|||
|
BaseFileSystem.prototype.symlink = function (srcpath, dstpath, type, cb) {
|
|||
|
cb(new ApiError(14 /* ENOTSUP */));
|
|||
|
};
|
|||
|
BaseFileSystem.prototype.symlinkSync = function (srcpath, dstpath, type) {
|
|||
|
throw new ApiError(14 /* ENOTSUP */);
|
|||
|
};
|
|||
|
BaseFileSystem.prototype.readlink = function (p, cb) {
|
|||
|
cb(new ApiError(14 /* ENOTSUP */));
|
|||
|
};
|
|||
|
BaseFileSystem.prototype.readlinkSync = function (p) {
|
|||
|
throw new ApiError(14 /* ENOTSUP */);
|
|||
|
};
|
|||
|
return BaseFileSystem;
|
|||
|
})();
|
|||
|
exports.BaseFileSystem = BaseFileSystem;
|
|||
|
|
|||
|
/**
|
|||
|
* Implements the asynchronous API in terms of the synchronous API.
|
|||
|
* @class SynchronousFileSystem
|
|||
|
*/
|
|||
|
var SynchronousFileSystem = (function (_super) {
|
|||
|
__extends(SynchronousFileSystem, _super);
|
|||
|
function SynchronousFileSystem() {
|
|||
|
_super.apply(this, arguments);
|
|||
|
}
|
|||
|
SynchronousFileSystem.prototype.supportsSynch = function () {
|
|||
|
return true;
|
|||
|
};
|
|||
|
|
|||
|
SynchronousFileSystem.prototype.rename = function (oldPath, newPath, cb) {
|
|||
|
try {
|
|||
|
this.renameSync(oldPath, newPath);
|
|||
|
cb();
|
|||
|
} catch (e) {
|
|||
|
cb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
SynchronousFileSystem.prototype.stat = function (p, isLstat, cb) {
|
|||
|
try {
|
|||
|
cb(null, this.statSync(p, isLstat));
|
|||
|
} catch (e) {
|
|||
|
cb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
SynchronousFileSystem.prototype.open = function (p, flags, mode, cb) {
|
|||
|
try {
|
|||
|
cb(null, this.openSync(p, flags, mode));
|
|||
|
} catch (e) {
|
|||
|
cb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
SynchronousFileSystem.prototype.unlink = function (p, cb) {
|
|||
|
try {
|
|||
|
this.unlinkSync(p);
|
|||
|
cb();
|
|||
|
} catch (e) {
|
|||
|
cb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
SynchronousFileSystem.prototype.rmdir = function (p, cb) {
|
|||
|
try {
|
|||
|
this.rmdirSync(p);
|
|||
|
cb();
|
|||
|
} catch (e) {
|
|||
|
cb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
SynchronousFileSystem.prototype.mkdir = function (p, mode, cb) {
|
|||
|
try {
|
|||
|
this.mkdirSync(p, mode);
|
|||
|
cb();
|
|||
|
} catch (e) {
|
|||
|
cb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
SynchronousFileSystem.prototype.readdir = function (p, cb) {
|
|||
|
try {
|
|||
|
cb(null, this.readdirSync(p));
|
|||
|
} catch (e) {
|
|||
|
cb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
SynchronousFileSystem.prototype.chmod = function (p, isLchmod, mode, cb) {
|
|||
|
try {
|
|||
|
this.chmodSync(p, isLchmod, mode);
|
|||
|
cb();
|
|||
|
} catch (e) {
|
|||
|
cb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
SynchronousFileSystem.prototype.chown = function (p, isLchown, uid, gid, cb) {
|
|||
|
try {
|
|||
|
this.chownSync(p, isLchown, uid, gid);
|
|||
|
cb();
|
|||
|
} catch (e) {
|
|||
|
cb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
SynchronousFileSystem.prototype.utimes = function (p, atime, mtime, cb) {
|
|||
|
try {
|
|||
|
this.utimesSync(p, atime, mtime);
|
|||
|
cb();
|
|||
|
} catch (e) {
|
|||
|
cb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
SynchronousFileSystem.prototype.link = function (srcpath, dstpath, cb) {
|
|||
|
try {
|
|||
|
this.linkSync(srcpath, dstpath);
|
|||
|
cb();
|
|||
|
} catch (e) {
|
|||
|
cb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
SynchronousFileSystem.prototype.symlink = function (srcpath, dstpath, type, cb) {
|
|||
|
try {
|
|||
|
this.symlinkSync(srcpath, dstpath, type);
|
|||
|
cb();
|
|||
|
} catch (e) {
|
|||
|
cb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
SynchronousFileSystem.prototype.readlink = function (p, cb) {
|
|||
|
try {
|
|||
|
cb(null, this.readlinkSync(p));
|
|||
|
} catch (e) {
|
|||
|
cb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
return SynchronousFileSystem;
|
|||
|
})(BaseFileSystem);
|
|||
|
exports.SynchronousFileSystem = SynchronousFileSystem;
|
|||
|
});
|
|||
|
//# sourceMappingURL=file_system.js.map
|
|||
|
;
|
|||
|
define('core/node_fs_stats',["require", "exports"], function(require, exports) {
|
|||
|
/**
|
|||
|
* Indicates the type of the given file. Applied to 'mode'.
|
|||
|
*/
|
|||
|
(function (FileType) {
|
|||
|
FileType[FileType["FILE"] = 0x8000] = "FILE";
|
|||
|
FileType[FileType["DIRECTORY"] = 0x4000] = "DIRECTORY";
|
|||
|
FileType[FileType["SYMLINK"] = 0xA000] = "SYMLINK";
|
|||
|
})(exports.FileType || (exports.FileType = {}));
|
|||
|
var FileType = exports.FileType;
|
|||
|
|
|||
|
/**
|
|||
|
* Emulation of Node's `fs.Stats` object.
|
|||
|
*
|
|||
|
* Attribute descriptions are from `man 2 stat'
|
|||
|
* @see http://nodejs.org/api/fs.html#fs_class_fs_stats
|
|||
|
* @see http://man7.org/linux/man-pages/man2/stat.2.html
|
|||
|
* @class
|
|||
|
*/
|
|||
|
var Stats = (function () {
|
|||
|
/**
|
|||
|
* Provides information about a particular entry in the file system.
|
|||
|
* @param [Number] item_type type of the item (FILE, DIRECTORY, SYMLINK, or SOCKET)
|
|||
|
* @param [Number] size Size of the item in bytes. For directories/symlinks,
|
|||
|
* this is normally the size of the struct that represents the item.
|
|||
|
* @param [Number] mode Unix-style file mode (e.g. 0o644)
|
|||
|
* @param [Date?] atime time of last access
|
|||
|
* @param [Date?] mtime time of last modification
|
|||
|
* @param [Date?] ctime time of creation
|
|||
|
*/
|
|||
|
function Stats(item_type, size, mode, atime, mtime, ctime) {
|
|||
|
if (typeof atime === "undefined") { atime = new Date(); }
|
|||
|
if (typeof mtime === "undefined") { mtime = new Date(); }
|
|||
|
if (typeof ctime === "undefined") { ctime = new Date(); }
|
|||
|
this.size = size;
|
|||
|
this.mode = mode;
|
|||
|
this.atime = atime;
|
|||
|
this.mtime = mtime;
|
|||
|
this.ctime = ctime;
|
|||
|
/**
|
|||
|
* UNSUPPORTED ATTRIBUTES
|
|||
|
* I assume no one is going to need these details, although we could fake
|
|||
|
* appropriate values if need be.
|
|||
|
*/
|
|||
|
// ID of device containing file
|
|||
|
this.dev = 0;
|
|||
|
// inode number
|
|||
|
this.ino = 0;
|
|||
|
// device ID (if special file)
|
|||
|
this.rdev = 0;
|
|||
|
// number of hard links
|
|||
|
this.nlink = 1;
|
|||
|
// blocksize for file system I/O
|
|||
|
this.blksize = 4096;
|
|||
|
// @todo Maybe support these? atm, it's a one-user filesystem.
|
|||
|
// user ID of owner
|
|||
|
this.uid = 0;
|
|||
|
// group ID of owner
|
|||
|
this.gid = 0;
|
|||
|
if (this.mode == null) {
|
|||
|
switch (item_type) {
|
|||
|
case 32768 /* FILE */:
|
|||
|
this.mode = 0x1a4;
|
|||
|
break;
|
|||
|
case 16384 /* DIRECTORY */:
|
|||
|
default:
|
|||
|
this.mode = 0x1ff;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
// number of 512B blocks allocated
|
|||
|
this.blocks = Math.ceil(size / 512);
|
|||
|
|
|||
|
// Check if mode also includes top-most bits, which indicate the file's
|
|||
|
// type.
|
|||
|
if (this.mode < 0x1000) {
|
|||
|
this.mode |= item_type;
|
|||
|
}
|
|||
|
}
|
|||
|
/**
|
|||
|
* **Nonstandard**: Clone the stats object.
|
|||
|
* @return [BrowserFS.node.fs.Stats]
|
|||
|
*/
|
|||
|
Stats.prototype.clone = function () {
|
|||
|
return new Stats(this.mode & 0xF000, this.size, this.mode & 0xFFF, this.atime, this.mtime, this.ctime);
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* @return [Boolean] True if this item is a file.
|
|||
|
*/
|
|||
|
Stats.prototype.isFile = function () {
|
|||
|
return (this.mode & 0xF000) === 32768 /* FILE */;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* @return [Boolean] True if this item is a directory.
|
|||
|
*/
|
|||
|
Stats.prototype.isDirectory = function () {
|
|||
|
return (this.mode & 0xF000) === 16384 /* DIRECTORY */;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* @return [Boolean] True if this item is a symbolic link (only valid through lstat)
|
|||
|
*/
|
|||
|
Stats.prototype.isSymbolicLink = function () {
|
|||
|
return (this.mode & 0xF000) === 40960 /* SYMLINK */;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Change the mode of the file. We use this helper function to prevent messing
|
|||
|
* up the type of the file, which is encoded in mode.
|
|||
|
*/
|
|||
|
Stats.prototype.chmod = function (mode) {
|
|||
|
this.mode = (this.mode & 0xF000) | mode;
|
|||
|
};
|
|||
|
|
|||
|
// We don't support the following types of files.
|
|||
|
Stats.prototype.isSocket = function () {
|
|||
|
return false;
|
|||
|
};
|
|||
|
|
|||
|
Stats.prototype.isBlockDevice = function () {
|
|||
|
return false;
|
|||
|
};
|
|||
|
|
|||
|
Stats.prototype.isCharacterDevice = function () {
|
|||
|
return false;
|
|||
|
};
|
|||
|
|
|||
|
Stats.prototype.isFIFO = function () {
|
|||
|
return false;
|
|||
|
};
|
|||
|
return Stats;
|
|||
|
})();
|
|||
|
exports.Stats = Stats;
|
|||
|
});
|
|||
|
//# sourceMappingURL=node_fs_stats.js.map
|
|||
|
;
|
|||
|
define('generic/inode',["require", "exports", '../core/node_fs_stats', '../core/buffer'], function(require, exports, node_fs_stats, buffer) {
|
|||
|
/**
|
|||
|
* Generic inode definition that can easily be serialized.
|
|||
|
*/
|
|||
|
var Inode = (function () {
|
|||
|
function Inode(id, size, mode, atime, mtime, ctime) {
|
|||
|
this.id = id;
|
|||
|
this.size = size;
|
|||
|
this.mode = mode;
|
|||
|
this.atime = atime;
|
|||
|
this.mtime = mtime;
|
|||
|
this.ctime = ctime;
|
|||
|
}
|
|||
|
/**
|
|||
|
* Handy function that converts the Inode to a Node Stats object.
|
|||
|
*/
|
|||
|
Inode.prototype.toStats = function () {
|
|||
|
return new node_fs_stats.Stats((this.mode & 0xF000) === 16384 /* DIRECTORY */ ? 16384 /* DIRECTORY */ : 32768 /* FILE */, this.size, this.mode, new Date(this.atime), new Date(this.mtime), new Date(this.ctime));
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Get the size of this Inode, in bytes.
|
|||
|
*/
|
|||
|
Inode.prototype.getSize = function () {
|
|||
|
// ASSUMPTION: ID is ASCII (1 byte per char).
|
|||
|
return 30 + this.id.length;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Writes the inode into the start of the buffer.
|
|||
|
*/
|
|||
|
Inode.prototype.toBuffer = function (buff) {
|
|||
|
if (typeof buff === "undefined") { buff = new buffer.Buffer(this.getSize()); }
|
|||
|
buff.writeUInt32LE(this.size, 0);
|
|||
|
buff.writeUInt16LE(this.mode, 4);
|
|||
|
buff.writeDoubleLE(this.atime, 6);
|
|||
|
buff.writeDoubleLE(this.mtime, 14);
|
|||
|
buff.writeDoubleLE(this.ctime, 22);
|
|||
|
buff.write(this.id, 30, this.id.length, 'ascii');
|
|||
|
return buff;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Updates the Inode using information from the stats object. Used by file
|
|||
|
* systems at sync time, e.g.:
|
|||
|
* - Program opens file and gets a File object.
|
|||
|
* - Program mutates file. File object is responsible for maintaining
|
|||
|
* metadata changes locally -- typically in a Stats object.
|
|||
|
* - Program closes file. File object's metadata changes are synced with the
|
|||
|
* file system.
|
|||
|
* @return True if any changes have occurred.
|
|||
|
*/
|
|||
|
Inode.prototype.update = function (stats) {
|
|||
|
var hasChanged = false;
|
|||
|
if (this.size !== stats.size) {
|
|||
|
this.size = stats.size;
|
|||
|
hasChanged = true;
|
|||
|
}
|
|||
|
|
|||
|
if (this.mode !== stats.mode) {
|
|||
|
this.mode = stats.mode;
|
|||
|
hasChanged = true;
|
|||
|
}
|
|||
|
|
|||
|
var atimeMs = stats.atime.getTime();
|
|||
|
if (this.atime !== atimeMs) {
|
|||
|
this.atime = atimeMs;
|
|||
|
hasChanged = true;
|
|||
|
}
|
|||
|
|
|||
|
var mtimeMs = stats.mtime.getTime();
|
|||
|
if (this.mtime !== mtimeMs) {
|
|||
|
this.mtime = mtimeMs;
|
|||
|
hasChanged = true;
|
|||
|
}
|
|||
|
|
|||
|
var ctimeMs = stats.ctime.getTime();
|
|||
|
if (this.ctime !== ctimeMs) {
|
|||
|
this.ctime = ctimeMs;
|
|||
|
hasChanged = true;
|
|||
|
}
|
|||
|
|
|||
|
return hasChanged;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Converts the buffer into an Inode.
|
|||
|
*/
|
|||
|
Inode.fromBuffer = function (buffer) {
|
|||
|
if (buffer === undefined) {
|
|||
|
throw new Error("NO");
|
|||
|
}
|
|||
|
return new Inode(buffer.toString('ascii', 30), buffer.readUInt32LE(0), buffer.readUInt16LE(4), buffer.readDoubleLE(6), buffer.readDoubleLE(14), buffer.readDoubleLE(22));
|
|||
|
};
|
|||
|
|
|||
|
// XXX: Copied from Stats. Should reconcile these two into something more
|
|||
|
// compact.
|
|||
|
/**
|
|||
|
* @return [Boolean] True if this item is a file.
|
|||
|
*/
|
|||
|
Inode.prototype.isFile = function () {
|
|||
|
return (this.mode & 0xF000) === 32768 /* FILE */;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* @return [Boolean] True if this item is a directory.
|
|||
|
*/
|
|||
|
Inode.prototype.isDirectory = function () {
|
|||
|
return (this.mode & 0xF000) === 16384 /* DIRECTORY */;
|
|||
|
};
|
|||
|
return Inode;
|
|||
|
})();
|
|||
|
|
|||
|
|
|||
|
return Inode;
|
|||
|
});
|
|||
|
//# sourceMappingURL=inode.js.map
|
|||
|
;
|
|||
|
define('core/file',["require", "exports", './api_error'], function(require, exports, api_error) {
|
|||
|
var ApiError = api_error.ApiError;
|
|||
|
var ErrorCode = api_error.ErrorCode;
|
|||
|
|
|||
|
/**
|
|||
|
* Base class that contains shared implementations of functions for the file
|
|||
|
* object.
|
|||
|
* @class
|
|||
|
*/
|
|||
|
var BaseFile = (function () {
|
|||
|
function BaseFile() {
|
|||
|
}
|
|||
|
BaseFile.prototype.sync = function (cb) {
|
|||
|
cb(new ApiError(14 /* ENOTSUP */));
|
|||
|
};
|
|||
|
BaseFile.prototype.syncSync = function () {
|
|||
|
throw new ApiError(14 /* ENOTSUP */);
|
|||
|
};
|
|||
|
BaseFile.prototype.datasync = function (cb) {
|
|||
|
this.sync(cb);
|
|||
|
};
|
|||
|
BaseFile.prototype.datasyncSync = function () {
|
|||
|
return this.syncSync();
|
|||
|
};
|
|||
|
BaseFile.prototype.chown = function (uid, gid, cb) {
|
|||
|
cb(new ApiError(14 /* ENOTSUP */));
|
|||
|
};
|
|||
|
BaseFile.prototype.chownSync = function (uid, gid) {
|
|||
|
throw new ApiError(14 /* ENOTSUP */);
|
|||
|
};
|
|||
|
BaseFile.prototype.chmod = function (mode, cb) {
|
|||
|
cb(new ApiError(14 /* ENOTSUP */));
|
|||
|
};
|
|||
|
BaseFile.prototype.chmodSync = function (mode) {
|
|||
|
throw new ApiError(14 /* ENOTSUP */);
|
|||
|
};
|
|||
|
BaseFile.prototype.utimes = function (atime, mtime, cb) {
|
|||
|
cb(new ApiError(14 /* ENOTSUP */));
|
|||
|
};
|
|||
|
BaseFile.prototype.utimesSync = function (atime, mtime) {
|
|||
|
throw new ApiError(14 /* ENOTSUP */);
|
|||
|
};
|
|||
|
return BaseFile;
|
|||
|
})();
|
|||
|
exports.BaseFile = BaseFile;
|
|||
|
});
|
|||
|
//# sourceMappingURL=file.js.map
|
|||
|
;
|
|||
|
var __extends = this.__extends || function (d, b) {
|
|||
|
for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p];
|
|||
|
function __() { this.constructor = d; }
|
|||
|
__.prototype = b.prototype;
|
|||
|
d.prototype = new __();
|
|||
|
};
|
|||
|
define('generic/preload_file',["require", "exports", '../core/file', '../core/buffer', '../core/api_error', '../core/node_fs'], function(require, exports, file, buffer, api_error, node_fs) {
|
|||
|
var ApiError = api_error.ApiError;
|
|||
|
var ErrorCode = api_error.ErrorCode;
|
|||
|
var fs = node_fs.fs;
|
|||
|
var Buffer = buffer.Buffer;
|
|||
|
|
|||
|
/**
|
|||
|
* An implementation of the File interface that operates on a file that is
|
|||
|
* completely in-memory. PreloadFiles are backed by a Buffer.
|
|||
|
*
|
|||
|
* This is also an abstract class, as it lacks an implementation of 'sync' and
|
|||
|
* 'close'. Each filesystem that wishes to use this file representation must
|
|||
|
* extend this class and implement those two methods.
|
|||
|
* @todo 'close' lever that disables functionality once closed.
|
|||
|
*/
|
|||
|
var PreloadFile = (function (_super) {
|
|||
|
__extends(PreloadFile, _super);
|
|||
|
/**
|
|||
|
* Creates a file with the given path and, optionally, the given contents. Note
|
|||
|
* that, if contents is specified, it will be mutated by the file!
|
|||
|
* @param [BrowserFS.FileSystem] _fs The file system that created the file.
|
|||
|
* @param [String] _path
|
|||
|
* @param [BrowserFS.FileMode] _mode The mode that the file was opened using.
|
|||
|
* Dictates permissions and where the file pointer starts.
|
|||
|
* @param [BrowserFS.node.fs.Stats] _stat The stats object for the given file.
|
|||
|
* PreloadFile will mutate this object. Note that this object must contain
|
|||
|
* the appropriate mode that the file was opened as.
|
|||
|
* @param [BrowserFS.node.Buffer?] contents A buffer containing the entire
|
|||
|
* contents of the file. PreloadFile will mutate this buffer. If not
|
|||
|
* specified, we assume it is a new file.
|
|||
|
*/
|
|||
|
function PreloadFile(_fs, _path, _flag, _stat, contents) {
|
|||
|
_super.call(this);
|
|||
|
this._pos = 0;
|
|||
|
this._fs = _fs;
|
|||
|
this._path = _path;
|
|||
|
this._flag = _flag;
|
|||
|
this._stat = _stat;
|
|||
|
if (contents != null) {
|
|||
|
this._buffer = contents;
|
|||
|
} else {
|
|||
|
// Empty buffer. It'll expand once we write stuff to it.
|
|||
|
this._buffer = new Buffer(0);
|
|||
|
}
|
|||
|
|
|||
|
// Note: This invariant is *not* maintained once the file starts getting
|
|||
|
// modified.
|
|||
|
if (this._stat.size !== this._buffer.length) {
|
|||
|
throw new Error("Invalid buffer: Buffer is " + this._buffer.length + " long, yet Stats object specifies that file is " + this._stat.size + " long.");
|
|||
|
}
|
|||
|
}
|
|||
|
/**
|
|||
|
* Get the path to this file.
|
|||
|
* @return [String] The path to the file.
|
|||
|
*/
|
|||
|
PreloadFile.prototype.getPath = function () {
|
|||
|
return this._path;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Get the current file position.
|
|||
|
*
|
|||
|
* We emulate the following bug mentioned in the Node documentation:
|
|||
|
* > On Linux, positional writes don't work when the file is opened in append
|
|||
|
* mode. The kernel ignores the position argument and always appends the data
|
|||
|
* to the end of the file.
|
|||
|
* @return [Number] The current file position.
|
|||
|
*/
|
|||
|
PreloadFile.prototype.getPos = function () {
|
|||
|
if (this._flag.isAppendable()) {
|
|||
|
return this._stat.size;
|
|||
|
}
|
|||
|
return this._pos;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Advance the current file position by the indicated number of positions.
|
|||
|
* @param [Number] delta
|
|||
|
*/
|
|||
|
PreloadFile.prototype.advancePos = function (delta) {
|
|||
|
return this._pos += delta;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Set the file position.
|
|||
|
* @param [Number] newPos
|
|||
|
*/
|
|||
|
PreloadFile.prototype.setPos = function (newPos) {
|
|||
|
return this._pos = newPos;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* **Core**: Asynchronous sync. Must be implemented by subclasses of this
|
|||
|
* class.
|
|||
|
* @param [Function(BrowserFS.ApiError)] cb
|
|||
|
*/
|
|||
|
PreloadFile.prototype.sync = function (cb) {
|
|||
|
try {
|
|||
|
this.syncSync();
|
|||
|
cb();
|
|||
|
} catch (e) {
|
|||
|
cb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* **Core**: Synchronous sync.
|
|||
|
*/
|
|||
|
PreloadFile.prototype.syncSync = function () {
|
|||
|
throw new ApiError(14 /* ENOTSUP */);
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* **Core**: Asynchronous close. Must be implemented by subclasses of this
|
|||
|
* class.
|
|||
|
* @param [Function(BrowserFS.ApiError)] cb
|
|||
|
*/
|
|||
|
PreloadFile.prototype.close = function (cb) {
|
|||
|
try {
|
|||
|
this.closeSync();
|
|||
|
cb();
|
|||
|
} catch (e) {
|
|||
|
cb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* **Core**: Synchronous close.
|
|||
|
*/
|
|||
|
PreloadFile.prototype.closeSync = function () {
|
|||
|
throw new ApiError(14 /* ENOTSUP */);
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Asynchronous `stat`.
|
|||
|
* @param [Function(BrowserFS.ApiError, BrowserFS.node.fs.Stats)] cb
|
|||
|
*/
|
|||
|
PreloadFile.prototype.stat = function (cb) {
|
|||
|
try {
|
|||
|
cb(null, this._stat.clone());
|
|||
|
} catch (e) {
|
|||
|
cb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Synchronous `stat`.
|
|||
|
*/
|
|||
|
PreloadFile.prototype.statSync = function () {
|
|||
|
return this._stat.clone();
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Asynchronous truncate.
|
|||
|
* @param [Number] len
|
|||
|
* @param [Function(BrowserFS.ApiError)] cb
|
|||
|
*/
|
|||
|
PreloadFile.prototype.truncate = function (len, cb) {
|
|||
|
try {
|
|||
|
this.truncateSync(len);
|
|||
|
if (this._flag.isSynchronous() && !fs.getRootFS().supportsSynch()) {
|
|||
|
this.sync(cb);
|
|||
|
}
|
|||
|
cb();
|
|||
|
} catch (e) {
|
|||
|
return cb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Synchronous truncate.
|
|||
|
* @param [Number] len
|
|||
|
*/
|
|||
|
PreloadFile.prototype.truncateSync = function (len) {
|
|||
|
if (!this._flag.isWriteable()) {
|
|||
|
throw new ApiError(0 /* EPERM */, 'File not opened with a writeable mode.');
|
|||
|
}
|
|||
|
this._stat.mtime = new Date();
|
|||
|
if (len > this._buffer.length) {
|
|||
|
var buf = new Buffer(len - this._buffer.length);
|
|||
|
buf.fill(0);
|
|||
|
|
|||
|
// Write will set @_stat.size for us.
|
|||
|
this.writeSync(buf, 0, buf.length, this._buffer.length);
|
|||
|
if (this._flag.isSynchronous() && fs.getRootFS().supportsSynch()) {
|
|||
|
this.syncSync();
|
|||
|
}
|
|||
|
return;
|
|||
|
}
|
|||
|
this._stat.size = len;
|
|||
|
|
|||
|
// Truncate buffer to 'len'.
|
|||
|
var newBuff = new Buffer(len);
|
|||
|
this._buffer.copy(newBuff, 0, 0, len);
|
|||
|
this._buffer = newBuff;
|
|||
|
if (this._flag.isSynchronous() && fs.getRootFS().supportsSynch()) {
|
|||
|
this.syncSync();
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Write buffer to the file.
|
|||
|
* Note that it is unsafe to use fs.write multiple times on the same file
|
|||
|
* without waiting for the callback.
|
|||
|
* @param [BrowserFS.node.Buffer] buffer Buffer containing the data to write to
|
|||
|
* the file.
|
|||
|
* @param [Number] offset Offset in the buffer to start reading data from.
|
|||
|
* @param [Number] length The amount of bytes to write to the file.
|
|||
|
* @param [Number] position Offset from the beginning of the file where this
|
|||
|
* data should be written. If position is null, the data will be written at
|
|||
|
* the current position.
|
|||
|
* @param [Function(BrowserFS.ApiError, Number, BrowserFS.node.Buffer)]
|
|||
|
* cb The number specifies the number of bytes written into the file.
|
|||
|
*/
|
|||
|
PreloadFile.prototype.write = function (buffer, offset, length, position, cb) {
|
|||
|
try {
|
|||
|
cb(null, this.writeSync(buffer, offset, length, position), buffer);
|
|||
|
} catch (e) {
|
|||
|
cb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Write buffer to the file.
|
|||
|
* Note that it is unsafe to use fs.writeSync multiple times on the same file
|
|||
|
* without waiting for the callback.
|
|||
|
* @param [BrowserFS.node.Buffer] buffer Buffer containing the data to write to
|
|||
|
* the file.
|
|||
|
* @param [Number] offset Offset in the buffer to start reading data from.
|
|||
|
* @param [Number] length The amount of bytes to write to the file.
|
|||
|
* @param [Number] position Offset from the beginning of the file where this
|
|||
|
* data should be written. If position is null, the data will be written at
|
|||
|
* the current position.
|
|||
|
* @return [Number]
|
|||
|
*/
|
|||
|
PreloadFile.prototype.writeSync = function (buffer, offset, length, position) {
|
|||
|
if (position == null) {
|
|||
|
position = this.getPos();
|
|||
|
}
|
|||
|
if (!this._flag.isWriteable()) {
|
|||
|
throw new ApiError(0 /* EPERM */, 'File not opened with a writeable mode.');
|
|||
|
}
|
|||
|
var endFp = position + length;
|
|||
|
if (endFp > this._stat.size) {
|
|||
|
this._stat.size = endFp;
|
|||
|
if (endFp > this._buffer.length) {
|
|||
|
// Extend the buffer!
|
|||
|
var newBuff = new Buffer(endFp);
|
|||
|
this._buffer.copy(newBuff);
|
|||
|
this._buffer = newBuff;
|
|||
|
}
|
|||
|
}
|
|||
|
var len = buffer.copy(this._buffer, position, offset, offset + length);
|
|||
|
this._stat.mtime = new Date();
|
|||
|
if (this._flag.isSynchronous()) {
|
|||
|
this.syncSync();
|
|||
|
return len;
|
|||
|
}
|
|||
|
this.setPos(position + len);
|
|||
|
return len;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Read data from the file.
|
|||
|
* @param [BrowserFS.node.Buffer] buffer The buffer that the data will be
|
|||
|
* written to.
|
|||
|
* @param [Number] offset The offset within the buffer where writing will
|
|||
|
* start.
|
|||
|
* @param [Number] length An integer specifying the number of bytes to read.
|
|||
|
* @param [Number] position An integer specifying where to begin reading from
|
|||
|
* in the file. If position is null, data will be read from the current file
|
|||
|
* position.
|
|||
|
* @param [Function(BrowserFS.ApiError, Number, BrowserFS.node.Buffer)] cb The
|
|||
|
* number is the number of bytes read
|
|||
|
*/
|
|||
|
PreloadFile.prototype.read = function (buffer, offset, length, position, cb) {
|
|||
|
try {
|
|||
|
cb(null, this.readSync(buffer, offset, length, position), buffer);
|
|||
|
} catch (e) {
|
|||
|
cb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Read data from the file.
|
|||
|
* @param [BrowserFS.node.Buffer] buffer The buffer that the data will be
|
|||
|
* written to.
|
|||
|
* @param [Number] offset The offset within the buffer where writing will
|
|||
|
* start.
|
|||
|
* @param [Number] length An integer specifying the number of bytes to read.
|
|||
|
* @param [Number] position An integer specifying where to begin reading from
|
|||
|
* in the file. If position is null, data will be read from the current file
|
|||
|
* position.
|
|||
|
* @return [Number]
|
|||
|
*/
|
|||
|
PreloadFile.prototype.readSync = function (buffer, offset, length, position) {
|
|||
|
if (!this._flag.isReadable()) {
|
|||
|
throw new ApiError(0 /* EPERM */, 'File not opened with a readable mode.');
|
|||
|
}
|
|||
|
if (position == null) {
|
|||
|
position = this.getPos();
|
|||
|
}
|
|||
|
var endRead = position + length;
|
|||
|
if (endRead > this._stat.size) {
|
|||
|
length = this._stat.size - position;
|
|||
|
}
|
|||
|
var rv = this._buffer.copy(buffer, offset, position, position + length);
|
|||
|
this._stat.atime = new Date();
|
|||
|
this._pos = position + length;
|
|||
|
return rv;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Asynchronous `fchmod`.
|
|||
|
* @param [Number|String] mode
|
|||
|
* @param [Function(BrowserFS.ApiError)] cb
|
|||
|
*/
|
|||
|
PreloadFile.prototype.chmod = function (mode, cb) {
|
|||
|
try {
|
|||
|
this.chmodSync(mode);
|
|||
|
cb();
|
|||
|
} catch (e) {
|
|||
|
cb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Asynchronous `fchmod`.
|
|||
|
* @param [Number] mode
|
|||
|
*/
|
|||
|
PreloadFile.prototype.chmodSync = function (mode) {
|
|||
|
if (!this._fs.supportsProps()) {
|
|||
|
throw new ApiError(14 /* ENOTSUP */);
|
|||
|
}
|
|||
|
this._stat.chmod(mode);
|
|||
|
this.syncSync();
|
|||
|
};
|
|||
|
return PreloadFile;
|
|||
|
})(file.BaseFile);
|
|||
|
exports.PreloadFile = PreloadFile;
|
|||
|
|
|||
|
/**
|
|||
|
* File class for the InMemory and XHR file systems.
|
|||
|
* Doesn't sync to anything, so it works nicely for memory-only files.
|
|||
|
*/
|
|||
|
var NoSyncFile = (function (_super) {
|
|||
|
__extends(NoSyncFile, _super);
|
|||
|
function NoSyncFile(_fs, _path, _flag, _stat, contents) {
|
|||
|
_super.call(this, _fs, _path, _flag, _stat, contents);
|
|||
|
}
|
|||
|
/**
|
|||
|
* Asynchronous sync. Doesn't do anything, simply calls the cb.
|
|||
|
* @param [Function(BrowserFS.ApiError)] cb
|
|||
|
*/
|
|||
|
NoSyncFile.prototype.sync = function (cb) {
|
|||
|
cb();
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Synchronous sync. Doesn't do anything.
|
|||
|
*/
|
|||
|
NoSyncFile.prototype.syncSync = function () {
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Asynchronous close. Doesn't do anything, simply calls the cb.
|
|||
|
* @param [Function(BrowserFS.ApiError)] cb
|
|||
|
*/
|
|||
|
NoSyncFile.prototype.close = function (cb) {
|
|||
|
cb();
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Synchronous close. Doesn't do anything.
|
|||
|
*/
|
|||
|
NoSyncFile.prototype.closeSync = function () {
|
|||
|
};
|
|||
|
return NoSyncFile;
|
|||
|
})(PreloadFile);
|
|||
|
exports.NoSyncFile = NoSyncFile;
|
|||
|
});
|
|||
|
//# sourceMappingURL=preload_file.js.map
|
|||
|
;
|
|||
|
var __extends = this.__extends || function (d, b) {
|
|||
|
for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p];
|
|||
|
function __() { this.constructor = d; }
|
|||
|
__.prototype = b.prototype;
|
|||
|
d.prototype = new __();
|
|||
|
};
|
|||
|
define('generic/key_value_filesystem',["require", "exports", '../core/file_system', '../core/api_error', '../core/node_fs_stats', '../core/node_path', '../generic/inode', '../core/buffer', '../generic/preload_file'], function(require, exports, file_system, api_error, node_fs_stats, node_path, Inode, buffer, preload_file) {
|
|||
|
var ROOT_NODE_ID = "/", path = node_path.path, ApiError = api_error.ApiError, Buffer = buffer.Buffer;
|
|||
|
|
|||
|
/**
|
|||
|
* Generates a random ID.
|
|||
|
*/
|
|||
|
function GenerateRandomID() {
|
|||
|
// From http://stackoverflow.com/questions/105034/how-to-create-a-guid-uuid-in-javascript
|
|||
|
return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, function (c) {
|
|||
|
var r = Math.random() * 16 | 0, v = c == 'x' ? r : (r & 0x3 | 0x8);
|
|||
|
return v.toString(16);
|
|||
|
});
|
|||
|
}
|
|||
|
|
|||
|
/**
|
|||
|
* Helper function. Checks if 'e' is defined. If so, it triggers the callback
|
|||
|
* with 'e' and returns false. Otherwise, returns true.
|
|||
|
*/
|
|||
|
function noError(e, cb) {
|
|||
|
if (e) {
|
|||
|
cb(e);
|
|||
|
return false;
|
|||
|
}
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
/**
|
|||
|
* Helper function. Checks if 'e' is defined. If so, it aborts the transaction,
|
|||
|
* triggers the callback with 'e', and returns false. Otherwise, returns true.
|
|||
|
*/
|
|||
|
function noErrorTx(e, tx, cb) {
|
|||
|
if (e) {
|
|||
|
tx.abort(function () {
|
|||
|
cb(e);
|
|||
|
});
|
|||
|
return false;
|
|||
|
}
|
|||
|
return true;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
|
|||
|
|
|||
|
|
|||
|
|
|||
|
|
|||
|
|
|||
|
|
|||
|
/**
|
|||
|
* A simple RW transaction for simple synchronous key-value stores.
|
|||
|
*/
|
|||
|
var SimpleSyncRWTransaction = (function () {
|
|||
|
function SimpleSyncRWTransaction(store) {
|
|||
|
this.store = store;
|
|||
|
/**
|
|||
|
* Stores data in the keys we modify prior to modifying them.
|
|||
|
* Allows us to roll back commits.
|
|||
|
*/
|
|||
|
this.originalData = {};
|
|||
|
/**
|
|||
|
* List of keys modified in this transaction, if any.
|
|||
|
*/
|
|||
|
this.modifiedKeys = [];
|
|||
|
}
|
|||
|
/**
|
|||
|
* Stashes given key value pair into `originalData` if it doesn't already
|
|||
|
* exist. Allows us to stash values the program is requesting anyway to
|
|||
|
* prevent needless `get` requests if the program modifies the data later
|
|||
|
* on during the transaction.
|
|||
|
*/
|
|||
|
SimpleSyncRWTransaction.prototype.stashOldValue = function (key, value) {
|
|||
|
// Keep only the earliest value in the transaction.
|
|||
|
if (!this.originalData.hasOwnProperty(key)) {
|
|||
|
this.originalData[key] = value;
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Marks the given key as modified, and stashes its value if it has not been
|
|||
|
* stashed already.
|
|||
|
*/
|
|||
|
SimpleSyncRWTransaction.prototype.markModified = function (key) {
|
|||
|
if (this.modifiedKeys.indexOf(key) === -1) {
|
|||
|
this.modifiedKeys.push(key);
|
|||
|
if (!this.originalData.hasOwnProperty(key)) {
|
|||
|
this.originalData[key] = this.store.get(key);
|
|||
|
}
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
SimpleSyncRWTransaction.prototype.get = function (key) {
|
|||
|
var val = this.store.get(key);
|
|||
|
this.stashOldValue(key, val);
|
|||
|
return val;
|
|||
|
};
|
|||
|
|
|||
|
SimpleSyncRWTransaction.prototype.put = function (key, data, overwrite) {
|
|||
|
this.markModified(key);
|
|||
|
return this.store.put(key, data, overwrite);
|
|||
|
};
|
|||
|
|
|||
|
SimpleSyncRWTransaction.prototype.delete = function (key) {
|
|||
|
this.markModified(key);
|
|||
|
this.store.delete(key);
|
|||
|
};
|
|||
|
|
|||
|
SimpleSyncRWTransaction.prototype.commit = function () {
|
|||
|
};
|
|||
|
SimpleSyncRWTransaction.prototype.abort = function () {
|
|||
|
// Rollback old values.
|
|||
|
var i, key, value;
|
|||
|
for (i = 0; i < this.modifiedKeys.length; i++) {
|
|||
|
key = this.modifiedKeys[i];
|
|||
|
value = this.originalData[key];
|
|||
|
if (value === null) {
|
|||
|
// Key didn't exist.
|
|||
|
this.store.delete(key);
|
|||
|
} else {
|
|||
|
// Key existed. Store old value.
|
|||
|
this.store.put(key, value, true);
|
|||
|
}
|
|||
|
}
|
|||
|
};
|
|||
|
return SimpleSyncRWTransaction;
|
|||
|
})();
|
|||
|
exports.SimpleSyncRWTransaction = SimpleSyncRWTransaction;
|
|||
|
|
|||
|
var SyncKeyValueFile = (function (_super) {
|
|||
|
__extends(SyncKeyValueFile, _super);
|
|||
|
function SyncKeyValueFile(_fs, _path, _flag, _stat, contents) {
|
|||
|
_super.call(this, _fs, _path, _flag, _stat, contents);
|
|||
|
}
|
|||
|
SyncKeyValueFile.prototype.syncSync = function () {
|
|||
|
this._fs._syncSync(this._path, this._buffer, this._stat);
|
|||
|
};
|
|||
|
|
|||
|
SyncKeyValueFile.prototype.closeSync = function () {
|
|||
|
this.syncSync();
|
|||
|
};
|
|||
|
return SyncKeyValueFile;
|
|||
|
})(preload_file.PreloadFile);
|
|||
|
exports.SyncKeyValueFile = SyncKeyValueFile;
|
|||
|
|
|||
|
/**
|
|||
|
* A "Synchronous key-value file system". Stores data to/retrieves data from an
|
|||
|
* underlying key-value store.
|
|||
|
*
|
|||
|
* We use a unique ID for each node in the file system. The root node has a
|
|||
|
* fixed ID.
|
|||
|
* @todo Introduce Node ID caching.
|
|||
|
* @todo Check modes.
|
|||
|
*/
|
|||
|
var SyncKeyValueFileSystem = (function (_super) {
|
|||
|
__extends(SyncKeyValueFileSystem, _super);
|
|||
|
function SyncKeyValueFileSystem(options) {
|
|||
|
_super.call(this);
|
|||
|
this.store = options.store;
|
|||
|
|
|||
|
// INVARIANT: Ensure that the root exists.
|
|||
|
this.makeRootDirectory();
|
|||
|
}
|
|||
|
SyncKeyValueFileSystem.isAvailable = function () {
|
|||
|
return true;
|
|||
|
};
|
|||
|
SyncKeyValueFileSystem.prototype.getName = function () {
|
|||
|
return this.store.name();
|
|||
|
};
|
|||
|
SyncKeyValueFileSystem.prototype.isReadOnly = function () {
|
|||
|
return false;
|
|||
|
};
|
|||
|
SyncKeyValueFileSystem.prototype.supportsSymlinks = function () {
|
|||
|
return false;
|
|||
|
};
|
|||
|
SyncKeyValueFileSystem.prototype.supportsProps = function () {
|
|||
|
return false;
|
|||
|
};
|
|||
|
SyncKeyValueFileSystem.prototype.supportsSynch = function () {
|
|||
|
return true;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Checks if the root directory exists. Creates it if it doesn't.
|
|||
|
*/
|
|||
|
SyncKeyValueFileSystem.prototype.makeRootDirectory = function () {
|
|||
|
var tx = this.store.beginTransaction('readwrite');
|
|||
|
if (tx.get(ROOT_NODE_ID) === undefined) {
|
|||
|
// Create new inode.
|
|||
|
var currTime = (new Date()).getTime(), dirInode = new Inode(GenerateRandomID(), 4096, 511 | 16384 /* DIRECTORY */, currTime, currTime, currTime);
|
|||
|
|
|||
|
// If the root doesn't exist, the first random ID shouldn't exist,
|
|||
|
// either.
|
|||
|
tx.put(dirInode.id, new Buffer("{}"), false);
|
|||
|
tx.put(ROOT_NODE_ID, dirInode.toBuffer(), false);
|
|||
|
tx.commit();
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Helper function for findINode.
|
|||
|
* @param parent The parent directory of the file we are attempting to find.
|
|||
|
* @param filename The filename of the inode we are attempting to find, minus
|
|||
|
* the parent.
|
|||
|
* @return string The ID of the file's inode in the file system.
|
|||
|
*/
|
|||
|
SyncKeyValueFileSystem.prototype._findINode = function (tx, parent, filename) {
|
|||
|
var _this = this;
|
|||
|
var read_directory = function (inode) {
|
|||
|
// Get the root's directory listing.
|
|||
|
var dirList = _this.getDirListing(tx, parent, inode);
|
|||
|
|
|||
|
// Get the file's ID.
|
|||
|
if (dirList[filename]) {
|
|||
|
return dirList[filename];
|
|||
|
} else {
|
|||
|
throw ApiError.ENOENT(path.resolve(parent, filename));
|
|||
|
}
|
|||
|
};
|
|||
|
if (parent === '/') {
|
|||
|
if (filename === '') {
|
|||
|
// BASE CASE #1: Return the root's ID.
|
|||
|
return ROOT_NODE_ID;
|
|||
|
} else {
|
|||
|
// BASE CASE #2: Find the item in the root ndoe.
|
|||
|
return read_directory(this.getINode(tx, parent, ROOT_NODE_ID));
|
|||
|
}
|
|||
|
} else {
|
|||
|
return read_directory(this.getINode(tx, parent + path.sep + filename, this._findINode(tx, path.dirname(parent), path.basename(parent))));
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Finds the Inode of the given path.
|
|||
|
* @param p The path to look up.
|
|||
|
* @return The Inode of the path p.
|
|||
|
* @todo memoize/cache
|
|||
|
*/
|
|||
|
SyncKeyValueFileSystem.prototype.findINode = function (tx, p) {
|
|||
|
return this.getINode(tx, p, this._findINode(tx, path.dirname(p), path.basename(p)));
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Given the ID of a node, retrieves the corresponding Inode.
|
|||
|
* @param tx The transaction to use.
|
|||
|
* @param p The corresponding path to the file (used for error messages).
|
|||
|
* @param id The ID to look up.
|
|||
|
*/
|
|||
|
SyncKeyValueFileSystem.prototype.getINode = function (tx, p, id) {
|
|||
|
var inode = tx.get(id);
|
|||
|
if (inode === undefined) {
|
|||
|
throw ApiError.ENOENT(p);
|
|||
|
}
|
|||
|
return Inode.fromBuffer(inode);
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Given the Inode of a directory, retrieves the corresponding directory
|
|||
|
* listing.
|
|||
|
*/
|
|||
|
SyncKeyValueFileSystem.prototype.getDirListing = function (tx, p, inode) {
|
|||
|
if (!inode.isDirectory()) {
|
|||
|
throw ApiError.ENOTDIR(p);
|
|||
|
}
|
|||
|
var data = tx.get(inode.id);
|
|||
|
if (data === undefined) {
|
|||
|
throw ApiError.ENOENT(p);
|
|||
|
}
|
|||
|
return JSON.parse(data.toString());
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Creates a new node under a random ID. Retries 5 times before giving up in
|
|||
|
* the exceedingly unlikely chance that we try to reuse a random GUID.
|
|||
|
* @return The GUID that the data was stored under.
|
|||
|
*/
|
|||
|
SyncKeyValueFileSystem.prototype.addNewNode = function (tx, data) {
|
|||
|
var retries = 0, currId;
|
|||
|
while (retries < 5) {
|
|||
|
try {
|
|||
|
currId = GenerateRandomID();
|
|||
|
tx.put(currId, data, false);
|
|||
|
return currId;
|
|||
|
} catch (e) {
|
|||
|
// Ignore and reroll.
|
|||
|
}
|
|||
|
}
|
|||
|
throw new ApiError(2 /* EIO */, 'Unable to commit data to key-value store.');
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Commits a new file (well, a FILE or a DIRECTORY) to the file system with
|
|||
|
* the given mode.
|
|||
|
* Note: This will commit the transaction.
|
|||
|
* @param p The path to the new file.
|
|||
|
* @param type The type of the new file.
|
|||
|
* @param mode The mode to create the new file with.
|
|||
|
* @param data The data to store at the file's data node.
|
|||
|
* @return The Inode for the new file.
|
|||
|
*/
|
|||
|
SyncKeyValueFileSystem.prototype.commitNewFile = function (tx, p, type, mode, data) {
|
|||
|
var parentDir = path.dirname(p), fname = path.basename(p), parentNode = this.findINode(tx, parentDir), dirListing = this.getDirListing(tx, parentDir, parentNode), currTime = (new Date()).getTime();
|
|||
|
|
|||
|
// Invariant: The root always exists.
|
|||
|
// If we don't check this prior to taking steps below, we will create a
|
|||
|
// file with name '' in root should p == '/'.
|
|||
|
if (p === '/') {
|
|||
|
throw ApiError.EEXIST(p);
|
|||
|
}
|
|||
|
|
|||
|
// Check if file already exists.
|
|||
|
if (dirListing[fname]) {
|
|||
|
throw ApiError.EEXIST(p);
|
|||
|
}
|
|||
|
|
|||
|
try {
|
|||
|
// Commit data.
|
|||
|
var dataId = this.addNewNode(tx, data), fileNode = new Inode(dataId, data.length, mode | type, currTime, currTime, currTime), fileNodeId = this.addNewNode(tx, fileNode.toBuffer());
|
|||
|
|
|||
|
// Update and commit parent directory listing.
|
|||
|
dirListing[fname] = fileNodeId;
|
|||
|
tx.put(parentNode.id, new Buffer(JSON.stringify(dirListing)), true);
|
|||
|
} catch (e) {
|
|||
|
tx.abort();
|
|||
|
throw e;
|
|||
|
}
|
|||
|
tx.commit();
|
|||
|
return fileNode;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Delete all contents stored in the file system.
|
|||
|
*/
|
|||
|
SyncKeyValueFileSystem.prototype.empty = function () {
|
|||
|
this.store.clear();
|
|||
|
|
|||
|
// INVARIANT: Root always exists.
|
|||
|
this.makeRootDirectory();
|
|||
|
};
|
|||
|
|
|||
|
SyncKeyValueFileSystem.prototype.renameSync = function (oldPath, newPath) {
|
|||
|
var tx = this.store.beginTransaction('readwrite'), oldParent = path.dirname(oldPath), oldName = path.basename(oldPath), newParent = path.dirname(newPath), newName = path.basename(newPath), oldDirNode = this.findINode(tx, oldParent), oldDirList = this.getDirListing(tx, oldParent, oldDirNode);
|
|||
|
if (!oldDirList[oldName]) {
|
|||
|
throw ApiError.ENOENT(oldPath);
|
|||
|
}
|
|||
|
var nodeId = oldDirList[oldName];
|
|||
|
delete oldDirList[oldName];
|
|||
|
|
|||
|
// Invariant: Can't move a folder inside itself.
|
|||
|
// This funny little hack ensures that the check passes only if oldPath
|
|||
|
// is a subpath of newParent. We append '/' to avoid matching folders that
|
|||
|
// are a substring of the bottom-most folder in the path.
|
|||
|
if ((newParent + '/').indexOf(oldPath + '/') === 0) {
|
|||
|
throw new ApiError(5 /* EBUSY */, oldParent);
|
|||
|
}
|
|||
|
|
|||
|
// Add newPath to parent's directory listing.
|
|||
|
var newDirNode, newDirList;
|
|||
|
if (newParent === oldParent) {
|
|||
|
// Prevent us from re-grabbing the same directory listing, which still
|
|||
|
// contains oldName.
|
|||
|
newDirNode = oldDirNode;
|
|||
|
newDirList = oldDirList;
|
|||
|
} else {
|
|||
|
newDirNode = this.findINode(tx, newParent);
|
|||
|
newDirList = this.getDirListing(tx, newParent, newDirNode);
|
|||
|
}
|
|||
|
|
|||
|
if (newDirList[newName]) {
|
|||
|
// If it's a file, delete it.
|
|||
|
var newNameNode = this.getINode(tx, newPath, newDirList[newName]);
|
|||
|
if (newNameNode.isFile()) {
|
|||
|
try {
|
|||
|
tx.delete(newNameNode.id);
|
|||
|
tx.delete(newDirList[newName]);
|
|||
|
} catch (e) {
|
|||
|
tx.abort();
|
|||
|
throw e;
|
|||
|
}
|
|||
|
} else {
|
|||
|
throw ApiError.EPERM(newPath);
|
|||
|
}
|
|||
|
}
|
|||
|
newDirList[newName] = nodeId;
|
|||
|
|
|||
|
try {
|
|||
|
tx.put(oldDirNode.id, new Buffer(JSON.stringify(oldDirList)), true);
|
|||
|
tx.put(newDirNode.id, new Buffer(JSON.stringify(newDirList)), true);
|
|||
|
} catch (e) {
|
|||
|
tx.abort();
|
|||
|
throw e;
|
|||
|
}
|
|||
|
|
|||
|
tx.commit();
|
|||
|
};
|
|||
|
|
|||
|
SyncKeyValueFileSystem.prototype.statSync = function (p, isLstat) {
|
|||
|
// Get the inode to the item, convert it into a Stats object.
|
|||
|
return this.findINode(this.store.beginTransaction('readonly'), p).toStats();
|
|||
|
};
|
|||
|
|
|||
|
SyncKeyValueFileSystem.prototype.createFileSync = function (p, flag, mode) {
|
|||
|
var tx = this.store.beginTransaction('readwrite'), data = new Buffer(0), newFile = this.commitNewFile(tx, p, 32768 /* FILE */, mode, data);
|
|||
|
|
|||
|
// Open the file.
|
|||
|
return new SyncKeyValueFile(this, p, flag, newFile.toStats(), data);
|
|||
|
};
|
|||
|
|
|||
|
SyncKeyValueFileSystem.prototype.openFileSync = function (p, flag) {
|
|||
|
var tx = this.store.beginTransaction('readonly'), node = this.findINode(tx, p), data = tx.get(node.id);
|
|||
|
if (data === undefined) {
|
|||
|
throw ApiError.ENOENT(p);
|
|||
|
}
|
|||
|
return new SyncKeyValueFile(this, p, flag, node.toStats(), data);
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Remove all traces of the given path from the file system.
|
|||
|
* @param p The path to remove from the file system.
|
|||
|
* @param isDir Does the path belong to a directory, or a file?
|
|||
|
* @todo Update mtime.
|
|||
|
*/
|
|||
|
SyncKeyValueFileSystem.prototype.removeEntry = function (p, isDir) {
|
|||
|
var tx = this.store.beginTransaction('readwrite'), parent = path.dirname(p), parentNode = this.findINode(tx, parent), parentListing = this.getDirListing(tx, parent, parentNode), fileName = path.basename(p);
|
|||
|
|
|||
|
if (!parentListing[fileName]) {
|
|||
|
throw ApiError.ENOENT(p);
|
|||
|
}
|
|||
|
|
|||
|
// Remove from directory listing of parent.
|
|||
|
var fileNodeId = parentListing[fileName];
|
|||
|
delete parentListing[fileName];
|
|||
|
|
|||
|
// Get file inode.
|
|||
|
var fileNode = this.getINode(tx, p, fileNodeId);
|
|||
|
if (!isDir && fileNode.isDirectory()) {
|
|||
|
throw ApiError.EISDIR(p);
|
|||
|
} else if (isDir && !fileNode.isDirectory()) {
|
|||
|
throw ApiError.ENOTDIR(p);
|
|||
|
}
|
|||
|
|
|||
|
try {
|
|||
|
// Delete data.
|
|||
|
tx.delete(fileNode.id);
|
|||
|
|
|||
|
// Delete node.
|
|||
|
tx.delete(fileNodeId);
|
|||
|
|
|||
|
// Update directory listing.
|
|||
|
tx.put(parentNode.id, new Buffer(JSON.stringify(parentListing)), true);
|
|||
|
} catch (e) {
|
|||
|
tx.abort();
|
|||
|
throw e;
|
|||
|
}
|
|||
|
|
|||
|
// Success.
|
|||
|
tx.commit();
|
|||
|
};
|
|||
|
|
|||
|
SyncKeyValueFileSystem.prototype.unlinkSync = function (p) {
|
|||
|
this.removeEntry(p, false);
|
|||
|
};
|
|||
|
|
|||
|
SyncKeyValueFileSystem.prototype.rmdirSync = function (p) {
|
|||
|
this.removeEntry(p, true);
|
|||
|
};
|
|||
|
|
|||
|
SyncKeyValueFileSystem.prototype.mkdirSync = function (p, mode) {
|
|||
|
var tx = this.store.beginTransaction('readwrite'), data = new Buffer('{}');
|
|||
|
this.commitNewFile(tx, p, 16384 /* DIRECTORY */, mode, data);
|
|||
|
};
|
|||
|
|
|||
|
SyncKeyValueFileSystem.prototype.readdirSync = function (p) {
|
|||
|
var tx = this.store.beginTransaction('readonly');
|
|||
|
return Object.keys(this.getDirListing(tx, p, this.findINode(tx, p)));
|
|||
|
};
|
|||
|
|
|||
|
SyncKeyValueFileSystem.prototype._syncSync = function (p, data, stats) {
|
|||
|
// @todo Ensure mtime updates properly, and use that to determine if a data
|
|||
|
// update is required.
|
|||
|
var tx = this.store.beginTransaction('readwrite'), fileInodeId = this._findINode(tx, path.dirname(p), path.basename(p)), fileInode = this.getINode(tx, p, fileInodeId), inodeChanged = fileInode.update(stats);
|
|||
|
|
|||
|
try {
|
|||
|
// Sync data.
|
|||
|
tx.put(fileInode.id, data, true);
|
|||
|
|
|||
|
// Sync metadata.
|
|||
|
if (inodeChanged) {
|
|||
|
tx.put(fileInodeId, fileInode.toBuffer(), true);
|
|||
|
}
|
|||
|
} catch (e) {
|
|||
|
tx.abort();
|
|||
|
throw e;
|
|||
|
}
|
|||
|
tx.commit();
|
|||
|
};
|
|||
|
return SyncKeyValueFileSystem;
|
|||
|
})(file_system.SynchronousFileSystem);
|
|||
|
exports.SyncKeyValueFileSystem = SyncKeyValueFileSystem;
|
|||
|
|
|||
|
|
|||
|
|
|||
|
|
|||
|
|
|||
|
|
|||
|
|
|||
|
var AsyncKeyValueFile = (function (_super) {
|
|||
|
__extends(AsyncKeyValueFile, _super);
|
|||
|
function AsyncKeyValueFile(_fs, _path, _flag, _stat, contents) {
|
|||
|
_super.call(this, _fs, _path, _flag, _stat, contents);
|
|||
|
}
|
|||
|
AsyncKeyValueFile.prototype.sync = function (cb) {
|
|||
|
this._fs._sync(this._path, this._buffer, this._stat, cb);
|
|||
|
};
|
|||
|
|
|||
|
AsyncKeyValueFile.prototype.close = function (cb) {
|
|||
|
this.sync(cb);
|
|||
|
};
|
|||
|
return AsyncKeyValueFile;
|
|||
|
})(preload_file.PreloadFile);
|
|||
|
exports.AsyncKeyValueFile = AsyncKeyValueFile;
|
|||
|
|
|||
|
/**
|
|||
|
* An "Asynchronous key-value file system". Stores data to/retrieves data from
|
|||
|
* an underlying asynchronous key-value store.
|
|||
|
*/
|
|||
|
var AsyncKeyValueFileSystem = (function (_super) {
|
|||
|
__extends(AsyncKeyValueFileSystem, _super);
|
|||
|
function AsyncKeyValueFileSystem() {
|
|||
|
_super.apply(this, arguments);
|
|||
|
}
|
|||
|
/**
|
|||
|
* Initializes the file system. Typically called by subclasses' async
|
|||
|
* constructors.
|
|||
|
*/
|
|||
|
AsyncKeyValueFileSystem.prototype.init = function (store, cb) {
|
|||
|
this.store = store;
|
|||
|
|
|||
|
// INVARIANT: Ensure that the root exists.
|
|||
|
this.makeRootDirectory(cb);
|
|||
|
};
|
|||
|
|
|||
|
AsyncKeyValueFileSystem.isAvailable = function () {
|
|||
|
return true;
|
|||
|
};
|
|||
|
AsyncKeyValueFileSystem.prototype.getName = function () {
|
|||
|
return this.store.name();
|
|||
|
};
|
|||
|
AsyncKeyValueFileSystem.prototype.isReadOnly = function () {
|
|||
|
return false;
|
|||
|
};
|
|||
|
AsyncKeyValueFileSystem.prototype.supportsSymlinks = function () {
|
|||
|
return false;
|
|||
|
};
|
|||
|
AsyncKeyValueFileSystem.prototype.supportsProps = function () {
|
|||
|
return false;
|
|||
|
};
|
|||
|
AsyncKeyValueFileSystem.prototype.supportsSynch = function () {
|
|||
|
return false;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Checks if the root directory exists. Creates it if it doesn't.
|
|||
|
*/
|
|||
|
AsyncKeyValueFileSystem.prototype.makeRootDirectory = function (cb) {
|
|||
|
var tx = this.store.beginTransaction('readwrite');
|
|||
|
tx.get(ROOT_NODE_ID, function (e, data) {
|
|||
|
if (e || data === undefined) {
|
|||
|
// Create new inode.
|
|||
|
var currTime = (new Date()).getTime(), dirInode = new Inode(GenerateRandomID(), 4096, 511 | 16384 /* DIRECTORY */, currTime, currTime, currTime);
|
|||
|
|
|||
|
// If the root doesn't exist, the first random ID shouldn't exist,
|
|||
|
// either.
|
|||
|
tx.put(dirInode.id, new Buffer("{}"), false, function (e) {
|
|||
|
if (noErrorTx(e, tx, cb)) {
|
|||
|
tx.put(ROOT_NODE_ID, dirInode.toBuffer(), false, function (e) {
|
|||
|
if (e) {
|
|||
|
tx.abort(function () {
|
|||
|
cb(e);
|
|||
|
});
|
|||
|
} else {
|
|||
|
tx.commit(cb);
|
|||
|
}
|
|||
|
});
|
|||
|
}
|
|||
|
});
|
|||
|
} else {
|
|||
|
// We're good.
|
|||
|
tx.commit(cb);
|
|||
|
}
|
|||
|
});
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Helper function for findINode.
|
|||
|
* @param parent The parent directory of the file we are attempting to find.
|
|||
|
* @param filename The filename of the inode we are attempting to find, minus
|
|||
|
* the parent.
|
|||
|
* @param cb Passed an error or the ID of the file's inode in the file system.
|
|||
|
*/
|
|||
|
AsyncKeyValueFileSystem.prototype._findINode = function (tx, parent, filename, cb) {
|
|||
|
var _this = this;
|
|||
|
var handle_directory_listings = function (e, inode, dirList) {
|
|||
|
if (e) {
|
|||
|
cb(e);
|
|||
|
} else if (dirList[filename]) {
|
|||
|
cb(null, dirList[filename]);
|
|||
|
} else {
|
|||
|
cb(ApiError.ENOENT(path.resolve(parent, filename)));
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
if (parent === '/') {
|
|||
|
if (filename === '') {
|
|||
|
// BASE CASE #1: Return the root's ID.
|
|||
|
cb(null, ROOT_NODE_ID);
|
|||
|
} else {
|
|||
|
// BASE CASE #2: Find the item in the root node.
|
|||
|
this.getINode(tx, parent, ROOT_NODE_ID, function (e, inode) {
|
|||
|
if (noError(e, cb)) {
|
|||
|
_this.getDirListing(tx, parent, inode, function (e, dirList) {
|
|||
|
// handle_directory_listings will handle e for us.
|
|||
|
handle_directory_listings(e, inode, dirList);
|
|||
|
});
|
|||
|
}
|
|||
|
});
|
|||
|
}
|
|||
|
} else {
|
|||
|
// Get the parent directory's INode, and find the file in its directory
|
|||
|
// listing.
|
|||
|
this.findINodeAndDirListing(tx, parent, handle_directory_listings);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Finds the Inode of the given path.
|
|||
|
* @param p The path to look up.
|
|||
|
* @param cb Passed an error or the Inode of the path p.
|
|||
|
* @todo memoize/cache
|
|||
|
*/
|
|||
|
AsyncKeyValueFileSystem.prototype.findINode = function (tx, p, cb) {
|
|||
|
var _this = this;
|
|||
|
this._findINode(tx, path.dirname(p), path.basename(p), function (e, id) {
|
|||
|
if (noError(e, cb)) {
|
|||
|
_this.getINode(tx, p, id, cb);
|
|||
|
}
|
|||
|
});
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Given the ID of a node, retrieves the corresponding Inode.
|
|||
|
* @param tx The transaction to use.
|
|||
|
* @param p The corresponding path to the file (used for error messages).
|
|||
|
* @param id The ID to look up.
|
|||
|
* @param cb Passed an error or the inode under the given id.
|
|||
|
*/
|
|||
|
AsyncKeyValueFileSystem.prototype.getINode = function (tx, p, id, cb) {
|
|||
|
tx.get(id, function (e, data) {
|
|||
|
if (noError(e, cb)) {
|
|||
|
if (data === undefined) {
|
|||
|
cb(ApiError.ENOENT(p));
|
|||
|
} else {
|
|||
|
cb(null, Inode.fromBuffer(data));
|
|||
|
}
|
|||
|
}
|
|||
|
});
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Given the Inode of a directory, retrieves the corresponding directory
|
|||
|
* listing.
|
|||
|
*/
|
|||
|
AsyncKeyValueFileSystem.prototype.getDirListing = function (tx, p, inode, cb) {
|
|||
|
if (!inode.isDirectory()) {
|
|||
|
cb(ApiError.ENOTDIR(p));
|
|||
|
} else {
|
|||
|
tx.get(inode.id, function (e, data) {
|
|||
|
if (noError(e, cb)) {
|
|||
|
try {
|
|||
|
cb(null, JSON.parse(data.toString()));
|
|||
|
} catch (e) {
|
|||
|
// Occurs when data is undefined, or corresponds to something other
|
|||
|
// than a directory listing. The latter should never occur unless
|
|||
|
// the file system is corrupted.
|
|||
|
cb(ApiError.ENOENT(p));
|
|||
|
}
|
|||
|
}
|
|||
|
});
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Given a path to a directory, retrieves the corresponding INode and
|
|||
|
* directory listing.
|
|||
|
*/
|
|||
|
AsyncKeyValueFileSystem.prototype.findINodeAndDirListing = function (tx, p, cb) {
|
|||
|
var _this = this;
|
|||
|
this.findINode(tx, p, function (e, inode) {
|
|||
|
if (noError(e, cb)) {
|
|||
|
_this.getDirListing(tx, p, inode, function (e, listing) {
|
|||
|
if (noError(e, cb)) {
|
|||
|
cb(null, inode, listing);
|
|||
|
}
|
|||
|
});
|
|||
|
}
|
|||
|
});
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Adds a new node under a random ID. Retries 5 times before giving up in
|
|||
|
* the exceedingly unlikely chance that we try to reuse a random GUID.
|
|||
|
* @param cb Passed an error or the GUID that the data was stored under.
|
|||
|
*/
|
|||
|
AsyncKeyValueFileSystem.prototype.addNewNode = function (tx, data, cb) {
|
|||
|
var retries = 0, currId, reroll = function () {
|
|||
|
if (++retries === 5) {
|
|||
|
// Max retries hit. Return with an error.
|
|||
|
cb(new ApiError(2 /* EIO */, 'Unable to commit data to key-value store.'));
|
|||
|
} else {
|
|||
|
// Try again.
|
|||
|
currId = GenerateRandomID();
|
|||
|
tx.put(currId, data, false, function (e, committed) {
|
|||
|
if (e || !committed) {
|
|||
|
reroll();
|
|||
|
} else {
|
|||
|
// Successfully stored under 'currId'.
|
|||
|
cb(null, currId);
|
|||
|
}
|
|||
|
});
|
|||
|
}
|
|||
|
};
|
|||
|
reroll();
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Commits a new file (well, a FILE or a DIRECTORY) to the file system with
|
|||
|
* the given mode.
|
|||
|
* Note: This will commit the transaction.
|
|||
|
* @param p The path to the new file.
|
|||
|
* @param type The type of the new file.
|
|||
|
* @param mode The mode to create the new file with.
|
|||
|
* @param data The data to store at the file's data node.
|
|||
|
* @param cb Passed an error or the Inode for the new file.
|
|||
|
*/
|
|||
|
AsyncKeyValueFileSystem.prototype.commitNewFile = function (tx, p, type, mode, data, cb) {
|
|||
|
var _this = this;
|
|||
|
var parentDir = path.dirname(p), fname = path.basename(p), currTime = (new Date()).getTime();
|
|||
|
|
|||
|
// Invariant: The root always exists.
|
|||
|
// If we don't check this prior to taking steps below, we will create a
|
|||
|
// file with name '' in root should p == '/'.
|
|||
|
if (p === '/') {
|
|||
|
return cb(ApiError.EEXIST(p));
|
|||
|
}
|
|||
|
|
|||
|
// Let's build a pyramid of code!
|
|||
|
// Step 1: Get the parent directory's inode and directory listing
|
|||
|
this.findINodeAndDirListing(tx, parentDir, function (e, parentNode, dirListing) {
|
|||
|
if (noErrorTx(e, tx, cb)) {
|
|||
|
if (dirListing[fname]) {
|
|||
|
// File already exists.
|
|||
|
tx.abort(function () {
|
|||
|
cb(ApiError.EEXIST(p));
|
|||
|
});
|
|||
|
} else {
|
|||
|
// Step 2: Commit data to store.
|
|||
|
_this.addNewNode(tx, data, function (e, dataId) {
|
|||
|
if (noErrorTx(e, tx, cb)) {
|
|||
|
// Step 3: Commit the file's inode to the store.
|
|||
|
var fileInode = new Inode(dataId, data.length, mode | type, currTime, currTime, currTime);
|
|||
|
_this.addNewNode(tx, fileInode.toBuffer(), function (e, fileInodeId) {
|
|||
|
if (noErrorTx(e, tx, cb)) {
|
|||
|
// Step 4: Update parent directory's listing.
|
|||
|
dirListing[fname] = fileInodeId;
|
|||
|
tx.put(parentNode.id, new Buffer(JSON.stringify(dirListing)), true, function (e) {
|
|||
|
if (noErrorTx(e, tx, cb)) {
|
|||
|
// Step 5: Commit and return the new inode.
|
|||
|
tx.commit(function (e) {
|
|||
|
if (noErrorTx(e, tx, cb)) {
|
|||
|
cb(null, fileInode);
|
|||
|
}
|
|||
|
});
|
|||
|
}
|
|||
|
});
|
|||
|
}
|
|||
|
});
|
|||
|
}
|
|||
|
});
|
|||
|
}
|
|||
|
}
|
|||
|
});
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Delete all contents stored in the file system.
|
|||
|
*/
|
|||
|
AsyncKeyValueFileSystem.prototype.empty = function (cb) {
|
|||
|
var _this = this;
|
|||
|
this.store.clear(function (e) {
|
|||
|
if (noError(e, cb)) {
|
|||
|
// INVARIANT: Root always exists.
|
|||
|
_this.makeRootDirectory(cb);
|
|||
|
}
|
|||
|
});
|
|||
|
};
|
|||
|
|
|||
|
AsyncKeyValueFileSystem.prototype.rename = function (oldPath, newPath, cb) {
|
|||
|
var _this = this;
|
|||
|
var tx = this.store.beginTransaction('readwrite'), oldParent = path.dirname(oldPath), oldName = path.basename(oldPath), newParent = path.dirname(newPath), newName = path.basename(newPath), inodes = {}, lists = {}, errorOccurred = false;
|
|||
|
|
|||
|
// Invariant: Can't move a folder inside itself.
|
|||
|
// This funny little hack ensures that the check passes only if oldPath
|
|||
|
// is a subpath of newParent. We append '/' to avoid matching folders that
|
|||
|
// are a substring of the bottom-most folder in the path.
|
|||
|
if ((newParent + '/').indexOf(oldPath + '/') === 0) {
|
|||
|
return cb(new ApiError(5 /* EBUSY */, oldParent));
|
|||
|
}
|
|||
|
|
|||
|
/**
|
|||
|
* Responsible for Phase 2 of the rename operation: Modifying and
|
|||
|
* committing the directory listings. Called once we have successfully
|
|||
|
* retrieved both the old and new parent's inodes and listings.
|
|||
|
*/
|
|||
|
var theOleSwitcharoo = function () {
|
|||
|
// Sanity check: Ensure both paths are present, and no error has occurred.
|
|||
|
if (errorOccurred || !lists.hasOwnProperty(oldParent) || !lists.hasOwnProperty(newParent)) {
|
|||
|
return;
|
|||
|
}
|
|||
|
var oldParentList = lists[oldParent], oldParentINode = inodes[oldParent], newParentList = lists[newParent], newParentINode = inodes[newParent];
|
|||
|
|
|||
|
// Delete file from old parent.
|
|||
|
if (!oldParentList[oldName]) {
|
|||
|
cb(ApiError.ENOENT(oldPath));
|
|||
|
} else {
|
|||
|
var fileId = oldParentList[oldName];
|
|||
|
delete oldParentList[oldName];
|
|||
|
|
|||
|
// Finishes off the renaming process by adding the file to the new
|
|||
|
// parent.
|
|||
|
var completeRename = function () {
|
|||
|
newParentList[newName] = fileId;
|
|||
|
|
|||
|
// Commit old parent's list.
|
|||
|
tx.put(oldParentINode.id, new Buffer(JSON.stringify(oldParentList)), true, function (e) {
|
|||
|
if (noErrorTx(e, tx, cb)) {
|
|||
|
if (oldParent === newParent) {
|
|||
|
// DONE!
|
|||
|
tx.commit(cb);
|
|||
|
} else {
|
|||
|
// Commit new parent's list.
|
|||
|
tx.put(newParentINode.id, new Buffer(JSON.stringify(newParentList)), true, function (e) {
|
|||
|
if (noErrorTx(e, tx, cb)) {
|
|||
|
tx.commit(cb);
|
|||
|
}
|
|||
|
});
|
|||
|
}
|
|||
|
}
|
|||
|
});
|
|||
|
};
|
|||
|
|
|||
|
if (newParentList[newName]) {
|
|||
|
// 'newPath' already exists. Check if it's a file or a directory, and
|
|||
|
// act accordingly.
|
|||
|
_this.getINode(tx, newPath, newParentList[newName], function (e, inode) {
|
|||
|
if (noErrorTx(e, tx, cb)) {
|
|||
|
if (inode.isFile()) {
|
|||
|
// Delete the file and continue.
|
|||
|
tx.delete(inode.id, function (e) {
|
|||
|
if (noErrorTx(e, tx, cb)) {
|
|||
|
tx.delete(newParentList[newName], function (e) {
|
|||
|
if (noErrorTx(e, tx, cb)) {
|
|||
|
completeRename();
|
|||
|
}
|
|||
|
});
|
|||
|
}
|
|||
|
});
|
|||
|
} else {
|
|||
|
// Can't overwrite a directory using rename.
|
|||
|
tx.abort(function (e) {
|
|||
|
cb(ApiError.EPERM(newPath));
|
|||
|
});
|
|||
|
}
|
|||
|
}
|
|||
|
});
|
|||
|
} else {
|
|||
|
completeRename();
|
|||
|
}
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Grabs a path's inode and directory listing, and shoves it into the
|
|||
|
* inodes and lists hashes.
|
|||
|
*/
|
|||
|
var processInodeAndListings = function (p) {
|
|||
|
_this.findINodeAndDirListing(tx, p, function (e, node, dirList) {
|
|||
|
if (e) {
|
|||
|
if (!errorOccurred) {
|
|||
|
errorOccurred = true;
|
|||
|
tx.abort(function () {
|
|||
|
cb(e);
|
|||
|
});
|
|||
|
}
|
|||
|
// If error has occurred already, just stop here.
|
|||
|
} else {
|
|||
|
inodes[p] = node;
|
|||
|
lists[p] = dirList;
|
|||
|
theOleSwitcharoo();
|
|||
|
}
|
|||
|
});
|
|||
|
};
|
|||
|
|
|||
|
processInodeAndListings(oldParent);
|
|||
|
if (oldParent !== newParent) {
|
|||
|
processInodeAndListings(newParent);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
AsyncKeyValueFileSystem.prototype.stat = function (p, isLstat, cb) {
|
|||
|
var tx = this.store.beginTransaction('readonly');
|
|||
|
this.findINode(tx, p, function (e, inode) {
|
|||
|
if (noError(e, cb)) {
|
|||
|
cb(null, inode.toStats());
|
|||
|
}
|
|||
|
});
|
|||
|
};
|
|||
|
|
|||
|
AsyncKeyValueFileSystem.prototype.createFile = function (p, flag, mode, cb) {
|
|||
|
var _this = this;
|
|||
|
var tx = this.store.beginTransaction('readwrite'), data = new Buffer(0);
|
|||
|
|
|||
|
this.commitNewFile(tx, p, 32768 /* FILE */, mode, data, function (e, newFile) {
|
|||
|
if (noError(e, cb)) {
|
|||
|
cb(null, new AsyncKeyValueFile(_this, p, flag, newFile.toStats(), data));
|
|||
|
}
|
|||
|
});
|
|||
|
};
|
|||
|
|
|||
|
AsyncKeyValueFileSystem.prototype.openFile = function (p, flag, cb) {
|
|||
|
var _this = this;
|
|||
|
var tx = this.store.beginTransaction('readonly');
|
|||
|
|
|||
|
// Step 1: Grab the file's inode.
|
|||
|
this.findINode(tx, p, function (e, inode) {
|
|||
|
if (noError(e, cb)) {
|
|||
|
// Step 2: Grab the file's data.
|
|||
|
tx.get(inode.id, function (e, data) {
|
|||
|
if (noError(e, cb)) {
|
|||
|
if (data === undefined) {
|
|||
|
cb(ApiError.ENOENT(p));
|
|||
|
} else {
|
|||
|
cb(null, new AsyncKeyValueFile(_this, p, flag, inode.toStats(), data));
|
|||
|
}
|
|||
|
}
|
|||
|
});
|
|||
|
}
|
|||
|
});
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Remove all traces of the given path from the file system.
|
|||
|
* @param p The path to remove from the file system.
|
|||
|
* @param isDir Does the path belong to a directory, or a file?
|
|||
|
* @todo Update mtime.
|
|||
|
*/
|
|||
|
AsyncKeyValueFileSystem.prototype.removeEntry = function (p, isDir, cb) {
|
|||
|
var _this = this;
|
|||
|
var tx = this.store.beginTransaction('readwrite'), parent = path.dirname(p), fileName = path.basename(p);
|
|||
|
|
|||
|
// Step 1: Get parent directory's node and directory listing.
|
|||
|
this.findINodeAndDirListing(tx, parent, function (e, parentNode, parentListing) {
|
|||
|
if (noErrorTx(e, tx, cb)) {
|
|||
|
if (!parentListing[fileName]) {
|
|||
|
tx.abort(function () {
|
|||
|
cb(ApiError.ENOENT(p));
|
|||
|
});
|
|||
|
} else {
|
|||
|
// Remove from directory listing of parent.
|
|||
|
var fileNodeId = parentListing[fileName];
|
|||
|
delete parentListing[fileName];
|
|||
|
|
|||
|
// Step 2: Get file inode.
|
|||
|
_this.getINode(tx, p, fileNodeId, function (e, fileNode) {
|
|||
|
if (noErrorTx(e, tx, cb)) {
|
|||
|
if (!isDir && fileNode.isDirectory()) {
|
|||
|
tx.abort(function () {
|
|||
|
cb(ApiError.EISDIR(p));
|
|||
|
});
|
|||
|
} else if (isDir && !fileNode.isDirectory()) {
|
|||
|
tx.abort(function () {
|
|||
|
cb(ApiError.ENOTDIR(p));
|
|||
|
});
|
|||
|
} else {
|
|||
|
// Step 3: Delete data.
|
|||
|
tx.delete(fileNode.id, function (e) {
|
|||
|
if (noErrorTx(e, tx, cb)) {
|
|||
|
// Step 4: Delete node.
|
|||
|
tx.delete(fileNodeId, function (e) {
|
|||
|
if (noErrorTx(e, tx, cb)) {
|
|||
|
// Step 5: Update directory listing.
|
|||
|
tx.put(parentNode.id, new Buffer(JSON.stringify(parentListing)), true, function (e) {
|
|||
|
if (noErrorTx(e, tx, cb)) {
|
|||
|
tx.commit(cb);
|
|||
|
}
|
|||
|
});
|
|||
|
}
|
|||
|
});
|
|||
|
}
|
|||
|
});
|
|||
|
}
|
|||
|
}
|
|||
|
});
|
|||
|
}
|
|||
|
}
|
|||
|
});
|
|||
|
};
|
|||
|
|
|||
|
AsyncKeyValueFileSystem.prototype.unlink = function (p, cb) {
|
|||
|
this.removeEntry(p, false, cb);
|
|||
|
};
|
|||
|
|
|||
|
AsyncKeyValueFileSystem.prototype.rmdir = function (p, cb) {
|
|||
|
this.removeEntry(p, true, cb);
|
|||
|
};
|
|||
|
|
|||
|
AsyncKeyValueFileSystem.prototype.mkdir = function (p, mode, cb) {
|
|||
|
var tx = this.store.beginTransaction('readwrite'), data = new Buffer('{}');
|
|||
|
this.commitNewFile(tx, p, 16384 /* DIRECTORY */, mode, data, cb);
|
|||
|
};
|
|||
|
|
|||
|
AsyncKeyValueFileSystem.prototype.readdir = function (p, cb) {
|
|||
|
var _this = this;
|
|||
|
var tx = this.store.beginTransaction('readonly');
|
|||
|
this.findINode(tx, p, function (e, inode) {
|
|||
|
if (noError(e, cb)) {
|
|||
|
_this.getDirListing(tx, p, inode, function (e, dirListing) {
|
|||
|
if (noError(e, cb)) {
|
|||
|
cb(null, Object.keys(dirListing));
|
|||
|
}
|
|||
|
});
|
|||
|
}
|
|||
|
});
|
|||
|
};
|
|||
|
|
|||
|
AsyncKeyValueFileSystem.prototype._sync = function (p, data, stats, cb) {
|
|||
|
var _this = this;
|
|||
|
// @todo Ensure mtime updates properly, and use that to determine if a data
|
|||
|
// update is required.
|
|||
|
var tx = this.store.beginTransaction('readwrite');
|
|||
|
|
|||
|
// Step 1: Get the file node's ID.
|
|||
|
this._findINode(tx, path.dirname(p), path.basename(p), function (e, fileInodeId) {
|
|||
|
if (noErrorTx(e, tx, cb)) {
|
|||
|
// Step 2: Get the file inode.
|
|||
|
_this.getINode(tx, p, fileInodeId, function (e, fileInode) {
|
|||
|
if (noErrorTx(e, tx, cb)) {
|
|||
|
var inodeChanged = fileInode.update(stats);
|
|||
|
|
|||
|
// Step 3: Sync the data.
|
|||
|
tx.put(fileInode.id, data, true, function (e) {
|
|||
|
if (noErrorTx(e, tx, cb)) {
|
|||
|
// Step 4: Sync the metadata (if it changed)!
|
|||
|
if (inodeChanged) {
|
|||
|
tx.put(fileInodeId, fileInode.toBuffer(), true, function (e) {
|
|||
|
if (noErrorTx(e, tx, cb)) {
|
|||
|
tx.commit(cb);
|
|||
|
}
|
|||
|
});
|
|||
|
} else {
|
|||
|
// No need to sync metadata; return.
|
|||
|
tx.commit(cb);
|
|||
|
}
|
|||
|
}
|
|||
|
});
|
|||
|
}
|
|||
|
});
|
|||
|
}
|
|||
|
});
|
|||
|
};
|
|||
|
return AsyncKeyValueFileSystem;
|
|||
|
})(file_system.BaseFileSystem);
|
|||
|
exports.AsyncKeyValueFileSystem = AsyncKeyValueFileSystem;
|
|||
|
});
|
|||
|
//# sourceMappingURL=key_value_filesystem.js.map
|
|||
|
;
|
|||
|
define('core/global',["require", "exports"], function(require, exports) {
|
|||
|
/**
|
|||
|
* Exports the global scope variable.
|
|||
|
* In the main browser thread, this is "window".
|
|||
|
* In a WebWorker, this is "self".
|
|||
|
* In Node, this is "global".
|
|||
|
*/
|
|||
|
var toExport;
|
|||
|
if (typeof (window) !== 'undefined') {
|
|||
|
toExport = window;
|
|||
|
} else if (typeof (self) !== 'undefined') {
|
|||
|
toExport = self;
|
|||
|
} else {
|
|||
|
toExport = global;
|
|||
|
}
|
|||
|
|
|||
|
return toExport;
|
|||
|
});
|
|||
|
//# sourceMappingURL=global.js.map
|
|||
|
;
|
|||
|
var __extends = this.__extends || function (d, b) {
|
|||
|
for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p];
|
|||
|
function __() { this.constructor = d; }
|
|||
|
__.prototype = b.prototype;
|
|||
|
d.prototype = new __();
|
|||
|
};
|
|||
|
define('backend/IndexedDB',["require", "exports", '../core/buffer', '../core/browserfs', '../generic/key_value_filesystem', '../core/api_error', '../core/buffer_core_arraybuffer', '../core/global'], function(require, exports, buffer, browserfs, kvfs, api_error, buffer_core_arraybuffer, global) {
|
|||
|
var Buffer = buffer.Buffer, ApiError = api_error.ApiError, ErrorCode = api_error.ErrorCode, indexedDB = global.indexedDB || global.mozIndexedDB || global.webkitIndexedDB || global.msIndexedDB;
|
|||
|
|
|||
|
/**
|
|||
|
* Converts a DOMException or a DOMError from an IndexedDB event into a
|
|||
|
* standardized BrowserFS API error.
|
|||
|
*/
|
|||
|
function convertError(e, message) {
|
|||
|
if (typeof message === "undefined") { message = e.toString(); }
|
|||
|
switch (e.name) {
|
|||
|
case "NotFoundError":
|
|||
|
return new ApiError(1 /* ENOENT */, message);
|
|||
|
case "QuotaExceededError":
|
|||
|
return new ApiError(11 /* ENOSPC */, message);
|
|||
|
default:
|
|||
|
// The rest do not seem to map cleanly to standard error codes.
|
|||
|
return new ApiError(2 /* EIO */, message);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
/**
|
|||
|
* Produces a new onerror handler for IDB. Our errors are always fatal, so we
|
|||
|
* handle them generically: Call the user-supplied callback with a translated
|
|||
|
* version of the error, and let the error bubble up.
|
|||
|
*/
|
|||
|
function onErrorHandler(cb, code, message) {
|
|||
|
if (typeof code === "undefined") { code = 2 /* EIO */; }
|
|||
|
if (typeof message === "undefined") { message = null; }
|
|||
|
return function (e) {
|
|||
|
// Prevent the error from canceling the transaction.
|
|||
|
e.preventDefault();
|
|||
|
cb(new ApiError(code, message));
|
|||
|
};
|
|||
|
}
|
|||
|
|
|||
|
/**
|
|||
|
* Converts a NodeBuffer into an ArrayBuffer.
|
|||
|
*/
|
|||
|
function buffer2arraybuffer(buffer) {
|
|||
|
// XXX: Typing hack.
|
|||
|
var backing_mem = buffer.getBufferCore();
|
|||
|
if (!(backing_mem instanceof buffer_core_arraybuffer.BufferCoreArrayBuffer)) {
|
|||
|
// Copy into an ArrayBuffer-backed Buffer.
|
|||
|
buffer = new Buffer(this._buffer.length);
|
|||
|
this._buffer.copy(buffer);
|
|||
|
backing_mem = buffer.getBufferCore();
|
|||
|
}
|
|||
|
|
|||
|
// Reach into the BC, grab the DV.
|
|||
|
var dv = backing_mem.getDataView();
|
|||
|
return dv.buffer;
|
|||
|
}
|
|||
|
|
|||
|
var IndexedDBROTransaction = (function () {
|
|||
|
function IndexedDBROTransaction(tx, store) {
|
|||
|
this.tx = tx;
|
|||
|
this.store = store;
|
|||
|
}
|
|||
|
IndexedDBROTransaction.prototype.get = function (key, cb) {
|
|||
|
try {
|
|||
|
var r = this.store.get(key);
|
|||
|
r.onerror = onErrorHandler(cb);
|
|||
|
r.onsuccess = function (event) {
|
|||
|
// IDB returns the value 'undefined' when you try to get keys that
|
|||
|
// don't exist. The caller expects this behavior.
|
|||
|
var result = event.target.result;
|
|||
|
if (result === undefined) {
|
|||
|
cb(null, result);
|
|||
|
} else {
|
|||
|
// IDB data is stored as an ArrayBuffer
|
|||
|
cb(null, new Buffer(result));
|
|||
|
}
|
|||
|
};
|
|||
|
} catch (e) {
|
|||
|
cb(convertError(e));
|
|||
|
}
|
|||
|
};
|
|||
|
return IndexedDBROTransaction;
|
|||
|
})();
|
|||
|
exports.IndexedDBROTransaction = IndexedDBROTransaction;
|
|||
|
|
|||
|
var IndexedDBRWTransaction = (function (_super) {
|
|||
|
__extends(IndexedDBRWTransaction, _super);
|
|||
|
function IndexedDBRWTransaction(tx, store) {
|
|||
|
_super.call(this, tx, store);
|
|||
|
}
|
|||
|
IndexedDBRWTransaction.prototype.put = function (key, data, overwrite, cb) {
|
|||
|
try {
|
|||
|
var arraybuffer = buffer2arraybuffer(data), r;
|
|||
|
if (overwrite) {
|
|||
|
r = this.store.put(arraybuffer, key);
|
|||
|
} else {
|
|||
|
// 'add' will never overwrite an existing key.
|
|||
|
r = this.store.add(arraybuffer, key);
|
|||
|
}
|
|||
|
|
|||
|
// XXX: NEED TO RETURN FALSE WHEN ADD HAS A KEY CONFLICT. NO ERROR.
|
|||
|
r.onerror = onErrorHandler(cb);
|
|||
|
r.onsuccess = function (event) {
|
|||
|
cb(null, true);
|
|||
|
};
|
|||
|
} catch (e) {
|
|||
|
cb(convertError(e));
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
IndexedDBRWTransaction.prototype.delete = function (key, cb) {
|
|||
|
try {
|
|||
|
var r = this.store.delete(key);
|
|||
|
r.onerror = onErrorHandler(cb);
|
|||
|
r.onsuccess = function (event) {
|
|||
|
cb();
|
|||
|
};
|
|||
|
} catch (e) {
|
|||
|
cb(convertError(e));
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
IndexedDBRWTransaction.prototype.commit = function (cb) {
|
|||
|
// Return to the event loop to commit the transaction.
|
|||
|
setTimeout(cb, 0);
|
|||
|
};
|
|||
|
|
|||
|
IndexedDBRWTransaction.prototype.abort = function (cb) {
|
|||
|
var _e;
|
|||
|
try {
|
|||
|
this.tx.abort();
|
|||
|
} catch (e) {
|
|||
|
_e = convertError(e);
|
|||
|
} finally {
|
|||
|
cb(_e);
|
|||
|
}
|
|||
|
};
|
|||
|
return IndexedDBRWTransaction;
|
|||
|
})(IndexedDBROTransaction);
|
|||
|
exports.IndexedDBRWTransaction = IndexedDBRWTransaction;
|
|||
|
|
|||
|
var IndexedDBStore = (function () {
|
|||
|
/**
|
|||
|
* Constructs an IndexedDB file system.
|
|||
|
* @param cb Called once the database is instantiated and ready for use.
|
|||
|
* Passes an error if there was an issue instantiating the database.
|
|||
|
* @param objectStoreName The name of this file system. You can have
|
|||
|
* multiple IndexedDB file systems operating at once, but each must have
|
|||
|
* a different name.
|
|||
|
*/
|
|||
|
function IndexedDBStore(cb, storeName) {
|
|||
|
if (typeof storeName === "undefined") { storeName = 'browserfs'; }
|
|||
|
var _this = this;
|
|||
|
this.storeName = storeName;
|
|||
|
var openReq = indexedDB.open(this.storeName, 1);
|
|||
|
|
|||
|
openReq.onupgradeneeded = function (event) {
|
|||
|
var db = event.target.result;
|
|||
|
|
|||
|
// Huh. This should never happen; we're at version 1. Why does another
|
|||
|
// database exist?
|
|||
|
if (db.objectStoreNames.contains(_this.storeName)) {
|
|||
|
db.deleteObjectStore(_this.storeName);
|
|||
|
}
|
|||
|
db.createObjectStore(_this.storeName);
|
|||
|
};
|
|||
|
|
|||
|
openReq.onsuccess = function (event) {
|
|||
|
_this.db = event.target.result;
|
|||
|
cb(null, _this);
|
|||
|
};
|
|||
|
|
|||
|
openReq.onerror = onErrorHandler(cb, 4 /* EACCES */);
|
|||
|
}
|
|||
|
IndexedDBStore.prototype.name = function () {
|
|||
|
return "IndexedDB - " + this.storeName;
|
|||
|
};
|
|||
|
|
|||
|
IndexedDBStore.prototype.clear = function (cb) {
|
|||
|
try {
|
|||
|
var tx = this.db.transaction(this.storeName, 'readwrite'), objectStore = tx.objectStore(this.storeName), r = objectStore.clear();
|
|||
|
r.onsuccess = function (event) {
|
|||
|
// Use setTimeout to commit transaction.
|
|||
|
setTimeout(cb, 0);
|
|||
|
};
|
|||
|
r.onerror = onErrorHandler(cb);
|
|||
|
} catch (e) {
|
|||
|
cb(convertError(e));
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
IndexedDBStore.prototype.beginTransaction = function (type) {
|
|||
|
if (typeof type === "undefined") { type = 'readonly'; }
|
|||
|
var tx = this.db.transaction(this.storeName, type), objectStore = tx.objectStore(this.storeName);
|
|||
|
if (type === 'readwrite') {
|
|||
|
return new IndexedDBRWTransaction(tx, objectStore);
|
|||
|
} else if (type === 'readonly') {
|
|||
|
return new IndexedDBROTransaction(tx, objectStore);
|
|||
|
} else {
|
|||
|
throw new ApiError(9 /* EINVAL */, 'Invalid transaction type.');
|
|||
|
}
|
|||
|
};
|
|||
|
return IndexedDBStore;
|
|||
|
})();
|
|||
|
exports.IndexedDBStore = IndexedDBStore;
|
|||
|
|
|||
|
/**
|
|||
|
* A file system that uses the IndexedDB key value file system.
|
|||
|
*/
|
|||
|
var IndexedDBFileSystem = (function (_super) {
|
|||
|
__extends(IndexedDBFileSystem, _super);
|
|||
|
function IndexedDBFileSystem(cb, storeName) {
|
|||
|
var _this = this;
|
|||
|
_super.call(this);
|
|||
|
new IndexedDBStore(function (e, store) {
|
|||
|
if (e) {
|
|||
|
cb(e);
|
|||
|
} else {
|
|||
|
_this.init(store, function (e) {
|
|||
|
cb(e, _this);
|
|||
|
});
|
|||
|
}
|
|||
|
}, storeName);
|
|||
|
}
|
|||
|
IndexedDBFileSystem.isAvailable = function () {
|
|||
|
return typeof indexedDB !== 'undefined';
|
|||
|
};
|
|||
|
return IndexedDBFileSystem;
|
|||
|
})(kvfs.AsyncKeyValueFileSystem);
|
|||
|
exports.IndexedDBFileSystem = IndexedDBFileSystem;
|
|||
|
|
|||
|
browserfs.registerFileSystem('IndexedDB', IndexedDBFileSystem);
|
|||
|
});
|
|||
|
//# sourceMappingURL=IndexedDB.js.map
|
|||
|
;
|
|||
|
define('generic/file_index',["require", "exports", '../core/node_fs_stats', '../core/node_path'], function(require, exports, node_fs_stats, node_path) {
|
|||
|
var Stats = node_fs_stats.Stats;
|
|||
|
var path = node_path.path;
|
|||
|
|
|||
|
/**
|
|||
|
* A simple class for storing a filesystem index. Assumes that all paths passed
|
|||
|
* to it are *absolute* paths.
|
|||
|
*
|
|||
|
* Can be used as a partial or a full index, although care must be taken if used
|
|||
|
* for the former purpose, especially when directories are concerned.
|
|||
|
*/
|
|||
|
var FileIndex = (function () {
|
|||
|
/**
|
|||
|
* Constructs a new FileIndex.
|
|||
|
*/
|
|||
|
function FileIndex() {
|
|||
|
// _index is a single-level key,value store that maps *directory* paths to
|
|||
|
// DirInodes. File information is only contained in DirInodes themselves.
|
|||
|
this._index = {};
|
|||
|
|
|||
|
// Create the root directory.
|
|||
|
this.addPath('/', new DirInode());
|
|||
|
}
|
|||
|
/**
|
|||
|
* Split into a (directory path, item name) pair
|
|||
|
*/
|
|||
|
FileIndex.prototype._split_path = function (p) {
|
|||
|
var dirpath = path.dirname(p);
|
|||
|
var itemname = p.substr(dirpath.length + (dirpath === "/" ? 0 : 1));
|
|||
|
return [dirpath, itemname];
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Runs the given function over all files in the index.
|
|||
|
*/
|
|||
|
FileIndex.prototype.fileIterator = function (cb) {
|
|||
|
for (var path in this._index) {
|
|||
|
var dir = this._index[path];
|
|||
|
var files = dir.getListing();
|
|||
|
for (var i = 0; i < files.length; i++) {
|
|||
|
var item = dir.getItem(files[i]);
|
|||
|
if (item.isFile()) {
|
|||
|
cb(item.getData());
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Adds the given absolute path to the index if it is not already in the index.
|
|||
|
* Creates any needed parent directories.
|
|||
|
* @param [String] path The path to add to the index.
|
|||
|
* @param [BrowserFS.FileInode | BrowserFS.DirInode] inode The inode for the
|
|||
|
* path to add.
|
|||
|
* @return [Boolean] 'True' if it was added or already exists, 'false' if there
|
|||
|
* was an issue adding it (e.g. item in path is a file, item exists but is
|
|||
|
* different).
|
|||
|
* @todo If adding fails and implicitly creates directories, we do not clean up
|
|||
|
* the new empty directories.
|
|||
|
*/
|
|||
|
FileIndex.prototype.addPath = function (path, inode) {
|
|||
|
if (inode == null) {
|
|||
|
throw new Error('Inode must be specified');
|
|||
|
}
|
|||
|
if (path[0] !== '/') {
|
|||
|
throw new Error('Path must be absolute, got: ' + path);
|
|||
|
}
|
|||
|
|
|||
|
// Check if it already exists.
|
|||
|
if (this._index.hasOwnProperty(path)) {
|
|||
|
return this._index[path] === inode;
|
|||
|
}
|
|||
|
|
|||
|
var splitPath = this._split_path(path);
|
|||
|
var dirpath = splitPath[0];
|
|||
|
var itemname = splitPath[1];
|
|||
|
|
|||
|
// Try to add to its parent directory first.
|
|||
|
var parent = this._index[dirpath];
|
|||
|
if (parent === undefined && path !== '/') {
|
|||
|
// Create parent.
|
|||
|
parent = new DirInode();
|
|||
|
if (!this.addPath(dirpath, parent)) {
|
|||
|
return false;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
// Add myself to my parent.
|
|||
|
if (path !== '/') {
|
|||
|
if (!parent.addItem(itemname, inode)) {
|
|||
|
return false;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
// If I'm a directory, add myself to the index.
|
|||
|
if (!inode.isFile()) {
|
|||
|
this._index[path] = inode;
|
|||
|
}
|
|||
|
return true;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Removes the given path. Can be a file or a directory.
|
|||
|
* @return [BrowserFS.FileInode | BrowserFS.DirInode | null] The removed item,
|
|||
|
* or null if it did not exist.
|
|||
|
*/
|
|||
|
FileIndex.prototype.removePath = function (path) {
|
|||
|
var splitPath = this._split_path(path);
|
|||
|
var dirpath = splitPath[0];
|
|||
|
var itemname = splitPath[1];
|
|||
|
|
|||
|
// Try to remove it from its parent directory first.
|
|||
|
var parent = this._index[dirpath];
|
|||
|
if (parent === undefined) {
|
|||
|
return null;
|
|||
|
}
|
|||
|
|
|||
|
// Remove myself from my parent.
|
|||
|
var inode = parent.remItem(itemname);
|
|||
|
if (inode === null) {
|
|||
|
return null;
|
|||
|
}
|
|||
|
|
|||
|
// If I'm a directory, remove myself from the index, and remove my children.
|
|||
|
if (!inode.isFile()) {
|
|||
|
var dirInode = inode;
|
|||
|
var children = dirInode.getListing();
|
|||
|
for (var i = 0; i < children.length; i++) {
|
|||
|
this.removePath(path + '/' + children[i]);
|
|||
|
}
|
|||
|
|
|||
|
// Remove the directory from the index, unless it's the root.
|
|||
|
if (path !== '/') {
|
|||
|
delete this._index[path];
|
|||
|
}
|
|||
|
}
|
|||
|
return inode;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Retrieves the directory listing of the given path.
|
|||
|
* @return [String[]] An array of files in the given path, or 'null' if it does
|
|||
|
* not exist.
|
|||
|
*/
|
|||
|
FileIndex.prototype.ls = function (path) {
|
|||
|
var item = this._index[path];
|
|||
|
if (item === undefined) {
|
|||
|
return null;
|
|||
|
}
|
|||
|
return item.getListing();
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Returns the inode of the given item.
|
|||
|
* @param [String] path
|
|||
|
* @return [BrowserFS.FileInode | BrowserFS.DirInode | null] Returns null if
|
|||
|
* the item does not exist.
|
|||
|
*/
|
|||
|
FileIndex.prototype.getInode = function (path) {
|
|||
|
var splitPath = this._split_path(path);
|
|||
|
var dirpath = splitPath[0];
|
|||
|
var itemname = splitPath[1];
|
|||
|
|
|||
|
// Retrieve from its parent directory.
|
|||
|
var parent = this._index[dirpath];
|
|||
|
if (parent === undefined) {
|
|||
|
return null;
|
|||
|
}
|
|||
|
|
|||
|
// Root case
|
|||
|
if (dirpath === path) {
|
|||
|
return parent;
|
|||
|
}
|
|||
|
return parent.getItem(itemname);
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Static method for constructing indices from a JSON listing.
|
|||
|
* @param [Object] listing Directory listing generated by tools/XHRIndexer.coffee
|
|||
|
* @return [BrowserFS.FileIndex] A new FileIndex object.
|
|||
|
*/
|
|||
|
FileIndex.from_listing = function (listing) {
|
|||
|
var idx = new FileIndex();
|
|||
|
|
|||
|
// Add a root DirNode.
|
|||
|
var rootInode = new DirInode();
|
|||
|
idx._index['/'] = rootInode;
|
|||
|
var queue = [['', listing, rootInode]];
|
|||
|
while (queue.length > 0) {
|
|||
|
var inode;
|
|||
|
var next = queue.pop();
|
|||
|
var pwd = next[0];
|
|||
|
var tree = next[1];
|
|||
|
var parent = next[2];
|
|||
|
for (var node in tree) {
|
|||
|
var children = tree[node];
|
|||
|
var name = "" + pwd + "/" + node;
|
|||
|
if (children != null) {
|
|||
|
idx._index[name] = inode = new DirInode();
|
|||
|
queue.push([name, children, inode]);
|
|||
|
} else {
|
|||
|
// This inode doesn't have correct size information, noted with -1.
|
|||
|
inode = new FileInode(new Stats(32768 /* FILE */, -1, 0x16D));
|
|||
|
}
|
|||
|
if (parent != null) {
|
|||
|
parent._ls[node] = inode;
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
return idx;
|
|||
|
};
|
|||
|
return FileIndex;
|
|||
|
})();
|
|||
|
exports.FileIndex = FileIndex;
|
|||
|
|
|||
|
|
|||
|
|
|||
|
/**
|
|||
|
* Inode for a file. Stores an arbitrary (filesystem-specific) data payload.
|
|||
|
*/
|
|||
|
var FileInode = (function () {
|
|||
|
function FileInode(data) {
|
|||
|
this.data = data;
|
|||
|
}
|
|||
|
FileInode.prototype.isFile = function () {
|
|||
|
return true;
|
|||
|
};
|
|||
|
FileInode.prototype.isDir = function () {
|
|||
|
return false;
|
|||
|
};
|
|||
|
FileInode.prototype.getData = function () {
|
|||
|
return this.data;
|
|||
|
};
|
|||
|
FileInode.prototype.setData = function (data) {
|
|||
|
this.data = data;
|
|||
|
};
|
|||
|
return FileInode;
|
|||
|
})();
|
|||
|
exports.FileInode = FileInode;
|
|||
|
|
|||
|
/**
|
|||
|
* Inode for a directory. Currently only contains the directory listing.
|
|||
|
*/
|
|||
|
var DirInode = (function () {
|
|||
|
/**
|
|||
|
* Constructs an inode for a directory.
|
|||
|
*/
|
|||
|
function DirInode() {
|
|||
|
this._ls = {};
|
|||
|
}
|
|||
|
DirInode.prototype.isFile = function () {
|
|||
|
return false;
|
|||
|
};
|
|||
|
DirInode.prototype.isDir = function () {
|
|||
|
return true;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Return a Stats object for this inode.
|
|||
|
* @todo Should probably remove this at some point. This isn't the
|
|||
|
* responsibility of the FileIndex.
|
|||
|
* @return [BrowserFS.node.fs.Stats]
|
|||
|
*/
|
|||
|
DirInode.prototype.getStats = function () {
|
|||
|
return new Stats(16384 /* DIRECTORY */, 4096, 0x16D);
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Returns the directory listing for this directory. Paths in the directory are
|
|||
|
* relative to the directory's path.
|
|||
|
* @return [String[]] The directory listing for this directory.
|
|||
|
*/
|
|||
|
DirInode.prototype.getListing = function () {
|
|||
|
return Object.keys(this._ls);
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Returns the inode for the indicated item, or null if it does not exist.
|
|||
|
* @param [String] p Name of item in this directory.
|
|||
|
* @return [BrowserFS.FileInode | BrowserFS.DirInode | null]
|
|||
|
*/
|
|||
|
DirInode.prototype.getItem = function (p) {
|
|||
|
var _ref;
|
|||
|
return (_ref = this._ls[p]) != null ? _ref : null;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Add the given item to the directory listing. Note that the given inode is
|
|||
|
* not copied, and will be mutated by the DirInode if it is a DirInode.
|
|||
|
* @param [String] p Item name to add to the directory listing.
|
|||
|
* @param [BrowserFS.FileInode | BrowserFS.DirInode] inode The inode for the
|
|||
|
* item to add to the directory inode.
|
|||
|
* @return [Boolean] True if it was added, false if it already existed.
|
|||
|
*/
|
|||
|
DirInode.prototype.addItem = function (p, inode) {
|
|||
|
if (p in this._ls) {
|
|||
|
return false;
|
|||
|
}
|
|||
|
this._ls[p] = inode;
|
|||
|
return true;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Removes the given item from the directory listing.
|
|||
|
* @param [String] p Name of item to remove from the directory listing.
|
|||
|
* @return [BrowserFS.FileInode | BrowserFS.DirInode | null] Returns the item
|
|||
|
* removed, or null if the item did not exist.
|
|||
|
*/
|
|||
|
DirInode.prototype.remItem = function (p) {
|
|||
|
var item = this._ls[p];
|
|||
|
if (item === undefined) {
|
|||
|
return null;
|
|||
|
}
|
|||
|
delete this._ls[p];
|
|||
|
return item;
|
|||
|
};
|
|||
|
return DirInode;
|
|||
|
})();
|
|||
|
exports.DirInode = DirInode;
|
|||
|
});
|
|||
|
//# sourceMappingURL=file_index.js.map
|
|||
|
;
|
|||
|
/**
|
|||
|
* Grab bag of utility functions used across the code.
|
|||
|
*/
|
|||
|
define('core/util',["require", "exports"], function(require, exports) {
|
|||
|
/**
|
|||
|
* Estimates the size of a JS object.
|
|||
|
* @param {Object} object - the object to measure.
|
|||
|
* @return {Number} estimated object size.
|
|||
|
* @see http://stackoverflow.com/a/11900218/10601
|
|||
|
*/
|
|||
|
function roughSizeOfObject(object) {
|
|||
|
var bytes, key, objectList, prop, stack, value;
|
|||
|
objectList = [];
|
|||
|
stack = [object];
|
|||
|
bytes = 0;
|
|||
|
while (stack.length !== 0) {
|
|||
|
value = stack.pop();
|
|||
|
if (typeof value === 'boolean') {
|
|||
|
bytes += 4;
|
|||
|
} else if (typeof value === 'string') {
|
|||
|
bytes += value.length * 2;
|
|||
|
} else if (typeof value === 'number') {
|
|||
|
bytes += 8;
|
|||
|
} else if (typeof value === 'object' && objectList.indexOf(value) < 0) {
|
|||
|
objectList.push(value);
|
|||
|
bytes += 4;
|
|||
|
for (key in value) {
|
|||
|
prop = value[key];
|
|||
|
bytes += key.length * 2;
|
|||
|
stack.push(prop);
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
return bytes;
|
|||
|
}
|
|||
|
exports.roughSizeOfObject = roughSizeOfObject;
|
|||
|
|
|||
|
/**
|
|||
|
* Checks for any IE version, including IE11 which removed MSIE from the
|
|||
|
* userAgent string.
|
|||
|
*/
|
|||
|
exports.isIE = (/(msie) ([\w.]+)/.exec(navigator.userAgent.toLowerCase()) != null || navigator.userAgent.indexOf('Trident') !== -1);
|
|||
|
});
|
|||
|
//# sourceMappingURL=util.js.map
|
|||
|
;
|
|||
|
/**
|
|||
|
* Contains utility methods for performing a variety of tasks with
|
|||
|
* XmlHttpRequest across browsers.
|
|||
|
*/
|
|||
|
define('generic/xhr',["require", "exports", '../core/util', '../core/buffer', '../core/api_error'], function(require, exports, util, buffer, api_error) {
|
|||
|
var ApiError = api_error.ApiError;
|
|||
|
var ErrorCode = api_error.ErrorCode;
|
|||
|
var Buffer = buffer.Buffer;
|
|||
|
|
|||
|
|
|||
|
|
|||
|
// Converts 'responseBody' in IE into the equivalent 'responseText' that other
|
|||
|
// browsers would generate.
|
|||
|
function getIEByteArray(IEByteArray) {
|
|||
|
var rawBytes = IEBinaryToArray_ByteStr(IEByteArray);
|
|||
|
var lastChr = IEBinaryToArray_ByteStr_Last(IEByteArray);
|
|||
|
var data_str = rawBytes.replace(/[\s\S]/g, function (match) {
|
|||
|
var v = match.charCodeAt(0);
|
|||
|
return String.fromCharCode(v & 0xff, v >> 8);
|
|||
|
}) + lastChr;
|
|||
|
var data_array = new Array(data_str.length);
|
|||
|
for (var i = 0; i < data_str.length; i++) {
|
|||
|
data_array[i] = data_str.charCodeAt(i);
|
|||
|
}
|
|||
|
return data_array;
|
|||
|
}
|
|||
|
|
|||
|
function downloadFileIE(async, p, type, cb) {
|
|||
|
switch (type) {
|
|||
|
case 'buffer':
|
|||
|
|
|||
|
case 'json':
|
|||
|
break;
|
|||
|
default:
|
|||
|
return cb(new ApiError(9 /* EINVAL */, "Invalid download type: " + type));
|
|||
|
}
|
|||
|
|
|||
|
var req = new XMLHttpRequest();
|
|||
|
req.open('GET', p, async);
|
|||
|
req.setRequestHeader("Accept-Charset", "x-user-defined");
|
|||
|
req.onreadystatechange = function (e) {
|
|||
|
var data_array;
|
|||
|
if (req.readyState === 4) {
|
|||
|
if (req.status === 200) {
|
|||
|
switch (type) {
|
|||
|
case 'buffer':
|
|||
|
data_array = getIEByteArray(req.responseBody);
|
|||
|
return cb(null, new Buffer(data_array));
|
|||
|
case 'json':
|
|||
|
return cb(null, JSON.parse(req.responseText));
|
|||
|
}
|
|||
|
} else {
|
|||
|
return cb(new ApiError(req.status, "XHR error."));
|
|||
|
}
|
|||
|
}
|
|||
|
};
|
|||
|
req.send();
|
|||
|
}
|
|||
|
|
|||
|
function asyncDownloadFileIE(p, type, cb) {
|
|||
|
downloadFileIE(true, p, type, cb);
|
|||
|
}
|
|||
|
|
|||
|
function syncDownloadFileIE(p, type) {
|
|||
|
var rv;
|
|||
|
downloadFileIE(false, p, type, function (err, data) {
|
|||
|
if (err)
|
|||
|
throw err;
|
|||
|
rv = data;
|
|||
|
});
|
|||
|
return rv;
|
|||
|
}
|
|||
|
|
|||
|
function asyncDownloadFileModern(p, type, cb) {
|
|||
|
var req = new XMLHttpRequest();
|
|||
|
req.open('GET', p, true);
|
|||
|
var jsonSupported = true;
|
|||
|
switch (type) {
|
|||
|
case 'buffer':
|
|||
|
req.responseType = 'arraybuffer';
|
|||
|
break;
|
|||
|
case 'json':
|
|||
|
try {
|
|||
|
req.responseType = 'json';
|
|||
|
jsonSupported = req.responseType === 'json';
|
|||
|
} catch (e) {
|
|||
|
jsonSupported = false;
|
|||
|
}
|
|||
|
break;
|
|||
|
default:
|
|||
|
return cb(new ApiError(9 /* EINVAL */, "Invalid download type: " + type));
|
|||
|
}
|
|||
|
req.onreadystatechange = function (e) {
|
|||
|
if (req.readyState === 4) {
|
|||
|
if (req.status === 200) {
|
|||
|
switch (type) {
|
|||
|
case 'buffer':
|
|||
|
// XXX: WebKit-based browsers return *null* when XHRing an empty file.
|
|||
|
return cb(null, new Buffer(req.response ? req.response : 0));
|
|||
|
case 'json':
|
|||
|
if (jsonSupported) {
|
|||
|
return cb(null, req.response);
|
|||
|
} else {
|
|||
|
return cb(null, JSON.parse(req.responseText));
|
|||
|
}
|
|||
|
}
|
|||
|
} else {
|
|||
|
return cb(new ApiError(req.status, "XHR error."));
|
|||
|
}
|
|||
|
}
|
|||
|
};
|
|||
|
req.send();
|
|||
|
}
|
|||
|
|
|||
|
function syncDownloadFileModern(p, type) {
|
|||
|
var req = new XMLHttpRequest();
|
|||
|
req.open('GET', p, false);
|
|||
|
|
|||
|
// On most platforms, we cannot set the responseType of synchronous downloads.
|
|||
|
// @todo Test for this; IE10 allows this, as do older versions of Chrome/FF.
|
|||
|
var data = null;
|
|||
|
var err = null;
|
|||
|
|
|||
|
// Classic hack to download binary data as a string.
|
|||
|
req.overrideMimeType('text/plain; charset=x-user-defined');
|
|||
|
req.onreadystatechange = function (e) {
|
|||
|
if (req.readyState === 4) {
|
|||
|
if (req.status === 200) {
|
|||
|
switch (type) {
|
|||
|
case 'buffer':
|
|||
|
// Convert the text into a buffer.
|
|||
|
var text = req.responseText;
|
|||
|
data = new Buffer(text.length);
|
|||
|
|
|||
|
for (var i = 0; i < text.length; i++) {
|
|||
|
// This will automatically throw away the upper bit of each
|
|||
|
// character for us.
|
|||
|
data.writeUInt8(text.charCodeAt(i), i);
|
|||
|
}
|
|||
|
return;
|
|||
|
case 'json':
|
|||
|
data = JSON.parse(req.responseText);
|
|||
|
return;
|
|||
|
}
|
|||
|
} else {
|
|||
|
err = new ApiError(req.status, "XHR error.");
|
|||
|
return;
|
|||
|
}
|
|||
|
}
|
|||
|
};
|
|||
|
req.send();
|
|||
|
if (err) {
|
|||
|
throw err;
|
|||
|
}
|
|||
|
return data;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
|
|||
|
function syncDownloadFileIE10(p, type) {
|
|||
|
var req = new XMLHttpRequest();
|
|||
|
req.open('GET', p, false);
|
|||
|
switch (type) {
|
|||
|
case 'buffer':
|
|||
|
req.responseType = 'arraybuffer';
|
|||
|
break;
|
|||
|
case 'json':
|
|||
|
break;
|
|||
|
default:
|
|||
|
throw new ApiError(9 /* EINVAL */, "Invalid download type: " + type);
|
|||
|
}
|
|||
|
var data;
|
|||
|
var err;
|
|||
|
req.onreadystatechange = function (e) {
|
|||
|
if (req.readyState === 4) {
|
|||
|
if (req.status === 200) {
|
|||
|
switch (type) {
|
|||
|
case 'buffer':
|
|||
|
data = new Buffer(req.response);
|
|||
|
break;
|
|||
|
case 'json':
|
|||
|
data = JSON.parse(req.response);
|
|||
|
break;
|
|||
|
}
|
|||
|
} else {
|
|||
|
err = new ApiError(req.status, "XHR error.");
|
|||
|
}
|
|||
|
}
|
|||
|
};
|
|||
|
req.send();
|
|||
|
if (err) {
|
|||
|
throw err;
|
|||
|
}
|
|||
|
return data;
|
|||
|
}
|
|||
|
|
|||
|
function getFileSize(async, p, cb) {
|
|||
|
var req = new XMLHttpRequest();
|
|||
|
req.open('HEAD', p, async);
|
|||
|
req.onreadystatechange = function (e) {
|
|||
|
if (req.readyState === 4) {
|
|||
|
if (req.status == 200) {
|
|||
|
try {
|
|||
|
return cb(null, parseInt(req.getResponseHeader('Content-Length'), 10));
|
|||
|
} catch (e) {
|
|||
|
// In the event that the header isn't present or there is an error...
|
|||
|
return cb(new ApiError(2 /* EIO */, "XHR HEAD error: Could not read content-length."));
|
|||
|
}
|
|||
|
} else {
|
|||
|
return cb(new ApiError(req.status, "XHR HEAD error."));
|
|||
|
}
|
|||
|
}
|
|||
|
};
|
|||
|
req.send();
|
|||
|
}
|
|||
|
|
|||
|
/**
|
|||
|
* Asynchronously download a file as a buffer or a JSON object.
|
|||
|
* Note that the third function signature with a non-specialized type is
|
|||
|
* invalid, but TypeScript requires it when you specialize string arguments to
|
|||
|
* constants.
|
|||
|
*/
|
|||
|
exports.asyncDownloadFile = (util.isIE && typeof Blob === 'undefined') ? asyncDownloadFileIE : asyncDownloadFileModern;
|
|||
|
|
|||
|
/**
|
|||
|
* Synchronously download a file as a buffer or a JSON object.
|
|||
|
* Note that the third function signature with a non-specialized type is
|
|||
|
* invalid, but TypeScript requires it when you specialize string arguments to
|
|||
|
* constants.
|
|||
|
*/
|
|||
|
exports.syncDownloadFile = (util.isIE && typeof Blob === 'undefined') ? syncDownloadFileIE : (util.isIE && typeof Blob !== 'undefined') ? syncDownloadFileIE10 : syncDownloadFileModern;
|
|||
|
|
|||
|
/**
|
|||
|
* Synchronously retrieves the size of the given file in bytes.
|
|||
|
*/
|
|||
|
function getFileSizeSync(p) {
|
|||
|
var rv;
|
|||
|
getFileSize(false, p, function (err, size) {
|
|||
|
if (err) {
|
|||
|
throw err;
|
|||
|
}
|
|||
|
rv = size;
|
|||
|
});
|
|||
|
return rv;
|
|||
|
}
|
|||
|
exports.getFileSizeSync = getFileSizeSync;
|
|||
|
|
|||
|
/**
|
|||
|
* Asynchronously retrieves the size of the given file in bytes.
|
|||
|
*/
|
|||
|
function getFileSizeAsync(p, cb) {
|
|||
|
getFileSize(true, p, cb);
|
|||
|
}
|
|||
|
exports.getFileSizeAsync = getFileSizeAsync;
|
|||
|
});
|
|||
|
//# sourceMappingURL=xhr.js.map
|
|||
|
;
|
|||
|
var __extends = this.__extends || function (d, b) {
|
|||
|
for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p];
|
|||
|
function __() { this.constructor = d; }
|
|||
|
__.prototype = b.prototype;
|
|||
|
d.prototype = new __();
|
|||
|
};
|
|||
|
define('backend/XmlHttpRequest',["require", "exports", '../core/file_system', '../generic/file_index', '../core/buffer', '../core/api_error', '../core/file_flag', '../generic/preload_file', '../core/browserfs', '../generic/xhr'], function(require, exports, file_system, file_index, buffer, api_error, file_flag, preload_file, browserfs, xhr) {
|
|||
|
var Buffer = buffer.Buffer;
|
|||
|
var ApiError = api_error.ApiError;
|
|||
|
var ErrorCode = api_error.ErrorCode;
|
|||
|
var FileFlag = file_flag.FileFlag;
|
|||
|
var ActionType = file_flag.ActionType;
|
|||
|
|
|||
|
/**
|
|||
|
* A simple filesystem backed by XmlHttpRequests.
|
|||
|
*/
|
|||
|
var XmlHttpRequest = (function (_super) {
|
|||
|
__extends(XmlHttpRequest, _super);
|
|||
|
/**
|
|||
|
* Constructs the file system.
|
|||
|
* @param [String] listing_url The path to the JSON file index generated by
|
|||
|
* tools/XHRIndexer.coffee. This can be relative to the current webpage URL
|
|||
|
* or absolutely specified.
|
|||
|
* @param [String] prefix_url The url prefix to use for all web-server requests.
|
|||
|
*/
|
|||
|
function XmlHttpRequest(listing_url, prefix_url) {
|
|||
|
if (typeof prefix_url === "undefined") { prefix_url = ''; }
|
|||
|
_super.call(this);
|
|||
|
this.prefix_url = prefix_url;
|
|||
|
if (listing_url == null) {
|
|||
|
listing_url = 'index.json';
|
|||
|
}
|
|||
|
|
|||
|
// prefix_url must end in a directory separator.
|
|||
|
if (prefix_url.length > 0 && prefix_url.charAt(prefix_url.length - 1) !== '/') {
|
|||
|
prefix_url = prefix_url + '/';
|
|||
|
}
|
|||
|
var listing = this._requestFileSync(listing_url, 'json');
|
|||
|
if (listing == null) {
|
|||
|
throw new Error("Unable to find listing at URL: " + listing_url);
|
|||
|
}
|
|||
|
this._index = file_index.FileIndex.from_listing(listing);
|
|||
|
}
|
|||
|
XmlHttpRequest.prototype.empty = function () {
|
|||
|
this._index.fileIterator(function (file) {
|
|||
|
file.file_data = null;
|
|||
|
});
|
|||
|
};
|
|||
|
|
|||
|
XmlHttpRequest.prototype.getXhrPath = function (filePath) {
|
|||
|
if (filePath.charAt(0) === '/') {
|
|||
|
filePath = filePath.slice(1);
|
|||
|
}
|
|||
|
return this.prefix_url + filePath;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Only requests the HEAD content, for the file size.
|
|||
|
*/
|
|||
|
XmlHttpRequest.prototype._requestFileSizeAsync = function (path, cb) {
|
|||
|
xhr.getFileSizeAsync(this.getXhrPath(path), cb);
|
|||
|
};
|
|||
|
XmlHttpRequest.prototype._requestFileSizeSync = function (path) {
|
|||
|
return xhr.getFileSizeSync(this.getXhrPath(path));
|
|||
|
};
|
|||
|
|
|||
|
XmlHttpRequest.prototype._requestFileAsync = function (p, type, cb) {
|
|||
|
xhr.asyncDownloadFile(this.getXhrPath(p), type, cb);
|
|||
|
};
|
|||
|
|
|||
|
XmlHttpRequest.prototype._requestFileSync = function (p, type) {
|
|||
|
return xhr.syncDownloadFile(this.getXhrPath(p), type);
|
|||
|
};
|
|||
|
|
|||
|
XmlHttpRequest.prototype.getName = function () {
|
|||
|
return 'XmlHttpRequest';
|
|||
|
};
|
|||
|
|
|||
|
XmlHttpRequest.isAvailable = function () {
|
|||
|
// @todo Older browsers use a different name for XHR, iirc.
|
|||
|
return typeof XMLHttpRequest !== "undefined" && XMLHttpRequest !== null;
|
|||
|
};
|
|||
|
|
|||
|
XmlHttpRequest.prototype.diskSpace = function (path, cb) {
|
|||
|
// Read-only file system. We could calculate the total space, but that's not
|
|||
|
// important right now.
|
|||
|
cb(0, 0);
|
|||
|
};
|
|||
|
|
|||
|
XmlHttpRequest.prototype.isReadOnly = function () {
|
|||
|
return true;
|
|||
|
};
|
|||
|
|
|||
|
XmlHttpRequest.prototype.supportsLinks = function () {
|
|||
|
return false;
|
|||
|
};
|
|||
|
|
|||
|
XmlHttpRequest.prototype.supportsProps = function () {
|
|||
|
return false;
|
|||
|
};
|
|||
|
|
|||
|
XmlHttpRequest.prototype.supportsSynch = function () {
|
|||
|
return true;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Special XHR function: Preload the given file into the index.
|
|||
|
* @param [String] path
|
|||
|
* @param [BrowserFS.Buffer] buffer
|
|||
|
*/
|
|||
|
XmlHttpRequest.prototype.preloadFile = function (path, buffer) {
|
|||
|
var inode = this._index.getInode(path);
|
|||
|
if (inode === null) {
|
|||
|
throw ApiError.ENOENT(path);
|
|||
|
}
|
|||
|
var stats = inode.getData();
|
|||
|
stats.size = buffer.length;
|
|||
|
stats.file_data = buffer;
|
|||
|
};
|
|||
|
|
|||
|
XmlHttpRequest.prototype.stat = function (path, isLstat, cb) {
|
|||
|
var inode = this._index.getInode(path);
|
|||
|
if (inode === null) {
|
|||
|
return cb(ApiError.ENOENT(path));
|
|||
|
}
|
|||
|
var stats;
|
|||
|
if (inode.isFile()) {
|
|||
|
stats = inode.getData();
|
|||
|
|
|||
|
// At this point, a non-opened file will still have default stats from the listing.
|
|||
|
if (stats.size < 0) {
|
|||
|
this._requestFileSizeAsync(path, function (e, size) {
|
|||
|
if (e) {
|
|||
|
return cb(e);
|
|||
|
}
|
|||
|
stats.size = size;
|
|||
|
cb(null, stats.clone());
|
|||
|
});
|
|||
|
} else {
|
|||
|
cb(null, stats.clone());
|
|||
|
}
|
|||
|
} else {
|
|||
|
stats = inode.getStats();
|
|||
|
cb(null, stats);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
XmlHttpRequest.prototype.statSync = function (path, isLstat) {
|
|||
|
var inode = this._index.getInode(path);
|
|||
|
if (inode === null) {
|
|||
|
throw ApiError.ENOENT(path);
|
|||
|
}
|
|||
|
var stats;
|
|||
|
if (inode.isFile()) {
|
|||
|
stats = inode.getData();
|
|||
|
|
|||
|
// At this point, a non-opened file will still have default stats from the listing.
|
|||
|
if (stats.size < 0) {
|
|||
|
stats.size = this._requestFileSizeSync(path);
|
|||
|
}
|
|||
|
} else {
|
|||
|
stats = inode.getStats();
|
|||
|
}
|
|||
|
return stats;
|
|||
|
};
|
|||
|
|
|||
|
XmlHttpRequest.prototype.open = function (path, flags, mode, cb) {
|
|||
|
// INVARIANT: You can't write to files on this file system.
|
|||
|
if (flags.isWriteable()) {
|
|||
|
return cb(new ApiError(0 /* EPERM */, path));
|
|||
|
}
|
|||
|
var _this = this;
|
|||
|
|
|||
|
// Check if the path exists, and is a file.
|
|||
|
var inode = this._index.getInode(path);
|
|||
|
if (inode === null) {
|
|||
|
return cb(ApiError.ENOENT(path));
|
|||
|
}
|
|||
|
if (inode.isDir()) {
|
|||
|
return cb(ApiError.EISDIR(path));
|
|||
|
}
|
|||
|
var stats = inode.getData();
|
|||
|
switch (flags.pathExistsAction()) {
|
|||
|
case 1 /* THROW_EXCEPTION */:
|
|||
|
case 2 /* TRUNCATE_FILE */:
|
|||
|
return cb(ApiError.EEXIST(path));
|
|||
|
case 0 /* NOP */:
|
|||
|
// Use existing file contents.
|
|||
|
// XXX: Uh, this maintains the previously-used flag.
|
|||
|
if (stats.file_data != null) {
|
|||
|
return cb(null, new preload_file.NoSyncFile(_this, path, flags, stats.clone(), stats.file_data));
|
|||
|
}
|
|||
|
|
|||
|
// @todo be lazier about actually requesting the file
|
|||
|
this._requestFileAsync(path, 'buffer', function (err, buffer) {
|
|||
|
if (err) {
|
|||
|
return cb(err);
|
|||
|
}
|
|||
|
|
|||
|
// we don't initially have file sizes
|
|||
|
stats.size = buffer.length;
|
|||
|
stats.file_data = buffer;
|
|||
|
return cb(null, new preload_file.NoSyncFile(_this, path, flags, stats.clone(), buffer));
|
|||
|
});
|
|||
|
break;
|
|||
|
default:
|
|||
|
return cb(new ApiError(9 /* EINVAL */, 'Invalid FileMode object.'));
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
XmlHttpRequest.prototype.openSync = function (path, flags, mode) {
|
|||
|
// INVARIANT: You can't write to files on this file system.
|
|||
|
if (flags.isWriteable()) {
|
|||
|
throw new ApiError(0 /* EPERM */, path);
|
|||
|
}
|
|||
|
|
|||
|
// Check if the path exists, and is a file.
|
|||
|
var inode = this._index.getInode(path);
|
|||
|
if (inode === null) {
|
|||
|
throw ApiError.ENOENT(path);
|
|||
|
}
|
|||
|
if (inode.isDir()) {
|
|||
|
throw ApiError.EISDIR(path);
|
|||
|
}
|
|||
|
var stats = inode.getData();
|
|||
|
switch (flags.pathExistsAction()) {
|
|||
|
case 1 /* THROW_EXCEPTION */:
|
|||
|
case 2 /* TRUNCATE_FILE */:
|
|||
|
throw ApiError.EEXIST(path);
|
|||
|
case 0 /* NOP */:
|
|||
|
// Use existing file contents.
|
|||
|
// XXX: Uh, this maintains the previously-used flag.
|
|||
|
if (stats.file_data != null) {
|
|||
|
return new preload_file.NoSyncFile(this, path, flags, stats.clone(), stats.file_data);
|
|||
|
}
|
|||
|
|
|||
|
// @todo be lazier about actually requesting the file
|
|||
|
var buffer = this._requestFileSync(path, 'buffer');
|
|||
|
|
|||
|
// we don't initially have file sizes
|
|||
|
stats.size = buffer.length;
|
|||
|
stats.file_data = buffer;
|
|||
|
return new preload_file.NoSyncFile(this, path, flags, stats.clone(), buffer);
|
|||
|
default:
|
|||
|
throw new ApiError(9 /* EINVAL */, 'Invalid FileMode object.');
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
XmlHttpRequest.prototype.readdir = function (path, cb) {
|
|||
|
try {
|
|||
|
cb(null, this.readdirSync(path));
|
|||
|
} catch (e) {
|
|||
|
cb(e);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
XmlHttpRequest.prototype.readdirSync = function (path) {
|
|||
|
// Check if it exists.
|
|||
|
var inode = this._index.getInode(path);
|
|||
|
if (inode === null) {
|
|||
|
throw ApiError.ENOENT(path);
|
|||
|
} else if (inode.isFile()) {
|
|||
|
throw ApiError.ENOTDIR(path);
|
|||
|
}
|
|||
|
return inode.getListing();
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* We have the entire file as a buffer; optimize readFile.
|
|||
|
*/
|
|||
|
XmlHttpRequest.prototype.readFile = function (fname, encoding, flag, cb) {
|
|||
|
// Wrap cb in file closing code.
|
|||
|
var oldCb = cb;
|
|||
|
|
|||
|
// Get file.
|
|||
|
this.open(fname, flag, 0x1a4, function (err, fd) {
|
|||
|
if (err) {
|
|||
|
return cb(err);
|
|||
|
}
|
|||
|
cb = function (err, arg) {
|
|||
|
fd.close(function (err2) {
|
|||
|
if (err == null) {
|
|||
|
err = err2;
|
|||
|
}
|
|||
|
return oldCb(err, arg);
|
|||
|
});
|
|||
|
};
|
|||
|
var fdCast = fd;
|
|||
|
var fdBuff = fdCast._buffer;
|
|||
|
if (encoding === null) {
|
|||
|
if (fdBuff.length > 0) {
|
|||
|
return cb(err, fdBuff.sliceCopy());
|
|||
|
} else {
|
|||
|
return cb(err, new buffer.Buffer(0));
|
|||
|
}
|
|||
|
}
|
|||
|
try {
|
|||
|
cb(null, fdBuff.toString(encoding));
|
|||
|
} catch (e) {
|
|||
|
cb(e);
|
|||
|
}
|
|||
|
});
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Specially-optimized readfile.
|
|||
|
*/
|
|||
|
XmlHttpRequest.prototype.readFileSync = function (fname, encoding, flag) {
|
|||
|
// Get file.
|
|||
|
var fd = this.openSync(fname, flag, 0x1a4);
|
|||
|
try {
|
|||
|
var fdCast = fd;
|
|||
|
var fdBuff = fdCast._buffer;
|
|||
|
if (encoding === null) {
|
|||
|
if (fdBuff.length > 0) {
|
|||
|
return fdBuff.sliceCopy();
|
|||
|
} else {
|
|||
|
return new buffer.Buffer(0);
|
|||
|
}
|
|||
|
}
|
|||
|
return fdBuff.toString(encoding);
|
|||
|
} finally {
|
|||
|
fd.closeSync();
|
|||
|
}
|
|||
|
};
|
|||
|
return XmlHttpRequest;
|
|||
|
})(file_system.BaseFileSystem);
|
|||
|
exports.XmlHttpRequest = XmlHttpRequest;
|
|||
|
|
|||
|
browserfs.registerFileSystem('XmlHttpRequest', XmlHttpRequest);
|
|||
|
});
|
|||
|
//# sourceMappingURL=XmlHttpRequest.js.map
|
|||
|
;
|
|||
|
/*global setImmediate: false, setTimeout: false, console: false */
|
|||
|
(function () {
|
|||
|
|
|||
|
var async = {};
|
|||
|
|
|||
|
// global on the server, window in the browser
|
|||
|
var root, previous_async;
|
|||
|
|
|||
|
root = this;
|
|||
|
if (root != null) {
|
|||
|
previous_async = root.async;
|
|||
|
}
|
|||
|
|
|||
|
async.noConflict = function () {
|
|||
|
root.async = previous_async;
|
|||
|
return async;
|
|||
|
};
|
|||
|
|
|||
|
function only_once(fn) {
|
|||
|
var called = false;
|
|||
|
return function() {
|
|||
|
if (called) throw new Error("Callback was already called.");
|
|||
|
called = true;
|
|||
|
fn.apply(root, arguments);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
//// cross-browser compatiblity functions ////
|
|||
|
|
|||
|
var _each = function (arr, iterator) {
|
|||
|
if (arr.forEach) {
|
|||
|
return arr.forEach(iterator);
|
|||
|
}
|
|||
|
for (var i = 0; i < arr.length; i += 1) {
|
|||
|
iterator(arr[i], i, arr);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
var _map = function (arr, iterator) {
|
|||
|
if (arr.map) {
|
|||
|
return arr.map(iterator);
|
|||
|
}
|
|||
|
var results = [];
|
|||
|
_each(arr, function (x, i, a) {
|
|||
|
results.push(iterator(x, i, a));
|
|||
|
});
|
|||
|
return results;
|
|||
|
};
|
|||
|
|
|||
|
var _reduce = function (arr, iterator, memo) {
|
|||
|
if (arr.reduce) {
|
|||
|
return arr.reduce(iterator, memo);
|
|||
|
}
|
|||
|
_each(arr, function (x, i, a) {
|
|||
|
memo = iterator(memo, x, i, a);
|
|||
|
});
|
|||
|
return memo;
|
|||
|
};
|
|||
|
|
|||
|
var _keys = function (obj) {
|
|||
|
if (Object.keys) {
|
|||
|
return Object.keys(obj);
|
|||
|
}
|
|||
|
var keys = [];
|
|||
|
for (var k in obj) {
|
|||
|
if (obj.hasOwnProperty(k)) {
|
|||
|
keys.push(k);
|
|||
|
}
|
|||
|
}
|
|||
|
return keys;
|
|||
|
};
|
|||
|
|
|||
|
//// exported async module functions ////
|
|||
|
|
|||
|
//// nextTick implementation with browser-compatible fallback ////
|
|||
|
if (typeof process === 'undefined' || !(process.nextTick)) {
|
|||
|
if (typeof setImmediate === 'function') {
|
|||
|
async.nextTick = function (fn) {
|
|||
|
// not a direct alias for IE10 compatibility
|
|||
|
setImmediate(fn);
|
|||
|
};
|
|||
|
async.setImmediate = async.nextTick;
|
|||
|
}
|
|||
|
else {
|
|||
|
async.nextTick = function (fn) {
|
|||
|
setTimeout(fn, 0);
|
|||
|
};
|
|||
|
async.setImmediate = async.nextTick;
|
|||
|
}
|
|||
|
}
|
|||
|
else {
|
|||
|
async.nextTick = process.nextTick;
|
|||
|
if (typeof setImmediate !== 'undefined') {
|
|||
|
async.setImmediate = function (fn) {
|
|||
|
// not a direct alias for IE10 compatibility
|
|||
|
setImmediate(fn);
|
|||
|
};
|
|||
|
}
|
|||
|
else {
|
|||
|
async.setImmediate = async.nextTick;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
async.each = function (arr, iterator, callback) {
|
|||
|
callback = callback || function () {};
|
|||
|
if (!arr.length) {
|
|||
|
return callback();
|
|||
|
}
|
|||
|
var completed = 0;
|
|||
|
_each(arr, function (x) {
|
|||
|
iterator(x, only_once(function (err) {
|
|||
|
if (err) {
|
|||
|
callback(err);
|
|||
|
callback = function () {};
|
|||
|
}
|
|||
|
else {
|
|||
|
completed += 1;
|
|||
|
if (completed >= arr.length) {
|
|||
|
callback(null);
|
|||
|
}
|
|||
|
}
|
|||
|
}));
|
|||
|
});
|
|||
|
};
|
|||
|
async.forEach = async.each;
|
|||
|
|
|||
|
async.eachSeries = function (arr, iterator, callback) {
|
|||
|
callback = callback || function () {};
|
|||
|
if (!arr.length) {
|
|||
|
return callback();
|
|||
|
}
|
|||
|
var completed = 0;
|
|||
|
var iterate = function () {
|
|||
|
iterator(arr[completed], function (err) {
|
|||
|
if (err) {
|
|||
|
callback(err);
|
|||
|
callback = function () {};
|
|||
|
}
|
|||
|
else {
|
|||
|
completed += 1;
|
|||
|
if (completed >= arr.length) {
|
|||
|
callback(null);
|
|||
|
}
|
|||
|
else {
|
|||
|
iterate();
|
|||
|
}
|
|||
|
}
|
|||
|
});
|
|||
|
};
|
|||
|
iterate();
|
|||
|
};
|
|||
|
async.forEachSeries = async.eachSeries;
|
|||
|
|
|||
|
async.eachLimit = function (arr, limit, iterator, callback) {
|
|||
|
var fn = _eachLimit(limit);
|
|||
|
fn.apply(null, [arr, iterator, callback]);
|
|||
|
};
|
|||
|
async.forEachLimit = async.eachLimit;
|
|||
|
|
|||
|
var _eachLimit = function (limit) {
|
|||
|
|
|||
|
return function (arr, iterator, callback) {
|
|||
|
callback = callback || function () {};
|
|||
|
if (!arr.length || limit <= 0) {
|
|||
|
return callback();
|
|||
|
}
|
|||
|
var completed = 0;
|
|||
|
var started = 0;
|
|||
|
var running = 0;
|
|||
|
|
|||
|
(function replenish () {
|
|||
|
if (completed >= arr.length) {
|
|||
|
return callback();
|
|||
|
}
|
|||
|
|
|||
|
while (running < limit && started < arr.length) {
|
|||
|
started += 1;
|
|||
|
running += 1;
|
|||
|
iterator(arr[started - 1], function (err) {
|
|||
|
if (err) {
|
|||
|
callback(err);
|
|||
|
callback = function () {};
|
|||
|
}
|
|||
|
else {
|
|||
|
completed += 1;
|
|||
|
running -= 1;
|
|||
|
if (completed >= arr.length) {
|
|||
|
callback();
|
|||
|
}
|
|||
|
else {
|
|||
|
replenish();
|
|||
|
}
|
|||
|
}
|
|||
|
});
|
|||
|
}
|
|||
|
})();
|
|||
|
};
|
|||
|
};
|
|||
|
|
|||
|
|
|||
|
var doParallel = function (fn) {
|
|||
|
return function () {
|
|||
|
var args = Array.prototype.slice.call(arguments);
|
|||
|
return fn.apply(null, [async.each].concat(args));
|
|||
|
};
|
|||
|
};
|
|||
|
var doParallelLimit = function(limit, fn) {
|
|||
|
return function () {
|
|||
|
var args = Array.prototype.slice.call(arguments);
|
|||
|
return fn.apply(null, [_eachLimit(limit)].concat(args));
|
|||
|
};
|
|||
|
};
|
|||
|
var doSeries = function (fn) {
|
|||
|
return function () {
|
|||
|
var args = Array.prototype.slice.call(arguments);
|
|||
|
return fn.apply(null, [async.eachSeries].concat(args));
|
|||
|
};
|
|||
|
};
|
|||
|
|
|||
|
|
|||
|
var _asyncMap = function (eachfn, arr, iterator, callback) {
|
|||
|
var results = [];
|
|||
|
arr = _map(arr, function (x, i) {
|
|||
|
return {index: i, value: x};
|
|||
|
});
|
|||
|
eachfn(arr, function (x, callback) {
|
|||
|
iterator(x.value, function (err, v) {
|
|||
|
results[x.index] = v;
|
|||
|
callback(err);
|
|||
|
});
|
|||
|
}, function (err) {
|
|||
|
callback(err, results);
|
|||
|
});
|
|||
|
};
|
|||
|
async.map = doParallel(_asyncMap);
|
|||
|
async.mapSeries = doSeries(_asyncMap);
|
|||
|
async.mapLimit = function (arr, limit, iterator, callback) {
|
|||
|
return _mapLimit(limit)(arr, iterator, callback);
|
|||
|
};
|
|||
|
|
|||
|
var _mapLimit = function(limit) {
|
|||
|
return doParallelLimit(limit, _asyncMap);
|
|||
|
};
|
|||
|
|
|||
|
// reduce only has a series version, as doing reduce in parallel won't
|
|||
|
// work in many situations.
|
|||
|
async.reduce = function (arr, memo, iterator, callback) {
|
|||
|
async.eachSeries(arr, function (x, callback) {
|
|||
|
iterator(memo, x, function (err, v) {
|
|||
|
memo = v;
|
|||
|
callback(err);
|
|||
|
});
|
|||
|
}, function (err) {
|
|||
|
callback(err, memo);
|
|||
|
});
|
|||
|
};
|
|||
|
// inject alias
|
|||
|
async.inject = async.reduce;
|
|||
|
// foldl alias
|
|||
|
async.foldl = async.reduce;
|
|||
|
|
|||
|
async.reduceRight = function (arr, memo, iterator, callback) {
|
|||
|
var reversed = _map(arr, function (x) {
|
|||
|
return x;
|
|||
|
}).reverse();
|
|||
|
async.reduce(reversed, memo, iterator, callback);
|
|||
|
};
|
|||
|
// foldr alias
|
|||
|
async.foldr = async.reduceRight;
|
|||
|
|
|||
|
var _filter = function (eachfn, arr, iterator, callback) {
|
|||
|
var results = [];
|
|||
|
arr = _map(arr, function (x, i) {
|
|||
|
return {index: i, value: x};
|
|||
|
});
|
|||
|
eachfn(arr, function (x, callback) {
|
|||
|
iterator(x.value, function (v) {
|
|||
|
if (v) {
|
|||
|
results.push(x);
|
|||
|
}
|
|||
|
callback();
|
|||
|
});
|
|||
|
}, function (err) {
|
|||
|
callback(_map(results.sort(function (a, b) {
|
|||
|
return a.index - b.index;
|
|||
|
}), function (x) {
|
|||
|
return x.value;
|
|||
|
}));
|
|||
|
});
|
|||
|
};
|
|||
|
async.filter = doParallel(_filter);
|
|||
|
async.filterSeries = doSeries(_filter);
|
|||
|
// select alias
|
|||
|
async.select = async.filter;
|
|||
|
async.selectSeries = async.filterSeries;
|
|||
|
|
|||
|
var _reject = function (eachfn, arr, iterator, callback) {
|
|||
|
var results = [];
|
|||
|
arr = _map(arr, function (x, i) {
|
|||
|
return {index: i, value: x};
|
|||
|
});
|
|||
|
eachfn(arr, function (x, callback) {
|
|||
|
iterator(x.value, function (v) {
|
|||
|
if (!v) {
|
|||
|
results.push(x);
|
|||
|
}
|
|||
|
callback();
|
|||
|
});
|
|||
|
}, function (err) {
|
|||
|
callback(_map(results.sort(function (a, b) {
|
|||
|
return a.index - b.index;
|
|||
|
}), function (x) {
|
|||
|
return x.value;
|
|||
|
}));
|
|||
|
});
|
|||
|
};
|
|||
|
async.reject = doParallel(_reject);
|
|||
|
async.rejectSeries = doSeries(_reject);
|
|||
|
|
|||
|
var _detect = function (eachfn, arr, iterator, main_callback) {
|
|||
|
eachfn(arr, function (x, callback) {
|
|||
|
iterator(x, function (result) {
|
|||
|
if (result) {
|
|||
|
main_callback(x);
|
|||
|
main_callback = function () {};
|
|||
|
}
|
|||
|
else {
|
|||
|
callback();
|
|||
|
}
|
|||
|
});
|
|||
|
}, function (err) {
|
|||
|
main_callback();
|
|||
|
});
|
|||
|
};
|
|||
|
async.detect = doParallel(_detect);
|
|||
|
async.detectSeries = doSeries(_detect);
|
|||
|
|
|||
|
async.some = function (arr, iterator, main_callback) {
|
|||
|
async.each(arr, function (x, callback) {
|
|||
|
iterator(x, function (v) {
|
|||
|
if (v) {
|
|||
|
main_callback(true);
|
|||
|
main_callback = function () {};
|
|||
|
}
|
|||
|
callback();
|
|||
|
});
|
|||
|
}, function (err) {
|
|||
|
main_callback(false);
|
|||
|
});
|
|||
|
};
|
|||
|
// any alias
|
|||
|
async.any = async.some;
|
|||
|
|
|||
|
async.every = function (arr, iterator, main_callback) {
|
|||
|
async.each(arr, function (x, callback) {
|
|||
|
iterator(x, function (v) {
|
|||
|
if (!v) {
|
|||
|
main_callback(false);
|
|||
|
main_callback = function () {};
|
|||
|
}
|
|||
|
callback();
|
|||
|
});
|
|||
|
}, function (err) {
|
|||
|
main_callback(true);
|
|||
|
});
|
|||
|
};
|
|||
|
// all alias
|
|||
|
async.all = async.every;
|
|||
|
|
|||
|
async.sortBy = function (arr, iterator, callback) {
|
|||
|
async.map(arr, function (x, callback) {
|
|||
|
iterator(x, function (err, criteria) {
|
|||
|
if (err) {
|
|||
|
callback(err);
|
|||
|
}
|
|||
|
else {
|
|||
|
callback(null, {value: x, criteria: criteria});
|
|||
|
}
|
|||
|
});
|
|||
|
}, function (err, results) {
|
|||
|
if (err) {
|
|||
|
return callback(err);
|
|||
|
}
|
|||
|
else {
|
|||
|
var fn = function (left, right) {
|
|||
|
var a = left.criteria, b = right.criteria;
|
|||
|
return a < b ? -1 : a > b ? 1 : 0;
|
|||
|
};
|
|||
|
callback(null, _map(results.sort(fn), function (x) {
|
|||
|
return x.value;
|
|||
|
}));
|
|||
|
}
|
|||
|
});
|
|||
|
};
|
|||
|
|
|||
|
async.auto = function (tasks, callback) {
|
|||
|
callback = callback || function () {};
|
|||
|
var keys = _keys(tasks);
|
|||
|
if (!keys.length) {
|
|||
|
return callback(null);
|
|||
|
}
|
|||
|
|
|||
|
var results = {};
|
|||
|
|
|||
|
var listeners = [];
|
|||
|
var addListener = function (fn) {
|
|||
|
listeners.unshift(fn);
|
|||
|
};
|
|||
|
var removeListener = function (fn) {
|
|||
|
for (var i = 0; i < listeners.length; i += 1) {
|
|||
|
if (listeners[i] === fn) {
|
|||
|
listeners.splice(i, 1);
|
|||
|
return;
|
|||
|
}
|
|||
|
}
|
|||
|
};
|
|||
|
var taskComplete = function () {
|
|||
|
_each(listeners.slice(0), function (fn) {
|
|||
|
fn();
|
|||
|
});
|
|||
|
};
|
|||
|
|
|||
|
addListener(function () {
|
|||
|
if (_keys(results).length === keys.length) {
|
|||
|
callback(null, results);
|
|||
|
callback = function () {};
|
|||
|
}
|
|||
|
});
|
|||
|
|
|||
|
_each(keys, function (k) {
|
|||
|
var task = (tasks[k] instanceof Function) ? [tasks[k]]: tasks[k];
|
|||
|
var taskCallback = function (err) {
|
|||
|
var args = Array.prototype.slice.call(arguments, 1);
|
|||
|
if (args.length <= 1) {
|
|||
|
args = args[0];
|
|||
|
}
|
|||
|
if (err) {
|
|||
|
var safeResults = {};
|
|||
|
_each(_keys(results), function(rkey) {
|
|||
|
safeResults[rkey] = results[rkey];
|
|||
|
});
|
|||
|
safeResults[k] = args;
|
|||
|
callback(err, safeResults);
|
|||
|
// stop subsequent errors hitting callback multiple times
|
|||
|
callback = function () {};
|
|||
|
}
|
|||
|
else {
|
|||
|
results[k] = args;
|
|||
|
async.setImmediate(taskComplete);
|
|||
|
}
|
|||
|
};
|
|||
|
var requires = task.slice(0, Math.abs(task.length - 1)) || [];
|
|||
|
var ready = function () {
|
|||
|
return _reduce(requires, function (a, x) {
|
|||
|
return (a && results.hasOwnProperty(x));
|
|||
|
}, true) && !results.hasOwnProperty(k);
|
|||
|
};
|
|||
|
if (ready()) {
|
|||
|
task[task.length - 1](taskCallback, results);
|
|||
|
}
|
|||
|
else {
|
|||
|
var listener = function () {
|
|||
|
if (ready()) {
|
|||
|
removeListener(listener);
|
|||
|
task[task.length - 1](taskCallback, results);
|
|||
|
}
|
|||
|
};
|
|||
|
addListener(listener);
|
|||
|
}
|
|||
|
});
|
|||
|
};
|
|||
|
|
|||
|
async.waterfall = function (tasks, callback) {
|
|||
|
callback = callback || function () {};
|
|||
|
if (tasks.constructor !== Array) {
|
|||
|
var err = new Error('First argument to waterfall must be an array of functions');
|
|||
|
return callback(err);
|
|||
|
}
|
|||
|
if (!tasks.length) {
|
|||
|
return callback();
|
|||
|
}
|
|||
|
var wrapIterator = function (iterator) {
|
|||
|
return function (err) {
|
|||
|
if (err) {
|
|||
|
callback.apply(null, arguments);
|
|||
|
callback = function () {};
|
|||
|
}
|
|||
|
else {
|
|||
|
var args = Array.prototype.slice.call(arguments, 1);
|
|||
|
var next = iterator.next();
|
|||
|
if (next) {
|
|||
|
args.push(wrapIterator(next));
|
|||
|
}
|
|||
|
else {
|
|||
|
args.push(callback);
|
|||
|
}
|
|||
|
async.setImmediate(function () {
|
|||
|
iterator.apply(null, args);
|
|||
|
});
|
|||
|
}
|
|||
|
};
|
|||
|
};
|
|||
|
wrapIterator(async.iterator(tasks))();
|
|||
|
};
|
|||
|
|
|||
|
var _parallel = function(eachfn, tasks, callback) {
|
|||
|
callback = callback || function () {};
|
|||
|
if (tasks.constructor === Array) {
|
|||
|
eachfn.map(tasks, function (fn, callback) {
|
|||
|
if (fn) {
|
|||
|
fn(function (err) {
|
|||
|
var args = Array.prototype.slice.call(arguments, 1);
|
|||
|
if (args.length <= 1) {
|
|||
|
args = args[0];
|
|||
|
}
|
|||
|
callback.call(null, err, args);
|
|||
|
});
|
|||
|
}
|
|||
|
}, callback);
|
|||
|
}
|
|||
|
else {
|
|||
|
var results = {};
|
|||
|
eachfn.each(_keys(tasks), function (k, callback) {
|
|||
|
tasks[k](function (err) {
|
|||
|
var args = Array.prototype.slice.call(arguments, 1);
|
|||
|
if (args.length <= 1) {
|
|||
|
args = args[0];
|
|||
|
}
|
|||
|
results[k] = args;
|
|||
|
callback(err);
|
|||
|
});
|
|||
|
}, function (err) {
|
|||
|
callback(err, results);
|
|||
|
});
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
async.parallel = function (tasks, callback) {
|
|||
|
_parallel({ map: async.map, each: async.each }, tasks, callback);
|
|||
|
};
|
|||
|
|
|||
|
async.parallelLimit = function(tasks, limit, callback) {
|
|||
|
_parallel({ map: _mapLimit(limit), each: _eachLimit(limit) }, tasks, callback);
|
|||
|
};
|
|||
|
|
|||
|
async.series = function (tasks, callback) {
|
|||
|
callback = callback || function () {};
|
|||
|
if (tasks.constructor === Array) {
|
|||
|
async.mapSeries(tasks, function (fn, callback) {
|
|||
|
if (fn) {
|
|||
|
fn(function (err) {
|
|||
|
var args = Array.prototype.slice.call(arguments, 1);
|
|||
|
if (args.length <= 1) {
|
|||
|
args = args[0];
|
|||
|
}
|
|||
|
callback.call(null, err, args);
|
|||
|
});
|
|||
|
}
|
|||
|
}, callback);
|
|||
|
}
|
|||
|
else {
|
|||
|
var results = {};
|
|||
|
async.eachSeries(_keys(tasks), function (k, callback) {
|
|||
|
tasks[k](function (err) {
|
|||
|
var args = Array.prototype.slice.call(arguments, 1);
|
|||
|
if (args.length <= 1) {
|
|||
|
args = args[0];
|
|||
|
}
|
|||
|
results[k] = args;
|
|||
|
callback(err);
|
|||
|
});
|
|||
|
}, function (err) {
|
|||
|
callback(err, results);
|
|||
|
});
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
async.iterator = function (tasks) {
|
|||
|
var makeCallback = function (index) {
|
|||
|
var fn = function () {
|
|||
|
if (tasks.length) {
|
|||
|
tasks[index].apply(null, arguments);
|
|||
|
}
|
|||
|
return fn.next();
|
|||
|
};
|
|||
|
fn.next = function () {
|
|||
|
return (index < tasks.length - 1) ? makeCallback(index + 1): null;
|
|||
|
};
|
|||
|
return fn;
|
|||
|
};
|
|||
|
return makeCallback(0);
|
|||
|
};
|
|||
|
|
|||
|
async.apply = function (fn) {
|
|||
|
var args = Array.prototype.slice.call(arguments, 1);
|
|||
|
return function () {
|
|||
|
return fn.apply(
|
|||
|
null, args.concat(Array.prototype.slice.call(arguments))
|
|||
|
);
|
|||
|
};
|
|||
|
};
|
|||
|
|
|||
|
var _concat = function (eachfn, arr, fn, callback) {
|
|||
|
var r = [];
|
|||
|
eachfn(arr, function (x, cb) {
|
|||
|
fn(x, function (err, y) {
|
|||
|
r = r.concat(y || []);
|
|||
|
cb(err);
|
|||
|
});
|
|||
|
}, function (err) {
|
|||
|
callback(err, r);
|
|||
|
});
|
|||
|
};
|
|||
|
async.concat = doParallel(_concat);
|
|||
|
async.concatSeries = doSeries(_concat);
|
|||
|
|
|||
|
async.whilst = function (test, iterator, callback) {
|
|||
|
if (test()) {
|
|||
|
iterator(function (err) {
|
|||
|
if (err) {
|
|||
|
return callback(err);
|
|||
|
}
|
|||
|
async.whilst(test, iterator, callback);
|
|||
|
});
|
|||
|
}
|
|||
|
else {
|
|||
|
callback();
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
async.doWhilst = function (iterator, test, callback) {
|
|||
|
iterator(function (err) {
|
|||
|
if (err) {
|
|||
|
return callback(err);
|
|||
|
}
|
|||
|
if (test()) {
|
|||
|
async.doWhilst(iterator, test, callback);
|
|||
|
}
|
|||
|
else {
|
|||
|
callback();
|
|||
|
}
|
|||
|
});
|
|||
|
};
|
|||
|
|
|||
|
async.until = function (test, iterator, callback) {
|
|||
|
if (!test()) {
|
|||
|
iterator(function (err) {
|
|||
|
if (err) {
|
|||
|
return callback(err);
|
|||
|
}
|
|||
|
async.until(test, iterator, callback);
|
|||
|
});
|
|||
|
}
|
|||
|
else {
|
|||
|
callback();
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
async.doUntil = function (iterator, test, callback) {
|
|||
|
iterator(function (err) {
|
|||
|
if (err) {
|
|||
|
return callback(err);
|
|||
|
}
|
|||
|
if (!test()) {
|
|||
|
async.doUntil(iterator, test, callback);
|
|||
|
}
|
|||
|
else {
|
|||
|
callback();
|
|||
|
}
|
|||
|
});
|
|||
|
};
|
|||
|
|
|||
|
async.queue = function (worker, concurrency) {
|
|||
|
if (concurrency === undefined) {
|
|||
|
concurrency = 1;
|
|||
|
}
|
|||
|
function _insert(q, data, pos, callback) {
|
|||
|
if(data.constructor !== Array) {
|
|||
|
data = [data];
|
|||
|
}
|
|||
|
_each(data, function(task) {
|
|||
|
var item = {
|
|||
|
data: task,
|
|||
|
callback: typeof callback === 'function' ? callback : null
|
|||
|
};
|
|||
|
|
|||
|
if (pos) {
|
|||
|
q.tasks.unshift(item);
|
|||
|
} else {
|
|||
|
q.tasks.push(item);
|
|||
|
}
|
|||
|
|
|||
|
if (q.saturated && q.tasks.length === concurrency) {
|
|||
|
q.saturated();
|
|||
|
}
|
|||
|
async.setImmediate(q.process);
|
|||
|
});
|
|||
|
}
|
|||
|
|
|||
|
var workers = 0;
|
|||
|
var q = {
|
|||
|
tasks: [],
|
|||
|
concurrency: concurrency,
|
|||
|
saturated: null,
|
|||
|
empty: null,
|
|||
|
drain: null,
|
|||
|
push: function (data, callback) {
|
|||
|
_insert(q, data, false, callback);
|
|||
|
},
|
|||
|
unshift: function (data, callback) {
|
|||
|
_insert(q, data, true, callback);
|
|||
|
},
|
|||
|
process: function () {
|
|||
|
if (workers < q.concurrency && q.tasks.length) {
|
|||
|
var task = q.tasks.shift();
|
|||
|
if (q.empty && q.tasks.length === 0) {
|
|||
|
q.empty();
|
|||
|
}
|
|||
|
workers += 1;
|
|||
|
var next = function () {
|
|||
|
workers -= 1;
|
|||
|
if (task.callback) {
|
|||
|
task.callback.apply(task, arguments);
|
|||
|
}
|
|||
|
if (q.drain && q.tasks.length + workers === 0) {
|
|||
|
q.drain();
|
|||
|
}
|
|||
|
q.process();
|
|||
|
};
|
|||
|
var cb = only_once(next);
|
|||
|
worker(task.data, cb);
|
|||
|
}
|
|||
|
},
|
|||
|
length: function () {
|
|||
|
return q.tasks.length;
|
|||
|
},
|
|||
|
running: function () {
|
|||
|
return workers;
|
|||
|
}
|
|||
|
};
|
|||
|
return q;
|
|||
|
};
|
|||
|
|
|||
|
async.cargo = function (worker, payload) {
|
|||
|
var working = false,
|
|||
|
tasks = [];
|
|||
|
|
|||
|
var cargo = {
|
|||
|
tasks: tasks,
|
|||
|
payload: payload,
|
|||
|
saturated: null,
|
|||
|
empty: null,
|
|||
|
drain: null,
|
|||
|
push: function (data, callback) {
|
|||
|
if(data.constructor !== Array) {
|
|||
|
data = [data];
|
|||
|
}
|
|||
|
_each(data, function(task) {
|
|||
|
tasks.push({
|
|||
|
data: task,
|
|||
|
callback: typeof callback === 'function' ? callback : null
|
|||
|
});
|
|||
|
if (cargo.saturated && tasks.length === payload) {
|
|||
|
cargo.saturated();
|
|||
|
}
|
|||
|
});
|
|||
|
async.setImmediate(cargo.process);
|
|||
|
},
|
|||
|
process: function process() {
|
|||
|
if (working) return;
|
|||
|
if (tasks.length === 0) {
|
|||
|
if(cargo.drain) cargo.drain();
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
var ts = typeof payload === 'number'
|
|||
|
? tasks.splice(0, payload)
|
|||
|
: tasks.splice(0);
|
|||
|
|
|||
|
var ds = _map(ts, function (task) {
|
|||
|
return task.data;
|
|||
|
});
|
|||
|
|
|||
|
if(cargo.empty) cargo.empty();
|
|||
|
working = true;
|
|||
|
worker(ds, function () {
|
|||
|
working = false;
|
|||
|
|
|||
|
var args = arguments;
|
|||
|
_each(ts, function (data) {
|
|||
|
if (data.callback) {
|
|||
|
data.callback.apply(null, args);
|
|||
|
}
|
|||
|
});
|
|||
|
|
|||
|
process();
|
|||
|
});
|
|||
|
},
|
|||
|
length: function () {
|
|||
|
return tasks.length;
|
|||
|
},
|
|||
|
running: function () {
|
|||
|
return working;
|
|||
|
}
|
|||
|
};
|
|||
|
return cargo;
|
|||
|
};
|
|||
|
|
|||
|
var _console_fn = function (name) {
|
|||
|
return function (fn) {
|
|||
|
var args = Array.prototype.slice.call(arguments, 1);
|
|||
|
fn.apply(null, args.concat([function (err) {
|
|||
|
var args = Array.prototype.slice.call(arguments, 1);
|
|||
|
if (typeof console !== 'undefined') {
|
|||
|
if (err) {
|
|||
|
if (console.error) {
|
|||
|
console.error(err);
|
|||
|
}
|
|||
|
}
|
|||
|
else if (console[name]) {
|
|||
|
_each(args, function (x) {
|
|||
|
console[name](x);
|
|||
|
});
|
|||
|
}
|
|||
|
}
|
|||
|
}]));
|
|||
|
};
|
|||
|
};
|
|||
|
async.log = _console_fn('log');
|
|||
|
async.dir = _console_fn('dir');
|
|||
|
/*async.info = _console_fn('info');
|
|||
|
async.warn = _console_fn('warn');
|
|||
|
async.error = _console_fn('error');*/
|
|||
|
|
|||
|
async.memoize = function (fn, hasher) {
|
|||
|
var memo = {};
|
|||
|
var queues = {};
|
|||
|
hasher = hasher || function (x) {
|
|||
|
return x;
|
|||
|
};
|
|||
|
var memoized = function () {
|
|||
|
var args = Array.prototype.slice.call(arguments);
|
|||
|
var callback = args.pop();
|
|||
|
var key = hasher.apply(null, args);
|
|||
|
if (key in memo) {
|
|||
|
callback.apply(null, memo[key]);
|
|||
|
}
|
|||
|
else if (key in queues) {
|
|||
|
queues[key].push(callback);
|
|||
|
}
|
|||
|
else {
|
|||
|
queues[key] = [callback];
|
|||
|
fn.apply(null, args.concat([function () {
|
|||
|
memo[key] = arguments;
|
|||
|
var q = queues[key];
|
|||
|
delete queues[key];
|
|||
|
for (var i = 0, l = q.length; i < l; i++) {
|
|||
|
q[i].apply(null, arguments);
|
|||
|
}
|
|||
|
}]));
|
|||
|
}
|
|||
|
};
|
|||
|
memoized.memo = memo;
|
|||
|
memoized.unmemoized = fn;
|
|||
|
return memoized;
|
|||
|
};
|
|||
|
|
|||
|
async.unmemoize = function (fn) {
|
|||
|
return function () {
|
|||
|
return (fn.unmemoized || fn).apply(null, arguments);
|
|||
|
};
|
|||
|
};
|
|||
|
|
|||
|
async.times = function (count, iterator, callback) {
|
|||
|
var counter = [];
|
|||
|
for (var i = 0; i < count; i++) {
|
|||
|
counter.push(i);
|
|||
|
}
|
|||
|
return async.map(counter, iterator, callback);
|
|||
|
};
|
|||
|
|
|||
|
async.timesSeries = function (count, iterator, callback) {
|
|||
|
var counter = [];
|
|||
|
for (var i = 0; i < count; i++) {
|
|||
|
counter.push(i);
|
|||
|
}
|
|||
|
return async.mapSeries(counter, iterator, callback);
|
|||
|
};
|
|||
|
|
|||
|
async.compose = function (/* functions... */) {
|
|||
|
var fns = Array.prototype.reverse.call(arguments);
|
|||
|
return function () {
|
|||
|
var that = this;
|
|||
|
var args = Array.prototype.slice.call(arguments);
|
|||
|
var callback = args.pop();
|
|||
|
async.reduce(fns, args, function (newargs, fn, cb) {
|
|||
|
fn.apply(that, newargs.concat([function () {
|
|||
|
var err = arguments[0];
|
|||
|
var nextargs = Array.prototype.slice.call(arguments, 1);
|
|||
|
cb(err, nextargs);
|
|||
|
}]))
|
|||
|
},
|
|||
|
function (err, results) {
|
|||
|
callback.apply(that, [err].concat(results));
|
|||
|
});
|
|||
|
};
|
|||
|
};
|
|||
|
|
|||
|
var _applyEach = function (eachfn, fns /*args...*/) {
|
|||
|
var go = function () {
|
|||
|
var that = this;
|
|||
|
var args = Array.prototype.slice.call(arguments);
|
|||
|
var callback = args.pop();
|
|||
|
return eachfn(fns, function (fn, cb) {
|
|||
|
fn.apply(that, args.concat([cb]));
|
|||
|
},
|
|||
|
callback);
|
|||
|
};
|
|||
|
if (arguments.length > 2) {
|
|||
|
var args = Array.prototype.slice.call(arguments, 2);
|
|||
|
return go.apply(this, args);
|
|||
|
}
|
|||
|
else {
|
|||
|
return go;
|
|||
|
}
|
|||
|
};
|
|||
|
async.applyEach = doParallel(_applyEach);
|
|||
|
async.applyEachSeries = doSeries(_applyEach);
|
|||
|
|
|||
|
async.forever = function (fn, callback) {
|
|||
|
function next(err) {
|
|||
|
if (err) {
|
|||
|
if (callback) {
|
|||
|
return callback(err);
|
|||
|
}
|
|||
|
throw err;
|
|||
|
}
|
|||
|
fn(next);
|
|||
|
}
|
|||
|
next();
|
|||
|
};
|
|||
|
|
|||
|
// AMD / RequireJS
|
|||
|
if (typeof define !== 'undefined' && define.amd) {
|
|||
|
define('async',[], function () {
|
|||
|
return async;
|
|||
|
});
|
|||
|
}
|
|||
|
// Node.js
|
|||
|
else if (typeof module !== 'undefined' && module.exports) {
|
|||
|
module.exports = async;
|
|||
|
}
|
|||
|
// included directly via <script> tag
|
|||
|
else {
|
|||
|
root.async = async;
|
|||
|
}
|
|||
|
|
|||
|
}());
|
|||
|
|
|||
|
var __extends = this.__extends || function (d, b) {
|
|||
|
for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p];
|
|||
|
function __() { this.constructor = d; }
|
|||
|
__.prototype = b.prototype;
|
|||
|
d.prototype = new __();
|
|||
|
};
|
|||
|
define('backend/dropbox',["require", "exports", '../generic/preload_file', '../core/file_system', '../core/node_fs_stats', '../core/buffer', '../core/api_error', '../core/node_path', '../core/browserfs', '../core/buffer_core_arraybuffer', "async"], function(require, exports, preload_file, file_system, node_fs_stats, buffer, api_error, node_path, browserfs, buffer_core_arraybuffer) {
|
|||
|
var Buffer = buffer.Buffer;
|
|||
|
var Stats = node_fs_stats.Stats;
|
|||
|
var ApiError = api_error.ApiError;
|
|||
|
var ErrorCode = api_error.ErrorCode;
|
|||
|
var path = node_path.path;
|
|||
|
var FileType = node_fs_stats.FileType;
|
|||
|
|
|||
|
// XXX: No typings available for the Dropbox client. :(
|
|||
|
// XXX: The typings for async on DefinitelyTyped are out of date.
|
|||
|
var async = require('async');
|
|||
|
var Buffer = buffer.Buffer;
|
|||
|
|
|||
|
var DropboxFile = (function (_super) {
|
|||
|
__extends(DropboxFile, _super);
|
|||
|
function DropboxFile(_fs, _path, _flag, _stat, contents) {
|
|||
|
_super.call(this, _fs, _path, _flag, _stat, contents);
|
|||
|
}
|
|||
|
DropboxFile.prototype.sync = function (cb) {
|
|||
|
var buffer = this._buffer;
|
|||
|
|
|||
|
// XXX: Typing hack.
|
|||
|
var backing_mem = this._buffer.getBufferCore();
|
|||
|
if (!(backing_mem instanceof buffer_core_arraybuffer.BufferCoreArrayBuffer)) {
|
|||
|
// Copy into an ArrayBuffer-backed Buffer.
|
|||
|
buffer = new Buffer(this._buffer.length);
|
|||
|
this._buffer.copy(buffer);
|
|||
|
backing_mem = buffer.getBufferCore();
|
|||
|
}
|
|||
|
|
|||
|
// Reach into the BC, grab the DV.
|
|||
|
var dv = backing_mem.getDataView();
|
|||
|
|
|||
|
// Create an appropriate view on the array buffer.
|
|||
|
var abv = new DataView(dv.buffer, dv.byteOffset + buffer.getOffset(), buffer.length);
|
|||
|
this._fs._writeFileStrict(this._path, abv, cb);
|
|||
|
};
|
|||
|
|
|||
|
DropboxFile.prototype.close = function (cb) {
|
|||
|
this.sync(cb);
|
|||
|
};
|
|||
|
return DropboxFile;
|
|||
|
})(preload_file.PreloadFile);
|
|||
|
exports.DropboxFile = DropboxFile;
|
|||
|
|
|||
|
var DropboxFileSystem = (function (_super) {
|
|||
|
__extends(DropboxFileSystem, _super);
|
|||
|
/**
|
|||
|
* Arguments: an authenticated Dropbox.js client
|
|||
|
*/
|
|||
|
function DropboxFileSystem(client) {
|
|||
|
_super.call(this);
|
|||
|
this.client = client;
|
|||
|
}
|
|||
|
DropboxFileSystem.prototype.getName = function () {
|
|||
|
return 'Dropbox';
|
|||
|
};
|
|||
|
|
|||
|
DropboxFileSystem.isAvailable = function () {
|
|||
|
// Checks if the Dropbox library is loaded.
|
|||
|
// @todo Check if the Dropbox library *can be used* in the current browser.
|
|||
|
return typeof Dropbox !== 'undefined';
|
|||
|
};
|
|||
|
|
|||
|
DropboxFileSystem.prototype.isReadOnly = function () {
|
|||
|
return false;
|
|||
|
};
|
|||
|
|
|||
|
// Dropbox doesn't support symlinks, properties, or synchronous calls
|
|||
|
DropboxFileSystem.prototype.supportsSymlinks = function () {
|
|||
|
return false;
|
|||
|
};
|
|||
|
|
|||
|
DropboxFileSystem.prototype.supportsProps = function () {
|
|||
|
return false;
|
|||
|
};
|
|||
|
|
|||
|
DropboxFileSystem.prototype.supportsSynch = function () {
|
|||
|
return false;
|
|||
|
};
|
|||
|
|
|||
|
DropboxFileSystem.prototype.empty = function (main_cb) {
|
|||
|
var _this = this;
|
|||
|
this.client.readdir('/', function (error, paths, dir, files) {
|
|||
|
if (error) {
|
|||
|
main_cb(_this.convert(error));
|
|||
|
} else {
|
|||
|
var deleteFile = function (file, cb) {
|
|||
|
_this.client.remove(file.path, function (err, stat) {
|
|||
|
cb(err ? _this.convert(err) : err);
|
|||
|
});
|
|||
|
};
|
|||
|
var finished = function (err) {
|
|||
|
if (err) {
|
|||
|
main_cb(_this.convert(err));
|
|||
|
} else {
|
|||
|
main_cb();
|
|||
|
}
|
|||
|
};
|
|||
|
async.each(files, deleteFile, finished);
|
|||
|
}
|
|||
|
});
|
|||
|
};
|
|||
|
|
|||
|
DropboxFileSystem.prototype.rename = function (oldPath, newPath, cb) {
|
|||
|
this.client.move(oldPath, newPath, function (error, stat) {
|
|||
|
if (error) {
|
|||
|
// XXX: Assume 404 for now.
|
|||
|
var missingPath = error.response.error.indexOf(oldPath) > -1 ? oldPath : newPath;
|
|||
|
cb(new ApiError(1 /* ENOENT */, missingPath + " doesn't exist"));
|
|||
|
} else {
|
|||
|
cb();
|
|||
|
}
|
|||
|
});
|
|||
|
};
|
|||
|
|
|||
|
DropboxFileSystem.prototype.stat = function (path, isLstat, cb) {
|
|||
|
var _this = this;
|
|||
|
// Ignore lstat case -- Dropbox doesn't support symlinks
|
|||
|
// Stat the file
|
|||
|
this.client.stat(path, function (error, stat) {
|
|||
|
// Dropbox keeps track of deleted files, so if a file has existed in the
|
|||
|
// past but doesn't any longer, you wont get an error
|
|||
|
if (error || ((stat != null) && stat.isRemoved)) {
|
|||
|
cb(new ApiError(1 /* ENOENT */, path + " doesn't exist"));
|
|||
|
} else {
|
|||
|
var stats = new Stats(_this._statType(stat), stat.size);
|
|||
|
return cb(null, stats);
|
|||
|
}
|
|||
|
});
|
|||
|
};
|
|||
|
|
|||
|
DropboxFileSystem.prototype.open = function (path, flags, mode, cb) {
|
|||
|
var _this = this;
|
|||
|
// Try and get the file's contents
|
|||
|
this.client.readFile(path, {
|
|||
|
arrayBuffer: true
|
|||
|
}, function (error, content, db_stat, range) {
|
|||
|
if (error) {
|
|||
|
// If the file's being opened for reading and doesn't exist, return an
|
|||
|
// error
|
|||
|
if (flags.isReadable()) {
|
|||
|
cb(new ApiError(1 /* ENOENT */, path + " doesn't exist"));
|
|||
|
} else {
|
|||
|
switch (error.status) {
|
|||
|
case 0:
|
|||
|
return console.error('No connection');
|
|||
|
|
|||
|
case 404:
|
|||
|
var ab = new ArrayBuffer(0);
|
|||
|
return _this._writeFileStrict(path, ab, function (error2, stat) {
|
|||
|
if (error2) {
|
|||
|
cb(error2);
|
|||
|
} else {
|
|||
|
var file = _this._makeFile(path, flags, stat, new Buffer(ab));
|
|||
|
cb(null, file);
|
|||
|
}
|
|||
|
});
|
|||
|
default:
|
|||
|
return console.log("Unhandled error: " + error);
|
|||
|
}
|
|||
|
}
|
|||
|
} else {
|
|||
|
// No error
|
|||
|
var buffer;
|
|||
|
|
|||
|
// Dropbox.js seems to set `content` to `null` rather than to an empty
|
|||
|
// buffer when reading an empty file. Not sure why this is.
|
|||
|
if (content === null) {
|
|||
|
buffer = new Buffer(0);
|
|||
|
} else {
|
|||
|
buffer = new Buffer(content);
|
|||
|
}
|
|||
|
var file = _this._makeFile(path, flags, db_stat, buffer);
|
|||
|
return cb(null, file);
|
|||
|
}
|
|||
|
});
|
|||
|
};
|
|||
|
|
|||
|
DropboxFileSystem.prototype._writeFileStrict = function (p, data, cb) {
|
|||
|
var _this = this;
|
|||
|
var parent = path.dirname(p);
|
|||
|
this.stat(parent, false, function (error, stat) {
|
|||
|
if (error) {
|
|||
|
cb(new ApiError(1 /* ENOENT */, "Can't create " + p + " because " + parent + " doesn't exist"));
|
|||
|
} else {
|
|||
|
_this.client.writeFile(p, data, function (error2, stat) {
|
|||
|
if (error2) {
|
|||
|
cb(_this.convert(error2));
|
|||
|
} else {
|
|||
|
cb(null, stat);
|
|||
|
}
|
|||
|
});
|
|||
|
}
|
|||
|
});
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Private
|
|||
|
* Returns a BrowserFS object representing the type of a Dropbox.js stat object
|
|||
|
*/
|
|||
|
DropboxFileSystem.prototype._statType = function (stat) {
|
|||
|
return stat.isFile ? 32768 /* FILE */ : 16384 /* DIRECTORY */;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Private
|
|||
|
* Returns a BrowserFS object representing a File, created from the data
|
|||
|
* returned by calls to the Dropbox API.
|
|||
|
*/
|
|||
|
DropboxFileSystem.prototype._makeFile = function (path, flag, stat, buffer) {
|
|||
|
var type = this._statType(stat);
|
|||
|
var stats = new Stats(type, stat.size);
|
|||
|
return new DropboxFile(this, path, flag, stats, buffer);
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Private
|
|||
|
* Delete a file or directory from Dropbox
|
|||
|
* isFile should reflect which call was made to remove the it (`unlink` or
|
|||
|
* `rmdir`). If this doesn't match what's actually at `path`, an error will be
|
|||
|
* returned
|
|||
|
*/
|
|||
|
DropboxFileSystem.prototype._remove = function (path, cb, isFile) {
|
|||
|
var _this = this;
|
|||
|
this.client.stat(path, function (error, stat) {
|
|||
|
var message = null;
|
|||
|
if (error) {
|
|||
|
cb(new ApiError(1 /* ENOENT */, path + " doesn't exist"));
|
|||
|
} else {
|
|||
|
if (stat.isFile && !isFile) {
|
|||
|
cb(new ApiError(7 /* ENOTDIR */, path + " is a file."));
|
|||
|
} else if (!stat.isFile && isFile) {
|
|||
|
cb(new ApiError(8 /* EISDIR */, path + " is a directory."));
|
|||
|
} else {
|
|||
|
_this.client.remove(path, function (error, stat) {
|
|||
|
if (error) {
|
|||
|
// @todo Make this more specific.
|
|||
|
cb(new ApiError(2 /* EIO */, "Failed to remove " + path));
|
|||
|
} else {
|
|||
|
cb(null);
|
|||
|
}
|
|||
|
});
|
|||
|
}
|
|||
|
}
|
|||
|
});
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Delete a file
|
|||
|
*/
|
|||
|
DropboxFileSystem.prototype.unlink = function (path, cb) {
|
|||
|
this._remove(path, cb, true);
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Delete a directory
|
|||
|
*/
|
|||
|
DropboxFileSystem.prototype.rmdir = function (path, cb) {
|
|||
|
this._remove(path, cb, false);
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Create a directory
|
|||
|
*/
|
|||
|
DropboxFileSystem.prototype.mkdir = function (p, mode, cb) {
|
|||
|
var _this = this;
|
|||
|
// Dropbox.js' client.mkdir() behaves like `mkdir -p`, i.e. it creates a
|
|||
|
// directory and all its ancestors if they don't exist.
|
|||
|
// Node's fs.mkdir() behaves like `mkdir`, i.e. it throws an error if an attempt
|
|||
|
// is made to create a directory without a parent.
|
|||
|
// To handle this inconsistency, a check for the existence of `path`'s parent
|
|||
|
// must be performed before it is created, and an error thrown if it does
|
|||
|
// not exist
|
|||
|
var parent = path.dirname(p);
|
|||
|
this.client.stat(parent, function (error, stat) {
|
|||
|
if (error) {
|
|||
|
cb(new ApiError(1 /* ENOENT */, "Can't create " + p + " because " + parent + " doesn't exist"));
|
|||
|
} else {
|
|||
|
_this.client.mkdir(p, function (error, stat) {
|
|||
|
if (error) {
|
|||
|
cb(new ApiError(6 /* EEXIST */, p + " already exists"));
|
|||
|
} else {
|
|||
|
cb(null);
|
|||
|
}
|
|||
|
});
|
|||
|
}
|
|||
|
});
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Get the names of the files in a directory
|
|||
|
*/
|
|||
|
DropboxFileSystem.prototype.readdir = function (path, cb) {
|
|||
|
var _this = this;
|
|||
|
this.client.readdir(path, function (error, files, dir_stat, content_stats) {
|
|||
|
if (error) {
|
|||
|
return cb(_this.convert(error));
|
|||
|
} else {
|
|||
|
return cb(null, files);
|
|||
|
}
|
|||
|
});
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Converts a Dropbox-JS error into a BFS error.
|
|||
|
*/
|
|||
|
DropboxFileSystem.prototype.convert = function (err, message) {
|
|||
|
if (typeof message === "undefined") { message = ""; }
|
|||
|
switch (err.status) {
|
|||
|
case 400:
|
|||
|
// INVALID_PARAM
|
|||
|
return new ApiError(9 /* EINVAL */, message);
|
|||
|
case 401:
|
|||
|
|
|||
|
case 403:
|
|||
|
// OAUTH_ERROR
|
|||
|
return new ApiError(2 /* EIO */, message);
|
|||
|
case 404:
|
|||
|
// NOT_FOUND
|
|||
|
return new ApiError(1 /* ENOENT */, message);
|
|||
|
case 405:
|
|||
|
// INVALID_METHOD
|
|||
|
return new ApiError(14 /* ENOTSUP */, message);
|
|||
|
|
|||
|
case 0:
|
|||
|
|
|||
|
case 304:
|
|||
|
|
|||
|
case 406:
|
|||
|
|
|||
|
case 409:
|
|||
|
|
|||
|
default:
|
|||
|
return new ApiError(2 /* EIO */, message);
|
|||
|
}
|
|||
|
};
|
|||
|
return DropboxFileSystem;
|
|||
|
})(file_system.BaseFileSystem);
|
|||
|
exports.DropboxFileSystem = DropboxFileSystem;
|
|||
|
|
|||
|
browserfs.registerFileSystem('Dropbox', DropboxFileSystem);
|
|||
|
});
|
|||
|
//# sourceMappingURL=dropbox.js.map
|
|||
|
;
|
|||
|
var __extends = this.__extends || function (d, b) {
|
|||
|
for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p];
|
|||
|
function __() { this.constructor = d; }
|
|||
|
__.prototype = b.prototype;
|
|||
|
d.prototype = new __();
|
|||
|
};
|
|||
|
define('backend/html5fs',["require", "exports", '../generic/preload_file', '../core/file_system', '../core/api_error', '../core/file_flag', '../core/node_fs_stats', '../core/buffer', '../core/browserfs', '../core/buffer_core_arraybuffer', '../core/node_path', '../core/global', "async"], function(require, exports, preload_file, file_system, api_error, file_flag, node_fs_stats, buffer, browserfs, buffer_core_arraybuffer, node_path, global) {
|
|||
|
var Buffer = buffer.Buffer;
|
|||
|
var Stats = node_fs_stats.Stats;
|
|||
|
var FileType = node_fs_stats.FileType;
|
|||
|
var ApiError = api_error.ApiError;
|
|||
|
var ErrorCode = api_error.ErrorCode;
|
|||
|
var ActionType = file_flag.ActionType;
|
|||
|
|
|||
|
// XXX: The typings for async on DefinitelyTyped are out of date.
|
|||
|
var async = require('async');
|
|||
|
|
|||
|
var _getFS = global.webkitRequestFileSystem || global.requestFileSystem || null;
|
|||
|
|
|||
|
function _requestQuota(type, size, success, errorCallback) {
|
|||
|
// We cast navigator and window to '<any>' because everything here is
|
|||
|
// nonstandard functionality, despite the fact that Chrome has the only
|
|||
|
// implementation of the HTML5FS and is likely driving the standardization
|
|||
|
// process. Thus, these objects defined off of navigator and window are not
|
|||
|
// present in the DefinitelyTyped TypeScript typings for FileSystem.
|
|||
|
if (typeof navigator['webkitPersistentStorage'] !== 'undefined') {
|
|||
|
switch (type) {
|
|||
|
case global.PERSISTENT:
|
|||
|
navigator.webkitPersistentStorage.requestQuota(size, success, errorCallback);
|
|||
|
break;
|
|||
|
case global.TEMPORARY:
|
|||
|
navigator.webkitTemporaryStorage.requestQuota(size, success, errorCallback);
|
|||
|
break;
|
|||
|
default:
|
|||
|
// TODO: Figure out how to construct a DOMException/DOMError.
|
|||
|
errorCallback(null);
|
|||
|
break;
|
|||
|
}
|
|||
|
} else {
|
|||
|
global.webkitStorageInfo.requestQuota(type, size, success, errorCallback);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
function _toArray(list) {
|
|||
|
return Array.prototype.slice.call(list || [], 0);
|
|||
|
}
|
|||
|
|
|||
|
// A note about getFile and getDirectory options:
|
|||
|
// These methods are called at numerous places in this file, and are passed
|
|||
|
// some combination of these two options:
|
|||
|
// - create: If true, the entry will be created if it doesn't exist.
|
|||
|
// If false, an error will be thrown if it doesn't exist.
|
|||
|
// - exclusive: If true, only create the entry if it doesn't already exist,
|
|||
|
// and throw an error if it does.
|
|||
|
var HTML5FSFile = (function (_super) {
|
|||
|
__extends(HTML5FSFile, _super);
|
|||
|
function HTML5FSFile(_fs, _path, _flag, _stat, contents) {
|
|||
|
_super.call(this, _fs, _path, _flag, _stat, contents);
|
|||
|
}
|
|||
|
HTML5FSFile.prototype.sync = function (cb) {
|
|||
|
var _this = this;
|
|||
|
// Don't create the file (it should already have been created by `open`)
|
|||
|
var opts = {
|
|||
|
create: false
|
|||
|
};
|
|||
|
var _fs = this._fs;
|
|||
|
var success = function (entry) {
|
|||
|
entry.createWriter(function (writer) {
|
|||
|
// XXX: Typing hack.
|
|||
|
var buffer = _this._buffer;
|
|||
|
var backing_mem = _this._buffer.getBufferCore();
|
|||
|
if (!(backing_mem instanceof buffer_core_arraybuffer.BufferCoreArrayBuffer)) {
|
|||
|
// Copy into an ArrayBuffer-backed Buffer.
|
|||
|
buffer = new Buffer(_this._buffer.length);
|
|||
|
_this._buffer.copy(buffer);
|
|||
|
backing_mem = buffer.getBufferCore();
|
|||
|
}
|
|||
|
|
|||
|
// Reach into the BC, grab the DV.
|
|||
|
var dv = backing_mem.getDataView();
|
|||
|
|
|||
|
// Create an appropriate view on the array buffer.
|
|||
|
var abv = new DataView(dv.buffer, dv.byteOffset + buffer.getOffset(), buffer.length);
|
|||
|
var blob = new Blob([abv]);
|
|||
|
var length = blob.size;
|
|||
|
writer.onwriteend = function (event) {
|
|||
|
writer.onwriteend = null;
|
|||
|
writer.truncate(length);
|
|||
|
cb();
|
|||
|
};
|
|||
|
writer.onerror = function (err) {
|
|||
|
cb(_fs.convert(err));
|
|||
|
};
|
|||
|
writer.write(blob);
|
|||
|
});
|
|||
|
};
|
|||
|
var error = function (err) {
|
|||
|
cb(_fs.convert(err));
|
|||
|
};
|
|||
|
_fs.fs.root.getFile(this._path, opts, success, error);
|
|||
|
};
|
|||
|
|
|||
|
HTML5FSFile.prototype.close = function (cb) {
|
|||
|
this.sync(cb);
|
|||
|
};
|
|||
|
return HTML5FSFile;
|
|||
|
})(preload_file.PreloadFile);
|
|||
|
exports.HTML5FSFile = HTML5FSFile;
|
|||
|
|
|||
|
var HTML5FS = (function (_super) {
|
|||
|
__extends(HTML5FS, _super);
|
|||
|
/**
|
|||
|
* Arguments:
|
|||
|
* - type: PERSISTENT or TEMPORARY
|
|||
|
* - size: storage quota to request, in megabytes. Allocated value may be less.
|
|||
|
*/
|
|||
|
function HTML5FS(size, type) {
|
|||
|
_super.call(this);
|
|||
|
this.size = size != null ? size : 5;
|
|||
|
this.type = type != null ? type : global.PERSISTENT;
|
|||
|
var kb = 1024;
|
|||
|
var mb = kb * kb;
|
|||
|
this.size *= mb;
|
|||
|
}
|
|||
|
HTML5FS.prototype.getName = function () {
|
|||
|
return 'HTML5 FileSystem';
|
|||
|
};
|
|||
|
|
|||
|
HTML5FS.isAvailable = function () {
|
|||
|
return _getFS != null;
|
|||
|
};
|
|||
|
|
|||
|
HTML5FS.prototype.isReadOnly = function () {
|
|||
|
return false;
|
|||
|
};
|
|||
|
|
|||
|
HTML5FS.prototype.supportsSymlinks = function () {
|
|||
|
return false;
|
|||
|
};
|
|||
|
|
|||
|
HTML5FS.prototype.supportsProps = function () {
|
|||
|
return false;
|
|||
|
};
|
|||
|
|
|||
|
HTML5FS.prototype.supportsSynch = function () {
|
|||
|
return false;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Converts the given DOMError into an appropriate ApiError.
|
|||
|
* Full list of values here:
|
|||
|
* https://developer.mozilla.org/en-US/docs/Web/API/DOMError
|
|||
|
* I've only implemented the most obvious ones, but more can be added to
|
|||
|
* make errors more descriptive in the future.
|
|||
|
*/
|
|||
|
HTML5FS.prototype.convert = function (err, message) {
|
|||
|
if (typeof message === "undefined") { message = ""; }
|
|||
|
switch (err.name) {
|
|||
|
case 'QuotaExceededError':
|
|||
|
return new ApiError(11 /* ENOSPC */, message);
|
|||
|
case 'NotFoundError':
|
|||
|
return new ApiError(1 /* ENOENT */, message);
|
|||
|
case 'SecurityError':
|
|||
|
return new ApiError(4 /* EACCES */, message);
|
|||
|
case 'InvalidModificationError':
|
|||
|
return new ApiError(0 /* EPERM */, message);
|
|||
|
case 'SyntaxError':
|
|||
|
case 'TypeMismatchError':
|
|||
|
return new ApiError(9 /* EINVAL */, message);
|
|||
|
default:
|
|||
|
return new ApiError(9 /* EINVAL */, message);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Converts the given ErrorEvent (from a FileReader) into an appropriate
|
|||
|
* APIError.
|
|||
|
*/
|
|||
|
HTML5FS.prototype.convertErrorEvent = function (err, message) {
|
|||
|
if (typeof message === "undefined") { message = ""; }
|
|||
|
return new ApiError(1 /* ENOENT */, err.message + "; " + message);
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Nonstandard
|
|||
|
* Requests a storage quota from the browser to back this FS.
|
|||
|
*/
|
|||
|
HTML5FS.prototype.allocate = function (cb) {
|
|||
|
var _this = this;
|
|||
|
if (typeof cb === "undefined") { cb = function () {
|
|||
|
}; }
|
|||
|
var success = function (fs) {
|
|||
|
_this.fs = fs;
|
|||
|
cb();
|
|||
|
};
|
|||
|
var error = function (err) {
|
|||
|
cb(_this.convert(err));
|
|||
|
};
|
|||
|
if (this.type === global.PERSISTENT) {
|
|||
|
_requestQuota(this.type, this.size, function (granted) {
|
|||
|
_getFS(_this.type, granted, success, error);
|
|||
|
}, error);
|
|||
|
} else {
|
|||
|
_getFS(this.type, this.size, success, error);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Nonstandard
|
|||
|
* Deletes everything in the FS. Used for testing.
|
|||
|
* Karma clears the storage after you quit it but not between runs of the test
|
|||
|
* suite, and the tests expect an empty FS every time.
|
|||
|
*/
|
|||
|
HTML5FS.prototype.empty = function (main_cb) {
|
|||
|
var _this = this;
|
|||
|
// Get a list of all entries in the root directory to delete them
|
|||
|
this._readdir('/', function (err, entries) {
|
|||
|
if (err) {
|
|||
|
console.error('Failed to empty FS');
|
|||
|
main_cb(err);
|
|||
|
} else {
|
|||
|
// Called when every entry has been operated on
|
|||
|
var finished = function (er) {
|
|||
|
if (err) {
|
|||
|
console.error("Failed to empty FS");
|
|||
|
main_cb(err);
|
|||
|
} else {
|
|||
|
main_cb();
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
// Removes files and recursively removes directories
|
|||
|
var deleteEntry = function (entry, cb) {
|
|||
|
var succ = function () {
|
|||
|
cb();
|
|||
|
};
|
|||
|
var error = function (err) {
|
|||
|
cb(_this.convert(err, entry.fullPath));
|
|||
|
};
|
|||
|
if (entry.isFile) {
|
|||
|
entry.remove(succ, error);
|
|||
|
} else {
|
|||
|
entry.removeRecursively(succ, error);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
// Loop through the entries and remove them, then call the callback
|
|||
|
// when they're all finished.
|
|||
|
async.each(entries, deleteEntry, finished);
|
|||
|
}
|
|||
|
});
|
|||
|
};
|
|||
|
|
|||
|
HTML5FS.prototype.rename = function (oldPath, newPath, cb) {
|
|||
|
var _this = this;
|
|||
|
var semaphore = 2, successCount = 0, root = this.fs.root, error = function (err) {
|
|||
|
if (--semaphore === 0) {
|
|||
|
cb(_this.convert(err, "Failed to rename " + oldPath + " to " + newPath + "."));
|
|||
|
}
|
|||
|
}, success = function (file) {
|
|||
|
if (++successCount === 2) {
|
|||
|
console.error("Something was identified as both a file and a directory. This should never happen.");
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
// SPECIAL CASE: If newPath === oldPath, and the path exists, then
|
|||
|
// this operation trivially succeeds.
|
|||
|
if (oldPath === newPath) {
|
|||
|
return cb();
|
|||
|
}
|
|||
|
|
|||
|
// Get the new parent directory.
|
|||
|
root.getDirectory(node_path.path.dirname(newPath), {}, function (parentDir) {
|
|||
|
file.moveTo(parentDir, node_path.path.basename(newPath), function (entry) {
|
|||
|
cb();
|
|||
|
}, function (err) {
|
|||
|
// SPECIAL CASE: If oldPath is a directory, and newPath is a
|
|||
|
// file, rename should delete the file and perform the move.
|
|||
|
if (file.isDirectory) {
|
|||
|
// Unlink only works on files. Try to delete newPath.
|
|||
|
_this.unlink(newPath, function (e) {
|
|||
|
if (e) {
|
|||
|
// newPath is probably a directory.
|
|||
|
error(err);
|
|||
|
} else {
|
|||
|
// Recurse, now that newPath doesn't exist.
|
|||
|
_this.rename(oldPath, newPath, cb);
|
|||
|
}
|
|||
|
});
|
|||
|
} else {
|
|||
|
error(err);
|
|||
|
}
|
|||
|
});
|
|||
|
}, error);
|
|||
|
};
|
|||
|
|
|||
|
// We don't know if oldPath is a *file* or a *directory*, and there's no
|
|||
|
// way to stat items. So launch both requests, see which one succeeds.
|
|||
|
root.getFile(oldPath, {}, success, error);
|
|||
|
root.getDirectory(oldPath, {}, success, error);
|
|||
|
};
|
|||
|
|
|||
|
HTML5FS.prototype.stat = function (path, isLstat, cb) {
|
|||
|
var _this = this;
|
|||
|
// Throw an error if the entry doesn't exist, because then there's nothing
|
|||
|
// to stat.
|
|||
|
var opts = {
|
|||
|
create: false
|
|||
|
};
|
|||
|
|
|||
|
// Called when the path has been successfully loaded as a file.
|
|||
|
var loadAsFile = function (entry) {
|
|||
|
var fileFromEntry = function (file) {
|
|||
|
var stat = new Stats(32768 /* FILE */, file.size);
|
|||
|
cb(null, stat);
|
|||
|
};
|
|||
|
entry.file(fileFromEntry, failedToLoad);
|
|||
|
};
|
|||
|
|
|||
|
// Called when the path has been successfully loaded as a directory.
|
|||
|
var loadAsDir = function (dir) {
|
|||
|
// Directory entry size can't be determined from the HTML5 FS API, and is
|
|||
|
// implementation-dependant anyway, so a dummy value is used.
|
|||
|
var size = 4096;
|
|||
|
var stat = new Stats(16384 /* DIRECTORY */, size);
|
|||
|
cb(null, stat);
|
|||
|
};
|
|||
|
|
|||
|
// Called when the path couldn't be opened as a directory or a file.
|
|||
|
var failedToLoad = function (err) {
|
|||
|
cb(_this.convert(err, path));
|
|||
|
};
|
|||
|
|
|||
|
// Called when the path couldn't be opened as a file, but might still be a
|
|||
|
// directory.
|
|||
|
var failedToLoadAsFile = function () {
|
|||
|
_this.fs.root.getDirectory(path, opts, loadAsDir, failedToLoad);
|
|||
|
};
|
|||
|
|
|||
|
// No method currently exists to determine whether a path refers to a
|
|||
|
// directory or a file, so this implementation tries both and uses the first
|
|||
|
// one that succeeds.
|
|||
|
this.fs.root.getFile(path, opts, loadAsFile, failedToLoadAsFile);
|
|||
|
};
|
|||
|
|
|||
|
HTML5FS.prototype.open = function (path, flags, mode, cb) {
|
|||
|
var _this = this;
|
|||
|
var opts = {
|
|||
|
create: flags.pathNotExistsAction() === 3 /* CREATE_FILE */,
|
|||
|
exclusive: flags.isExclusive()
|
|||
|
};
|
|||
|
var error = function (err) {
|
|||
|
cb(_this.convertErrorEvent(err, path));
|
|||
|
};
|
|||
|
var error2 = function (err) {
|
|||
|
cb(_this.convert(err, path));
|
|||
|
};
|
|||
|
var success = function (entry) {
|
|||
|
var success2 = function (file) {
|
|||
|
var reader = new FileReader();
|
|||
|
reader.onloadend = function (event) {
|
|||
|
var bfs_file = _this._makeFile(path, flags, file, reader.result);
|
|||
|
cb(null, bfs_file);
|
|||
|
};
|
|||
|
reader.onerror = error;
|
|||
|
reader.readAsArrayBuffer(file);
|
|||
|
};
|
|||
|
entry.file(success2, error2);
|
|||
|
};
|
|||
|
this.fs.root.getFile(path, opts, success, error);
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Returns a BrowserFS object representing the type of a Dropbox.js stat object
|
|||
|
*/
|
|||
|
HTML5FS.prototype._statType = function (stat) {
|
|||
|
return stat.isFile ? 32768 /* FILE */ : 16384 /* DIRECTORY */;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Returns a BrowserFS object representing a File, created from the data
|
|||
|
* returned by calls to the Dropbox API.
|
|||
|
*/
|
|||
|
HTML5FS.prototype._makeFile = function (path, flag, stat, data) {
|
|||
|
if (typeof data === "undefined") { data = new ArrayBuffer(0); }
|
|||
|
var stats = new Stats(32768 /* FILE */, stat.size);
|
|||
|
var buffer = new Buffer(data);
|
|||
|
return new HTML5FSFile(this, path, flag, stats, buffer);
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Delete a file or directory from the file system
|
|||
|
* isFile should reflect which call was made to remove the it (`unlink` or
|
|||
|
* `rmdir`). If this doesn't match what's actually at `path`, an error will be
|
|||
|
* returned
|
|||
|
*/
|
|||
|
HTML5FS.prototype._remove = function (path, cb, isFile) {
|
|||
|
var _this = this;
|
|||
|
var success = function (entry) {
|
|||
|
var succ = function () {
|
|||
|
cb();
|
|||
|
};
|
|||
|
var err = function (err) {
|
|||
|
cb(_this.convert(err, path));
|
|||
|
};
|
|||
|
entry.remove(succ, err);
|
|||
|
};
|
|||
|
var error = function (err) {
|
|||
|
cb(_this.convert(err, path));
|
|||
|
};
|
|||
|
|
|||
|
// Deleting the entry, so don't create it
|
|||
|
var opts = {
|
|||
|
create: false
|
|||
|
};
|
|||
|
|
|||
|
if (isFile) {
|
|||
|
this.fs.root.getFile(path, opts, success, error);
|
|||
|
} else {
|
|||
|
this.fs.root.getDirectory(path, opts, success, error);
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
HTML5FS.prototype.unlink = function (path, cb) {
|
|||
|
this._remove(path, cb, true);
|
|||
|
};
|
|||
|
|
|||
|
HTML5FS.prototype.rmdir = function (path, cb) {
|
|||
|
this._remove(path, cb, false);
|
|||
|
};
|
|||
|
|
|||
|
HTML5FS.prototype.mkdir = function (path, mode, cb) {
|
|||
|
var _this = this;
|
|||
|
// Create the directory, but throw an error if it already exists, as per
|
|||
|
// mkdir(1)
|
|||
|
var opts = {
|
|||
|
create: true,
|
|||
|
exclusive: true
|
|||
|
};
|
|||
|
var success = function (dir) {
|
|||
|
cb();
|
|||
|
};
|
|||
|
var error = function (err) {
|
|||
|
cb(_this.convert(err, path));
|
|||
|
};
|
|||
|
this.fs.root.getDirectory(path, opts, success, error);
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Returns an array of `FileEntry`s. Used internally by empty and readdir.
|
|||
|
*/
|
|||
|
HTML5FS.prototype._readdir = function (path, cb) {
|
|||
|
var _this = this;
|
|||
|
// Grab the requested directory.
|
|||
|
this.fs.root.getDirectory(path, { create: false }, function (dirEntry) {
|
|||
|
var reader = dirEntry.createReader();
|
|||
|
var entries = [];
|
|||
|
var error = function (err) {
|
|||
|
cb(_this.convert(err, path));
|
|||
|
};
|
|||
|
|
|||
|
// Call the reader.readEntries() until no more results are returned.
|
|||
|
var readEntries = function () {
|
|||
|
reader.readEntries((function (results) {
|
|||
|
if (results.length) {
|
|||
|
entries = entries.concat(_toArray(results));
|
|||
|
readEntries();
|
|||
|
} else {
|
|||
|
cb(null, entries);
|
|||
|
}
|
|||
|
}), error);
|
|||
|
};
|
|||
|
readEntries();
|
|||
|
});
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Map _readdir's list of `FileEntry`s to their names and return that.
|
|||
|
*/
|
|||
|
HTML5FS.prototype.readdir = function (path, cb) {
|
|||
|
this._readdir(path, function (e, entries) {
|
|||
|
if (e != null) {
|
|||
|
return cb(e);
|
|||
|
}
|
|||
|
var rv = [];
|
|||
|
for (var i = 0; i < entries.length; i++) {
|
|||
|
rv.push(entries[i].name);
|
|||
|
}
|
|||
|
cb(null, rv);
|
|||
|
});
|
|||
|
};
|
|||
|
return HTML5FS;
|
|||
|
})(file_system.BaseFileSystem);
|
|||
|
exports.HTML5FS = HTML5FS;
|
|||
|
|
|||
|
browserfs.registerFileSystem('HTML5FS', HTML5FS);
|
|||
|
});
|
|||
|
//# sourceMappingURL=html5fs.js.map
|
|||
|
;
|
|||
|
var __extends = this.__extends || function (d, b) {
|
|||
|
for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p];
|
|||
|
function __() { this.constructor = d; }
|
|||
|
__.prototype = b.prototype;
|
|||
|
d.prototype = new __();
|
|||
|
};
|
|||
|
define('backend/in_memory',["require", "exports", '../generic/key_value_filesystem', '../core/browserfs'], function(require, exports, kvfs, browserfs) {
|
|||
|
/**
|
|||
|
* A simple in-memory key-value store backed by a JavaScript object.
|
|||
|
*/
|
|||
|
var InMemoryStore = (function () {
|
|||
|
function InMemoryStore() {
|
|||
|
this.store = {};
|
|||
|
}
|
|||
|
InMemoryStore.prototype.name = function () {
|
|||
|
return 'In-memory';
|
|||
|
};
|
|||
|
InMemoryStore.prototype.clear = function () {
|
|||
|
this.store = {};
|
|||
|
};
|
|||
|
|
|||
|
InMemoryStore.prototype.beginTransaction = function (type) {
|
|||
|
return new kvfs.SimpleSyncRWTransaction(this);
|
|||
|
};
|
|||
|
|
|||
|
InMemoryStore.prototype.get = function (key) {
|
|||
|
return this.store[key];
|
|||
|
};
|
|||
|
|
|||
|
InMemoryStore.prototype.put = function (key, data, overwrite) {
|
|||
|
if (!overwrite && this.store.hasOwnProperty(key)) {
|
|||
|
return false;
|
|||
|
}
|
|||
|
this.store[key] = data;
|
|||
|
return true;
|
|||
|
};
|
|||
|
|
|||
|
InMemoryStore.prototype.delete = function (key) {
|
|||
|
delete this.store[key];
|
|||
|
};
|
|||
|
return InMemoryStore;
|
|||
|
})();
|
|||
|
exports.InMemoryStore = InMemoryStore;
|
|||
|
|
|||
|
/**
|
|||
|
* A simple in-memory file system backed by an InMemoryStore.
|
|||
|
*/
|
|||
|
var InMemoryFileSystem = (function (_super) {
|
|||
|
__extends(InMemoryFileSystem, _super);
|
|||
|
function InMemoryFileSystem() {
|
|||
|
_super.call(this, { store: new InMemoryStore() });
|
|||
|
}
|
|||
|
return InMemoryFileSystem;
|
|||
|
})(kvfs.SyncKeyValueFileSystem);
|
|||
|
exports.InMemoryFileSystem = InMemoryFileSystem;
|
|||
|
|
|||
|
browserfs.registerFileSystem('InMemory', InMemoryFileSystem);
|
|||
|
});
|
|||
|
//# sourceMappingURL=in_memory.js.map
|
|||
|
;
|
|||
|
var __extends = this.__extends || function (d, b) {
|
|||
|
for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p];
|
|||
|
function __() { this.constructor = d; }
|
|||
|
__.prototype = b.prototype;
|
|||
|
d.prototype = new __();
|
|||
|
};
|
|||
|
define('backend/localStorage',["require", "exports", '../core/buffer', '../core/browserfs', '../generic/key_value_filesystem', '../core/api_error', '../core/global'], function(require, exports, buffer, browserfs, kvfs, api_error, global) {
|
|||
|
var Buffer = buffer.Buffer, ApiError = api_error.ApiError, ErrorCode = api_error.ErrorCode;
|
|||
|
|
|||
|
// Some versions of FF and all versions of IE do not support the full range of
|
|||
|
// 16-bit numbers encoded as characters, as they enforce UTF-16 restrictions.
|
|||
|
// http://stackoverflow.com/questions/11170716/are-there-any-characters-that-are-not-allowed-in-localstorage/11173673#11173673
|
|||
|
var supportsBinaryString = false, binaryEncoding;
|
|||
|
try {
|
|||
|
global.localStorage.setItem("__test__", String.fromCharCode(0xD800));
|
|||
|
supportsBinaryString = global.localStorage.getItem("__test__") === String.fromCharCode(0xD800);
|
|||
|
} catch (e) {
|
|||
|
// IE throws an exception.
|
|||
|
supportsBinaryString = false;
|
|||
|
}
|
|||
|
binaryEncoding = supportsBinaryString ? 'binary_string' : 'binary_string_ie';
|
|||
|
|
|||
|
/**
|
|||
|
* A synchronous key-value store backed by localStorage.
|
|||
|
*/
|
|||
|
var LocalStorageStore = (function () {
|
|||
|
function LocalStorageStore() {
|
|||
|
}
|
|||
|
LocalStorageStore.prototype.name = function () {
|
|||
|
return 'LocalStorage';
|
|||
|
};
|
|||
|
|
|||
|
LocalStorageStore.prototype.clear = function () {
|
|||
|
global.localStorage.clear();
|
|||
|
};
|
|||
|
|
|||
|
LocalStorageStore.prototype.beginTransaction = function (type) {
|
|||
|
// No need to differentiate.
|
|||
|
return new kvfs.SimpleSyncRWTransaction(this);
|
|||
|
};
|
|||
|
|
|||
|
LocalStorageStore.prototype.get = function (key) {
|
|||
|
try {
|
|||
|
var data = global.localStorage.getItem(key);
|
|||
|
if (data !== null) {
|
|||
|
return new Buffer(data, binaryEncoding);
|
|||
|
}
|
|||
|
} catch (e) {
|
|||
|
}
|
|||
|
|
|||
|
// Key doesn't exist, or a failure occurred.
|
|||
|
return undefined;
|
|||
|
};
|
|||
|
|
|||
|
LocalStorageStore.prototype.put = function (key, data, overwrite) {
|
|||
|
try {
|
|||
|
if (!overwrite && global.localStorage.getItem(key) !== null) {
|
|||
|
// Don't want to overwrite the key!
|
|||
|
return false;
|
|||
|
}
|
|||
|
global.localStorage.setItem(key, data.toString(binaryEncoding));
|
|||
|
return true;
|
|||
|
} catch (e) {
|
|||
|
throw new ApiError(11 /* ENOSPC */, "LocalStorage is full.");
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
LocalStorageStore.prototype.delete = function (key) {
|
|||
|
try {
|
|||
|
global.localStorage.removeItem(key);
|
|||
|
} catch (e) {
|
|||
|
throw new ApiError(2 /* EIO */, "Unable to delete key " + key + ": " + e);
|
|||
|
}
|
|||
|
};
|
|||
|
return LocalStorageStore;
|
|||
|
})();
|
|||
|
exports.LocalStorageStore = LocalStorageStore;
|
|||
|
|
|||
|
/**
|
|||
|
* A synchronous file system backed by localStorage. Connects our
|
|||
|
* LocalStorageStore to our SyncKeyValueFileSystem.
|
|||
|
*/
|
|||
|
var LocalStorageFileSystem = (function (_super) {
|
|||
|
__extends(LocalStorageFileSystem, _super);
|
|||
|
function LocalStorageFileSystem() {
|
|||
|
_super.call(this, { store: new LocalStorageStore() });
|
|||
|
}
|
|||
|
LocalStorageFileSystem.isAvailable = function () {
|
|||
|
return typeof global.localStorage !== 'undefined';
|
|||
|
};
|
|||
|
return LocalStorageFileSystem;
|
|||
|
})(kvfs.SyncKeyValueFileSystem);
|
|||
|
exports.LocalStorageFileSystem = LocalStorageFileSystem;
|
|||
|
|
|||
|
browserfs.registerFileSystem('LocalStorage', LocalStorageFileSystem);
|
|||
|
});
|
|||
|
//# sourceMappingURL=localStorage.js.map
|
|||
|
;
|
|||
|
var __extends = this.__extends || function (d, b) {
|
|||
|
for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p];
|
|||
|
function __() { this.constructor = d; }
|
|||
|
__.prototype = b.prototype;
|
|||
|
d.prototype = new __();
|
|||
|
};
|
|||
|
define('backend/mountable_file_system',["require", "exports", '../core/file_system', './in_memory', '../core/api_error', '../core/node_fs', '../core/browserfs'], function(require, exports, file_system, in_memory, api_error, node_fs, browserfs) {
|
|||
|
var ApiError = api_error.ApiError;
|
|||
|
var ErrorCode = api_error.ErrorCode;
|
|||
|
var fs = node_fs.fs;
|
|||
|
|
|||
|
/**
|
|||
|
* The MountableFileSystem allows you to mount multiple backend types or
|
|||
|
* multiple instantiations of the same backend into a single file system tree.
|
|||
|
* The file systems do not need to know about each other; all interactions are
|
|||
|
* automatically facilitated through this interface.
|
|||
|
*
|
|||
|
* For example, if a file system is mounted at /mnt/blah, and a request came in
|
|||
|
* for /mnt/blah/foo.txt, the file system would see a request for /foo.txt.
|
|||
|
*/
|
|||
|
var MountableFileSystem = (function (_super) {
|
|||
|
__extends(MountableFileSystem, _super);
|
|||
|
function MountableFileSystem() {
|
|||
|
_super.call(this);
|
|||
|
this.mntMap = {};
|
|||
|
|
|||
|
// The InMemory file system serves purely to provide directory listings for
|
|||
|
// mounted file systems.
|
|||
|
this.rootFs = new in_memory.InMemoryFileSystem();
|
|||
|
}
|
|||
|
/**
|
|||
|
* Mounts the file system at the given mount point.
|
|||
|
*/
|
|||
|
MountableFileSystem.prototype.mount = function (mnt_pt, fs) {
|
|||
|
if (this.mntMap[mnt_pt]) {
|
|||
|
throw new ApiError(9 /* EINVAL */, "Mount point " + mnt_pt + " is already taken.");
|
|||
|
}
|
|||
|
|
|||
|
// @todo Ensure new mount path is not subsumed by active mount paths.
|
|||
|
this.rootFs.mkdirSync(mnt_pt, 0x1ff);
|
|||
|
this.mntMap[mnt_pt] = fs;
|
|||
|
};
|
|||
|
|
|||
|
MountableFileSystem.prototype.umount = function (mnt_pt) {
|
|||
|
if (!this.mntMap[mnt_pt]) {
|
|||
|
throw new ApiError(9 /* EINVAL */, "Mount point " + mnt_pt + " is already unmounted.");
|
|||
|
}
|
|||
|
delete this.mntMap[mnt_pt];
|
|||
|
this.rootFs.rmdirSync(mnt_pt);
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Returns the file system that the path points to.
|
|||
|
*/
|
|||
|
MountableFileSystem.prototype._get_fs = function (path) {
|
|||
|
for (var mnt_pt in this.mntMap) {
|
|||
|
var fs = this.mntMap[mnt_pt];
|
|||
|
if (path.indexOf(mnt_pt) === 0) {
|
|||
|
path = path.substr(mnt_pt.length > 1 ? mnt_pt.length : 0);
|
|||
|
if (path === '') {
|
|||
|
path = '/';
|
|||
|
}
|
|||
|
return { fs: fs, path: path };
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
// Query our root file system.
|
|||
|
return { fs: this.rootFs, path: path };
|
|||
|
};
|
|||
|
|
|||
|
// Global information methods
|
|||
|
MountableFileSystem.prototype.getName = function () {
|
|||
|
return 'MountableFileSystem';
|
|||
|
};
|
|||
|
|
|||
|
MountableFileSystem.isAvailable = function () {
|
|||
|
return true;
|
|||
|
};
|
|||
|
|
|||
|
MountableFileSystem.prototype.diskSpace = function (path, cb) {
|
|||
|
cb(0, 0);
|
|||
|
};
|
|||
|
|
|||
|
MountableFileSystem.prototype.isReadOnly = function () {
|
|||
|
return false;
|
|||
|
};
|
|||
|
|
|||
|
MountableFileSystem.prototype.supportsLinks = function () {
|
|||
|
// I'm not ready for cross-FS links yet.
|
|||
|
return false;
|
|||
|
};
|
|||
|
|
|||
|
MountableFileSystem.prototype.supportsProps = function () {
|
|||
|
return false;
|
|||
|
};
|
|||
|
|
|||
|
MountableFileSystem.prototype.supportsSynch = function () {
|
|||
|
return true;
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Fixes up error messages so they mention the mounted file location relative
|
|||
|
* to the MFS root, not to the particular FS's root.
|
|||
|
* Mutates the input error, and returns it.
|
|||
|
*/
|
|||
|
MountableFileSystem.prototype.standardizeError = function (err, path, realPath) {
|
|||
|
var index;
|
|||
|
if (-1 !== (index = err.message.indexOf(path))) {
|
|||
|
err.message = err.message.substr(0, index) + realPath + err.message.substr(index + path.length);
|
|||
|
}
|
|||
|
return err;
|
|||
|
};
|
|||
|
|
|||
|
// The following methods involve multiple file systems, and thus have custom
|
|||
|
// logic.
|
|||
|
// Note that we go through the Node API to use its robust default argument
|
|||
|
// processing.
|
|||
|
MountableFileSystem.prototype.rename = function (oldPath, newPath, cb) {
|
|||
|
// Scenario 1: old and new are on same FS.
|
|||
|
var fs1_rv = this._get_fs(oldPath);
|
|||
|
var fs2_rv = this._get_fs(newPath);
|
|||
|
if (fs1_rv.fs === fs2_rv.fs) {
|
|||
|
var _this = this;
|
|||
|
return fs1_rv.fs.rename(fs1_rv.path, fs2_rv.path, function (e) {
|
|||
|
if (e)
|
|||
|
_this.standardizeError(_this.standardizeError(e, fs1_rv.path, oldPath), fs2_rv.path, newPath);
|
|||
|
cb(e);
|
|||
|
});
|
|||
|
}
|
|||
|
|
|||
|
// Scenario 2: Different file systems.
|
|||
|
// Read old file, write new file, delete old file.
|
|||
|
return fs.readFile(oldPath, function (err, data) {
|
|||
|
if (err) {
|
|||
|
return cb(err);
|
|||
|
}
|
|||
|
fs.writeFile(newPath, data, function (err) {
|
|||
|
if (err) {
|
|||
|
return cb(err);
|
|||
|
}
|
|||
|
fs.unlink(oldPath, cb);
|
|||
|
});
|
|||
|
});
|
|||
|
};
|
|||
|
|
|||
|
MountableFileSystem.prototype.renameSync = function (oldPath, newPath) {
|
|||
|
// Scenario 1: old and new are on same FS.
|
|||
|
var fs1_rv = this._get_fs(oldPath);
|
|||
|
var fs2_rv = this._get_fs(newPath);
|
|||
|
if (fs1_rv.fs === fs2_rv.fs) {
|
|||
|
try {
|
|||
|
return fs1_rv.fs.renameSync(fs1_rv.path, fs2_rv.path);
|
|||
|
} catch (e) {
|
|||
|
this.standardizeError(this.standardizeError(e, fs1_rv.path, oldPath), fs2_rv.path, newPath);
|
|||
|
throw e;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
// Scenario 2: Different file systems.
|
|||
|
var data = fs.readFileSync(oldPath);
|
|||
|
fs.writeFileSync(newPath, data);
|
|||
|
return fs.unlinkSync(oldPath);
|
|||
|
};
|
|||
|
return MountableFileSystem;
|
|||
|
})(file_system.BaseFileSystem);
|
|||
|
exports.MountableFileSystem = MountableFileSystem;
|
|||
|
|
|||
|
/**
|
|||
|
* Tricky: Define all of the functions that merely forward arguments to the
|
|||
|
* relevant file system, or return/throw an error.
|
|||
|
* Take advantage of the fact that the *first* argument is always the path, and
|
|||
|
* the *last* is the callback function (if async).
|
|||
|
*/
|
|||
|
function defineFcn(name, isSync, numArgs) {
|
|||
|
if (isSync) {
|
|||
|
return function () {
|
|||
|
var args = [];
|
|||
|
for (var _i = 0; _i < (arguments.length - 0); _i++) {
|
|||
|
args[_i] = arguments[_i + 0];
|
|||
|
}
|
|||
|
var path = args[0];
|
|||
|
var rv = this._get_fs(path);
|
|||
|
args[0] = rv.path;
|
|||
|
try {
|
|||
|
return rv.fs[name].apply(rv.fs, args);
|
|||
|
} catch (e) {
|
|||
|
this.standardizeError(e, rv.path, path);
|
|||
|
throw e;
|
|||
|
}
|
|||
|
};
|
|||
|
} else {
|
|||
|
return function () {
|
|||
|
var args = [];
|
|||
|
for (var _i = 0; _i < (arguments.length - 0); _i++) {
|
|||
|
args[_i] = arguments[_i + 0];
|
|||
|
}
|
|||
|
var path = args[0];
|
|||
|
var rv = this._get_fs(path);
|
|||
|
args[0] = rv.path;
|
|||
|
if (typeof args[args.length - 1] === 'function') {
|
|||
|
var cb = args[args.length - 1];
|
|||
|
var _this = this;
|
|||
|
args[args.length - 1] = function () {
|
|||
|
var args = [];
|
|||
|
for (var _i = 0; _i < (arguments.length - 0); _i++) {
|
|||
|
args[_i] = arguments[_i + 0];
|
|||
|
}
|
|||
|
if (args.length > 0 && args[0] instanceof api_error.ApiError) {
|
|||
|
_this.standardizeError(args[0], rv.path, path);
|
|||
|
}
|
|||
|
cb.apply(null, args);
|
|||
|
};
|
|||
|
}
|
|||
|
return rv.fs[name].apply(rv.fs, args);
|
|||
|
};
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
var fsCmdMap = [
|
|||
|
['readdir', 'exists', 'unlink', 'rmdir', 'readlink'],
|
|||
|
['stat', 'mkdir', 'realpath', 'truncate'],
|
|||
|
['open', 'readFile', 'chmod', 'utimes'],
|
|||
|
['chown'],
|
|||
|
['writeFile', 'appendFile']];
|
|||
|
|
|||
|
for (var i = 0; i < fsCmdMap.length; i++) {
|
|||
|
var cmds = fsCmdMap[i];
|
|||
|
for (var j = 0; j < cmds.length; j++) {
|
|||
|
var fnName = cmds[j];
|
|||
|
MountableFileSystem.prototype[fnName] = defineFcn(fnName, false, i + 1);
|
|||
|
MountableFileSystem.prototype[fnName + 'Sync'] = defineFcn(fnName + 'Sync', true, i + 1);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
browserfs.registerFileSystem('MountableFileSystem', MountableFileSystem);
|
|||
|
});
|
|||
|
//# sourceMappingURL=mountable_file_system.js.map
|
|||
|
;
|
|||
|
/** @license zlib.js 2012 - imaya [ https://github.com/imaya/zlib.js ] The MIT License */(function() {var l=void 0,p=this;function q(c,d){var a=c.split("."),b=p;!(a[0]in b)&&b.execScript&&b.execScript("var "+a[0]);for(var e;a.length&&(e=a.shift());)!a.length&&d!==l?b[e]=d:b=b[e]?b[e]:b[e]={}};var r="undefined"!==typeof Uint8Array&&"undefined"!==typeof Uint16Array&&"undefined"!==typeof Uint32Array;function u(c){var d=c.length,a=0,b=Number.POSITIVE_INFINITY,e,f,g,h,k,m,s,n,t;for(n=0;n<d;++n)c[n]>a&&(a=c[n]),c[n]<b&&(b=c[n]);e=1<<a;f=new (r?Uint32Array:Array)(e);g=1;h=0;for(k=2;g<=a;){for(n=0;n<d;++n)if(c[n]===g){m=0;s=h;for(t=0;t<g;++t)m=m<<1|s&1,s>>=1;for(t=m;t<e;t+=k)f[t]=g<<16|n;++h}++g;h<<=1;k<<=1}return[f,a,b]};function v(c,d){this.g=[];this.h=32768;this.c=this.f=this.d=this.k=0;this.input=r?new Uint8Array(c):c;this.l=!1;this.i=w;this.p=!1;if(d||!(d={}))d.index&&(this.d=d.index),d.bufferSize&&(this.h=d.bufferSize),d.bufferType&&(this.i=d.bufferType),d.resize&&(this.p=d.resize);switch(this.i){case x:this.a=32768;this.b=new (r?Uint8Array:Array)(32768+this.h+258);break;case w:this.a=0;this.b=new (r?Uint8Array:Array)(this.h);this.e=this.u;this.m=this.r;this.j=this.s;break;default:throw Error("invalid inflate mode");
|
|||
|
}}var x=0,w=1;
|
|||
|
v.prototype.t=function(){for(;!this.l;){var c=y(this,3);c&1&&(this.l=!0);c>>>=1;switch(c){case 0:var d=this.input,a=this.d,b=this.b,e=this.a,f=l,g=l,h=l,k=b.length,m=l;this.c=this.f=0;f=d[a++];if(f===l)throw Error("invalid uncompressed block header: LEN (first byte)");g=f;f=d[a++];if(f===l)throw Error("invalid uncompressed block header: LEN (second byte)");g|=f<<8;f=d[a++];if(f===l)throw Error("invalid uncompressed block header: NLEN (first byte)");h=f;f=d[a++];if(f===l)throw Error("invalid uncompressed block header: NLEN (second byte)");h|=
|
|||
|
f<<8;if(g===~h)throw Error("invalid uncompressed block header: length verify");if(a+g>d.length)throw Error("input buffer is broken");switch(this.i){case x:for(;e+g>b.length;){m=k-e;g-=m;if(r)b.set(d.subarray(a,a+m),e),e+=m,a+=m;else for(;m--;)b[e++]=d[a++];this.a=e;b=this.e();e=this.a}break;case w:for(;e+g>b.length;)b=this.e({o:2});break;default:throw Error("invalid inflate mode");}if(r)b.set(d.subarray(a,a+g),e),e+=g,a+=g;else for(;g--;)b[e++]=d[a++];this.d=a;this.a=e;this.b=b;break;case 1:this.j(z,
|
|||
|
A);break;case 2:B(this);break;default:throw Error("unknown BTYPE: "+c);}}return this.m()};
|
|||
|
var C=[16,17,18,0,8,7,9,6,10,5,11,4,12,3,13,2,14,1,15],D=r?new Uint16Array(C):C,E=[3,4,5,6,7,8,9,10,11,13,15,17,19,23,27,31,35,43,51,59,67,83,99,115,131,163,195,227,258,258,258],F=r?new Uint16Array(E):E,G=[0,0,0,0,0,0,0,0,1,1,1,1,2,2,2,2,3,3,3,3,4,4,4,4,5,5,5,5,0,0,0],H=r?new Uint8Array(G):G,I=[1,2,3,4,5,7,9,13,17,25,33,49,65,97,129,193,257,385,513,769,1025,1537,2049,3073,4097,6145,8193,12289,16385,24577],J=r?new Uint16Array(I):I,K=[0,0,0,0,1,1,2,2,3,3,4,4,5,5,6,6,7,7,8,8,9,9,10,10,11,11,12,12,13,
|
|||
|
13],L=r?new Uint8Array(K):K,M=new (r?Uint8Array:Array)(288),N,O;N=0;for(O=M.length;N<O;++N)M[N]=143>=N?8:255>=N?9:279>=N?7:8;var z=u(M),P=new (r?Uint8Array:Array)(30),Q,R;Q=0;for(R=P.length;Q<R;++Q)P[Q]=5;var A=u(P);function y(c,d){for(var a=c.f,b=c.c,e=c.input,f=c.d,g;b<d;){g=e[f++];if(g===l)throw Error("input buffer is broken");a|=g<<b;b+=8}g=a&(1<<d)-1;c.f=a>>>d;c.c=b-d;c.d=f;return g}
|
|||
|
function S(c,d){for(var a=c.f,b=c.c,e=c.input,f=c.d,g=d[0],h=d[1],k,m,s;b<h;){k=e[f++];if(k===l)break;a|=k<<b;b+=8}m=g[a&(1<<h)-1];s=m>>>16;c.f=a>>s;c.c=b-s;c.d=f;return m&65535}
|
|||
|
function B(c){function d(a,c,b){var d,f,e,g;for(g=0;g<a;)switch(d=S(this,c),d){case 16:for(e=3+y(this,2);e--;)b[g++]=f;break;case 17:for(e=3+y(this,3);e--;)b[g++]=0;f=0;break;case 18:for(e=11+y(this,7);e--;)b[g++]=0;f=0;break;default:f=b[g++]=d}return b}var a=y(c,5)+257,b=y(c,5)+1,e=y(c,4)+4,f=new (r?Uint8Array:Array)(D.length),g,h,k,m;for(m=0;m<e;++m)f[D[m]]=y(c,3);g=u(f);h=new (r?Uint8Array:Array)(a);k=new (r?Uint8Array:Array)(b);c.j(u(d.call(c,a,g,h)),u(d.call(c,b,g,k)))}
|
|||
|
v.prototype.j=function(c,d){var a=this.b,b=this.a;this.n=c;for(var e=a.length-258,f,g,h,k;256!==(f=S(this,c));)if(256>f)b>=e&&(this.a=b,a=this.e(),b=this.a),a[b++]=f;else{g=f-257;k=F[g];0<H[g]&&(k+=y(this,H[g]));f=S(this,d);h=J[f];0<L[f]&&(h+=y(this,L[f]));b>=e&&(this.a=b,a=this.e(),b=this.a);for(;k--;)a[b]=a[b++-h]}for(;8<=this.c;)this.c-=8,this.d--;this.a=b};
|
|||
|
v.prototype.s=function(c,d){var a=this.b,b=this.a;this.n=c;for(var e=a.length,f,g,h,k;256!==(f=S(this,c));)if(256>f)b>=e&&(a=this.e(),e=a.length),a[b++]=f;else{g=f-257;k=F[g];0<H[g]&&(k+=y(this,H[g]));f=S(this,d);h=J[f];0<L[f]&&(h+=y(this,L[f]));b+k>e&&(a=this.e(),e=a.length);for(;k--;)a[b]=a[b++-h]}for(;8<=this.c;)this.c-=8,this.d--;this.a=b};
|
|||
|
v.prototype.e=function(){var c=new (r?Uint8Array:Array)(this.a-32768),d=this.a-32768,a,b,e=this.b;if(r)c.set(e.subarray(32768,c.length));else{a=0;for(b=c.length;a<b;++a)c[a]=e[a+32768]}this.g.push(c);this.k+=c.length;if(r)e.set(e.subarray(d,d+32768));else for(a=0;32768>a;++a)e[a]=e[d+a];this.a=32768;return e};
|
|||
|
v.prototype.u=function(c){var d,a=this.input.length/this.d+1|0,b,e,f,g=this.input,h=this.b;c&&("number"===typeof c.o&&(a=c.o),"number"===typeof c.q&&(a+=c.q));2>a?(b=(g.length-this.d)/this.n[2],f=258*(b/2)|0,e=f<h.length?h.length+f:h.length<<1):e=h.length*a;r?(d=new Uint8Array(e),d.set(h)):d=h;return this.b=d};
|
|||
|
v.prototype.m=function(){var c=0,d=this.b,a=this.g,b,e=new (r?Uint8Array:Array)(this.k+(this.a-32768)),f,g,h,k;if(0===a.length)return r?this.b.subarray(32768,this.a):this.b.slice(32768,this.a);f=0;for(g=a.length;f<g;++f){b=a[f];h=0;for(k=b.length;h<k;++h)e[c++]=b[h]}f=32768;for(g=this.a;f<g;++f)e[c++]=d[f];this.g=[];return this.buffer=e};
|
|||
|
v.prototype.r=function(){var c,d=this.a;r?this.p?(c=new Uint8Array(d),c.set(this.b.subarray(0,d))):c=this.b.subarray(0,d):(this.b.length>d&&(this.b.length=d),c=this.b);return this.buffer=c};q("Zlib.RawInflate",v);q("Zlib.RawInflate.prototype.decompress",v.prototype.t);var T={ADAPTIVE:w,BLOCK:x},U,V,W,X;if(Object.keys)U=Object.keys(T);else for(V in U=[],W=0,T)U[W++]=V;W=0;for(X=U.length;W<X;++W)V=U[W],q("Zlib.RawInflate.BufferType."+V,T[V]);}).call(this); //@ sourceMappingURL=rawinflate.min.js.map
|
|||
|
;
|
|||
|
define("zlib", (function (global) {
|
|||
|
return function () {
|
|||
|
var ret, fn;
|
|||
|
return ret || global.Zlib.RawInflate;
|
|||
|
};
|
|||
|
}(this)));
|
|||
|
|
|||
|
/// <amd-dependency path="zlib" />
|
|||
|
var __extends = this.__extends || function (d, b) {
|
|||
|
for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p];
|
|||
|
function __() { this.constructor = d; }
|
|||
|
__.prototype = b.prototype;
|
|||
|
d.prototype = new __();
|
|||
|
};
|
|||
|
define('backend/zipfs',["require", "exports", '../core/buffer', '../core/api_error', '../generic/file_index', '../core/browserfs', '../core/node_fs_stats', '../core/file_system', '../core/file_flag', '../core/buffer_core_arraybuffer', '../generic/preload_file', "zlib"], function(require, exports, buffer, api_error, file_index, browserfs, node_fs_stats, file_system, file_flag, buffer_core_arraybuffer, preload_file) {
|
|||
|
var ApiError = api_error.ApiError;
|
|||
|
var ErrorCode = api_error.ErrorCode;
|
|||
|
var ActionType = file_flag.ActionType;
|
|||
|
|
|||
|
|
|||
|
|
|||
|
var RawInflate = Zlib.RawInflate;
|
|||
|
|
|||
|
/**
|
|||
|
* 4.4.2.2: Indicates the compatibiltiy of a file's external attributes.
|
|||
|
*/
|
|||
|
(function (ExternalFileAttributeType) {
|
|||
|
ExternalFileAttributeType[ExternalFileAttributeType["MSDOS"] = 0] = "MSDOS";
|
|||
|
ExternalFileAttributeType[ExternalFileAttributeType["AMIGA"] = 1] = "AMIGA";
|
|||
|
ExternalFileAttributeType[ExternalFileAttributeType["OPENVMS"] = 2] = "OPENVMS";
|
|||
|
ExternalFileAttributeType[ExternalFileAttributeType["UNIX"] = 3] = "UNIX";
|
|||
|
ExternalFileAttributeType[ExternalFileAttributeType["VM_CMS"] = 4] = "VM_CMS";
|
|||
|
ExternalFileAttributeType[ExternalFileAttributeType["ATARI_ST"] = 5] = "ATARI_ST";
|
|||
|
ExternalFileAttributeType[ExternalFileAttributeType["OS2_HPFS"] = 6] = "OS2_HPFS";
|
|||
|
ExternalFileAttributeType[ExternalFileAttributeType["MAC"] = 7] = "MAC";
|
|||
|
ExternalFileAttributeType[ExternalFileAttributeType["Z_SYSTEM"] = 8] = "Z_SYSTEM";
|
|||
|
ExternalFileAttributeType[ExternalFileAttributeType["CP_M"] = 9] = "CP_M";
|
|||
|
ExternalFileAttributeType[ExternalFileAttributeType["NTFS"] = 10] = "NTFS";
|
|||
|
ExternalFileAttributeType[ExternalFileAttributeType["MVS"] = 11] = "MVS";
|
|||
|
ExternalFileAttributeType[ExternalFileAttributeType["VSE"] = 12] = "VSE";
|
|||
|
ExternalFileAttributeType[ExternalFileAttributeType["ACORN_RISC"] = 13] = "ACORN_RISC";
|
|||
|
ExternalFileAttributeType[ExternalFileAttributeType["VFAT"] = 14] = "VFAT";
|
|||
|
ExternalFileAttributeType[ExternalFileAttributeType["ALT_MVS"] = 15] = "ALT_MVS";
|
|||
|
ExternalFileAttributeType[ExternalFileAttributeType["BEOS"] = 16] = "BEOS";
|
|||
|
ExternalFileAttributeType[ExternalFileAttributeType["TANDEM"] = 17] = "TANDEM";
|
|||
|
ExternalFileAttributeType[ExternalFileAttributeType["OS_400"] = 18] = "OS_400";
|
|||
|
ExternalFileAttributeType[ExternalFileAttributeType["OSX"] = 19] = "OSX";
|
|||
|
})(exports.ExternalFileAttributeType || (exports.ExternalFileAttributeType = {}));
|
|||
|
var ExternalFileAttributeType = exports.ExternalFileAttributeType;
|
|||
|
|
|||
|
/**
|
|||
|
* 4.4.5
|
|||
|
*/
|
|||
|
(function (CompressionMethod) {
|
|||
|
CompressionMethod[CompressionMethod["STORED"] = 0] = "STORED";
|
|||
|
CompressionMethod[CompressionMethod["SHRUNK"] = 1] = "SHRUNK";
|
|||
|
CompressionMethod[CompressionMethod["REDUCED_1"] = 2] = "REDUCED_1";
|
|||
|
CompressionMethod[CompressionMethod["REDUCED_2"] = 3] = "REDUCED_2";
|
|||
|
CompressionMethod[CompressionMethod["REDUCED_3"] = 4] = "REDUCED_3";
|
|||
|
CompressionMethod[CompressionMethod["REDUCED_4"] = 5] = "REDUCED_4";
|
|||
|
CompressionMethod[CompressionMethod["IMPLODE"] = 6] = "IMPLODE";
|
|||
|
CompressionMethod[CompressionMethod["DEFLATE"] = 8] = "DEFLATE";
|
|||
|
CompressionMethod[CompressionMethod["DEFLATE64"] = 9] = "DEFLATE64";
|
|||
|
CompressionMethod[CompressionMethod["TERSE_OLD"] = 10] = "TERSE_OLD";
|
|||
|
CompressionMethod[CompressionMethod["BZIP2"] = 12] = "BZIP2";
|
|||
|
CompressionMethod[CompressionMethod["LZMA"] = 14] = "LZMA";
|
|||
|
CompressionMethod[CompressionMethod["TERSE_NEW"] = 18] = "TERSE_NEW";
|
|||
|
CompressionMethod[CompressionMethod["LZ77"] = 19] = "LZ77";
|
|||
|
CompressionMethod[CompressionMethod["WAVPACK"] = 97] = "WAVPACK";
|
|||
|
CompressionMethod[CompressionMethod["PPMD"] = 98] = "PPMD";
|
|||
|
})(exports.CompressionMethod || (exports.CompressionMethod = {}));
|
|||
|
var CompressionMethod = exports.CompressionMethod;
|
|||
|
|
|||
|
/**
|
|||
|
* Converts the input time and date in MS-DOS format into a JavaScript Date
|
|||
|
* object.
|
|||
|
*/
|
|||
|
function msdos2date(time, date) {
|
|||
|
// MS-DOS Date
|
|||
|
//|0 0 0 0 0|0 0 0 0|0 0 0 0 0 0 0
|
|||
|
// D (1-31) M (1-23) Y (from 1980)
|
|||
|
var day = date & 0x1F;
|
|||
|
|
|||
|
// JS date is 0-indexed, DOS is 1-indexed.
|
|||
|
var month = ((date >> 5) & 0xF) - 1;
|
|||
|
var year = (date >> 9) + 1980;
|
|||
|
|
|||
|
// MS DOS Time
|
|||
|
//|0 0 0 0 0|0 0 0 0 0 0|0 0 0 0 0
|
|||
|
// Second Minute Hour
|
|||
|
var second = time & 0x1F;
|
|||
|
var minute = (time >> 5) & 0x3F;
|
|||
|
var hour = time >> 11;
|
|||
|
return new Date(year, month, day, hour, minute, second);
|
|||
|
}
|
|||
|
|
|||
|
/**
|
|||
|
* Safely returns the string from the buffer, even if it is 0 bytes long.
|
|||
|
* (Normally, calling toString() on a buffer with start === end causes an
|
|||
|
* exception).
|
|||
|
*/
|
|||
|
function safeToString(buff, useUTF8, start, length) {
|
|||
|
return length === 0 ? "" : buff.toString(useUTF8 ? 'utf8' : 'extended_ascii', start, start + length);
|
|||
|
}
|
|||
|
|
|||
|
/*
|
|||
|
4.3.6 Overall .ZIP file format:
|
|||
|
|
|||
|
[local file header 1]
|
|||
|
[encryption header 1]
|
|||
|
[file data 1]
|
|||
|
[data descriptor 1]
|
|||
|
.
|
|||
|
.
|
|||
|
.
|
|||
|
[local file header n]
|
|||
|
[encryption header n]
|
|||
|
[file data n]
|
|||
|
[data descriptor n]
|
|||
|
[archive decryption header]
|
|||
|
[archive extra data record]
|
|||
|
[central directory header 1]
|
|||
|
.
|
|||
|
.
|
|||
|
.
|
|||
|
[central directory header n]
|
|||
|
[zip64 end of central directory record]
|
|||
|
[zip64 end of central directory locator]
|
|||
|
[end of central directory record]
|
|||
|
*/
|
|||
|
/*
|
|||
|
4.3.7 Local file header:
|
|||
|
|
|||
|
local file header signature 4 bytes (0x04034b50)
|
|||
|
version needed to extract 2 bytes
|
|||
|
general purpose bit flag 2 bytes
|
|||
|
compression method 2 bytes
|
|||
|
last mod file time 2 bytes
|
|||
|
last mod file date 2 bytes
|
|||
|
crc-32 4 bytes
|
|||
|
compressed size 4 bytes
|
|||
|
uncompressed size 4 bytes
|
|||
|
file name length 2 bytes
|
|||
|
extra field length 2 bytes
|
|||
|
|
|||
|
file name (variable size)
|
|||
|
extra field (variable size)
|
|||
|
*/
|
|||
|
var FileHeader = (function () {
|
|||
|
function FileHeader(data) {
|
|||
|
this.data = data;
|
|||
|
if (data.readUInt32LE(0) !== 0x04034b50) {
|
|||
|
throw new ApiError(9 /* EINVAL */, "Invalid Zip file: Local file header has invalid signature: " + this.data.readUInt32LE(0));
|
|||
|
}
|
|||
|
}
|
|||
|
FileHeader.prototype.versionNeeded = function () {
|
|||
|
return this.data.readUInt16LE(4);
|
|||
|
};
|
|||
|
FileHeader.prototype.flags = function () {
|
|||
|
return this.data.readUInt16LE(6);
|
|||
|
};
|
|||
|
FileHeader.prototype.compressionMethod = function () {
|
|||
|
return this.data.readUInt16LE(8);
|
|||
|
};
|
|||
|
FileHeader.prototype.lastModFileTime = function () {
|
|||
|
// Time and date is in MS-DOS format.
|
|||
|
return msdos2date(this.data.readUInt16LE(10), this.data.readUInt16LE(12));
|
|||
|
};
|
|||
|
FileHeader.prototype.crc32 = function () {
|
|||
|
return this.data.readUInt32LE(14);
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* These two values are COMPLETELY USELESS.
|
|||
|
*
|
|||
|
* Section 4.4.9:
|
|||
|
* If bit 3 of the general purpose bit flag is set,
|
|||
|
* these fields are set to zero in the local header and the
|
|||
|
* correct values are put in the data descriptor and
|
|||
|
* in the central directory.
|
|||
|
*
|
|||
|
* So we'll just use the central directory's values.
|
|||
|
*/
|
|||
|
// public compressedSize(): number { return this.data.readUInt32LE(18); }
|
|||
|
// public uncompressedSize(): number { return this.data.readUInt32LE(22); }
|
|||
|
FileHeader.prototype.fileNameLength = function () {
|
|||
|
return this.data.readUInt16LE(26);
|
|||
|
};
|
|||
|
FileHeader.prototype.extraFieldLength = function () {
|
|||
|
return this.data.readUInt16LE(28);
|
|||
|
};
|
|||
|
FileHeader.prototype.fileName = function () {
|
|||
|
return safeToString(this.data, this.useUTF8(), 30, this.fileNameLength());
|
|||
|
};
|
|||
|
FileHeader.prototype.extraField = function () {
|
|||
|
var start = 30 + this.fileNameLength();
|
|||
|
return this.data.slice(start, start + this.extraFieldLength());
|
|||
|
};
|
|||
|
FileHeader.prototype.totalSize = function () {
|
|||
|
return 30 + this.fileNameLength() + this.extraFieldLength();
|
|||
|
};
|
|||
|
FileHeader.prototype.useUTF8 = function () {
|
|||
|
return (this.flags() & 0x800) === 0x800;
|
|||
|
};
|
|||
|
return FileHeader;
|
|||
|
})();
|
|||
|
exports.FileHeader = FileHeader;
|
|||
|
|
|||
|
/**
|
|||
|
4.3.8 File data
|
|||
|
|
|||
|
Immediately following the local header for a file
|
|||
|
SHOULD be placed the compressed or stored data for the file.
|
|||
|
If the file is encrypted, the encryption header for the file
|
|||
|
SHOULD be placed after the local header and before the file
|
|||
|
data. The series of [local file header][encryption header]
|
|||
|
[file data][data descriptor] repeats for each file in the
|
|||
|
.ZIP archive.
|
|||
|
|
|||
|
Zero-byte files, directories, and other file types that
|
|||
|
contain no content MUST not include file data.
|
|||
|
*/
|
|||
|
var FileData = (function () {
|
|||
|
function FileData(header, record, data) {
|
|||
|
this.header = header;
|
|||
|
this.record = record;
|
|||
|
this.data = data;
|
|||
|
}
|
|||
|
FileData.prototype.decompress = function () {
|
|||
|
var buff = this.data;
|
|||
|
|
|||
|
// Check the compression
|
|||
|
var compressionMethod = this.header.compressionMethod();
|
|||
|
switch (compressionMethod) {
|
|||
|
case 8 /* DEFLATE */:
|
|||
|
// Convert to Uint8Array or an array of bytes for the library.
|
|||
|
if (buff.getBufferCore() instanceof buffer_core_arraybuffer.BufferCoreArrayBuffer) {
|
|||
|
// Grab a slice of the zip file that contains the compressed data
|
|||
|
// (avoids copying).
|
|||
|
// XXX: Does RawInflate mutate the buffer? I hope not.
|
|||
|
var bcore = buff.getBufferCore();
|
|||
|
var dview = bcore.getDataView();
|
|||
|
var start = dview.byteOffset + buff.getOffset();
|
|||
|
var uarray = (new Uint8Array(dview.buffer)).subarray(start, start + this.record.compressedSize());
|
|||
|
var data = (new RawInflate(uarray)).decompress();
|
|||
|
return new buffer.Buffer(new buffer_core_arraybuffer.BufferCoreArrayBuffer(data.buffer), data.byteOffset, data.byteOffset + data.length);
|
|||
|
} else {
|
|||
|
// Convert to an array of bytes and decompress, then write into a new
|
|||
|
// buffer :(
|
|||
|
var newBuff = buff.slice(0, this.record.compressedSize());
|
|||
|
return new buffer.Buffer((new RawInflate(newBuff.toJSON().data)).decompress());
|
|||
|
}
|
|||
|
case 0 /* STORED */:
|
|||
|
// Grab and copy.
|
|||
|
return buff.sliceCopy(0, this.record.uncompressedSize());
|
|||
|
default:
|
|||
|
var name = CompressionMethod[compressionMethod];
|
|||
|
name = name ? name : "Unknown: " + compressionMethod;
|
|||
|
throw new ApiError(9 /* EINVAL */, "Invalid compression method on file '" + this.header.fileName() + "': " + name);
|
|||
|
}
|
|||
|
};
|
|||
|
return FileData;
|
|||
|
})();
|
|||
|
exports.FileData = FileData;
|
|||
|
|
|||
|
/*
|
|||
|
4.3.9 Data descriptor:
|
|||
|
|
|||
|
crc-32 4 bytes
|
|||
|
compressed size 4 bytes
|
|||
|
uncompressed size 4 bytes
|
|||
|
*/
|
|||
|
var DataDescriptor = (function () {
|
|||
|
function DataDescriptor(data) {
|
|||
|
this.data = data;
|
|||
|
}
|
|||
|
DataDescriptor.prototype.crc32 = function () {
|
|||
|
return this.data.readUInt32LE(0);
|
|||
|
};
|
|||
|
DataDescriptor.prototype.compressedSize = function () {
|
|||
|
return this.data.readUInt32LE(4);
|
|||
|
};
|
|||
|
DataDescriptor.prototype.uncompressedSize = function () {
|
|||
|
return this.data.readUInt32LE(8);
|
|||
|
};
|
|||
|
return DataDescriptor;
|
|||
|
})();
|
|||
|
exports.DataDescriptor = DataDescriptor;
|
|||
|
|
|||
|
/*
|
|||
|
` 4.3.10 Archive decryption header:
|
|||
|
|
|||
|
4.3.10.1 The Archive Decryption Header is introduced in version 6.2
|
|||
|
of the ZIP format specification. This record exists in support
|
|||
|
of the Central Directory Encryption Feature implemented as part of
|
|||
|
the Strong Encryption Specification as described in this document.
|
|||
|
When the Central Directory Structure is encrypted, this decryption
|
|||
|
header MUST precede the encrypted data segment.
|
|||
|
*/
|
|||
|
/*
|
|||
|
4.3.11 Archive extra data record:
|
|||
|
|
|||
|
archive extra data signature 4 bytes (0x08064b50)
|
|||
|
extra field length 4 bytes
|
|||
|
extra field data (variable size)
|
|||
|
|
|||
|
4.3.11.1 The Archive Extra Data Record is introduced in version 6.2
|
|||
|
of the ZIP format specification. This record MAY be used in support
|
|||
|
of the Central Directory Encryption Feature implemented as part of
|
|||
|
the Strong Encryption Specification as described in this document.
|
|||
|
When present, this record MUST immediately precede the central
|
|||
|
directory data structure.
|
|||
|
*/
|
|||
|
var ArchiveExtraDataRecord = (function () {
|
|||
|
function ArchiveExtraDataRecord(data) {
|
|||
|
this.data = data;
|
|||
|
if (this.data.readUInt32LE(0) !== 0x08064b50) {
|
|||
|
throw new ApiError(9 /* EINVAL */, "Invalid archive extra data record signature: " + this.data.readUInt32LE(0));
|
|||
|
}
|
|||
|
}
|
|||
|
ArchiveExtraDataRecord.prototype.length = function () {
|
|||
|
return this.data.readUInt32LE(4);
|
|||
|
};
|
|||
|
ArchiveExtraDataRecord.prototype.extraFieldData = function () {
|
|||
|
return this.data.slice(8, 8 + this.length());
|
|||
|
};
|
|||
|
return ArchiveExtraDataRecord;
|
|||
|
})();
|
|||
|
exports.ArchiveExtraDataRecord = ArchiveExtraDataRecord;
|
|||
|
|
|||
|
/*
|
|||
|
4.3.13 Digital signature:
|
|||
|
|
|||
|
header signature 4 bytes (0x05054b50)
|
|||
|
size of data 2 bytes
|
|||
|
signature data (variable size)
|
|||
|
|
|||
|
With the introduction of the Central Directory Encryption
|
|||
|
feature in version 6.2 of this specification, the Central
|
|||
|
Directory Structure MAY be stored both compressed and encrypted.
|
|||
|
Although not required, it is assumed when encrypting the
|
|||
|
Central Directory Structure, that it will be compressed
|
|||
|
for greater storage efficiency. Information on the
|
|||
|
Central Directory Encryption feature can be found in the section
|
|||
|
describing the Strong Encryption Specification. The Digital
|
|||
|
Signature record will be neither compressed nor encrypted.
|
|||
|
*/
|
|||
|
var DigitalSignature = (function () {
|
|||
|
function DigitalSignature(data) {
|
|||
|
this.data = data;
|
|||
|
if (this.data.readUInt32LE(0) !== 0x05054b50) {
|
|||
|
throw new ApiError(9 /* EINVAL */, "Invalid digital signature signature: " + this.data.readUInt32LE(0));
|
|||
|
}
|
|||
|
}
|
|||
|
DigitalSignature.prototype.size = function () {
|
|||
|
return this.data.readUInt16LE(4);
|
|||
|
};
|
|||
|
DigitalSignature.prototype.signatureData = function () {
|
|||
|
return this.data.slice(6, 6 + this.size());
|
|||
|
};
|
|||
|
return DigitalSignature;
|
|||
|
})();
|
|||
|
exports.DigitalSignature = DigitalSignature;
|
|||
|
|
|||
|
/*
|
|||
|
4.3.12 Central directory structure:
|
|||
|
|
|||
|
central file header signature 4 bytes (0x02014b50)
|
|||
|
version made by 2 bytes
|
|||
|
version needed to extract 2 bytes
|
|||
|
general purpose bit flag 2 bytes
|
|||
|
compression method 2 bytes
|
|||
|
last mod file time 2 bytes
|
|||
|
last mod file date 2 bytes
|
|||
|
crc-32 4 bytes
|
|||
|
compressed size 4 bytes
|
|||
|
uncompressed size 4 bytes
|
|||
|
file name length 2 bytes
|
|||
|
extra field length 2 bytes
|
|||
|
file comment length 2 bytes
|
|||
|
disk number start 2 bytes
|
|||
|
internal file attributes 2 bytes
|
|||
|
external file attributes 4 bytes
|
|||
|
relative offset of local header 4 bytes
|
|||
|
|
|||
|
file name (variable size)
|
|||
|
extra field (variable size)
|
|||
|
file comment (variable size)
|
|||
|
*/
|
|||
|
var CentralDirectory = (function () {
|
|||
|
function CentralDirectory(zipData, data) {
|
|||
|
this.zipData = zipData;
|
|||
|
this.data = data;
|
|||
|
// Sanity check.
|
|||
|
if (this.data.readUInt32LE(0) !== 0x02014b50)
|
|||
|
throw new ApiError(9 /* EINVAL */, "Invalid Zip file: Central directory record has invalid signature: " + this.data.readUInt32LE(0));
|
|||
|
}
|
|||
|
CentralDirectory.prototype.versionMadeBy = function () {
|
|||
|
return this.data.readUInt16LE(4);
|
|||
|
};
|
|||
|
CentralDirectory.prototype.versionNeeded = function () {
|
|||
|
return this.data.readUInt16LE(6);
|
|||
|
};
|
|||
|
CentralDirectory.prototype.flag = function () {
|
|||
|
return this.data.readUInt16LE(8);
|
|||
|
};
|
|||
|
CentralDirectory.prototype.compressionMethod = function () {
|
|||
|
return this.data.readUInt16LE(10);
|
|||
|
};
|
|||
|
CentralDirectory.prototype.lastModFileTime = function () {
|
|||
|
// Time and date is in MS-DOS format.
|
|||
|
return msdos2date(this.data.readUInt16LE(12), this.data.readUInt16LE(14));
|
|||
|
};
|
|||
|
CentralDirectory.prototype.crc32 = function () {
|
|||
|
return this.data.readUInt32LE(16);
|
|||
|
};
|
|||
|
CentralDirectory.prototype.compressedSize = function () {
|
|||
|
return this.data.readUInt32LE(20);
|
|||
|
};
|
|||
|
CentralDirectory.prototype.uncompressedSize = function () {
|
|||
|
return this.data.readUInt32LE(24);
|
|||
|
};
|
|||
|
CentralDirectory.prototype.fileNameLength = function () {
|
|||
|
return this.data.readUInt16LE(28);
|
|||
|
};
|
|||
|
CentralDirectory.prototype.extraFieldLength = function () {
|
|||
|
return this.data.readUInt16LE(30);
|
|||
|
};
|
|||
|
CentralDirectory.prototype.fileCommentLength = function () {
|
|||
|
return this.data.readUInt16LE(32);
|
|||
|
};
|
|||
|
CentralDirectory.prototype.diskNumberStart = function () {
|
|||
|
return this.data.readUInt16LE(34);
|
|||
|
};
|
|||
|
CentralDirectory.prototype.internalAttributes = function () {
|
|||
|
return this.data.readUInt16LE(36);
|
|||
|
};
|
|||
|
CentralDirectory.prototype.externalAttributes = function () {
|
|||
|
return this.data.readUInt32LE(38);
|
|||
|
};
|
|||
|
CentralDirectory.prototype.headerRelativeOffset = function () {
|
|||
|
return this.data.readUInt32LE(42);
|
|||
|
};
|
|||
|
CentralDirectory.prototype.fileName = function () {
|
|||
|
/*
|
|||
|
4.4.17.1 claims:
|
|||
|
* All slashes are forward ('/') slashes.
|
|||
|
* Filename doesn't begin with a slash.
|
|||
|
* No drive letters or any nonsense like that.
|
|||
|
* If filename is missing, the input came from standard input.
|
|||
|
|
|||
|
Unfortunately, this isn't true in practice. Some Windows zip utilities use
|
|||
|
a backslash here, but the correct Unix-style path in file headers.
|
|||
|
|
|||
|
To avoid seeking all over the file to recover the known-good filenames
|
|||
|
from file headers, we simply convert '/' to '\' here.
|
|||
|
*/
|
|||
|
var fileName = safeToString(this.data, this.useUTF8(), 46, this.fileNameLength());
|
|||
|
return fileName.replace(/\\/g, "/");
|
|||
|
};
|
|||
|
CentralDirectory.prototype.extraField = function () {
|
|||
|
var start = 44 + this.fileNameLength();
|
|||
|
return this.data.slice(start, start + this.extraFieldLength());
|
|||
|
};
|
|||
|
CentralDirectory.prototype.fileComment = function () {
|
|||
|
var start = 46 + this.fileNameLength() + this.extraFieldLength();
|
|||
|
return safeToString(this.data, this.useUTF8(), start, this.fileCommentLength());
|
|||
|
};
|
|||
|
CentralDirectory.prototype.totalSize = function () {
|
|||
|
return 46 + this.fileNameLength() + this.extraFieldLength() + this.fileCommentLength();
|
|||
|
};
|
|||
|
CentralDirectory.prototype.isDirectory = function () {
|
|||
|
// NOTE: This assumes that the zip file implementation uses the lower byte
|
|||
|
// of external attributes for DOS attributes for
|
|||
|
// backwards-compatibility. This is not mandated, but appears to be
|
|||
|
// commonplace.
|
|||
|
// According to the spec, the layout of external attributes is
|
|||
|
// platform-dependent.
|
|||
|
// If that fails, we also check if the name of the file ends in '/',
|
|||
|
// which is what Java's ZipFile implementation does.
|
|||
|
var fileName = this.fileName();
|
|||
|
return (this.externalAttributes() & 0x10 ? true : false) || (fileName.charAt(fileName.length - 1) === '/');
|
|||
|
};
|
|||
|
CentralDirectory.prototype.isFile = function () {
|
|||
|
return !this.isDirectory();
|
|||
|
};
|
|||
|
CentralDirectory.prototype.useUTF8 = function () {
|
|||
|
return (this.flag() & 0x800) === 0x800;
|
|||
|
};
|
|||
|
CentralDirectory.prototype.isEncrypted = function () {
|
|||
|
return (this.flag() & 0x1) === 0x1;
|
|||
|
};
|
|||
|
CentralDirectory.prototype.getData = function () {
|
|||
|
// Need to grab the header before we can figure out where the actual
|
|||
|
// compressed data starts.
|
|||
|
var start = this.headerRelativeOffset();
|
|||
|
var header = new FileHeader(this.zipData.slice(start));
|
|||
|
var filedata = new FileData(header, this, this.zipData.slice(start + header.totalSize()));
|
|||
|
return filedata.decompress();
|
|||
|
};
|
|||
|
CentralDirectory.prototype.getStats = function () {
|
|||
|
return new node_fs_stats.Stats(32768 /* FILE */, this.uncompressedSize(), 0x16D, new Date(), this.lastModFileTime());
|
|||
|
};
|
|||
|
return CentralDirectory;
|
|||
|
})();
|
|||
|
exports.CentralDirectory = CentralDirectory;
|
|||
|
|
|||
|
/*
|
|||
|
4.3.16: end of central directory record
|
|||
|
end of central dir signature 4 bytes (0x06054b50)
|
|||
|
number of this disk 2 bytes
|
|||
|
number of the disk with the
|
|||
|
start of the central directory 2 bytes
|
|||
|
total number of entries in the
|
|||
|
central directory on this disk 2 bytes
|
|||
|
total number of entries in
|
|||
|
the central directory 2 bytes
|
|||
|
size of the central directory 4 bytes
|
|||
|
offset of start of central
|
|||
|
directory with respect to
|
|||
|
the starting disk number 4 bytes
|
|||
|
.ZIP file comment length 2 bytes
|
|||
|
.ZIP file comment (variable size)
|
|||
|
*/
|
|||
|
var EndOfCentralDirectory = (function () {
|
|||
|
function EndOfCentralDirectory(data) {
|
|||
|
this.data = data;
|
|||
|
if (this.data.readUInt32LE(0) !== 0x06054b50)
|
|||
|
throw new ApiError(9 /* EINVAL */, "Invalid Zip file: End of central directory record has invalid signature: " + this.data.readUInt32LE(0));
|
|||
|
}
|
|||
|
EndOfCentralDirectory.prototype.diskNumber = function () {
|
|||
|
return this.data.readUInt16LE(4);
|
|||
|
};
|
|||
|
EndOfCentralDirectory.prototype.cdDiskNumber = function () {
|
|||
|
return this.data.readUInt16LE(6);
|
|||
|
};
|
|||
|
EndOfCentralDirectory.prototype.cdDiskEntryCount = function () {
|
|||
|
return this.data.readUInt16LE(8);
|
|||
|
};
|
|||
|
EndOfCentralDirectory.prototype.cdTotalEntryCount = function () {
|
|||
|
return this.data.readUInt16LE(10);
|
|||
|
};
|
|||
|
EndOfCentralDirectory.prototype.cdSize = function () {
|
|||
|
return this.data.readUInt32LE(12);
|
|||
|
};
|
|||
|
EndOfCentralDirectory.prototype.cdOffset = function () {
|
|||
|
return this.data.readUInt32LE(16);
|
|||
|
};
|
|||
|
EndOfCentralDirectory.prototype.cdZipComment = function () {
|
|||
|
// Assuming UTF-8. The specification doesn't specify.
|
|||
|
return safeToString(this.data, true, 22, this.data.readUInt16LE(20));
|
|||
|
};
|
|||
|
return EndOfCentralDirectory;
|
|||
|
})();
|
|||
|
exports.EndOfCentralDirectory = EndOfCentralDirectory;
|
|||
|
|
|||
|
var ZipFS = (function (_super) {
|
|||
|
__extends(ZipFS, _super);
|
|||
|
/**
|
|||
|
* Constructs a ZipFS from the given zip file data. Name is optional, and is
|
|||
|
* used primarily for our unit tests' purposes to differentiate different
|
|||
|
* test zip files in test output.
|
|||
|
*/
|
|||
|
function ZipFS(data, name) {
|
|||
|
if (typeof name === "undefined") { name = ''; }
|
|||
|
_super.call(this);
|
|||
|
this.data = data;
|
|||
|
this.name = name;
|
|||
|
this._index = new file_index.FileIndex();
|
|||
|
this.populateIndex();
|
|||
|
}
|
|||
|
ZipFS.prototype.getName = function () {
|
|||
|
return 'ZipFS' + (this.name !== '' ? ' ' + this.name : '');
|
|||
|
};
|
|||
|
|
|||
|
ZipFS.isAvailable = function () {
|
|||
|
return true;
|
|||
|
};
|
|||
|
|
|||
|
ZipFS.prototype.diskSpace = function (path, cb) {
|
|||
|
// Read-only file system.
|
|||
|
cb(this.data.length, 0);
|
|||
|
};
|
|||
|
|
|||
|
ZipFS.prototype.isReadOnly = function () {
|
|||
|
return true;
|
|||
|
};
|
|||
|
|
|||
|
ZipFS.prototype.supportsLinks = function () {
|
|||
|
return false;
|
|||
|
};
|
|||
|
|
|||
|
ZipFS.prototype.supportsProps = function () {
|
|||
|
return false;
|
|||
|
};
|
|||
|
|
|||
|
ZipFS.prototype.supportsSynch = function () {
|
|||
|
return true;
|
|||
|
};
|
|||
|
|
|||
|
ZipFS.prototype.statSync = function (path, isLstat) {
|
|||
|
var inode = this._index.getInode(path);
|
|||
|
if (inode === null) {
|
|||
|
throw new ApiError(1 /* ENOENT */, "" + path + " not found.");
|
|||
|
}
|
|||
|
var stats;
|
|||
|
if (inode.isFile()) {
|
|||
|
stats = inode.getData().getStats();
|
|||
|
} else {
|
|||
|
stats = inode.getStats();
|
|||
|
}
|
|||
|
return stats;
|
|||
|
};
|
|||
|
|
|||
|
ZipFS.prototype.openSync = function (path, flags, mode) {
|
|||
|
// INVARIANT: Cannot write to RO file systems.
|
|||
|
if (flags.isWriteable()) {
|
|||
|
throw new ApiError(0 /* EPERM */, path);
|
|||
|
}
|
|||
|
|
|||
|
// Check if the path exists, and is a file.
|
|||
|
var inode = this._index.getInode(path);
|
|||
|
if (inode === null) {
|
|||
|
throw new ApiError(1 /* ENOENT */, "" + path + " is not in the FileIndex.");
|
|||
|
}
|
|||
|
if (inode.isDir()) {
|
|||
|
throw new ApiError(8 /* EISDIR */, "" + path + " is a directory.");
|
|||
|
}
|
|||
|
var cdRecord = inode.getData();
|
|||
|
var stats = cdRecord.getStats();
|
|||
|
switch (flags.pathExistsAction()) {
|
|||
|
case 1 /* THROW_EXCEPTION */:
|
|||
|
case 2 /* TRUNCATE_FILE */:
|
|||
|
throw new ApiError(6 /* EEXIST */, "" + path + " already exists.");
|
|||
|
case 0 /* NOP */:
|
|||
|
return new preload_file.NoSyncFile(this, path, flags, stats, cdRecord.getData());
|
|||
|
default:
|
|||
|
throw new ApiError(9 /* EINVAL */, 'Invalid FileMode object.');
|
|||
|
}
|
|||
|
return null;
|
|||
|
};
|
|||
|
|
|||
|
ZipFS.prototype.readdirSync = function (path) {
|
|||
|
// Check if it exists.
|
|||
|
var inode = this._index.getInode(path);
|
|||
|
if (inode === null) {
|
|||
|
throw new ApiError(1 /* ENOENT */, "" + path + " not found.");
|
|||
|
} else if (inode.isFile()) {
|
|||
|
throw new ApiError(7 /* ENOTDIR */, "" + path + " is a file, not a directory.");
|
|||
|
}
|
|||
|
return inode.getListing();
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Specially-optimized readfile.
|
|||
|
*/
|
|||
|
ZipFS.prototype.readFileSync = function (fname, encoding, flag) {
|
|||
|
// Get file.
|
|||
|
var fd = this.openSync(fname, flag, 0x1a4);
|
|||
|
try {
|
|||
|
var fdCast = fd;
|
|||
|
var fdBuff = fdCast._buffer;
|
|||
|
if (encoding === null) {
|
|||
|
if (fdBuff.length > 0) {
|
|||
|
return fdBuff.sliceCopy();
|
|||
|
} else {
|
|||
|
return new buffer.Buffer(0);
|
|||
|
}
|
|||
|
}
|
|||
|
return fdBuff.toString(encoding);
|
|||
|
} finally {
|
|||
|
fd.closeSync();
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
/**
|
|||
|
* Locates the end of central directory record at the end of the file.
|
|||
|
* Throws an exception if it cannot be found.
|
|||
|
*/
|
|||
|
ZipFS.prototype.getEOCD = function () {
|
|||
|
// Unfortunately, the comment is variable size and up to 64K in size.
|
|||
|
// We assume that the magic signature does not appear in the comment, and
|
|||
|
// in the bytes between the comment and the signature. Other ZIP
|
|||
|
// implementations make this same assumption, since the alternative is to
|
|||
|
// read thread every entry in the file to get to it. :(
|
|||
|
// These are *negative* offsets from the end of the file.
|
|||
|
var startOffset = 22;
|
|||
|
var endOffset = Math.min(startOffset + 0xFFFF, this.data.length - 1);
|
|||
|
|
|||
|
for (var i = startOffset; i < endOffset; i++) {
|
|||
|
// Magic number: EOCD Signature
|
|||
|
if (this.data.readUInt32LE(this.data.length - i) === 0x06054b50) {
|
|||
|
return new EndOfCentralDirectory(this.data.slice(this.data.length - i));
|
|||
|
}
|
|||
|
}
|
|||
|
throw new ApiError(9 /* EINVAL */, "Invalid ZIP file: Could not locate End of Central Directory signature.");
|
|||
|
};
|
|||
|
|
|||
|
ZipFS.prototype.populateIndex = function () {
|
|||
|
var eocd = this.getEOCD();
|
|||
|
if (eocd.diskNumber() !== eocd.cdDiskNumber())
|
|||
|
throw new ApiError(9 /* EINVAL */, "ZipFS does not support spanned zip files.");
|
|||
|
|
|||
|
var cdPtr = eocd.cdOffset();
|
|||
|
if (cdPtr === 0xFFFFFFFF)
|
|||
|
throw new ApiError(9 /* EINVAL */, "ZipFS does not support Zip64.");
|
|||
|
var cdEnd = cdPtr + eocd.cdSize();
|
|||
|
while (cdPtr < cdEnd) {
|
|||
|
var cd = new CentralDirectory(this.data, this.data.slice(cdPtr));
|
|||
|
cdPtr += cd.totalSize();
|
|||
|
|
|||
|
// Paths must be absolute, yet zip file paths are always relative to the
|
|||
|
// zip root. So we append '/' and call it a day.
|
|||
|
var filename = cd.fileName();
|
|||
|
if (filename.charAt(0) === '/')
|
|||
|
throw new Error("WHY IS THIS ABSOLUTE");
|
|||
|
|
|||
|
// XXX: For the file index, strip the trailing '/'.
|
|||
|
if (filename.charAt(filename.length - 1) === '/') {
|
|||
|
filename = filename.substr(0, filename.length - 1);
|
|||
|
}
|
|||
|
if (cd.isDirectory()) {
|
|||
|
this._index.addPath('/' + filename, new file_index.DirInode());
|
|||
|
} else {
|
|||
|
this._index.addPath('/' + filename, new file_index.FileInode(cd));
|
|||
|
}
|
|||
|
}
|
|||
|
};
|
|||
|
return ZipFS;
|
|||
|
})(file_system.SynchronousFileSystem);
|
|||
|
exports.ZipFS = ZipFS;
|
|||
|
|
|||
|
browserfs.registerFileSystem('ZipFS', ZipFS);
|
|||
|
});
|
|||
|
//# sourceMappingURL=zipfs.js.map
|
|||
|
;require('core/global').BrowserFS=require('core/browserfs');require('generic/emscripten_fs');require('backend/IndexedDB');require('backend/XmlHttpRequest');require('backend/dropbox');require('backend/html5fs');require('backend/in_memory');require('backend/localStorage');require('backend/mountable_file_system');require('backend/zipfs');})();
|
|||
|
//# sourceMappingURL=browserfs.js.map
|