42354 lines
1.0 MiB
42354 lines
1.0 MiB
;(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);throw new Error("Cannot find module '"+o+"'")}var f=n[o]={exports:{}};t[o][0].call(f.exports,function(e){var n=t[o][1][e];return s(n?n:e)},f,f.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})({1:[function(require,module,exports){
|
|
var process=require("__browserify_process");/*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 = setImmediate;
|
|
}
|
|
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([], function () {
|
|
return async;
|
|
});
|
|
}
|
|
// Node.js
|
|
else if (typeof module !== 'undefined' && module.exports) {
|
|
module.exports = async;
|
|
}
|
|
// included directly via <script> tag
|
|
else {
|
|
root.async = async;
|
|
}
|
|
|
|
}());
|
|
|
|
},{"__browserify_process":15}],2:[function(require,module,exports){
|
|
|
|
|
|
//
|
|
// The shims in this file are not fully implemented shims for the ES5
|
|
// features, but do work for the particular usecases there is in
|
|
// the other modules.
|
|
//
|
|
|
|
var toString = Object.prototype.toString;
|
|
var hasOwnProperty = Object.prototype.hasOwnProperty;
|
|
|
|
// Array.isArray is supported in IE9
|
|
function isArray(xs) {
|
|
return toString.call(xs) === '[object Array]';
|
|
}
|
|
exports.isArray = typeof Array.isArray === 'function' ? Array.isArray : isArray;
|
|
|
|
// Array.prototype.indexOf is supported in IE9
|
|
exports.indexOf = function indexOf(xs, x) {
|
|
if (xs.indexOf) return xs.indexOf(x);
|
|
for (var i = 0; i < xs.length; i++) {
|
|
if (x === xs[i]) return i;
|
|
}
|
|
return -1;
|
|
};
|
|
|
|
// Array.prototype.filter is supported in IE9
|
|
exports.filter = function filter(xs, fn) {
|
|
if (xs.filter) return xs.filter(fn);
|
|
var res = [];
|
|
for (var i = 0; i < xs.length; i++) {
|
|
if (fn(xs[i], i, xs)) res.push(xs[i]);
|
|
}
|
|
return res;
|
|
};
|
|
|
|
// Array.prototype.forEach is supported in IE9
|
|
exports.forEach = function forEach(xs, fn, self) {
|
|
if (xs.forEach) return xs.forEach(fn, self);
|
|
for (var i = 0; i < xs.length; i++) {
|
|
fn.call(self, xs[i], i, xs);
|
|
}
|
|
};
|
|
|
|
// Array.prototype.map is supported in IE9
|
|
exports.map = function map(xs, fn) {
|
|
if (xs.map) return xs.map(fn);
|
|
var out = new Array(xs.length);
|
|
for (var i = 0; i < xs.length; i++) {
|
|
out[i] = fn(xs[i], i, xs);
|
|
}
|
|
return out;
|
|
};
|
|
|
|
// Array.prototype.reduce is supported in IE9
|
|
exports.reduce = function reduce(array, callback, opt_initialValue) {
|
|
if (array.reduce) return array.reduce(callback, opt_initialValue);
|
|
var value, isValueSet = false;
|
|
|
|
if (2 < arguments.length) {
|
|
value = opt_initialValue;
|
|
isValueSet = true;
|
|
}
|
|
for (var i = 0, l = array.length; l > i; ++i) {
|
|
if (array.hasOwnProperty(i)) {
|
|
if (isValueSet) {
|
|
value = callback(value, array[i], i, array);
|
|
}
|
|
else {
|
|
value = array[i];
|
|
isValueSet = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
return value;
|
|
};
|
|
|
|
// String.prototype.substr - negative index don't work in IE8
|
|
if ('ab'.substr(-1) !== 'b') {
|
|
exports.substr = function (str, start, length) {
|
|
// did we get a negative start, calculate how much it is from the beginning of the string
|
|
if (start < 0) start = str.length + start;
|
|
|
|
// call the original function
|
|
return str.substr(start, length);
|
|
};
|
|
} else {
|
|
exports.substr = function (str, start, length) {
|
|
return str.substr(start, length);
|
|
};
|
|
}
|
|
|
|
// String.prototype.trim is supported in IE9
|
|
exports.trim = function (str) {
|
|
if (str.trim) return str.trim();
|
|
return str.replace(/^\s+|\s+$/g, '');
|
|
};
|
|
|
|
// Function.prototype.bind is supported in IE9
|
|
exports.bind = function () {
|
|
var args = Array.prototype.slice.call(arguments);
|
|
var fn = args.shift();
|
|
if (fn.bind) return fn.bind.apply(fn, args);
|
|
var self = args.shift();
|
|
return function () {
|
|
fn.apply(self, args.concat([Array.prototype.slice.call(arguments)]));
|
|
};
|
|
};
|
|
|
|
// Object.create is supported in IE9
|
|
function create(prototype, properties) {
|
|
var object;
|
|
if (prototype === null) {
|
|
object = { '__proto__' : null };
|
|
}
|
|
else {
|
|
if (typeof prototype !== 'object') {
|
|
throw new TypeError(
|
|
'typeof prototype[' + (typeof prototype) + '] != \'object\''
|
|
);
|
|
}
|
|
var Type = function () {};
|
|
Type.prototype = prototype;
|
|
object = new Type();
|
|
object.__proto__ = prototype;
|
|
}
|
|
if (typeof properties !== 'undefined' && Object.defineProperties) {
|
|
Object.defineProperties(object, properties);
|
|
}
|
|
return object;
|
|
}
|
|
exports.create = typeof Object.create === 'function' ? Object.create : create;
|
|
|
|
// Object.keys and Object.getOwnPropertyNames is supported in IE9 however
|
|
// they do show a description and number property on Error objects
|
|
function notObject(object) {
|
|
return ((typeof object != "object" && typeof object != "function") || object === null);
|
|
}
|
|
|
|
function keysShim(object) {
|
|
if (notObject(object)) {
|
|
throw new TypeError("Object.keys called on a non-object");
|
|
}
|
|
|
|
var result = [];
|
|
for (var name in object) {
|
|
if (hasOwnProperty.call(object, name)) {
|
|
result.push(name);
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
// getOwnPropertyNames is almost the same as Object.keys one key feature
|
|
// is that it returns hidden properties, since that can't be implemented,
|
|
// this feature gets reduced so it just shows the length property on arrays
|
|
function propertyShim(object) {
|
|
if (notObject(object)) {
|
|
throw new TypeError("Object.getOwnPropertyNames called on a non-object");
|
|
}
|
|
|
|
var result = keysShim(object);
|
|
if (exports.isArray(object) && exports.indexOf(object, 'length') === -1) {
|
|
result.push('length');
|
|
}
|
|
return result;
|
|
}
|
|
|
|
var keys = typeof Object.keys === 'function' ? Object.keys : keysShim;
|
|
var getOwnPropertyNames = typeof Object.getOwnPropertyNames === 'function' ?
|
|
Object.getOwnPropertyNames : propertyShim;
|
|
|
|
if (new Error().hasOwnProperty('description')) {
|
|
var ERROR_PROPERTY_FILTER = function (obj, array) {
|
|
if (toString.call(obj) === '[object Error]') {
|
|
array = exports.filter(array, function (name) {
|
|
return name !== 'description' && name !== 'number' && name !== 'message';
|
|
});
|
|
}
|
|
return array;
|
|
};
|
|
|
|
exports.keys = function (object) {
|
|
return ERROR_PROPERTY_FILTER(object, keys(object));
|
|
};
|
|
exports.getOwnPropertyNames = function (object) {
|
|
return ERROR_PROPERTY_FILTER(object, getOwnPropertyNames(object));
|
|
};
|
|
} else {
|
|
exports.keys = keys;
|
|
exports.getOwnPropertyNames = getOwnPropertyNames;
|
|
}
|
|
|
|
// Object.getOwnPropertyDescriptor - supported in IE8 but only on dom elements
|
|
function valueObject(value, key) {
|
|
return { value: value[key] };
|
|
}
|
|
|
|
if (typeof Object.getOwnPropertyDescriptor === 'function') {
|
|
try {
|
|
Object.getOwnPropertyDescriptor({'a': 1}, 'a');
|
|
exports.getOwnPropertyDescriptor = Object.getOwnPropertyDescriptor;
|
|
} catch (e) {
|
|
// IE8 dom element issue - use a try catch and default to valueObject
|
|
exports.getOwnPropertyDescriptor = function (value, key) {
|
|
try {
|
|
return Object.getOwnPropertyDescriptor(value, key);
|
|
} catch (e) {
|
|
return valueObject(value, key);
|
|
}
|
|
};
|
|
}
|
|
} else {
|
|
exports.getOwnPropertyDescriptor = valueObject;
|
|
}
|
|
|
|
},{}],3:[function(require,module,exports){
|
|
// Copyright Joyent, Inc. and other Node contributors.
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining a
|
|
// copy of this software and associated documentation files (the
|
|
// "Software"), to deal in the Software without restriction, including
|
|
// without limitation the rights to use, copy, modify, merge, publish,
|
|
// distribute, sublicense, and/or sell copies of the Software, and to permit
|
|
// persons to whom the Software is furnished to do so, subject to the
|
|
// following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be included
|
|
// in all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
|
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
|
|
// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
|
|
// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
|
|
// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
|
|
// USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
// UTILITY
|
|
var util = require('util');
|
|
var shims = require('_shims');
|
|
var pSlice = Array.prototype.slice;
|
|
|
|
// 1. The assert module provides functions that throw
|
|
// AssertionError's when particular conditions are not met. The
|
|
// assert module must conform to the following interface.
|
|
|
|
var assert = module.exports = ok;
|
|
|
|
// 2. The AssertionError is defined in assert.
|
|
// new assert.AssertionError({ message: message,
|
|
// actual: actual,
|
|
// expected: expected })
|
|
|
|
assert.AssertionError = function AssertionError(options) {
|
|
this.name = 'AssertionError';
|
|
this.actual = options.actual;
|
|
this.expected = options.expected;
|
|
this.operator = options.operator;
|
|
this.message = options.message || getMessage(this);
|
|
};
|
|
|
|
// assert.AssertionError instanceof Error
|
|
util.inherits(assert.AssertionError, Error);
|
|
|
|
function replacer(key, value) {
|
|
if (util.isUndefined(value)) {
|
|
return '' + value;
|
|
}
|
|
if (util.isNumber(value) && (isNaN(value) || !isFinite(value))) {
|
|
return value.toString();
|
|
}
|
|
if (util.isFunction(value) || util.isRegExp(value)) {
|
|
return value.toString();
|
|
}
|
|
return value;
|
|
}
|
|
|
|
function truncate(s, n) {
|
|
if (util.isString(s)) {
|
|
return s.length < n ? s : s.slice(0, n);
|
|
} else {
|
|
return s;
|
|
}
|
|
}
|
|
|
|
function getMessage(self) {
|
|
return truncate(JSON.stringify(self.actual, replacer), 128) + ' ' +
|
|
self.operator + ' ' +
|
|
truncate(JSON.stringify(self.expected, replacer), 128);
|
|
}
|
|
|
|
// At present only the three keys mentioned above are used and
|
|
// understood by the spec. Implementations or sub modules can pass
|
|
// other keys to the AssertionError's constructor - they will be
|
|
// ignored.
|
|
|
|
// 3. All of the following functions must throw an AssertionError
|
|
// when a corresponding condition is not met, with a message that
|
|
// may be undefined if not provided. All assertion methods provide
|
|
// both the actual and expected values to the assertion error for
|
|
// display purposes.
|
|
|
|
function fail(actual, expected, message, operator, stackStartFunction) {
|
|
throw new assert.AssertionError({
|
|
message: message,
|
|
actual: actual,
|
|
expected: expected,
|
|
operator: operator,
|
|
stackStartFunction: stackStartFunction
|
|
});
|
|
}
|
|
|
|
// EXTENSION! allows for well behaved errors defined elsewhere.
|
|
assert.fail = fail;
|
|
|
|
// 4. Pure assertion tests whether a value is truthy, as determined
|
|
// by !!guard.
|
|
// assert.ok(guard, message_opt);
|
|
// This statement is equivalent to assert.equal(true, !!guard,
|
|
// message_opt);. To test strictly for the value true, use
|
|
// assert.strictEqual(true, guard, message_opt);.
|
|
|
|
function ok(value, message) {
|
|
if (!value) fail(value, true, message, '==', assert.ok);
|
|
}
|
|
assert.ok = ok;
|
|
|
|
// 5. The equality assertion tests shallow, coercive equality with
|
|
// ==.
|
|
// assert.equal(actual, expected, message_opt);
|
|
|
|
assert.equal = function equal(actual, expected, message) {
|
|
if (actual != expected) fail(actual, expected, message, '==', assert.equal);
|
|
};
|
|
|
|
// 6. The non-equality assertion tests for whether two objects are not equal
|
|
// with != assert.notEqual(actual, expected, message_opt);
|
|
|
|
assert.notEqual = function notEqual(actual, expected, message) {
|
|
if (actual == expected) {
|
|
fail(actual, expected, message, '!=', assert.notEqual);
|
|
}
|
|
};
|
|
|
|
// 7. The equivalence assertion tests a deep equality relation.
|
|
// assert.deepEqual(actual, expected, message_opt);
|
|
|
|
assert.deepEqual = function deepEqual(actual, expected, message) {
|
|
if (!_deepEqual(actual, expected)) {
|
|
fail(actual, expected, message, 'deepEqual', assert.deepEqual);
|
|
}
|
|
};
|
|
|
|
function _deepEqual(actual, expected) {
|
|
// 7.1. All identical values are equivalent, as determined by ===.
|
|
if (actual === expected) {
|
|
return true;
|
|
|
|
} else if (util.isBuffer(actual) && util.isBuffer(expected)) {
|
|
if (actual.length != expected.length) return false;
|
|
|
|
for (var i = 0; i < actual.length; i++) {
|
|
if (actual[i] !== expected[i]) return false;
|
|
}
|
|
|
|
return true;
|
|
|
|
// 7.2. If the expected value is a Date object, the actual value is
|
|
// equivalent if it is also a Date object that refers to the same time.
|
|
} else if (util.isDate(actual) && util.isDate(expected)) {
|
|
return actual.getTime() === expected.getTime();
|
|
|
|
// 7.3 If the expected value is a RegExp object, the actual value is
|
|
// equivalent if it is also a RegExp object with the same source and
|
|
// properties (`global`, `multiline`, `lastIndex`, `ignoreCase`).
|
|
} else if (util.isRegExp(actual) && util.isRegExp(expected)) {
|
|
return actual.source === expected.source &&
|
|
actual.global === expected.global &&
|
|
actual.multiline === expected.multiline &&
|
|
actual.lastIndex === expected.lastIndex &&
|
|
actual.ignoreCase === expected.ignoreCase;
|
|
|
|
// 7.4. Other pairs that do not both pass typeof value == 'object',
|
|
// equivalence is determined by ==.
|
|
} else if (!util.isObject(actual) && !util.isObject(expected)) {
|
|
return actual == expected;
|
|
|
|
// 7.5 For all other Object pairs, including Array objects, equivalence is
|
|
// determined by having the same number of owned properties (as verified
|
|
// with Object.prototype.hasOwnProperty.call), the same set of keys
|
|
// (although not necessarily the same order), equivalent values for every
|
|
// corresponding key, and an identical 'prototype' property. Note: this
|
|
// accounts for both named and indexed properties on Arrays.
|
|
} else {
|
|
return objEquiv(actual, expected);
|
|
}
|
|
}
|
|
|
|
function isArguments(object) {
|
|
return Object.prototype.toString.call(object) == '[object Arguments]';
|
|
}
|
|
|
|
function objEquiv(a, b) {
|
|
if (util.isNullOrUndefined(a) || util.isNullOrUndefined(b))
|
|
return false;
|
|
// an identical 'prototype' property.
|
|
if (a.prototype !== b.prototype) return false;
|
|
//~~~I've managed to break Object.keys through screwy arguments passing.
|
|
// Converting to array solves the problem.
|
|
if (isArguments(a)) {
|
|
if (!isArguments(b)) {
|
|
return false;
|
|
}
|
|
a = pSlice.call(a);
|
|
b = pSlice.call(b);
|
|
return _deepEqual(a, b);
|
|
}
|
|
try {
|
|
var ka = shims.keys(a),
|
|
kb = shims.keys(b),
|
|
key, i;
|
|
} catch (e) {//happens when one is a string literal and the other isn't
|
|
return false;
|
|
}
|
|
// having the same number of owned properties (keys incorporates
|
|
// hasOwnProperty)
|
|
if (ka.length != kb.length)
|
|
return false;
|
|
//the same set of keys (although not necessarily the same order),
|
|
ka.sort();
|
|
kb.sort();
|
|
//~~~cheap key test
|
|
for (i = ka.length - 1; i >= 0; i--) {
|
|
if (ka[i] != kb[i])
|
|
return false;
|
|
}
|
|
//equivalent values for every corresponding key, and
|
|
//~~~possibly expensive deep test
|
|
for (i = ka.length - 1; i >= 0; i--) {
|
|
key = ka[i];
|
|
if (!_deepEqual(a[key], b[key])) return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
// 8. The non-equivalence assertion tests for any deep inequality.
|
|
// assert.notDeepEqual(actual, expected, message_opt);
|
|
|
|
assert.notDeepEqual = function notDeepEqual(actual, expected, message) {
|
|
if (_deepEqual(actual, expected)) {
|
|
fail(actual, expected, message, 'notDeepEqual', assert.notDeepEqual);
|
|
}
|
|
};
|
|
|
|
// 9. The strict equality assertion tests strict equality, as determined by ===.
|
|
// assert.strictEqual(actual, expected, message_opt);
|
|
|
|
assert.strictEqual = function strictEqual(actual, expected, message) {
|
|
if (actual !== expected) {
|
|
fail(actual, expected, message, '===', assert.strictEqual);
|
|
}
|
|
};
|
|
|
|
// 10. The strict non-equality assertion tests for strict inequality, as
|
|
// determined by !==. assert.notStrictEqual(actual, expected, message_opt);
|
|
|
|
assert.notStrictEqual = function notStrictEqual(actual, expected, message) {
|
|
if (actual === expected) {
|
|
fail(actual, expected, message, '!==', assert.notStrictEqual);
|
|
}
|
|
};
|
|
|
|
function expectedException(actual, expected) {
|
|
if (!actual || !expected) {
|
|
return false;
|
|
}
|
|
|
|
if (Object.prototype.toString.call(expected) == '[object RegExp]') {
|
|
return expected.test(actual);
|
|
} else if (actual instanceof expected) {
|
|
return true;
|
|
} else if (expected.call({}, actual) === true) {
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
function _throws(shouldThrow, block, expected, message) {
|
|
var actual;
|
|
|
|
if (util.isString(expected)) {
|
|
message = expected;
|
|
expected = null;
|
|
}
|
|
|
|
try {
|
|
block();
|
|
} catch (e) {
|
|
actual = e;
|
|
}
|
|
|
|
message = (expected && expected.name ? ' (' + expected.name + ').' : '.') +
|
|
(message ? ' ' + message : '.');
|
|
|
|
if (shouldThrow && !actual) {
|
|
fail(actual, expected, 'Missing expected exception' + message);
|
|
}
|
|
|
|
if (!shouldThrow && expectedException(actual, expected)) {
|
|
fail(actual, expected, 'Got unwanted exception' + message);
|
|
}
|
|
|
|
if ((shouldThrow && actual && expected &&
|
|
!expectedException(actual, expected)) || (!shouldThrow && actual)) {
|
|
throw actual;
|
|
}
|
|
}
|
|
|
|
// 11. Expected to throw an error:
|
|
// assert.throws(block, Error_opt, message_opt);
|
|
|
|
assert.throws = function(block, /*optional*/error, /*optional*/message) {
|
|
_throws.apply(this, [true].concat(pSlice.call(arguments)));
|
|
};
|
|
|
|
// EXTENSION! This is annoying to write outside this module.
|
|
assert.doesNotThrow = function(block, /*optional*/message) {
|
|
_throws.apply(this, [false].concat(pSlice.call(arguments)));
|
|
};
|
|
|
|
assert.ifError = function(err) { if (err) {throw err;}};
|
|
},{"_shims":2,"util":10}],4:[function(require,module,exports){
|
|
|
|
// not implemented
|
|
// The reason for having an empty file and not throwing is to allow
|
|
// untraditional implementation of this module.
|
|
|
|
},{}],5:[function(require,module,exports){
|
|
// Copyright Joyent, Inc. and other Node contributors.
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining a
|
|
// copy of this software and associated documentation files (the
|
|
// "Software"), to deal in the Software without restriction, including
|
|
// without limitation the rights to use, copy, modify, merge, publish,
|
|
// distribute, sublicense, and/or sell copies of the Software, and to permit
|
|
// persons to whom the Software is furnished to do so, subject to the
|
|
// following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be included
|
|
// in all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
|
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
|
|
// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
|
|
// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
|
|
// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
|
|
// USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
var util = require('util');
|
|
|
|
function EventEmitter() {
|
|
this._events = this._events || {};
|
|
this._maxListeners = this._maxListeners || undefined;
|
|
}
|
|
module.exports = EventEmitter;
|
|
|
|
// Backwards-compat with node 0.10.x
|
|
EventEmitter.EventEmitter = EventEmitter;
|
|
|
|
EventEmitter.prototype._events = undefined;
|
|
EventEmitter.prototype._maxListeners = undefined;
|
|
|
|
// By default EventEmitters will print a warning if more than 10 listeners are
|
|
// added to it. This is a useful default which helps finding memory leaks.
|
|
EventEmitter.defaultMaxListeners = 10;
|
|
|
|
// Obviously not all Emitters should be limited to 10. This function allows
|
|
// that to be increased. Set to zero for unlimited.
|
|
EventEmitter.prototype.setMaxListeners = function(n) {
|
|
if (!util.isNumber(n) || n < 0)
|
|
throw TypeError('n must be a positive number');
|
|
this._maxListeners = n;
|
|
return this;
|
|
};
|
|
|
|
EventEmitter.prototype.emit = function(type) {
|
|
var er, handler, len, args, i, listeners;
|
|
|
|
if (!this._events)
|
|
this._events = {};
|
|
|
|
// If there is no 'error' event listener then throw.
|
|
if (type === 'error') {
|
|
if (!this._events.error ||
|
|
(util.isObject(this._events.error) && !this._events.error.length)) {
|
|
er = arguments[1];
|
|
if (er instanceof Error) {
|
|
throw er; // Unhandled 'error' event
|
|
} else {
|
|
throw TypeError('Uncaught, unspecified "error" event.');
|
|
}
|
|
return false;
|
|
}
|
|
}
|
|
|
|
handler = this._events[type];
|
|
|
|
if (util.isUndefined(handler))
|
|
return false;
|
|
|
|
if (util.isFunction(handler)) {
|
|
switch (arguments.length) {
|
|
// fast cases
|
|
case 1:
|
|
handler.call(this);
|
|
break;
|
|
case 2:
|
|
handler.call(this, arguments[1]);
|
|
break;
|
|
case 3:
|
|
handler.call(this, arguments[1], arguments[2]);
|
|
break;
|
|
// slower
|
|
default:
|
|
len = arguments.length;
|
|
args = new Array(len - 1);
|
|
for (i = 1; i < len; i++)
|
|
args[i - 1] = arguments[i];
|
|
handler.apply(this, args);
|
|
}
|
|
} else if (util.isObject(handler)) {
|
|
len = arguments.length;
|
|
args = new Array(len - 1);
|
|
for (i = 1; i < len; i++)
|
|
args[i - 1] = arguments[i];
|
|
|
|
listeners = handler.slice();
|
|
len = listeners.length;
|
|
for (i = 0; i < len; i++)
|
|
listeners[i].apply(this, args);
|
|
}
|
|
|
|
return true;
|
|
};
|
|
|
|
EventEmitter.prototype.addListener = function(type, listener) {
|
|
var m;
|
|
|
|
if (!util.isFunction(listener))
|
|
throw TypeError('listener must be a function');
|
|
|
|
if (!this._events)
|
|
this._events = {};
|
|
|
|
// To avoid recursion in the case that type === "newListener"! Before
|
|
// adding it to the listeners, first emit "newListener".
|
|
if (this._events.newListener)
|
|
this.emit('newListener', type,
|
|
util.isFunction(listener.listener) ?
|
|
listener.listener : listener);
|
|
|
|
if (!this._events[type])
|
|
// Optimize the case of one listener. Don't need the extra array object.
|
|
this._events[type] = listener;
|
|
else if (util.isObject(this._events[type]))
|
|
// If we've already got an array, just append.
|
|
this._events[type].push(listener);
|
|
else
|
|
// Adding the second element, need to change to array.
|
|
this._events[type] = [this._events[type], listener];
|
|
|
|
// Check for listener leak
|
|
if (util.isObject(this._events[type]) && !this._events[type].warned) {
|
|
var m;
|
|
if (!util.isUndefined(this._maxListeners)) {
|
|
m = this._maxListeners;
|
|
} else {
|
|
m = EventEmitter.defaultMaxListeners;
|
|
}
|
|
|
|
if (m && m > 0 && this._events[type].length > m) {
|
|
this._events[type].warned = true;
|
|
console.error('(node) warning: possible EventEmitter memory ' +
|
|
'leak detected. %d listeners added. ' +
|
|
'Use emitter.setMaxListeners() to increase limit.',
|
|
this._events[type].length);
|
|
console.trace();
|
|
}
|
|
}
|
|
|
|
return this;
|
|
};
|
|
|
|
EventEmitter.prototype.on = EventEmitter.prototype.addListener;
|
|
|
|
EventEmitter.prototype.once = function(type, listener) {
|
|
if (!util.isFunction(listener))
|
|
throw TypeError('listener must be a function');
|
|
|
|
function g() {
|
|
this.removeListener(type, g);
|
|
listener.apply(this, arguments);
|
|
}
|
|
|
|
g.listener = listener;
|
|
this.on(type, g);
|
|
|
|
return this;
|
|
};
|
|
|
|
// emits a 'removeListener' event iff the listener was removed
|
|
EventEmitter.prototype.removeListener = function(type, listener) {
|
|
var list, position, length, i;
|
|
|
|
if (!util.isFunction(listener))
|
|
throw TypeError('listener must be a function');
|
|
|
|
if (!this._events || !this._events[type])
|
|
return this;
|
|
|
|
list = this._events[type];
|
|
length = list.length;
|
|
position = -1;
|
|
|
|
if (list === listener ||
|
|
(util.isFunction(list.listener) && list.listener === listener)) {
|
|
delete this._events[type];
|
|
if (this._events.removeListener)
|
|
this.emit('removeListener', type, listener);
|
|
|
|
} else if (util.isObject(list)) {
|
|
for (i = length; i-- > 0;) {
|
|
if (list[i] === listener ||
|
|
(list[i].listener && list[i].listener === listener)) {
|
|
position = i;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (position < 0)
|
|
return this;
|
|
|
|
if (list.length === 1) {
|
|
list.length = 0;
|
|
delete this._events[type];
|
|
} else {
|
|
list.splice(position, 1);
|
|
}
|
|
|
|
if (this._events.removeListener)
|
|
this.emit('removeListener', type, listener);
|
|
}
|
|
|
|
return this;
|
|
};
|
|
|
|
EventEmitter.prototype.removeAllListeners = function(type) {
|
|
var key, listeners;
|
|
|
|
if (!this._events)
|
|
return this;
|
|
|
|
// not listening for removeListener, no need to emit
|
|
if (!this._events.removeListener) {
|
|
if (arguments.length === 0)
|
|
this._events = {};
|
|
else if (this._events[type])
|
|
delete this._events[type];
|
|
return this;
|
|
}
|
|
|
|
// emit removeListener for all listeners on all events
|
|
if (arguments.length === 0) {
|
|
for (key in this._events) {
|
|
if (key === 'removeListener') continue;
|
|
this.removeAllListeners(key);
|
|
}
|
|
this.removeAllListeners('removeListener');
|
|
this._events = {};
|
|
return this;
|
|
}
|
|
|
|
listeners = this._events[type];
|
|
|
|
if (util.isFunction(listeners)) {
|
|
this.removeListener(type, listeners);
|
|
} else {
|
|
// LIFO order
|
|
while (listeners.length)
|
|
this.removeListener(type, listeners[listeners.length - 1]);
|
|
}
|
|
delete this._events[type];
|
|
|
|
return this;
|
|
};
|
|
|
|
EventEmitter.prototype.listeners = function(type) {
|
|
var ret;
|
|
if (!this._events || !this._events[type])
|
|
ret = [];
|
|
else if (util.isFunction(this._events[type]))
|
|
ret = [this._events[type]];
|
|
else
|
|
ret = this._events[type].slice();
|
|
return ret;
|
|
};
|
|
|
|
EventEmitter.listenerCount = function(emitter, type) {
|
|
var ret;
|
|
if (!emitter._events || !emitter._events[type])
|
|
ret = 0;
|
|
else if (util.isFunction(emitter._events[type]))
|
|
ret = 1;
|
|
else
|
|
ret = emitter._events[type].length;
|
|
return ret;
|
|
};
|
|
},{"util":10}],6:[function(require,module,exports){
|
|
module.exports=require(4)
|
|
},{}],7:[function(require,module,exports){
|
|
var process=require("__browserify_process");// Copyright Joyent, Inc. and other Node contributors.
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining a
|
|
// copy of this software and associated documentation files (the
|
|
// "Software"), to deal in the Software without restriction, including
|
|
// without limitation the rights to use, copy, modify, merge, publish,
|
|
// distribute, sublicense, and/or sell copies of the Software, and to permit
|
|
// persons to whom the Software is furnished to do so, subject to the
|
|
// following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be included
|
|
// in all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
|
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
|
|
// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
|
|
// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
|
|
// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
|
|
// USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
var util = require('util');
|
|
var shims = require('_shims');
|
|
|
|
// resolves . and .. elements in a path array with directory names there
|
|
// must be no slashes, empty elements, or device names (c:\) in the array
|
|
// (so also no leading and trailing slashes - it does not distinguish
|
|
// relative and absolute paths)
|
|
function normalizeArray(parts, allowAboveRoot) {
|
|
// if the path tries to go above the root, `up` ends up > 0
|
|
var up = 0;
|
|
for (var i = parts.length - 1; i >= 0; i--) {
|
|
var last = parts[i];
|
|
if (last === '.') {
|
|
parts.splice(i, 1);
|
|
} else if (last === '..') {
|
|
parts.splice(i, 1);
|
|
up++;
|
|
} else if (up) {
|
|
parts.splice(i, 1);
|
|
up--;
|
|
}
|
|
}
|
|
|
|
// if the path is allowed to go above the root, restore leading ..s
|
|
if (allowAboveRoot) {
|
|
for (; up--; up) {
|
|
parts.unshift('..');
|
|
}
|
|
}
|
|
|
|
return parts;
|
|
}
|
|
|
|
// Split a filename into [root, dir, basename, ext], unix version
|
|
// 'root' is just a slash, or nothing.
|
|
var splitPathRe =
|
|
/^(\/?|)([\s\S]*?)((?:\.{1,2}|[^\/]+?|)(\.[^.\/]*|))(?:[\/]*)$/;
|
|
var splitPath = function(filename) {
|
|
return splitPathRe.exec(filename).slice(1);
|
|
};
|
|
|
|
// path.resolve([from ...], to)
|
|
// posix version
|
|
exports.resolve = function() {
|
|
var resolvedPath = '',
|
|
resolvedAbsolute = false;
|
|
|
|
for (var i = arguments.length - 1; i >= -1 && !resolvedAbsolute; i--) {
|
|
var path = (i >= 0) ? arguments[i] : process.cwd();
|
|
|
|
// Skip empty and invalid entries
|
|
if (!util.isString(path)) {
|
|
throw new TypeError('Arguments to path.resolve must be strings');
|
|
} else if (!path) {
|
|
continue;
|
|
}
|
|
|
|
resolvedPath = path + '/' + resolvedPath;
|
|
resolvedAbsolute = path.charAt(0) === '/';
|
|
}
|
|
|
|
// At this point the path should be resolved to a full absolute path, but
|
|
// handle relative paths to be safe (might happen when process.cwd() fails)
|
|
|
|
// Normalize the path
|
|
resolvedPath = normalizeArray(shims.filter(resolvedPath.split('/'), function(p) {
|
|
return !!p;
|
|
}), !resolvedAbsolute).join('/');
|
|
|
|
return ((resolvedAbsolute ? '/' : '') + resolvedPath) || '.';
|
|
};
|
|
|
|
// path.normalize(path)
|
|
// posix version
|
|
exports.normalize = function(path) {
|
|
var isAbsolute = exports.isAbsolute(path),
|
|
trailingSlash = shims.substr(path, -1) === '/';
|
|
|
|
// Normalize the path
|
|
path = normalizeArray(shims.filter(path.split('/'), function(p) {
|
|
return !!p;
|
|
}), !isAbsolute).join('/');
|
|
|
|
if (!path && !isAbsolute) {
|
|
path = '.';
|
|
}
|
|
if (path && trailingSlash) {
|
|
path += '/';
|
|
}
|
|
|
|
return (isAbsolute ? '/' : '') + path;
|
|
};
|
|
|
|
// posix version
|
|
exports.isAbsolute = function(path) {
|
|
return path.charAt(0) === '/';
|
|
};
|
|
|
|
// posix version
|
|
exports.join = function() {
|
|
var paths = Array.prototype.slice.call(arguments, 0);
|
|
return exports.normalize(shims.filter(paths, function(p, index) {
|
|
if (!util.isString(p)) {
|
|
throw new TypeError('Arguments to path.join must be strings');
|
|
}
|
|
return p;
|
|
}).join('/'));
|
|
};
|
|
|
|
|
|
// path.relative(from, to)
|
|
// posix version
|
|
exports.relative = function(from, to) {
|
|
from = exports.resolve(from).substr(1);
|
|
to = exports.resolve(to).substr(1);
|
|
|
|
function trim(arr) {
|
|
var start = 0;
|
|
for (; start < arr.length; start++) {
|
|
if (arr[start] !== '') break;
|
|
}
|
|
|
|
var end = arr.length - 1;
|
|
for (; end >= 0; end--) {
|
|
if (arr[end] !== '') break;
|
|
}
|
|
|
|
if (start > end) return [];
|
|
return arr.slice(start, end - start + 1);
|
|
}
|
|
|
|
var fromParts = trim(from.split('/'));
|
|
var toParts = trim(to.split('/'));
|
|
|
|
var length = Math.min(fromParts.length, toParts.length);
|
|
var samePartsLength = length;
|
|
for (var i = 0; i < length; i++) {
|
|
if (fromParts[i] !== toParts[i]) {
|
|
samePartsLength = i;
|
|
break;
|
|
}
|
|
}
|
|
|
|
var outputParts = [];
|
|
for (var i = samePartsLength; i < fromParts.length; i++) {
|
|
outputParts.push('..');
|
|
}
|
|
|
|
outputParts = outputParts.concat(toParts.slice(samePartsLength));
|
|
|
|
return outputParts.join('/');
|
|
};
|
|
|
|
exports.sep = '/';
|
|
exports.delimiter = ':';
|
|
|
|
exports.dirname = function(path) {
|
|
var result = splitPath(path),
|
|
root = result[0],
|
|
dir = result[1];
|
|
|
|
if (!root && !dir) {
|
|
// No dirname whatsoever
|
|
return '.';
|
|
}
|
|
|
|
if (dir) {
|
|
// It has a dirname, strip trailing slash
|
|
dir = dir.substr(0, dir.length - 1);
|
|
}
|
|
|
|
return root + dir;
|
|
};
|
|
|
|
|
|
exports.basename = function(path, ext) {
|
|
var f = splitPath(path)[2];
|
|
// TODO: make this comparison case-insensitive on windows?
|
|
if (ext && f.substr(-1 * ext.length) === ext) {
|
|
f = f.substr(0, f.length - ext.length);
|
|
}
|
|
return f;
|
|
};
|
|
|
|
|
|
exports.extname = function(path) {
|
|
return splitPath(path)[3];
|
|
};
|
|
|
|
},{"__browserify_process":15,"_shims":2,"util":10}],8:[function(require,module,exports){
|
|
// Copyright Joyent, Inc. and other Node contributors.
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining a
|
|
// copy of this software and associated documentation files (the
|
|
// "Software"), to deal in the Software without restriction, including
|
|
// without limitation the rights to use, copy, modify, merge, publish,
|
|
// distribute, sublicense, and/or sell copies of the Software, and to permit
|
|
// persons to whom the Software is furnished to do so, subject to the
|
|
// following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be included
|
|
// in all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
|
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
|
|
// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
|
|
// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
|
|
// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
|
|
// USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
// Query String Utilities
|
|
|
|
var QueryString = exports;
|
|
var util = require('util');
|
|
var shims = require('_shims');
|
|
var Buffer = require('buffer').Buffer;
|
|
|
|
// If obj.hasOwnProperty has been overridden, then calling
|
|
// obj.hasOwnProperty(prop) will break.
|
|
// See: https://github.com/joyent/node/issues/1707
|
|
function hasOwnProperty(obj, prop) {
|
|
return Object.prototype.hasOwnProperty.call(obj, prop);
|
|
}
|
|
|
|
|
|
function charCode(c) {
|
|
return c.charCodeAt(0);
|
|
}
|
|
|
|
|
|
// a safe fast alternative to decodeURIComponent
|
|
QueryString.unescapeBuffer = function(s, decodeSpaces) {
|
|
var out = new Buffer(s.length);
|
|
var state = 'CHAR'; // states: CHAR, HEX0, HEX1
|
|
var n, m, hexchar;
|
|
|
|
for (var inIndex = 0, outIndex = 0; inIndex <= s.length; inIndex++) {
|
|
var c = s.charCodeAt(inIndex);
|
|
switch (state) {
|
|
case 'CHAR':
|
|
switch (c) {
|
|
case charCode('%'):
|
|
n = 0;
|
|
m = 0;
|
|
state = 'HEX0';
|
|
break;
|
|
case charCode('+'):
|
|
if (decodeSpaces) c = charCode(' ');
|
|
// pass thru
|
|
default:
|
|
out[outIndex++] = c;
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case 'HEX0':
|
|
state = 'HEX1';
|
|
hexchar = c;
|
|
if (charCode('0') <= c && c <= charCode('9')) {
|
|
n = c - charCode('0');
|
|
} else if (charCode('a') <= c && c <= charCode('f')) {
|
|
n = c - charCode('a') + 10;
|
|
} else if (charCode('A') <= c && c <= charCode('F')) {
|
|
n = c - charCode('A') + 10;
|
|
} else {
|
|
out[outIndex++] = charCode('%');
|
|
out[outIndex++] = c;
|
|
state = 'CHAR';
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case 'HEX1':
|
|
state = 'CHAR';
|
|
if (charCode('0') <= c && c <= charCode('9')) {
|
|
m = c - charCode('0');
|
|
} else if (charCode('a') <= c && c <= charCode('f')) {
|
|
m = c - charCode('a') + 10;
|
|
} else if (charCode('A') <= c && c <= charCode('F')) {
|
|
m = c - charCode('A') + 10;
|
|
} else {
|
|
out[outIndex++] = charCode('%');
|
|
out[outIndex++] = hexchar;
|
|
out[outIndex++] = c;
|
|
break;
|
|
}
|
|
out[outIndex++] = 16 * n + m;
|
|
break;
|
|
}
|
|
}
|
|
|
|
// TODO support returning arbitrary buffers.
|
|
|
|
return out.slice(0, outIndex - 1);
|
|
};
|
|
|
|
|
|
QueryString.unescape = function(s, decodeSpaces) {
|
|
return QueryString.unescapeBuffer(s, decodeSpaces).toString();
|
|
};
|
|
|
|
|
|
QueryString.escape = function(str) {
|
|
return encodeURIComponent(str);
|
|
};
|
|
|
|
var stringifyPrimitive = function(v) {
|
|
if (util.isString(v))
|
|
return v;
|
|
if (util.isBoolean(v))
|
|
return v ? 'true' : 'false';
|
|
if (util.isNumber(v))
|
|
return isFinite(v) ? v : '';
|
|
return '';
|
|
};
|
|
|
|
|
|
QueryString.stringify = QueryString.encode = function(obj, sep, eq, name) {
|
|
sep = sep || '&';
|
|
eq = eq || '=';
|
|
if (util.isNull(obj)) {
|
|
obj = undefined;
|
|
}
|
|
|
|
if (util.isObject(obj)) {
|
|
return shims.map(shims.keys(obj), function(k) {
|
|
var ks = QueryString.escape(stringifyPrimitive(k)) + eq;
|
|
if (util.isArray(obj[k])) {
|
|
return shims.map(obj[k], function(v) {
|
|
return ks + QueryString.escape(stringifyPrimitive(v));
|
|
}).join(sep);
|
|
} else {
|
|
return ks + QueryString.escape(stringifyPrimitive(obj[k]));
|
|
}
|
|
}).join(sep);
|
|
|
|
}
|
|
|
|
if (!name) return '';
|
|
return QueryString.escape(stringifyPrimitive(name)) + eq +
|
|
QueryString.escape(stringifyPrimitive(obj));
|
|
};
|
|
|
|
// Parse a key=val string.
|
|
QueryString.parse = QueryString.decode = function(qs, sep, eq, options) {
|
|
sep = sep || '&';
|
|
eq = eq || '=';
|
|
var obj = {};
|
|
|
|
if (!util.isString(qs) || qs.length === 0) {
|
|
return obj;
|
|
}
|
|
|
|
var regexp = /\+/g;
|
|
qs = qs.split(sep);
|
|
|
|
var maxKeys = 1000;
|
|
if (options && util.isNumber(options.maxKeys)) {
|
|
maxKeys = options.maxKeys;
|
|
}
|
|
|
|
var len = qs.length;
|
|
// maxKeys <= 0 means that we should not limit keys count
|
|
if (maxKeys > 0 && len > maxKeys) {
|
|
len = maxKeys;
|
|
}
|
|
|
|
for (var i = 0; i < len; ++i) {
|
|
var x = qs[i].replace(regexp, '%20'),
|
|
idx = x.indexOf(eq),
|
|
kstr, vstr, k, v;
|
|
|
|
if (idx >= 0) {
|
|
kstr = x.substr(0, idx);
|
|
vstr = x.substr(idx + 1);
|
|
} else {
|
|
kstr = x;
|
|
vstr = '';
|
|
}
|
|
|
|
try {
|
|
k = decodeURIComponent(kstr);
|
|
v = decodeURIComponent(vstr);
|
|
} catch (e) {
|
|
k = QueryString.unescape(kstr, true);
|
|
v = QueryString.unescape(vstr, true);
|
|
}
|
|
|
|
if (!hasOwnProperty(obj, k)) {
|
|
obj[k] = v;
|
|
} else if (util.isArray(obj[k])) {
|
|
obj[k].push(v);
|
|
} else {
|
|
obj[k] = [obj[k], v];
|
|
}
|
|
}
|
|
|
|
return obj;
|
|
};
|
|
},{"_shims":2,"buffer":12,"util":10}],9:[function(require,module,exports){
|
|
// Copyright Joyent, Inc. and other Node contributors.
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining a
|
|
// copy of this software and associated documentation files (the
|
|
// "Software"), to deal in the Software without restriction, including
|
|
// without limitation the rights to use, copy, modify, merge, publish,
|
|
// distribute, sublicense, and/or sell copies of the Software, and to permit
|
|
// persons to whom the Software is furnished to do so, subject to the
|
|
// following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be included
|
|
// in all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
|
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
|
|
// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
|
|
// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
|
|
// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
|
|
// USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
var punycode = { encode : function (s) { return s } };
|
|
var util = require('util');
|
|
var shims = require('_shims');
|
|
|
|
exports.parse = urlParse;
|
|
exports.resolve = urlResolve;
|
|
exports.resolveObject = urlResolveObject;
|
|
exports.format = urlFormat;
|
|
|
|
exports.Url = Url;
|
|
|
|
function Url() {
|
|
this.protocol = null;
|
|
this.slashes = null;
|
|
this.auth = null;
|
|
this.host = null;
|
|
this.port = null;
|
|
this.hostname = null;
|
|
this.hash = null;
|
|
this.search = null;
|
|
this.query = null;
|
|
this.pathname = null;
|
|
this.path = null;
|
|
this.href = null;
|
|
}
|
|
|
|
// Reference: RFC 3986, RFC 1808, RFC 2396
|
|
|
|
// define these here so at least they only have to be
|
|
// compiled once on the first module load.
|
|
var protocolPattern = /^([a-z0-9.+-]+:)/i,
|
|
portPattern = /:[0-9]*$/,
|
|
|
|
// RFC 2396: characters reserved for delimiting URLs.
|
|
// We actually just auto-escape these.
|
|
delims = ['<', '>', '"', '`', ' ', '\r', '\n', '\t'],
|
|
|
|
// RFC 2396: characters not allowed for various reasons.
|
|
unwise = ['{', '}', '|', '\\', '^', '`'].concat(delims),
|
|
|
|
// Allowed by RFCs, but cause of XSS attacks. Always escape these.
|
|
autoEscape = ['\''].concat(unwise),
|
|
// Characters that are never ever allowed in a hostname.
|
|
// Note that any invalid chars are also handled, but these
|
|
// are the ones that are *expected* to be seen, so we fast-path
|
|
// them.
|
|
nonHostChars = ['%', '/', '?', ';', '#'].concat(autoEscape),
|
|
hostEndingChars = ['/', '?', '#'],
|
|
hostnameMaxLen = 255,
|
|
hostnamePartPattern = /^[a-z0-9A-Z_-]{0,63}$/,
|
|
hostnamePartStart = /^([a-z0-9A-Z_-]{0,63})(.*)$/,
|
|
// protocols that can allow "unsafe" and "unwise" chars.
|
|
unsafeProtocol = {
|
|
'javascript': true,
|
|
'javascript:': true
|
|
},
|
|
// protocols that never have a hostname.
|
|
hostlessProtocol = {
|
|
'javascript': true,
|
|
'javascript:': true
|
|
},
|
|
// protocols that always contain a // bit.
|
|
slashedProtocol = {
|
|
'http': true,
|
|
'https': true,
|
|
'ftp': true,
|
|
'gopher': true,
|
|
'file': true,
|
|
'http:': true,
|
|
'https:': true,
|
|
'ftp:': true,
|
|
'gopher:': true,
|
|
'file:': true
|
|
},
|
|
querystring = require('querystring');
|
|
|
|
function urlParse(url, parseQueryString, slashesDenoteHost) {
|
|
if (url && util.isObject(url) && url instanceof Url) return url;
|
|
|
|
var u = new Url;
|
|
u.parse(url, parseQueryString, slashesDenoteHost);
|
|
return u;
|
|
}
|
|
|
|
Url.prototype.parse = function(url, parseQueryString, slashesDenoteHost) {
|
|
if (!util.isString(url)) {
|
|
throw new TypeError("Parameter 'url' must be a string, not " + typeof url);
|
|
}
|
|
|
|
var rest = url;
|
|
|
|
// trim before proceeding.
|
|
// This is to support parse stuff like " http://foo.com \n"
|
|
rest = shims.trim(rest);
|
|
|
|
var proto = protocolPattern.exec(rest);
|
|
if (proto) {
|
|
proto = proto[0];
|
|
var lowerProto = proto.toLowerCase();
|
|
this.protocol = lowerProto;
|
|
rest = rest.substr(proto.length);
|
|
}
|
|
|
|
// figure out if it's got a host
|
|
// user@server is *always* interpreted as a hostname, and url
|
|
// resolution will treat //foo/bar as host=foo,path=bar because that's
|
|
// how the browser resolves relative URLs.
|
|
if (slashesDenoteHost || proto || rest.match(/^\/\/[^@\/]+@[^@\/]+/)) {
|
|
var slashes = rest.substr(0, 2) === '//';
|
|
if (slashes && !(proto && hostlessProtocol[proto])) {
|
|
rest = rest.substr(2);
|
|
this.slashes = true;
|
|
}
|
|
}
|
|
|
|
if (!hostlessProtocol[proto] &&
|
|
(slashes || (proto && !slashedProtocol[proto]))) {
|
|
|
|
// there's a hostname.
|
|
// the first instance of /, ?, ;, or # ends the host.
|
|
//
|
|
// If there is an @ in the hostname, then non-host chars *are* allowed
|
|
// to the left of the last @ sign, unless some host-ending character
|
|
// comes *before* the @-sign.
|
|
// URLs are obnoxious.
|
|
//
|
|
// ex:
|
|
// http://a@b@c/ => user:a@b host:c
|
|
// http://a@b?@c => user:a host:c path:/?@c
|
|
|
|
// v0.12 TODO(isaacs): This is not quite how Chrome does things.
|
|
// Review our test case against browsers more comprehensively.
|
|
|
|
// find the first instance of any hostEndingChars
|
|
var hostEnd = -1;
|
|
for (var i = 0; i < hostEndingChars.length; i++) {
|
|
var hec = rest.indexOf(hostEndingChars[i]);
|
|
if (hec !== -1 && (hostEnd === -1 || hec < hostEnd))
|
|
hostEnd = hec;
|
|
}
|
|
|
|
// at this point, either we have an explicit point where the
|
|
// auth portion cannot go past, or the last @ char is the decider.
|
|
var auth, atSign;
|
|
if (hostEnd === -1) {
|
|
// atSign can be anywhere.
|
|
atSign = rest.lastIndexOf('@');
|
|
} else {
|
|
// atSign must be in auth portion.
|
|
// http://a@b/c@d => host:b auth:a path:/c@d
|
|
atSign = rest.lastIndexOf('@', hostEnd);
|
|
}
|
|
|
|
// Now we have a portion which is definitely the auth.
|
|
// Pull that off.
|
|
if (atSign !== -1) {
|
|
auth = rest.slice(0, atSign);
|
|
rest = rest.slice(atSign + 1);
|
|
this.auth = decodeURIComponent(auth);
|
|
}
|
|
|
|
// the host is the remaining to the left of the first non-host char
|
|
hostEnd = -1;
|
|
for (var i = 0; i < nonHostChars.length; i++) {
|
|
var hec = rest.indexOf(nonHostChars[i]);
|
|
if (hec !== -1 && (hostEnd === -1 || hec < hostEnd))
|
|
hostEnd = hec;
|
|
}
|
|
// if we still have not hit it, then the entire thing is a host.
|
|
if (hostEnd === -1)
|
|
hostEnd = rest.length;
|
|
|
|
this.host = rest.slice(0, hostEnd);
|
|
rest = rest.slice(hostEnd);
|
|
|
|
// pull out port.
|
|
this.parseHost();
|
|
|
|
// we've indicated that there is a hostname,
|
|
// so even if it's empty, it has to be present.
|
|
this.hostname = this.hostname || '';
|
|
|
|
// if hostname begins with [ and ends with ]
|
|
// assume that it's an IPv6 address.
|
|
var ipv6Hostname = this.hostname[0] === '[' &&
|
|
this.hostname[this.hostname.length - 1] === ']';
|
|
|
|
// validate a little.
|
|
if (!ipv6Hostname) {
|
|
var hostparts = this.hostname.split(/\./);
|
|
for (var i = 0, l = hostparts.length; i < l; i++) {
|
|
var part = hostparts[i];
|
|
if (!part) continue;
|
|
if (!part.match(hostnamePartPattern)) {
|
|
var newpart = '';
|
|
for (var j = 0, k = part.length; j < k; j++) {
|
|
if (part.charCodeAt(j) > 127) {
|
|
// we replace non-ASCII char with a temporary placeholder
|
|
// we need this to make sure size of hostname is not
|
|
// broken by replacing non-ASCII by nothing
|
|
newpart += 'x';
|
|
} else {
|
|
newpart += part[j];
|
|
}
|
|
}
|
|
// we test again with ASCII char only
|
|
if (!newpart.match(hostnamePartPattern)) {
|
|
var validParts = hostparts.slice(0, i);
|
|
var notHost = hostparts.slice(i + 1);
|
|
var bit = part.match(hostnamePartStart);
|
|
if (bit) {
|
|
validParts.push(bit[1]);
|
|
notHost.unshift(bit[2]);
|
|
}
|
|
if (notHost.length) {
|
|
rest = '/' + notHost.join('.') + rest;
|
|
}
|
|
this.hostname = validParts.join('.');
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (this.hostname.length > hostnameMaxLen) {
|
|
this.hostname = '';
|
|
} else {
|
|
// hostnames are always lower case.
|
|
this.hostname = this.hostname.toLowerCase();
|
|
}
|
|
|
|
if (!ipv6Hostname) {
|
|
// IDNA Support: Returns a puny coded representation of "domain".
|
|
// It only converts the part of the domain name that
|
|
// has non ASCII characters. I.e. it dosent matter if
|
|
// you call it with a domain that already is in ASCII.
|
|
var domainArray = this.hostname.split('.');
|
|
var newOut = [];
|
|
for (var i = 0; i < domainArray.length; ++i) {
|
|
var s = domainArray[i];
|
|
newOut.push(s.match(/[^A-Za-z0-9_-]/) ?
|
|
'xn--' + punycode.encode(s) : s);
|
|
}
|
|
this.hostname = newOut.join('.');
|
|
}
|
|
|
|
var p = this.port ? ':' + this.port : '';
|
|
var h = this.hostname || '';
|
|
this.host = h + p;
|
|
this.href += this.host;
|
|
|
|
// strip [ and ] from the hostname
|
|
// the host field still retains them, though
|
|
if (ipv6Hostname) {
|
|
this.hostname = this.hostname.substr(1, this.hostname.length - 2);
|
|
if (rest[0] !== '/') {
|
|
rest = '/' + rest;
|
|
}
|
|
}
|
|
}
|
|
|
|
// now rest is set to the post-host stuff.
|
|
// chop off any delim chars.
|
|
if (!unsafeProtocol[lowerProto]) {
|
|
|
|
// First, make 100% sure that any "autoEscape" chars get
|
|
// escaped, even if encodeURIComponent doesn't think they
|
|
// need to be.
|
|
for (var i = 0, l = autoEscape.length; i < l; i++) {
|
|
var ae = autoEscape[i];
|
|
var esc = encodeURIComponent(ae);
|
|
if (esc === ae) {
|
|
esc = escape(ae);
|
|
}
|
|
rest = rest.split(ae).join(esc);
|
|
}
|
|
}
|
|
|
|
|
|
// chop off from the tail first.
|
|
var hash = rest.indexOf('#');
|
|
if (hash !== -1) {
|
|
// got a fragment string.
|
|
this.hash = rest.substr(hash);
|
|
rest = rest.slice(0, hash);
|
|
}
|
|
var qm = rest.indexOf('?');
|
|
if (qm !== -1) {
|
|
this.search = rest.substr(qm);
|
|
this.query = rest.substr(qm + 1);
|
|
if (parseQueryString) {
|
|
this.query = querystring.parse(this.query);
|
|
}
|
|
rest = rest.slice(0, qm);
|
|
} else if (parseQueryString) {
|
|
// no query string, but parseQueryString still requested
|
|
this.search = '';
|
|
this.query = {};
|
|
}
|
|
if (rest) this.pathname = rest;
|
|
if (slashedProtocol[lowerProto] &&
|
|
this.hostname && !this.pathname) {
|
|
this.pathname = '/';
|
|
}
|
|
|
|
//to support http.request
|
|
if (this.pathname || this.search) {
|
|
var p = this.pathname || '';
|
|
var s = this.search || '';
|
|
this.path = p + s;
|
|
}
|
|
|
|
// finally, reconstruct the href based on what has been validated.
|
|
this.href = this.format();
|
|
return this;
|
|
};
|
|
|
|
// format a parsed object into a url string
|
|
function urlFormat(obj) {
|
|
// ensure it's an object, and not a string url.
|
|
// If it's an obj, this is a no-op.
|
|
// this way, you can call url_format() on strings
|
|
// to clean up potentially wonky urls.
|
|
if (util.isString(obj)) obj = urlParse(obj);
|
|
if (!(obj instanceof Url)) return Url.prototype.format.call(obj);
|
|
return obj.format();
|
|
}
|
|
|
|
Url.prototype.format = function() {
|
|
var auth = this.auth || '';
|
|
if (auth) {
|
|
auth = encodeURIComponent(auth);
|
|
auth = auth.replace(/%3A/i, ':');
|
|
auth += '@';
|
|
}
|
|
|
|
var protocol = this.protocol || '',
|
|
pathname = this.pathname || '',
|
|
hash = this.hash || '',
|
|
host = false,
|
|
query = '';
|
|
|
|
if (this.host) {
|
|
host = auth + this.host;
|
|
} else if (this.hostname) {
|
|
host = auth + (this.hostname.indexOf(':') === -1 ?
|
|
this.hostname :
|
|
'[' + this.hostname + ']');
|
|
if (this.port) {
|
|
host += ':' + this.port;
|
|
}
|
|
}
|
|
|
|
if (this.query &&
|
|
util.isObject(this.query) &&
|
|
shims.keys(this.query).length) {
|
|
query = querystring.stringify(this.query);
|
|
}
|
|
|
|
var search = this.search || (query && ('?' + query)) || '';
|
|
|
|
if (protocol && shims.substr(protocol, -1) !== ':') protocol += ':';
|
|
|
|
// only the slashedProtocols get the //. Not mailto:, xmpp:, etc.
|
|
// unless they had them to begin with.
|
|
if (this.slashes ||
|
|
(!protocol || slashedProtocol[protocol]) && host !== false) {
|
|
host = '//' + (host || '');
|
|
if (pathname && pathname.charAt(0) !== '/') pathname = '/' + pathname;
|
|
} else if (!host) {
|
|
host = '';
|
|
}
|
|
|
|
if (hash && hash.charAt(0) !== '#') hash = '#' + hash;
|
|
if (search && search.charAt(0) !== '?') search = '?' + search;
|
|
|
|
pathname = pathname.replace(/[?#]/g, function(match) {
|
|
return encodeURIComponent(match);
|
|
});
|
|
search = search.replace('#', '%23');
|
|
|
|
return protocol + host + pathname + search + hash;
|
|
};
|
|
|
|
function urlResolve(source, relative) {
|
|
return urlParse(source, false, true).resolve(relative);
|
|
}
|
|
|
|
Url.prototype.resolve = function(relative) {
|
|
return this.resolveObject(urlParse(relative, false, true)).format();
|
|
};
|
|
|
|
function urlResolveObject(source, relative) {
|
|
if (!source) return relative;
|
|
return urlParse(source, false, true).resolveObject(relative);
|
|
}
|
|
|
|
Url.prototype.resolveObject = function(relative) {
|
|
if (util.isString(relative)) {
|
|
var rel = new Url();
|
|
rel.parse(relative, false, true);
|
|
relative = rel;
|
|
}
|
|
|
|
var result = new Url();
|
|
shims.forEach(shims.keys(this), function(k) {
|
|
result[k] = this[k];
|
|
}, this);
|
|
|
|
// hash is always overridden, no matter what.
|
|
// even href="" will remove it.
|
|
result.hash = relative.hash;
|
|
|
|
// if the relative url is empty, then there's nothing left to do here.
|
|
if (relative.href === '') {
|
|
result.href = result.format();
|
|
return result;
|
|
}
|
|
|
|
// hrefs like //foo/bar always cut to the protocol.
|
|
if (relative.slashes && !relative.protocol) {
|
|
// take everything except the protocol from relative
|
|
shims.forEach(shims.keys(relative), function(k) {
|
|
if (k !== 'protocol')
|
|
result[k] = relative[k];
|
|
});
|
|
|
|
//urlParse appends trailing / to urls like http://www.example.com
|
|
if (slashedProtocol[result.protocol] &&
|
|
result.hostname && !result.pathname) {
|
|
result.path = result.pathname = '/';
|
|
}
|
|
|
|
result.href = result.format();
|
|
return result;
|
|
}
|
|
|
|
if (relative.protocol && relative.protocol !== result.protocol) {
|
|
// if it's a known url protocol, then changing
|
|
// the protocol does weird things
|
|
// first, if it's not file:, then we MUST have a host,
|
|
// and if there was a path
|
|
// to begin with, then we MUST have a path.
|
|
// if it is file:, then the host is dropped,
|
|
// because that's known to be hostless.
|
|
// anything else is assumed to be absolute.
|
|
if (!slashedProtocol[relative.protocol]) {
|
|
shims.forEach(shims.keys(relative), function(k) {
|
|
result[k] = relative[k];
|
|
});
|
|
result.href = result.format();
|
|
return result;
|
|
}
|
|
|
|
result.protocol = relative.protocol;
|
|
if (!relative.host && !hostlessProtocol[relative.protocol]) {
|
|
var relPath = (relative.pathname || '').split('/');
|
|
while (relPath.length && !(relative.host = relPath.shift()));
|
|
if (!relative.host) relative.host = '';
|
|
if (!relative.hostname) relative.hostname = '';
|
|
if (relPath[0] !== '') relPath.unshift('');
|
|
if (relPath.length < 2) relPath.unshift('');
|
|
result.pathname = relPath.join('/');
|
|
} else {
|
|
result.pathname = relative.pathname;
|
|
}
|
|
result.search = relative.search;
|
|
result.query = relative.query;
|
|
result.host = relative.host || '';
|
|
result.auth = relative.auth;
|
|
result.hostname = relative.hostname || relative.host;
|
|
result.port = relative.port;
|
|
// to support http.request
|
|
if (result.pathname || result.search) {
|
|
var p = result.pathname || '';
|
|
var s = result.search || '';
|
|
result.path = p + s;
|
|
}
|
|
result.slashes = result.slashes || relative.slashes;
|
|
result.href = result.format();
|
|
return result;
|
|
}
|
|
|
|
var isSourceAbs = (result.pathname && result.pathname.charAt(0) === '/'),
|
|
isRelAbs = (
|
|
relative.host ||
|
|
relative.pathname && relative.pathname.charAt(0) === '/'
|
|
),
|
|
mustEndAbs = (isRelAbs || isSourceAbs ||
|
|
(result.host && relative.pathname)),
|
|
removeAllDots = mustEndAbs,
|
|
srcPath = result.pathname && result.pathname.split('/') || [],
|
|
relPath = relative.pathname && relative.pathname.split('/') || [],
|
|
psychotic = result.protocol && !slashedProtocol[result.protocol];
|
|
|
|
// if the url is a non-slashed url, then relative
|
|
// links like ../.. should be able
|
|
// to crawl up to the hostname, as well. This is strange.
|
|
// result.protocol has already been set by now.
|
|
// Later on, put the first path part into the host field.
|
|
if (psychotic) {
|
|
result.hostname = '';
|
|
result.port = null;
|
|
if (result.host) {
|
|
if (srcPath[0] === '') srcPath[0] = result.host;
|
|
else srcPath.unshift(result.host);
|
|
}
|
|
result.host = '';
|
|
if (relative.protocol) {
|
|
relative.hostname = null;
|
|
relative.port = null;
|
|
if (relative.host) {
|
|
if (relPath[0] === '') relPath[0] = relative.host;
|
|
else relPath.unshift(relative.host);
|
|
}
|
|
relative.host = null;
|
|
}
|
|
mustEndAbs = mustEndAbs && (relPath[0] === '' || srcPath[0] === '');
|
|
}
|
|
|
|
if (isRelAbs) {
|
|
// it's absolute.
|
|
result.host = (relative.host || relative.host === '') ?
|
|
relative.host : result.host;
|
|
result.hostname = (relative.hostname || relative.hostname === '') ?
|
|
relative.hostname : result.hostname;
|
|
result.search = relative.search;
|
|
result.query = relative.query;
|
|
srcPath = relPath;
|
|
// fall through to the dot-handling below.
|
|
} else if (relPath.length) {
|
|
// it's relative
|
|
// throw away the existing file, and take the new path instead.
|
|
if (!srcPath) srcPath = [];
|
|
srcPath.pop();
|
|
srcPath = srcPath.concat(relPath);
|
|
result.search = relative.search;
|
|
result.query = relative.query;
|
|
} else if (!util.isNullOrUndefined(relative.search)) {
|
|
// just pull out the search.
|
|
// like href='?foo'.
|
|
// Put this after the other two cases because it simplifies the booleans
|
|
if (psychotic) {
|
|
result.hostname = result.host = srcPath.shift();
|
|
//occationaly the auth can get stuck only in host
|
|
//this especialy happens in cases like
|
|
//url.resolveObject('mailto:local1@domain1', 'local2@domain2')
|
|
var authInHost = result.host && result.host.indexOf('@') > 0 ?
|
|
result.host.split('@') : false;
|
|
if (authInHost) {
|
|
result.auth = authInHost.shift();
|
|
result.host = result.hostname = authInHost.shift();
|
|
}
|
|
}
|
|
result.search = relative.search;
|
|
result.query = relative.query;
|
|
//to support http.request
|
|
if (!util.isNull(result.pathname) || !util.isNull(result.search)) {
|
|
result.path = (result.pathname ? result.pathname : '') +
|
|
(result.search ? result.search : '');
|
|
}
|
|
result.href = result.format();
|
|
return result;
|
|
}
|
|
|
|
if (!srcPath.length) {
|
|
// no path at all. easy.
|
|
// we've already handled the other stuff above.
|
|
result.pathname = null;
|
|
//to support http.request
|
|
if (result.search) {
|
|
result.path = '/' + result.search;
|
|
} else {
|
|
result.path = null;
|
|
}
|
|
result.href = result.format();
|
|
return result;
|
|
}
|
|
|
|
// if a url ENDs in . or .., then it must get a trailing slash.
|
|
// however, if it ends in anything else non-slashy,
|
|
// then it must NOT get a trailing slash.
|
|
var last = srcPath.slice(-1)[0];
|
|
var hasTrailingSlash = (
|
|
(result.host || relative.host) && (last === '.' || last === '..') ||
|
|
last === '');
|
|
|
|
// strip single dots, resolve double dots to parent dir
|
|
// if the path tries to go above the root, `up` ends up > 0
|
|
var up = 0;
|
|
for (var i = srcPath.length; i >= 0; i--) {
|
|
last = srcPath[i];
|
|
if (last == '.') {
|
|
srcPath.splice(i, 1);
|
|
} else if (last === '..') {
|
|
srcPath.splice(i, 1);
|
|
up++;
|
|
} else if (up) {
|
|
srcPath.splice(i, 1);
|
|
up--;
|
|
}
|
|
}
|
|
|
|
// if the path is allowed to go above the root, restore leading ..s
|
|
if (!mustEndAbs && !removeAllDots) {
|
|
for (; up--; up) {
|
|
srcPath.unshift('..');
|
|
}
|
|
}
|
|
|
|
if (mustEndAbs && srcPath[0] !== '' &&
|
|
(!srcPath[0] || srcPath[0].charAt(0) !== '/')) {
|
|
srcPath.unshift('');
|
|
}
|
|
|
|
if (hasTrailingSlash && (shims.substr(srcPath.join('/'), -1) !== '/')) {
|
|
srcPath.push('');
|
|
}
|
|
|
|
var isAbsolute = srcPath[0] === '' ||
|
|
(srcPath[0] && srcPath[0].charAt(0) === '/');
|
|
|
|
// put the host back
|
|
if (psychotic) {
|
|
result.hostname = result.host = isAbsolute ? '' :
|
|
srcPath.length ? srcPath.shift() : '';
|
|
//occationaly the auth can get stuck only in host
|
|
//this especialy happens in cases like
|
|
//url.resolveObject('mailto:local1@domain1', 'local2@domain2')
|
|
var authInHost = result.host && result.host.indexOf('@') > 0 ?
|
|
result.host.split('@') : false;
|
|
if (authInHost) {
|
|
result.auth = authInHost.shift();
|
|
result.host = result.hostname = authInHost.shift();
|
|
}
|
|
}
|
|
|
|
mustEndAbs = mustEndAbs || (result.host && srcPath.length);
|
|
|
|
if (mustEndAbs && !isAbsolute) {
|
|
srcPath.unshift('');
|
|
}
|
|
|
|
if (!srcPath.length) {
|
|
result.pathname = null;
|
|
result.path = null;
|
|
} else {
|
|
result.pathname = srcPath.join('/');
|
|
}
|
|
|
|
//to support request.http
|
|
if (!util.isNull(result.pathname) || !util.isNull(result.search)) {
|
|
result.path = (result.pathname ? result.pathname : '') +
|
|
(result.search ? result.search : '');
|
|
}
|
|
result.auth = relative.auth || result.auth;
|
|
result.slashes = result.slashes || relative.slashes;
|
|
result.href = result.format();
|
|
return result;
|
|
};
|
|
|
|
Url.prototype.parseHost = function() {
|
|
var host = this.host;
|
|
var port = portPattern.exec(host);
|
|
if (port) {
|
|
port = port[0];
|
|
if (port !== ':') {
|
|
this.port = port.substr(1);
|
|
}
|
|
host = host.substr(0, host.length - port.length);
|
|
}
|
|
if (host) this.hostname = host;
|
|
};
|
|
},{"_shims":2,"querystring":8,"util":10}],10:[function(require,module,exports){
|
|
// Copyright Joyent, Inc. and other Node contributors.
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining a
|
|
// copy of this software and associated documentation files (the
|
|
// "Software"), to deal in the Software without restriction, including
|
|
// without limitation the rights to use, copy, modify, merge, publish,
|
|
// distribute, sublicense, and/or sell copies of the Software, and to permit
|
|
// persons to whom the Software is furnished to do so, subject to the
|
|
// following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be included
|
|
// in all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
|
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
|
|
// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
|
|
// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
|
|
// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
|
|
// USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
var shims = require('_shims');
|
|
|
|
var formatRegExp = /%[sdj%]/g;
|
|
exports.format = function(f) {
|
|
if (!isString(f)) {
|
|
var objects = [];
|
|
for (var i = 0; i < arguments.length; i++) {
|
|
objects.push(inspect(arguments[i]));
|
|
}
|
|
return objects.join(' ');
|
|
}
|
|
|
|
var i = 1;
|
|
var args = arguments;
|
|
var len = args.length;
|
|
var str = String(f).replace(formatRegExp, function(x) {
|
|
if (x === '%%') return '%';
|
|
if (i >= len) return x;
|
|
switch (x) {
|
|
case '%s': return String(args[i++]);
|
|
case '%d': return Number(args[i++]);
|
|
case '%j':
|
|
try {
|
|
return JSON.stringify(args[i++]);
|
|
} catch (_) {
|
|
return '[Circular]';
|
|
}
|
|
default:
|
|
return x;
|
|
}
|
|
});
|
|
for (var x = args[i]; i < len; x = args[++i]) {
|
|
if (isNull(x) || !isObject(x)) {
|
|
str += ' ' + x;
|
|
} else {
|
|
str += ' ' + inspect(x);
|
|
}
|
|
}
|
|
return str;
|
|
};
|
|
|
|
/**
|
|
* Echos the value of a value. Trys to print the value out
|
|
* in the best way possible given the different types.
|
|
*
|
|
* @param {Object} obj The object to print out.
|
|
* @param {Object} opts Optional options object that alters the output.
|
|
*/
|
|
/* legacy: obj, showHidden, depth, colors*/
|
|
function inspect(obj, opts) {
|
|
// default options
|
|
var ctx = {
|
|
seen: [],
|
|
stylize: stylizeNoColor
|
|
};
|
|
// legacy...
|
|
if (arguments.length >= 3) ctx.depth = arguments[2];
|
|
if (arguments.length >= 4) ctx.colors = arguments[3];
|
|
if (isBoolean(opts)) {
|
|
// legacy...
|
|
ctx.showHidden = opts;
|
|
} else if (opts) {
|
|
// got an "options" object
|
|
exports._extend(ctx, opts);
|
|
}
|
|
// set default options
|
|
if (isUndefined(ctx.showHidden)) ctx.showHidden = false;
|
|
if (isUndefined(ctx.depth)) ctx.depth = 2;
|
|
if (isUndefined(ctx.colors)) ctx.colors = false;
|
|
if (isUndefined(ctx.customInspect)) ctx.customInspect = true;
|
|
if (ctx.colors) ctx.stylize = stylizeWithColor;
|
|
return formatValue(ctx, obj, ctx.depth);
|
|
}
|
|
exports.inspect = inspect;
|
|
|
|
|
|
// http://en.wikipedia.org/wiki/ANSI_escape_code#graphics
|
|
inspect.colors = {
|
|
'bold' : [1, 22],
|
|
'italic' : [3, 23],
|
|
'underline' : [4, 24],
|
|
'inverse' : [7, 27],
|
|
'white' : [37, 39],
|
|
'grey' : [90, 39],
|
|
'black' : [30, 39],
|
|
'blue' : [34, 39],
|
|
'cyan' : [36, 39],
|
|
'green' : [32, 39],
|
|
'magenta' : [35, 39],
|
|
'red' : [31, 39],
|
|
'yellow' : [33, 39]
|
|
};
|
|
|
|
// Don't use 'blue' not visible on cmd.exe
|
|
inspect.styles = {
|
|
'special': 'cyan',
|
|
'number': 'yellow',
|
|
'boolean': 'yellow',
|
|
'undefined': 'grey',
|
|
'null': 'bold',
|
|
'string': 'green',
|
|
'date': 'magenta',
|
|
// "name": intentionally not styling
|
|
'regexp': 'red'
|
|
};
|
|
|
|
|
|
function stylizeWithColor(str, styleType) {
|
|
var style = inspect.styles[styleType];
|
|
|
|
if (style) {
|
|
return '\u001b[' + inspect.colors[style][0] + 'm' + str +
|
|
'\u001b[' + inspect.colors[style][1] + 'm';
|
|
} else {
|
|
return str;
|
|
}
|
|
}
|
|
|
|
|
|
function stylizeNoColor(str, styleType) {
|
|
return str;
|
|
}
|
|
|
|
|
|
function arrayToHash(array) {
|
|
var hash = {};
|
|
|
|
shims.forEach(array, function(val, idx) {
|
|
hash[val] = true;
|
|
});
|
|
|
|
return hash;
|
|
}
|
|
|
|
|
|
function formatValue(ctx, value, recurseTimes) {
|
|
// Provide a hook for user-specified inspect functions.
|
|
// Check that value is an object with an inspect function on it
|
|
if (ctx.customInspect &&
|
|
value &&
|
|
isFunction(value.inspect) &&
|
|
// Filter out the util module, it's inspect function is special
|
|
value.inspect !== exports.inspect &&
|
|
// Also filter out any prototype objects using the circular check.
|
|
!(value.constructor && value.constructor.prototype === value)) {
|
|
var ret = value.inspect(recurseTimes);
|
|
if (!isString(ret)) {
|
|
ret = formatValue(ctx, ret, recurseTimes);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
// Primitive types cannot have properties
|
|
var primitive = formatPrimitive(ctx, value);
|
|
if (primitive) {
|
|
return primitive;
|
|
}
|
|
|
|
// Look up the keys of the object.
|
|
var keys = shims.keys(value);
|
|
var visibleKeys = arrayToHash(keys);
|
|
|
|
if (ctx.showHidden) {
|
|
keys = shims.getOwnPropertyNames(value);
|
|
}
|
|
|
|
// Some type of object without properties can be shortcutted.
|
|
if (keys.length === 0) {
|
|
if (isFunction(value)) {
|
|
var name = value.name ? ': ' + value.name : '';
|
|
return ctx.stylize('[Function' + name + ']', 'special');
|
|
}
|
|
if (isRegExp(value)) {
|
|
return ctx.stylize(RegExp.prototype.toString.call(value), 'regexp');
|
|
}
|
|
if (isDate(value)) {
|
|
return ctx.stylize(Date.prototype.toString.call(value), 'date');
|
|
}
|
|
if (isError(value)) {
|
|
return formatError(value);
|
|
}
|
|
}
|
|
|
|
var base = '', array = false, braces = ['{', '}'];
|
|
|
|
// Make Array say that they are Array
|
|
if (isArray(value)) {
|
|
array = true;
|
|
braces = ['[', ']'];
|
|
}
|
|
|
|
// Make functions say that they are functions
|
|
if (isFunction(value)) {
|
|
var n = value.name ? ': ' + value.name : '';
|
|
base = ' [Function' + n + ']';
|
|
}
|
|
|
|
// Make RegExps say that they are RegExps
|
|
if (isRegExp(value)) {
|
|
base = ' ' + RegExp.prototype.toString.call(value);
|
|
}
|
|
|
|
// Make dates with properties first say the date
|
|
if (isDate(value)) {
|
|
base = ' ' + Date.prototype.toUTCString.call(value);
|
|
}
|
|
|
|
// Make error with message first say the error
|
|
if (isError(value)) {
|
|
base = ' ' + formatError(value);
|
|
}
|
|
|
|
if (keys.length === 0 && (!array || value.length == 0)) {
|
|
return braces[0] + base + braces[1];
|
|
}
|
|
|
|
if (recurseTimes < 0) {
|
|
if (isRegExp(value)) {
|
|
return ctx.stylize(RegExp.prototype.toString.call(value), 'regexp');
|
|
} else {
|
|
return ctx.stylize('[Object]', 'special');
|
|
}
|
|
}
|
|
|
|
ctx.seen.push(value);
|
|
|
|
var output;
|
|
if (array) {
|
|
output = formatArray(ctx, value, recurseTimes, visibleKeys, keys);
|
|
} else {
|
|
output = keys.map(function(key) {
|
|
return formatProperty(ctx, value, recurseTimes, visibleKeys, key, array);
|
|
});
|
|
}
|
|
|
|
ctx.seen.pop();
|
|
|
|
return reduceToSingleString(output, base, braces);
|
|
}
|
|
|
|
|
|
function formatPrimitive(ctx, value) {
|
|
if (isUndefined(value))
|
|
return ctx.stylize('undefined', 'undefined');
|
|
if (isString(value)) {
|
|
var simple = '\'' + JSON.stringify(value).replace(/^"|"$/g, '')
|
|
.replace(/'/g, "\\'")
|
|
.replace(/\\"/g, '"') + '\'';
|
|
return ctx.stylize(simple, 'string');
|
|
}
|
|
if (isNumber(value))
|
|
return ctx.stylize('' + value, 'number');
|
|
if (isBoolean(value))
|
|
return ctx.stylize('' + value, 'boolean');
|
|
// For some reason typeof null is "object", so special case here.
|
|
if (isNull(value))
|
|
return ctx.stylize('null', 'null');
|
|
}
|
|
|
|
|
|
function formatError(value) {
|
|
return '[' + Error.prototype.toString.call(value) + ']';
|
|
}
|
|
|
|
|
|
function formatArray(ctx, value, recurseTimes, visibleKeys, keys) {
|
|
var output = [];
|
|
for (var i = 0, l = value.length; i < l; ++i) {
|
|
if (hasOwnProperty(value, String(i))) {
|
|
output.push(formatProperty(ctx, value, recurseTimes, visibleKeys,
|
|
String(i), true));
|
|
} else {
|
|
output.push('');
|
|
}
|
|
}
|
|
|
|
shims.forEach(keys, function(key) {
|
|
if (!key.match(/^\d+$/)) {
|
|
output.push(formatProperty(ctx, value, recurseTimes, visibleKeys,
|
|
key, true));
|
|
}
|
|
});
|
|
return output;
|
|
}
|
|
|
|
|
|
function formatProperty(ctx, value, recurseTimes, visibleKeys, key, array) {
|
|
var name, str, desc;
|
|
desc = shims.getOwnPropertyDescriptor(value, key) || { value: value[key] };
|
|
if (desc.get) {
|
|
if (desc.set) {
|
|
str = ctx.stylize('[Getter/Setter]', 'special');
|
|
} else {
|
|
str = ctx.stylize('[Getter]', 'special');
|
|
}
|
|
} else {
|
|
if (desc.set) {
|
|
str = ctx.stylize('[Setter]', 'special');
|
|
}
|
|
}
|
|
|
|
if (!hasOwnProperty(visibleKeys, key)) {
|
|
name = '[' + key + ']';
|
|
}
|
|
if (!str) {
|
|
if (shims.indexOf(ctx.seen, desc.value) < 0) {
|
|
if (isNull(recurseTimes)) {
|
|
str = formatValue(ctx, desc.value, null);
|
|
} else {
|
|
str = formatValue(ctx, desc.value, recurseTimes - 1);
|
|
}
|
|
if (str.indexOf('\n') > -1) {
|
|
if (array) {
|
|
str = str.split('\n').map(function(line) {
|
|
return ' ' + line;
|
|
}).join('\n').substr(2);
|
|
} else {
|
|
str = '\n' + str.split('\n').map(function(line) {
|
|
return ' ' + line;
|
|
}).join('\n');
|
|
}
|
|
}
|
|
} else {
|
|
str = ctx.stylize('[Circular]', 'special');
|
|
}
|
|
}
|
|
if (isUndefined(name)) {
|
|
if (array && key.match(/^\d+$/)) {
|
|
return str;
|
|
}
|
|
name = JSON.stringify('' + key);
|
|
if (name.match(/^"([a-zA-Z_][a-zA-Z_0-9]*)"$/)) {
|
|
name = name.substr(1, name.length - 2);
|
|
name = ctx.stylize(name, 'name');
|
|
} else {
|
|
name = name.replace(/'/g, "\\'")
|
|
.replace(/\\"/g, '"')
|
|
.replace(/(^"|"$)/g, "'");
|
|
name = ctx.stylize(name, 'string');
|
|
}
|
|
}
|
|
|
|
return name + ': ' + str;
|
|
}
|
|
|
|
|
|
function reduceToSingleString(output, base, braces) {
|
|
var numLinesEst = 0;
|
|
var length = shims.reduce(output, function(prev, cur) {
|
|
numLinesEst++;
|
|
if (cur.indexOf('\n') >= 0) numLinesEst++;
|
|
return prev + cur.replace(/\u001b\[\d\d?m/g, '').length + 1;
|
|
}, 0);
|
|
|
|
if (length > 60) {
|
|
return braces[0] +
|
|
(base === '' ? '' : base + '\n ') +
|
|
' ' +
|
|
output.join(',\n ') +
|
|
' ' +
|
|
braces[1];
|
|
}
|
|
|
|
return braces[0] + base + ' ' + output.join(', ') + ' ' + braces[1];
|
|
}
|
|
|
|
|
|
// NOTE: These type checking functions intentionally don't use `instanceof`
|
|
// because it is fragile and can be easily faked with `Object.create()`.
|
|
function isArray(ar) {
|
|
return shims.isArray(ar);
|
|
}
|
|
exports.isArray = isArray;
|
|
|
|
function isBoolean(arg) {
|
|
return typeof arg === 'boolean';
|
|
}
|
|
exports.isBoolean = isBoolean;
|
|
|
|
function isNull(arg) {
|
|
return arg === null;
|
|
}
|
|
exports.isNull = isNull;
|
|
|
|
function isNullOrUndefined(arg) {
|
|
return arg == null;
|
|
}
|
|
exports.isNullOrUndefined = isNullOrUndefined;
|
|
|
|
function isNumber(arg) {
|
|
return typeof arg === 'number';
|
|
}
|
|
exports.isNumber = isNumber;
|
|
|
|
function isString(arg) {
|
|
return typeof arg === 'string';
|
|
}
|
|
exports.isString = isString;
|
|
|
|
function isSymbol(arg) {
|
|
return typeof arg === 'symbol';
|
|
}
|
|
exports.isSymbol = isSymbol;
|
|
|
|
function isUndefined(arg) {
|
|
return arg === void 0;
|
|
}
|
|
exports.isUndefined = isUndefined;
|
|
|
|
function isRegExp(re) {
|
|
return isObject(re) && objectToString(re) === '[object RegExp]';
|
|
}
|
|
exports.isRegExp = isRegExp;
|
|
|
|
function isObject(arg) {
|
|
return typeof arg === 'object' && arg;
|
|
}
|
|
exports.isObject = isObject;
|
|
|
|
function isDate(d) {
|
|
return isObject(d) && objectToString(d) === '[object Date]';
|
|
}
|
|
exports.isDate = isDate;
|
|
|
|
function isError(e) {
|
|
return isObject(e) && objectToString(e) === '[object Error]';
|
|
}
|
|
exports.isError = isError;
|
|
|
|
function isFunction(arg) {
|
|
return typeof arg === 'function';
|
|
}
|
|
exports.isFunction = isFunction;
|
|
|
|
function isPrimitive(arg) {
|
|
return arg === null ||
|
|
typeof arg === 'boolean' ||
|
|
typeof arg === 'number' ||
|
|
typeof arg === 'string' ||
|
|
typeof arg === 'symbol' || // ES6 symbol
|
|
typeof arg === 'undefined';
|
|
}
|
|
exports.isPrimitive = isPrimitive;
|
|
|
|
function isBuffer(arg) {
|
|
return arg && typeof arg === 'object'
|
|
&& typeof arg.copy === 'function'
|
|
&& typeof arg.fill === 'function'
|
|
&& typeof arg.binarySlice === 'function'
|
|
;
|
|
}
|
|
exports.isBuffer = isBuffer;
|
|
|
|
function objectToString(o) {
|
|
return Object.prototype.toString.call(o);
|
|
}
|
|
|
|
|
|
function pad(n) {
|
|
return n < 10 ? '0' + n.toString(10) : n.toString(10);
|
|
}
|
|
|
|
|
|
var months = ['Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug', 'Sep',
|
|
'Oct', 'Nov', 'Dec'];
|
|
|
|
// 26 Feb 16:19:34
|
|
function timestamp() {
|
|
var d = new Date();
|
|
var time = [pad(d.getHours()),
|
|
pad(d.getMinutes()),
|
|
pad(d.getSeconds())].join(':');
|
|
return [d.getDate(), months[d.getMonth()], time].join(' ');
|
|
}
|
|
|
|
|
|
// log is just a thin wrapper to console.log that prepends a timestamp
|
|
exports.log = function() {
|
|
console.log('%s - %s', timestamp(), exports.format.apply(exports, arguments));
|
|
};
|
|
|
|
|
|
/**
|
|
* Inherit the prototype methods from one constructor into another.
|
|
*
|
|
* The Function.prototype.inherits from lang.js rewritten as a standalone
|
|
* function (not on Function.prototype). NOTE: If this file is to be loaded
|
|
* during bootstrapping this function needs to be rewritten using some native
|
|
* functions as prototype setup using normal JavaScript does not work as
|
|
* expected during bootstrapping (see mirror.js in r114903).
|
|
*
|
|
* @param {function} ctor Constructor function which needs to inherit the
|
|
* prototype.
|
|
* @param {function} superCtor Constructor function to inherit prototype from.
|
|
*/
|
|
exports.inherits = function(ctor, superCtor) {
|
|
ctor.super_ = superCtor;
|
|
ctor.prototype = shims.create(superCtor.prototype, {
|
|
constructor: {
|
|
value: ctor,
|
|
enumerable: false,
|
|
writable: true,
|
|
configurable: true
|
|
}
|
|
});
|
|
};
|
|
|
|
exports._extend = function(origin, add) {
|
|
// Don't do anything if add isn't an object
|
|
if (!add || !isObject(add)) return origin;
|
|
|
|
var keys = shims.keys(add);
|
|
var i = keys.length;
|
|
while (i--) {
|
|
origin[keys[i]] = add[keys[i]];
|
|
}
|
|
return origin;
|
|
};
|
|
|
|
function hasOwnProperty(obj, prop) {
|
|
return Object.prototype.hasOwnProperty.call(obj, prop);
|
|
}
|
|
|
|
},{"_shims":2}],11:[function(require,module,exports){
|
|
exports.readIEEE754 = function(buffer, offset, isBE, mLen, nBytes) {
|
|
var e, m,
|
|
eLen = nBytes * 8 - mLen - 1,
|
|
eMax = (1 << eLen) - 1,
|
|
eBias = eMax >> 1,
|
|
nBits = -7,
|
|
i = isBE ? 0 : (nBytes - 1),
|
|
d = isBE ? 1 : -1,
|
|
s = buffer[offset + i];
|
|
|
|
i += d;
|
|
|
|
e = s & ((1 << (-nBits)) - 1);
|
|
s >>= (-nBits);
|
|
nBits += eLen;
|
|
for (; nBits > 0; e = e * 256 + buffer[offset + i], i += d, nBits -= 8);
|
|
|
|
m = e & ((1 << (-nBits)) - 1);
|
|
e >>= (-nBits);
|
|
nBits += mLen;
|
|
for (; nBits > 0; m = m * 256 + buffer[offset + i], i += d, nBits -= 8);
|
|
|
|
if (e === 0) {
|
|
e = 1 - eBias;
|
|
} else if (e === eMax) {
|
|
return m ? NaN : ((s ? -1 : 1) * Infinity);
|
|
} else {
|
|
m = m + Math.pow(2, mLen);
|
|
e = e - eBias;
|
|
}
|
|
return (s ? -1 : 1) * m * Math.pow(2, e - mLen);
|
|
};
|
|
|
|
exports.writeIEEE754 = function(buffer, value, offset, isBE, mLen, nBytes) {
|
|
var e, m, c,
|
|
eLen = nBytes * 8 - mLen - 1,
|
|
eMax = (1 << eLen) - 1,
|
|
eBias = eMax >> 1,
|
|
rt = (mLen === 23 ? Math.pow(2, -24) - Math.pow(2, -77) : 0),
|
|
i = isBE ? (nBytes - 1) : 0,
|
|
d = isBE ? -1 : 1,
|
|
s = value < 0 || (value === 0 && 1 / value < 0) ? 1 : 0;
|
|
|
|
value = Math.abs(value);
|
|
|
|
if (isNaN(value) || value === Infinity) {
|
|
m = isNaN(value) ? 1 : 0;
|
|
e = eMax;
|
|
} else {
|
|
e = Math.floor(Math.log(value) / Math.LN2);
|
|
if (value * (c = Math.pow(2, -e)) < 1) {
|
|
e--;
|
|
c *= 2;
|
|
}
|
|
if (e + eBias >= 1) {
|
|
value += rt / c;
|
|
} else {
|
|
value += rt * Math.pow(2, 1 - eBias);
|
|
}
|
|
if (value * c >= 2) {
|
|
e++;
|
|
c /= 2;
|
|
}
|
|
|
|
if (e + eBias >= eMax) {
|
|
m = 0;
|
|
e = eMax;
|
|
} else if (e + eBias >= 1) {
|
|
m = (value * c - 1) * Math.pow(2, mLen);
|
|
e = e + eBias;
|
|
} else {
|
|
m = value * Math.pow(2, eBias - 1) * Math.pow(2, mLen);
|
|
e = 0;
|
|
}
|
|
}
|
|
|
|
for (; mLen >= 8; buffer[offset + i] = m & 0xff, i += d, m /= 256, mLen -= 8);
|
|
|
|
e = (e << mLen) | m;
|
|
eLen += mLen;
|
|
for (; eLen > 0; buffer[offset + i] = e & 0xff, i += d, e /= 256, eLen -= 8);
|
|
|
|
buffer[offset + i - d] |= s * 128;
|
|
};
|
|
|
|
},{}],12:[function(require,module,exports){
|
|
var assert;
|
|
exports.Buffer = Buffer;
|
|
exports.SlowBuffer = Buffer;
|
|
Buffer.poolSize = 8192;
|
|
exports.INSPECT_MAX_BYTES = 50;
|
|
|
|
function stringtrim(str) {
|
|
if (str.trim) return str.trim();
|
|
return str.replace(/^\s+|\s+$/g, '');
|
|
}
|
|
|
|
function Buffer(subject, encoding, offset) {
|
|
if(!assert) assert= require('assert');
|
|
if (!(this instanceof Buffer)) {
|
|
return new Buffer(subject, encoding, offset);
|
|
}
|
|
this.parent = this;
|
|
this.offset = 0;
|
|
|
|
// Work-around: node's base64 implementation
|
|
// allows for non-padded strings while base64-js
|
|
// does not..
|
|
if (encoding == "base64" && typeof subject == "string") {
|
|
subject = stringtrim(subject);
|
|
while (subject.length % 4 != 0) {
|
|
subject = subject + "=";
|
|
}
|
|
}
|
|
|
|
var type;
|
|
|
|
// Are we slicing?
|
|
if (typeof offset === 'number') {
|
|
this.length = coerce(encoding);
|
|
// slicing works, with limitations (no parent tracking/update)
|
|
// check https://github.com/toots/buffer-browserify/issues/19
|
|
for (var i = 0; i < this.length; i++) {
|
|
this[i] = subject.get(i+offset);
|
|
}
|
|
} else {
|
|
// Find the length
|
|
switch (type = typeof subject) {
|
|
case 'number':
|
|
this.length = coerce(subject);
|
|
break;
|
|
|
|
case 'string':
|
|
this.length = Buffer.byteLength(subject, encoding);
|
|
break;
|
|
|
|
case 'object': // Assume object is an array
|
|
this.length = coerce(subject.length);
|
|
break;
|
|
|
|
default:
|
|
throw new Error('First argument needs to be a number, ' +
|
|
'array or string.');
|
|
}
|
|
|
|
// Treat array-ish objects as a byte array.
|
|
if (isArrayIsh(subject)) {
|
|
for (var i = 0; i < this.length; i++) {
|
|
if (subject instanceof Buffer) {
|
|
this[i] = subject.readUInt8(i);
|
|
}
|
|
else {
|
|
this[i] = subject[i];
|
|
}
|
|
}
|
|
} else if (type == 'string') {
|
|
// We are a string
|
|
this.length = this.write(subject, 0, encoding);
|
|
} else if (type === 'number') {
|
|
for (var i = 0; i < this.length; i++) {
|
|
this[i] = 0;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
Buffer.prototype.get = function get(i) {
|
|
if (i < 0 || i >= this.length) throw new Error('oob');
|
|
return this[i];
|
|
};
|
|
|
|
Buffer.prototype.set = function set(i, v) {
|
|
if (i < 0 || i >= this.length) throw new Error('oob');
|
|
return this[i] = v;
|
|
};
|
|
|
|
Buffer.byteLength = function (str, encoding) {
|
|
switch (encoding || "utf8") {
|
|
case 'hex':
|
|
return str.length / 2;
|
|
|
|
case 'utf8':
|
|
case 'utf-8':
|
|
return utf8ToBytes(str).length;
|
|
|
|
case 'ascii':
|
|
case 'binary':
|
|
return str.length;
|
|
|
|
case 'base64':
|
|
return base64ToBytes(str).length;
|
|
|
|
default:
|
|
throw new Error('Unknown encoding');
|
|
}
|
|
};
|
|
|
|
Buffer.prototype.utf8Write = function (string, offset, length) {
|
|
var bytes, pos;
|
|
return Buffer._charsWritten = blitBuffer(utf8ToBytes(string), this, offset, length);
|
|
};
|
|
|
|
Buffer.prototype.asciiWrite = function (string, offset, length) {
|
|
var bytes, pos;
|
|
return Buffer._charsWritten = blitBuffer(asciiToBytes(string), this, offset, length);
|
|
};
|
|
|
|
Buffer.prototype.binaryWrite = Buffer.prototype.asciiWrite;
|
|
|
|
Buffer.prototype.base64Write = function (string, offset, length) {
|
|
var bytes, pos;
|
|
return Buffer._charsWritten = blitBuffer(base64ToBytes(string), this, offset, length);
|
|
};
|
|
|
|
Buffer.prototype.base64Slice = function (start, end) {
|
|
var bytes = Array.prototype.slice.apply(this, arguments)
|
|
return require("base64-js").fromByteArray(bytes);
|
|
};
|
|
|
|
Buffer.prototype.utf8Slice = function () {
|
|
var bytes = Array.prototype.slice.apply(this, arguments);
|
|
var res = "";
|
|
var tmp = "";
|
|
var i = 0;
|
|
while (i < bytes.length) {
|
|
if (bytes[i] <= 0x7F) {
|
|
res += decodeUtf8Char(tmp) + String.fromCharCode(bytes[i]);
|
|
tmp = "";
|
|
} else
|
|
tmp += "%" + bytes[i].toString(16);
|
|
|
|
i++;
|
|
}
|
|
|
|
return res + decodeUtf8Char(tmp);
|
|
}
|
|
|
|
Buffer.prototype.asciiSlice = function () {
|
|
var bytes = Array.prototype.slice.apply(this, arguments);
|
|
var ret = "";
|
|
for (var i = 0; i < bytes.length; i++)
|
|
ret += String.fromCharCode(bytes[i]);
|
|
return ret;
|
|
}
|
|
|
|
Buffer.prototype.binarySlice = Buffer.prototype.asciiSlice;
|
|
|
|
Buffer.prototype.inspect = function() {
|
|
var out = [],
|
|
len = this.length;
|
|
for (var i = 0; i < len; i++) {
|
|
out[i] = toHex(this[i]);
|
|
if (i == exports.INSPECT_MAX_BYTES) {
|
|
out[i + 1] = '...';
|
|
break;
|
|
}
|
|
}
|
|
return '<Buffer ' + out.join(' ') + '>';
|
|
};
|
|
|
|
|
|
Buffer.prototype.hexSlice = function(start, end) {
|
|
var len = this.length;
|
|
|
|
if (!start || start < 0) start = 0;
|
|
if (!end || end < 0 || end > len) end = len;
|
|
|
|
var out = '';
|
|
for (var i = start; i < end; i++) {
|
|
out += toHex(this[i]);
|
|
}
|
|
return out;
|
|
};
|
|
|
|
|
|
Buffer.prototype.toString = function(encoding, start, end) {
|
|
encoding = String(encoding || 'utf8').toLowerCase();
|
|
start = +start || 0;
|
|
if (typeof end == 'undefined') end = this.length;
|
|
|
|
// Fastpath empty strings
|
|
if (+end == start) {
|
|
return '';
|
|
}
|
|
|
|
switch (encoding) {
|
|
case 'hex':
|
|
return this.hexSlice(start, end);
|
|
|
|
case 'utf8':
|
|
case 'utf-8':
|
|
return this.utf8Slice(start, end);
|
|
|
|
case 'ascii':
|
|
return this.asciiSlice(start, end);
|
|
|
|
case 'binary':
|
|
return this.binarySlice(start, end);
|
|
|
|
case 'base64':
|
|
return this.base64Slice(start, end);
|
|
|
|
case 'ucs2':
|
|
case 'ucs-2':
|
|
return this.ucs2Slice(start, end);
|
|
|
|
default:
|
|
throw new Error('Unknown encoding');
|
|
}
|
|
};
|
|
|
|
|
|
Buffer.prototype.hexWrite = function(string, offset, length) {
|
|
offset = +offset || 0;
|
|
var remaining = this.length - offset;
|
|
if (!length) {
|
|
length = remaining;
|
|
} else {
|
|
length = +length;
|
|
if (length > remaining) {
|
|
length = remaining;
|
|
}
|
|
}
|
|
|
|
// must be an even number of digits
|
|
var strLen = string.length;
|
|
if (strLen % 2) {
|
|
throw new Error('Invalid hex string');
|
|
}
|
|
if (length > strLen / 2) {
|
|
length = strLen / 2;
|
|
}
|
|
for (var i = 0; i < length; i++) {
|
|
var byte = parseInt(string.substr(i * 2, 2), 16);
|
|
if (isNaN(byte)) throw new Error('Invalid hex string');
|
|
this[offset + i] = byte;
|
|
}
|
|
Buffer._charsWritten = i * 2;
|
|
return i;
|
|
};
|
|
|
|
|
|
Buffer.prototype.write = function(string, offset, length, encoding) {
|
|
// Support both (string, offset, length, encoding)
|
|
// and the legacy (string, encoding, offset, length)
|
|
if (isFinite(offset)) {
|
|
if (!isFinite(length)) {
|
|
encoding = length;
|
|
length = undefined;
|
|
}
|
|
} else { // legacy
|
|
var swap = encoding;
|
|
encoding = offset;
|
|
offset = length;
|
|
length = swap;
|
|
}
|
|
|
|
offset = +offset || 0;
|
|
var remaining = this.length - offset;
|
|
if (!length) {
|
|
length = remaining;
|
|
} else {
|
|
length = +length;
|
|
if (length > remaining) {
|
|
length = remaining;
|
|
}
|
|
}
|
|
encoding = String(encoding || 'utf8').toLowerCase();
|
|
|
|
switch (encoding) {
|
|
case 'hex':
|
|
return this.hexWrite(string, offset, length);
|
|
|
|
case 'utf8':
|
|
case 'utf-8':
|
|
return this.utf8Write(string, offset, length);
|
|
|
|
case 'ascii':
|
|
return this.asciiWrite(string, offset, length);
|
|
|
|
case 'binary':
|
|
return this.binaryWrite(string, offset, length);
|
|
|
|
case 'base64':
|
|
return this.base64Write(string, offset, length);
|
|
|
|
case 'ucs2':
|
|
case 'ucs-2':
|
|
return this.ucs2Write(string, offset, length);
|
|
|
|
default:
|
|
throw new Error('Unknown encoding');
|
|
}
|
|
};
|
|
|
|
// slice(start, end)
|
|
function clamp(index, len, defaultValue) {
|
|
if (typeof index !== 'number') return defaultValue;
|
|
index = ~~index; // Coerce to integer.
|
|
if (index >= len) return len;
|
|
if (index >= 0) return index;
|
|
index += len;
|
|
if (index >= 0) return index;
|
|
return 0;
|
|
}
|
|
|
|
Buffer.prototype.slice = function(start, end) {
|
|
var len = this.length;
|
|
start = clamp(start, len, 0);
|
|
end = clamp(end, len, len);
|
|
return new Buffer(this, end - start, +start);
|
|
};
|
|
|
|
// copy(targetBuffer, targetStart=0, sourceStart=0, sourceEnd=buffer.length)
|
|
Buffer.prototype.copy = function(target, target_start, start, end) {
|
|
var source = this;
|
|
start || (start = 0);
|
|
if (end === undefined || isNaN(end)) {
|
|
end = this.length;
|
|
}
|
|
target_start || (target_start = 0);
|
|
|
|
if (end < start) throw new Error('sourceEnd < sourceStart');
|
|
|
|
// Copy 0 bytes; we're done
|
|
if (end === start) return 0;
|
|
if (target.length == 0 || source.length == 0) return 0;
|
|
|
|
if (target_start < 0 || target_start >= target.length) {
|
|
throw new Error('targetStart out of bounds');
|
|
}
|
|
|
|
if (start < 0 || start >= source.length) {
|
|
throw new Error('sourceStart out of bounds');
|
|
}
|
|
|
|
if (end < 0 || end > source.length) {
|
|
throw new Error('sourceEnd out of bounds');
|
|
}
|
|
|
|
// Are we oob?
|
|
if (end > this.length) {
|
|
end = this.length;
|
|
}
|
|
|
|
if (target.length - target_start < end - start) {
|
|
end = target.length - target_start + start;
|
|
}
|
|
|
|
var temp = [];
|
|
for (var i=start; i<end; i++) {
|
|
assert.ok(typeof this[i] !== 'undefined', "copying undefined buffer bytes!");
|
|
temp.push(this[i]);
|
|
}
|
|
|
|
for (var i=target_start; i<target_start+temp.length; i++) {
|
|
target[i] = temp[i-target_start];
|
|
}
|
|
};
|
|
|
|
// fill(value, start=0, end=buffer.length)
|
|
Buffer.prototype.fill = function fill(value, start, end) {
|
|
value || (value = 0);
|
|
start || (start = 0);
|
|
end || (end = this.length);
|
|
|
|
if (typeof value === 'string') {
|
|
value = value.charCodeAt(0);
|
|
}
|
|
if (!(typeof value === 'number') || isNaN(value)) {
|
|
throw new Error('value is not a number');
|
|
}
|
|
|
|
if (end < start) throw new Error('end < start');
|
|
|
|
// Fill 0 bytes; we're done
|
|
if (end === start) return 0;
|
|
if (this.length == 0) return 0;
|
|
|
|
if (start < 0 || start >= this.length) {
|
|
throw new Error('start out of bounds');
|
|
}
|
|
|
|
if (end < 0 || end > this.length) {
|
|
throw new Error('end out of bounds');
|
|
}
|
|
|
|
for (var i = start; i < end; i++) {
|
|
this[i] = value;
|
|
}
|
|
}
|
|
|
|
// Static methods
|
|
Buffer.isBuffer = function isBuffer(b) {
|
|
return b instanceof Buffer || b instanceof Buffer;
|
|
};
|
|
|
|
Buffer.concat = function (list, totalLength) {
|
|
if (!isArray(list)) {
|
|
throw new Error("Usage: Buffer.concat(list, [totalLength])\n \
|
|
list should be an Array.");
|
|
}
|
|
|
|
if (list.length === 0) {
|
|
return new Buffer(0);
|
|
} else if (list.length === 1) {
|
|
return list[0];
|
|
}
|
|
|
|
if (typeof totalLength !== 'number') {
|
|
totalLength = 0;
|
|
for (var i = 0; i < list.length; i++) {
|
|
var buf = list[i];
|
|
totalLength += buf.length;
|
|
}
|
|
}
|
|
|
|
var buffer = new Buffer(totalLength);
|
|
var pos = 0;
|
|
for (var i = 0; i < list.length; i++) {
|
|
var buf = list[i];
|
|
buf.copy(buffer, pos);
|
|
pos += buf.length;
|
|
}
|
|
return buffer;
|
|
};
|
|
|
|
Buffer.isEncoding = function(encoding) {
|
|
switch ((encoding + '').toLowerCase()) {
|
|
case 'hex':
|
|
case 'utf8':
|
|
case 'utf-8':
|
|
case 'ascii':
|
|
case 'binary':
|
|
case 'base64':
|
|
case 'ucs2':
|
|
case 'ucs-2':
|
|
case 'utf16le':
|
|
case 'utf-16le':
|
|
case 'raw':
|
|
return true;
|
|
|
|
default:
|
|
return false;
|
|
}
|
|
};
|
|
|
|
// helpers
|
|
|
|
function coerce(length) {
|
|
// Coerce length to a number (possibly NaN), round up
|
|
// in case it's fractional (e.g. 123.456) then do a
|
|
// double negate to coerce a NaN to 0. Easy, right?
|
|
length = ~~Math.ceil(+length);
|
|
return length < 0 ? 0 : length;
|
|
}
|
|
|
|
function isArray(subject) {
|
|
return (Array.isArray ||
|
|
function(subject){
|
|
return {}.toString.apply(subject) == '[object Array]'
|
|
})
|
|
(subject)
|
|
}
|
|
|
|
function isArrayIsh(subject) {
|
|
return isArray(subject) || Buffer.isBuffer(subject) ||
|
|
subject && typeof subject === 'object' &&
|
|
typeof subject.length === 'number';
|
|
}
|
|
|
|
function toHex(n) {
|
|
if (n < 16) return '0' + n.toString(16);
|
|
return n.toString(16);
|
|
}
|
|
|
|
function utf8ToBytes(str) {
|
|
var byteArray = [];
|
|
for (var i = 0; i < str.length; i++)
|
|
if (str.charCodeAt(i) <= 0x7F)
|
|
byteArray.push(str.charCodeAt(i));
|
|
else {
|
|
var h = encodeURIComponent(str.charAt(i)).substr(1).split('%');
|
|
for (var j = 0; j < h.length; j++)
|
|
byteArray.push(parseInt(h[j], 16));
|
|
}
|
|
|
|
return byteArray;
|
|
}
|
|
|
|
function asciiToBytes(str) {
|
|
var byteArray = []
|
|
for (var i = 0; i < str.length; i++ )
|
|
// Node's code seems to be doing this and not & 0x7F..
|
|
byteArray.push( str.charCodeAt(i) & 0xFF );
|
|
|
|
return byteArray;
|
|
}
|
|
|
|
function base64ToBytes(str) {
|
|
return require("base64-js").toByteArray(str);
|
|
}
|
|
|
|
function blitBuffer(src, dst, offset, length) {
|
|
var pos, i = 0;
|
|
while (i < length) {
|
|
if ((i+offset >= dst.length) || (i >= src.length))
|
|
break;
|
|
|
|
dst[i + offset] = src[i];
|
|
i++;
|
|
}
|
|
return i;
|
|
}
|
|
|
|
function decodeUtf8Char(str) {
|
|
try {
|
|
return decodeURIComponent(str);
|
|
} catch (err) {
|
|
return String.fromCharCode(0xFFFD); // UTF 8 invalid char
|
|
}
|
|
}
|
|
|
|
// read/write bit-twiddling
|
|
|
|
Buffer.prototype.readUInt8 = function(offset, noAssert) {
|
|
var buffer = this;
|
|
|
|
if (!noAssert) {
|
|
assert.ok(offset !== undefined && offset !== null,
|
|
'missing offset');
|
|
|
|
assert.ok(offset < buffer.length,
|
|
'Trying to read beyond buffer length');
|
|
}
|
|
|
|
if (offset >= buffer.length) return;
|
|
|
|
return buffer[offset];
|
|
};
|
|
|
|
function readUInt16(buffer, offset, isBigEndian, noAssert) {
|
|
var val = 0;
|
|
|
|
|
|
if (!noAssert) {
|
|
assert.ok(typeof (isBigEndian) === 'boolean',
|
|
'missing or invalid endian');
|
|
|
|
assert.ok(offset !== undefined && offset !== null,
|
|
'missing offset');
|
|
|
|
assert.ok(offset + 1 < buffer.length,
|
|
'Trying to read beyond buffer length');
|
|
}
|
|
|
|
if (offset >= buffer.length) return 0;
|
|
|
|
if (isBigEndian) {
|
|
val = buffer[offset] << 8;
|
|
if (offset + 1 < buffer.length) {
|
|
val |= buffer[offset + 1];
|
|
}
|
|
} else {
|
|
val = buffer[offset];
|
|
if (offset + 1 < buffer.length) {
|
|
val |= buffer[offset + 1] << 8;
|
|
}
|
|
}
|
|
|
|
return val;
|
|
}
|
|
|
|
Buffer.prototype.readUInt16LE = function(offset, noAssert) {
|
|
return readUInt16(this, offset, false, noAssert);
|
|
};
|
|
|
|
Buffer.prototype.readUInt16BE = function(offset, noAssert) {
|
|
return readUInt16(this, offset, true, noAssert);
|
|
};
|
|
|
|
function readUInt32(buffer, offset, isBigEndian, noAssert) {
|
|
var val = 0;
|
|
|
|
if (!noAssert) {
|
|
assert.ok(typeof (isBigEndian) === 'boolean',
|
|
'missing or invalid endian');
|
|
|
|
assert.ok(offset !== undefined && offset !== null,
|
|
'missing offset');
|
|
|
|
assert.ok(offset + 3 < buffer.length,
|
|
'Trying to read beyond buffer length');
|
|
}
|
|
|
|
if (offset >= buffer.length) return 0;
|
|
|
|
if (isBigEndian) {
|
|
if (offset + 1 < buffer.length)
|
|
val = buffer[offset + 1] << 16;
|
|
if (offset + 2 < buffer.length)
|
|
val |= buffer[offset + 2] << 8;
|
|
if (offset + 3 < buffer.length)
|
|
val |= buffer[offset + 3];
|
|
val = val + (buffer[offset] << 24 >>> 0);
|
|
} else {
|
|
if (offset + 2 < buffer.length)
|
|
val = buffer[offset + 2] << 16;
|
|
if (offset + 1 < buffer.length)
|
|
val |= buffer[offset + 1] << 8;
|
|
val |= buffer[offset];
|
|
if (offset + 3 < buffer.length)
|
|
val = val + (buffer[offset + 3] << 24 >>> 0);
|
|
}
|
|
|
|
return val;
|
|
}
|
|
|
|
Buffer.prototype.readUInt32LE = function(offset, noAssert) {
|
|
return readUInt32(this, offset, false, noAssert);
|
|
};
|
|
|
|
Buffer.prototype.readUInt32BE = function(offset, noAssert) {
|
|
return readUInt32(this, offset, true, noAssert);
|
|
};
|
|
|
|
|
|
/*
|
|
* Signed integer types, yay team! A reminder on how two's complement actually
|
|
* works. The first bit is the signed bit, i.e. tells us whether or not the
|
|
* number should be positive or negative. If the two's complement value is
|
|
* positive, then we're done, as it's equivalent to the unsigned representation.
|
|
*
|
|
* Now if the number is positive, you're pretty much done, you can just leverage
|
|
* the unsigned translations and return those. Unfortunately, negative numbers
|
|
* aren't quite that straightforward.
|
|
*
|
|
* At first glance, one might be inclined to use the traditional formula to
|
|
* translate binary numbers between the positive and negative values in two's
|
|
* complement. (Though it doesn't quite work for the most negative value)
|
|
* Mainly:
|
|
* - invert all the bits
|
|
* - add one to the result
|
|
*
|
|
* Of course, this doesn't quite work in Javascript. Take for example the value
|
|
* of -128. This could be represented in 16 bits (big-endian) as 0xff80. But of
|
|
* course, Javascript will do the following:
|
|
*
|
|
* > ~0xff80
|
|
* -65409
|
|
*
|
|
* Whoh there, Javascript, that's not quite right. But wait, according to
|
|
* Javascript that's perfectly correct. When Javascript ends up seeing the
|
|
* constant 0xff80, it has no notion that it is actually a signed number. It
|
|
* assumes that we've input the unsigned value 0xff80. Thus, when it does the
|
|
* binary negation, it casts it into a signed value, (positive 0xff80). Then
|
|
* when you perform binary negation on that, it turns it into a negative number.
|
|
*
|
|
* Instead, we're going to have to use the following general formula, that works
|
|
* in a rather Javascript friendly way. I'm glad we don't support this kind of
|
|
* weird numbering scheme in the kernel.
|
|
*
|
|
* (BIT-MAX - (unsigned)val + 1) * -1
|
|
*
|
|
* The astute observer, may think that this doesn't make sense for 8-bit numbers
|
|
* (really it isn't necessary for them). However, when you get 16-bit numbers,
|
|
* you do. Let's go back to our prior example and see how this will look:
|
|
*
|
|
* (0xffff - 0xff80 + 1) * -1
|
|
* (0x007f + 1) * -1
|
|
* (0x0080) * -1
|
|
*/
|
|
Buffer.prototype.readInt8 = function(offset, noAssert) {
|
|
var buffer = this;
|
|
var neg;
|
|
|
|
if (!noAssert) {
|
|
assert.ok(offset !== undefined && offset !== null,
|
|
'missing offset');
|
|
|
|
assert.ok(offset < buffer.length,
|
|
'Trying to read beyond buffer length');
|
|
}
|
|
|
|
if (offset >= buffer.length) return;
|
|
|
|
neg = buffer[offset] & 0x80;
|
|
if (!neg) {
|
|
return (buffer[offset]);
|
|
}
|
|
|
|
return ((0xff - buffer[offset] + 1) * -1);
|
|
};
|
|
|
|
function readInt16(buffer, offset, isBigEndian, noAssert) {
|
|
var neg, val;
|
|
|
|
if (!noAssert) {
|
|
assert.ok(typeof (isBigEndian) === 'boolean',
|
|
'missing or invalid endian');
|
|
|
|
assert.ok(offset !== undefined && offset !== null,
|
|
'missing offset');
|
|
|
|
assert.ok(offset + 1 < buffer.length,
|
|
'Trying to read beyond buffer length');
|
|
}
|
|
|
|
val = readUInt16(buffer, offset, isBigEndian, noAssert);
|
|
neg = val & 0x8000;
|
|
if (!neg) {
|
|
return val;
|
|
}
|
|
|
|
return (0xffff - val + 1) * -1;
|
|
}
|
|
|
|
Buffer.prototype.readInt16LE = function(offset, noAssert) {
|
|
return readInt16(this, offset, false, noAssert);
|
|
};
|
|
|
|
Buffer.prototype.readInt16BE = function(offset, noAssert) {
|
|
return readInt16(this, offset, true, noAssert);
|
|
};
|
|
|
|
function readInt32(buffer, offset, isBigEndian, noAssert) {
|
|
var neg, val;
|
|
|
|
if (!noAssert) {
|
|
assert.ok(typeof (isBigEndian) === 'boolean',
|
|
'missing or invalid endian');
|
|
|
|
assert.ok(offset !== undefined && offset !== null,
|
|
'missing offset');
|
|
|
|
assert.ok(offset + 3 < buffer.length,
|
|
'Trying to read beyond buffer length');
|
|
}
|
|
|
|
val = readUInt32(buffer, offset, isBigEndian, noAssert);
|
|
neg = val & 0x80000000;
|
|
if (!neg) {
|
|
return (val);
|
|
}
|
|
|
|
return (0xffffffff - val + 1) * -1;
|
|
}
|
|
|
|
Buffer.prototype.readInt32LE = function(offset, noAssert) {
|
|
return readInt32(this, offset, false, noAssert);
|
|
};
|
|
|
|
Buffer.prototype.readInt32BE = function(offset, noAssert) {
|
|
return readInt32(this, offset, true, noAssert);
|
|
};
|
|
|
|
function readFloat(buffer, offset, isBigEndian, noAssert) {
|
|
if (!noAssert) {
|
|
assert.ok(typeof (isBigEndian) === 'boolean',
|
|
'missing or invalid endian');
|
|
|
|
assert.ok(offset + 3 < buffer.length,
|
|
'Trying to read beyond buffer length');
|
|
}
|
|
|
|
return require('./buffer_ieee754').readIEEE754(buffer, offset, isBigEndian,
|
|
23, 4);
|
|
}
|
|
|
|
Buffer.prototype.readFloatLE = function(offset, noAssert) {
|
|
return readFloat(this, offset, false, noAssert);
|
|
};
|
|
|
|
Buffer.prototype.readFloatBE = function(offset, noAssert) {
|
|
return readFloat(this, offset, true, noAssert);
|
|
};
|
|
|
|
function readDouble(buffer, offset, isBigEndian, noAssert) {
|
|
if (!noAssert) {
|
|
assert.ok(typeof (isBigEndian) === 'boolean',
|
|
'missing or invalid endian');
|
|
|
|
assert.ok(offset + 7 < buffer.length,
|
|
'Trying to read beyond buffer length');
|
|
}
|
|
|
|
return require('./buffer_ieee754').readIEEE754(buffer, offset, isBigEndian,
|
|
52, 8);
|
|
}
|
|
|
|
Buffer.prototype.readDoubleLE = function(offset, noAssert) {
|
|
return readDouble(this, offset, false, noAssert);
|
|
};
|
|
|
|
Buffer.prototype.readDoubleBE = function(offset, noAssert) {
|
|
return readDouble(this, offset, true, noAssert);
|
|
};
|
|
|
|
|
|
/*
|
|
* We have to make sure that the value is a valid integer. This means that it is
|
|
* non-negative. It has no fractional component and that it does not exceed the
|
|
* maximum allowed value.
|
|
*
|
|
* value The number to check for validity
|
|
*
|
|
* max The maximum value
|
|
*/
|
|
function verifuint(value, max) {
|
|
assert.ok(typeof (value) == 'number',
|
|
'cannot write a non-number as a number');
|
|
|
|
assert.ok(value >= 0,
|
|
'specified a negative value for writing an unsigned value');
|
|
|
|
assert.ok(value <= max, 'value is larger than maximum value for type');
|
|
|
|
assert.ok(Math.floor(value) === value, 'value has a fractional component');
|
|
}
|
|
|
|
Buffer.prototype.writeUInt8 = function(value, offset, noAssert) {
|
|
var buffer = this;
|
|
|
|
if (!noAssert) {
|
|
assert.ok(value !== undefined && value !== null,
|
|
'missing value');
|
|
|
|
assert.ok(offset !== undefined && offset !== null,
|
|
'missing offset');
|
|
|
|
assert.ok(offset < buffer.length,
|
|
'trying to write beyond buffer length');
|
|
|
|
verifuint(value, 0xff);
|
|
}
|
|
|
|
if (offset < buffer.length) {
|
|
buffer[offset] = value;
|
|
}
|
|
};
|
|
|
|
function writeUInt16(buffer, value, offset, isBigEndian, noAssert) {
|
|
if (!noAssert) {
|
|
assert.ok(value !== undefined && value !== null,
|
|
'missing value');
|
|
|
|
assert.ok(typeof (isBigEndian) === 'boolean',
|
|
'missing or invalid endian');
|
|
|
|
assert.ok(offset !== undefined && offset !== null,
|
|
'missing offset');
|
|
|
|
assert.ok(offset + 1 < buffer.length,
|
|
'trying to write beyond buffer length');
|
|
|
|
verifuint(value, 0xffff);
|
|
}
|
|
|
|
for (var i = 0; i < Math.min(buffer.length - offset, 2); i++) {
|
|
buffer[offset + i] =
|
|
(value & (0xff << (8 * (isBigEndian ? 1 - i : i)))) >>>
|
|
(isBigEndian ? 1 - i : i) * 8;
|
|
}
|
|
|
|
}
|
|
|
|
Buffer.prototype.writeUInt16LE = function(value, offset, noAssert) {
|
|
writeUInt16(this, value, offset, false, noAssert);
|
|
};
|
|
|
|
Buffer.prototype.writeUInt16BE = function(value, offset, noAssert) {
|
|
writeUInt16(this, value, offset, true, noAssert);
|
|
};
|
|
|
|
function writeUInt32(buffer, value, offset, isBigEndian, noAssert) {
|
|
if (!noAssert) {
|
|
assert.ok(value !== undefined && value !== null,
|
|
'missing value');
|
|
|
|
assert.ok(typeof (isBigEndian) === 'boolean',
|
|
'missing or invalid endian');
|
|
|
|
assert.ok(offset !== undefined && offset !== null,
|
|
'missing offset');
|
|
|
|
assert.ok(offset + 3 < buffer.length,
|
|
'trying to write beyond buffer length');
|
|
|
|
verifuint(value, 0xffffffff);
|
|
}
|
|
|
|
for (var i = 0; i < Math.min(buffer.length - offset, 4); i++) {
|
|
buffer[offset + i] =
|
|
(value >>> (isBigEndian ? 3 - i : i) * 8) & 0xff;
|
|
}
|
|
}
|
|
|
|
Buffer.prototype.writeUInt32LE = function(value, offset, noAssert) {
|
|
writeUInt32(this, value, offset, false, noAssert);
|
|
};
|
|
|
|
Buffer.prototype.writeUInt32BE = function(value, offset, noAssert) {
|
|
writeUInt32(this, value, offset, true, noAssert);
|
|
};
|
|
|
|
|
|
/*
|
|
* We now move onto our friends in the signed number category. Unlike unsigned
|
|
* numbers, we're going to have to worry a bit more about how we put values into
|
|
* arrays. Since we are only worrying about signed 32-bit values, we're in
|
|
* slightly better shape. Unfortunately, we really can't do our favorite binary
|
|
* & in this system. It really seems to do the wrong thing. For example:
|
|
*
|
|
* > -32 & 0xff
|
|
* 224
|
|
*
|
|
* What's happening above is really: 0xe0 & 0xff = 0xe0. However, the results of
|
|
* this aren't treated as a signed number. Ultimately a bad thing.
|
|
*
|
|
* What we're going to want to do is basically create the unsigned equivalent of
|
|
* our representation and pass that off to the wuint* functions. To do that
|
|
* we're going to do the following:
|
|
*
|
|
* - if the value is positive
|
|
* we can pass it directly off to the equivalent wuint
|
|
* - if the value is negative
|
|
* we do the following computation:
|
|
* mb + val + 1, where
|
|
* mb is the maximum unsigned value in that byte size
|
|
* val is the Javascript negative integer
|
|
*
|
|
*
|
|
* As a concrete value, take -128. In signed 16 bits this would be 0xff80. If
|
|
* you do out the computations:
|
|
*
|
|
* 0xffff - 128 + 1
|
|
* 0xffff - 127
|
|
* 0xff80
|
|
*
|
|
* You can then encode this value as the signed version. This is really rather
|
|
* hacky, but it should work and get the job done which is our goal here.
|
|
*/
|
|
|
|
/*
|
|
* A series of checks to make sure we actually have a signed 32-bit number
|
|
*/
|
|
function verifsint(value, max, min) {
|
|
assert.ok(typeof (value) == 'number',
|
|
'cannot write a non-number as a number');
|
|
|
|
assert.ok(value <= max, 'value larger than maximum allowed value');
|
|
|
|
assert.ok(value >= min, 'value smaller than minimum allowed value');
|
|
|
|
assert.ok(Math.floor(value) === value, 'value has a fractional component');
|
|
}
|
|
|
|
function verifIEEE754(value, max, min) {
|
|
assert.ok(typeof (value) == 'number',
|
|
'cannot write a non-number as a number');
|
|
|
|
assert.ok(value <= max, 'value larger than maximum allowed value');
|
|
|
|
assert.ok(value >= min, 'value smaller than minimum allowed value');
|
|
}
|
|
|
|
Buffer.prototype.writeInt8 = function(value, offset, noAssert) {
|
|
var buffer = this;
|
|
|
|
if (!noAssert) {
|
|
assert.ok(value !== undefined && value !== null,
|
|
'missing value');
|
|
|
|
assert.ok(offset !== undefined && offset !== null,
|
|
'missing offset');
|
|
|
|
assert.ok(offset < buffer.length,
|
|
'Trying to write beyond buffer length');
|
|
|
|
verifsint(value, 0x7f, -0x80);
|
|
}
|
|
|
|
if (value >= 0) {
|
|
buffer.writeUInt8(value, offset, noAssert);
|
|
} else {
|
|
buffer.writeUInt8(0xff + value + 1, offset, noAssert);
|
|
}
|
|
};
|
|
|
|
function writeInt16(buffer, value, offset, isBigEndian, noAssert) {
|
|
if (!noAssert) {
|
|
assert.ok(value !== undefined && value !== null,
|
|
'missing value');
|
|
|
|
assert.ok(typeof (isBigEndian) === 'boolean',
|
|
'missing or invalid endian');
|
|
|
|
assert.ok(offset !== undefined && offset !== null,
|
|
'missing offset');
|
|
|
|
assert.ok(offset + 1 < buffer.length,
|
|
'Trying to write beyond buffer length');
|
|
|
|
verifsint(value, 0x7fff, -0x8000);
|
|
}
|
|
|
|
if (value >= 0) {
|
|
writeUInt16(buffer, value, offset, isBigEndian, noAssert);
|
|
} else {
|
|
writeUInt16(buffer, 0xffff + value + 1, offset, isBigEndian, noAssert);
|
|
}
|
|
}
|
|
|
|
Buffer.prototype.writeInt16LE = function(value, offset, noAssert) {
|
|
writeInt16(this, value, offset, false, noAssert);
|
|
};
|
|
|
|
Buffer.prototype.writeInt16BE = function(value, offset, noAssert) {
|
|
writeInt16(this, value, offset, true, noAssert);
|
|
};
|
|
|
|
function writeInt32(buffer, value, offset, isBigEndian, noAssert) {
|
|
if (!noAssert) {
|
|
assert.ok(value !== undefined && value !== null,
|
|
'missing value');
|
|
|
|
assert.ok(typeof (isBigEndian) === 'boolean',
|
|
'missing or invalid endian');
|
|
|
|
assert.ok(offset !== undefined && offset !== null,
|
|
'missing offset');
|
|
|
|
assert.ok(offset + 3 < buffer.length,
|
|
'Trying to write beyond buffer length');
|
|
|
|
verifsint(value, 0x7fffffff, -0x80000000);
|
|
}
|
|
|
|
if (value >= 0) {
|
|
writeUInt32(buffer, value, offset, isBigEndian, noAssert);
|
|
} else {
|
|
writeUInt32(buffer, 0xffffffff + value + 1, offset, isBigEndian, noAssert);
|
|
}
|
|
}
|
|
|
|
Buffer.prototype.writeInt32LE = function(value, offset, noAssert) {
|
|
writeInt32(this, value, offset, false, noAssert);
|
|
};
|
|
|
|
Buffer.prototype.writeInt32BE = function(value, offset, noAssert) {
|
|
writeInt32(this, value, offset, true, noAssert);
|
|
};
|
|
|
|
function writeFloat(buffer, value, offset, isBigEndian, noAssert) {
|
|
if (!noAssert) {
|
|
assert.ok(value !== undefined && value !== null,
|
|
'missing value');
|
|
|
|
assert.ok(typeof (isBigEndian) === 'boolean',
|
|
'missing or invalid endian');
|
|
|
|
assert.ok(offset !== undefined && offset !== null,
|
|
'missing offset');
|
|
|
|
assert.ok(offset + 3 < buffer.length,
|
|
'Trying to write beyond buffer length');
|
|
|
|
verifIEEE754(value, 3.4028234663852886e+38, -3.4028234663852886e+38);
|
|
}
|
|
|
|
require('./buffer_ieee754').writeIEEE754(buffer, value, offset, isBigEndian,
|
|
23, 4);
|
|
}
|
|
|
|
Buffer.prototype.writeFloatLE = function(value, offset, noAssert) {
|
|
writeFloat(this, value, offset, false, noAssert);
|
|
};
|
|
|
|
Buffer.prototype.writeFloatBE = function(value, offset, noAssert) {
|
|
writeFloat(this, value, offset, true, noAssert);
|
|
};
|
|
|
|
function writeDouble(buffer, value, offset, isBigEndian, noAssert) {
|
|
if (!noAssert) {
|
|
assert.ok(value !== undefined && value !== null,
|
|
'missing value');
|
|
|
|
assert.ok(typeof (isBigEndian) === 'boolean',
|
|
'missing or invalid endian');
|
|
|
|
assert.ok(offset !== undefined && offset !== null,
|
|
'missing offset');
|
|
|
|
assert.ok(offset + 7 < buffer.length,
|
|
'Trying to write beyond buffer length');
|
|
|
|
verifIEEE754(value, 1.7976931348623157E+308, -1.7976931348623157E+308);
|
|
}
|
|
|
|
require('./buffer_ieee754').writeIEEE754(buffer, value, offset, isBigEndian,
|
|
52, 8);
|
|
}
|
|
|
|
Buffer.prototype.writeDoubleLE = function(value, offset, noAssert) {
|
|
writeDouble(this, value, offset, false, noAssert);
|
|
};
|
|
|
|
Buffer.prototype.writeDoubleBE = function(value, offset, noAssert) {
|
|
writeDouble(this, value, offset, true, noAssert);
|
|
};
|
|
|
|
},{"./buffer_ieee754":11,"assert":3,"base64-js":13}],13:[function(require,module,exports){
|
|
(function (exports) {
|
|
'use strict';
|
|
|
|
var lookup = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/';
|
|
|
|
function b64ToByteArray(b64) {
|
|
var i, j, l, tmp, placeHolders, arr;
|
|
|
|
if (b64.length % 4 > 0) {
|
|
throw 'Invalid string. Length must be a multiple of 4';
|
|
}
|
|
|
|
// the number of equal signs (place holders)
|
|
// if there are two placeholders, than the two characters before it
|
|
// represent one byte
|
|
// if there is only one, then the three characters before it represent 2 bytes
|
|
// this is just a cheap hack to not do indexOf twice
|
|
placeHolders = b64.indexOf('=');
|
|
placeHolders = placeHolders > 0 ? b64.length - placeHolders : 0;
|
|
|
|
// base64 is 4/3 + up to two characters of the original data
|
|
arr = [];//new Uint8Array(b64.length * 3 / 4 - placeHolders);
|
|
|
|
// if there are placeholders, only get up to the last complete 4 chars
|
|
l = placeHolders > 0 ? b64.length - 4 : b64.length;
|
|
|
|
for (i = 0, j = 0; i < l; i += 4, j += 3) {
|
|
tmp = (lookup.indexOf(b64[i]) << 18) | (lookup.indexOf(b64[i + 1]) << 12) | (lookup.indexOf(b64[i + 2]) << 6) | lookup.indexOf(b64[i + 3]);
|
|
arr.push((tmp & 0xFF0000) >> 16);
|
|
arr.push((tmp & 0xFF00) >> 8);
|
|
arr.push(tmp & 0xFF);
|
|
}
|
|
|
|
if (placeHolders === 2) {
|
|
tmp = (lookup.indexOf(b64[i]) << 2) | (lookup.indexOf(b64[i + 1]) >> 4);
|
|
arr.push(tmp & 0xFF);
|
|
} else if (placeHolders === 1) {
|
|
tmp = (lookup.indexOf(b64[i]) << 10) | (lookup.indexOf(b64[i + 1]) << 4) | (lookup.indexOf(b64[i + 2]) >> 2);
|
|
arr.push((tmp >> 8) & 0xFF);
|
|
arr.push(tmp & 0xFF);
|
|
}
|
|
|
|
return arr;
|
|
}
|
|
|
|
function uint8ToBase64(uint8) {
|
|
var i,
|
|
extraBytes = uint8.length % 3, // if we have 1 byte left, pad 2 bytes
|
|
output = "",
|
|
temp, length;
|
|
|
|
function tripletToBase64 (num) {
|
|
return lookup[num >> 18 & 0x3F] + lookup[num >> 12 & 0x3F] + lookup[num >> 6 & 0x3F] + lookup[num & 0x3F];
|
|
};
|
|
|
|
// go through the array every three bytes, we'll deal with trailing stuff later
|
|
for (i = 0, length = uint8.length - extraBytes; i < length; i += 3) {
|
|
temp = (uint8[i] << 16) + (uint8[i + 1] << 8) + (uint8[i + 2]);
|
|
output += tripletToBase64(temp);
|
|
}
|
|
|
|
// pad the end with zeros, but make sure to not forget the extra bytes
|
|
switch (extraBytes) {
|
|
case 1:
|
|
temp = uint8[uint8.length - 1];
|
|
output += lookup[temp >> 2];
|
|
output += lookup[(temp << 4) & 0x3F];
|
|
output += '==';
|
|
break;
|
|
case 2:
|
|
temp = (uint8[uint8.length - 2] << 8) + (uint8[uint8.length - 1]);
|
|
output += lookup[temp >> 10];
|
|
output += lookup[(temp >> 4) & 0x3F];
|
|
output += lookup[(temp << 2) & 0x3F];
|
|
output += '=';
|
|
break;
|
|
}
|
|
|
|
return output;
|
|
}
|
|
|
|
module.exports.toByteArray = b64ToByteArray;
|
|
module.exports.fromByteArray = uint8ToBase64;
|
|
}());
|
|
|
|
},{}],14:[function(require,module,exports){
|
|
require=(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);throw new Error("Cannot find module '"+o+"'")}var f=n[o]={exports:{}};t[o][0].call(f.exports,function(e){var n=t[o][1][e];return s(n?n:e)},f,f.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})({1:[function(require,module,exports){
|
|
exports.readIEEE754 = function(buffer, offset, isBE, mLen, nBytes) {
|
|
var e, m,
|
|
eLen = nBytes * 8 - mLen - 1,
|
|
eMax = (1 << eLen) - 1,
|
|
eBias = eMax >> 1,
|
|
nBits = -7,
|
|
i = isBE ? 0 : (nBytes - 1),
|
|
d = isBE ? 1 : -1,
|
|
s = buffer[offset + i];
|
|
|
|
i += d;
|
|
|
|
e = s & ((1 << (-nBits)) - 1);
|
|
s >>= (-nBits);
|
|
nBits += eLen;
|
|
for (; nBits > 0; e = e * 256 + buffer[offset + i], i += d, nBits -= 8);
|
|
|
|
m = e & ((1 << (-nBits)) - 1);
|
|
e >>= (-nBits);
|
|
nBits += mLen;
|
|
for (; nBits > 0; m = m * 256 + buffer[offset + i], i += d, nBits -= 8);
|
|
|
|
if (e === 0) {
|
|
e = 1 - eBias;
|
|
} else if (e === eMax) {
|
|
return m ? NaN : ((s ? -1 : 1) * Infinity);
|
|
} else {
|
|
m = m + Math.pow(2, mLen);
|
|
e = e - eBias;
|
|
}
|
|
return (s ? -1 : 1) * m * Math.pow(2, e - mLen);
|
|
};
|
|
|
|
exports.writeIEEE754 = function(buffer, value, offset, isBE, mLen, nBytes) {
|
|
var e, m, c,
|
|
eLen = nBytes * 8 - mLen - 1,
|
|
eMax = (1 << eLen) - 1,
|
|
eBias = eMax >> 1,
|
|
rt = (mLen === 23 ? Math.pow(2, -24) - Math.pow(2, -77) : 0),
|
|
i = isBE ? (nBytes - 1) : 0,
|
|
d = isBE ? -1 : 1,
|
|
s = value < 0 || (value === 0 && 1 / value < 0) ? 1 : 0;
|
|
|
|
value = Math.abs(value);
|
|
|
|
if (isNaN(value) || value === Infinity) {
|
|
m = isNaN(value) ? 1 : 0;
|
|
e = eMax;
|
|
} else {
|
|
e = Math.floor(Math.log(value) / Math.LN2);
|
|
if (value * (c = Math.pow(2, -e)) < 1) {
|
|
e--;
|
|
c *= 2;
|
|
}
|
|
if (e + eBias >= 1) {
|
|
value += rt / c;
|
|
} else {
|
|
value += rt * Math.pow(2, 1 - eBias);
|
|
}
|
|
if (value * c >= 2) {
|
|
e++;
|
|
c /= 2;
|
|
}
|
|
|
|
if (e + eBias >= eMax) {
|
|
m = 0;
|
|
e = eMax;
|
|
} else if (e + eBias >= 1) {
|
|
m = (value * c - 1) * Math.pow(2, mLen);
|
|
e = e + eBias;
|
|
} else {
|
|
m = value * Math.pow(2, eBias - 1) * Math.pow(2, mLen);
|
|
e = 0;
|
|
}
|
|
}
|
|
|
|
for (; mLen >= 8; buffer[offset + i] = m & 0xff, i += d, m /= 256, mLen -= 8);
|
|
|
|
e = (e << mLen) | m;
|
|
eLen += mLen;
|
|
for (; eLen > 0; buffer[offset + i] = e & 0xff, i += d, e /= 256, eLen -= 8);
|
|
|
|
buffer[offset + i - d] |= s * 128;
|
|
};
|
|
|
|
},{}],"q9TxCC":[function(require,module,exports){
|
|
var assert;
|
|
exports.Buffer = Buffer;
|
|
exports.SlowBuffer = Buffer;
|
|
Buffer.poolSize = 8192;
|
|
exports.INSPECT_MAX_BYTES = 50;
|
|
|
|
function stringtrim(str) {
|
|
if (str.trim) return str.trim();
|
|
return str.replace(/^\s+|\s+$/g, '');
|
|
}
|
|
|
|
function Buffer(subject, encoding, offset) {
|
|
if(!assert) assert= require('assert');
|
|
if (!(this instanceof Buffer)) {
|
|
return new Buffer(subject, encoding, offset);
|
|
}
|
|
this.parent = this;
|
|
this.offset = 0;
|
|
|
|
// Work-around: node's base64 implementation
|
|
// allows for non-padded strings while base64-js
|
|
// does not..
|
|
if (encoding == "base64" && typeof subject == "string") {
|
|
subject = stringtrim(subject);
|
|
while (subject.length % 4 != 0) {
|
|
subject = subject + "=";
|
|
}
|
|
}
|
|
|
|
var type;
|
|
|
|
// Are we slicing?
|
|
if (typeof offset === 'number') {
|
|
this.length = coerce(encoding);
|
|
// slicing works, with limitations (no parent tracking/update)
|
|
// check https://github.com/toots/buffer-browserify/issues/19
|
|
for (var i = 0; i < this.length; i++) {
|
|
this[i] = subject.get(i+offset);
|
|
}
|
|
} else {
|
|
// Find the length
|
|
switch (type = typeof subject) {
|
|
case 'number':
|
|
this.length = coerce(subject);
|
|
break;
|
|
|
|
case 'string':
|
|
this.length = Buffer.byteLength(subject, encoding);
|
|
break;
|
|
|
|
case 'object': // Assume object is an array
|
|
this.length = coerce(subject.length);
|
|
break;
|
|
|
|
default:
|
|
throw new Error('First argument needs to be a number, ' +
|
|
'array or string.');
|
|
}
|
|
|
|
// Treat array-ish objects as a byte array.
|
|
if (isArrayIsh(subject)) {
|
|
for (var i = 0; i < this.length; i++) {
|
|
if (subject instanceof Buffer) {
|
|
this[i] = subject.readUInt8(i);
|
|
}
|
|
else {
|
|
this[i] = subject[i];
|
|
}
|
|
}
|
|
} else if (type == 'string') {
|
|
// We are a string
|
|
this.length = this.write(subject, 0, encoding);
|
|
} else if (type === 'number') {
|
|
for (var i = 0; i < this.length; i++) {
|
|
this[i] = 0;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
Buffer.prototype.get = function get(i) {
|
|
if (i < 0 || i >= this.length) throw new Error('oob');
|
|
return this[i];
|
|
};
|
|
|
|
Buffer.prototype.set = function set(i, v) {
|
|
if (i < 0 || i >= this.length) throw new Error('oob');
|
|
return this[i] = v;
|
|
};
|
|
|
|
Buffer.byteLength = function (str, encoding) {
|
|
switch (encoding || "utf8") {
|
|
case 'hex':
|
|
return str.length / 2;
|
|
|
|
case 'utf8':
|
|
case 'utf-8':
|
|
return utf8ToBytes(str).length;
|
|
|
|
case 'ascii':
|
|
case 'binary':
|
|
return str.length;
|
|
|
|
case 'base64':
|
|
return base64ToBytes(str).length;
|
|
|
|
default:
|
|
throw new Error('Unknown encoding');
|
|
}
|
|
};
|
|
|
|
Buffer.prototype.utf8Write = function (string, offset, length) {
|
|
var bytes, pos;
|
|
return Buffer._charsWritten = blitBuffer(utf8ToBytes(string), this, offset, length);
|
|
};
|
|
|
|
Buffer.prototype.asciiWrite = function (string, offset, length) {
|
|
var bytes, pos;
|
|
return Buffer._charsWritten = blitBuffer(asciiToBytes(string), this, offset, length);
|
|
};
|
|
|
|
Buffer.prototype.binaryWrite = Buffer.prototype.asciiWrite;
|
|
|
|
Buffer.prototype.base64Write = function (string, offset, length) {
|
|
var bytes, pos;
|
|
return Buffer._charsWritten = blitBuffer(base64ToBytes(string), this, offset, length);
|
|
};
|
|
|
|
Buffer.prototype.base64Slice = function (start, end) {
|
|
var bytes = Array.prototype.slice.apply(this, arguments)
|
|
return require("base64-js").fromByteArray(bytes);
|
|
};
|
|
|
|
Buffer.prototype.utf8Slice = function () {
|
|
var bytes = Array.prototype.slice.apply(this, arguments);
|
|
var res = "";
|
|
var tmp = "";
|
|
var i = 0;
|
|
while (i < bytes.length) {
|
|
if (bytes[i] <= 0x7F) {
|
|
res += decodeUtf8Char(tmp) + String.fromCharCode(bytes[i]);
|
|
tmp = "";
|
|
} else
|
|
tmp += "%" + bytes[i].toString(16);
|
|
|
|
i++;
|
|
}
|
|
|
|
return res + decodeUtf8Char(tmp);
|
|
}
|
|
|
|
Buffer.prototype.asciiSlice = function () {
|
|
var bytes = Array.prototype.slice.apply(this, arguments);
|
|
var ret = "";
|
|
for (var i = 0; i < bytes.length; i++)
|
|
ret += String.fromCharCode(bytes[i]);
|
|
return ret;
|
|
}
|
|
|
|
Buffer.prototype.binarySlice = Buffer.prototype.asciiSlice;
|
|
|
|
Buffer.prototype.inspect = function() {
|
|
var out = [],
|
|
len = this.length;
|
|
for (var i = 0; i < len; i++) {
|
|
out[i] = toHex(this[i]);
|
|
if (i == exports.INSPECT_MAX_BYTES) {
|
|
out[i + 1] = '...';
|
|
break;
|
|
}
|
|
}
|
|
return '<Buffer ' + out.join(' ') + '>';
|
|
};
|
|
|
|
|
|
Buffer.prototype.hexSlice = function(start, end) {
|
|
var len = this.length;
|
|
|
|
if (!start || start < 0) start = 0;
|
|
if (!end || end < 0 || end > len) end = len;
|
|
|
|
var out = '';
|
|
for (var i = start; i < end; i++) {
|
|
out += toHex(this[i]);
|
|
}
|
|
return out;
|
|
};
|
|
|
|
|
|
Buffer.prototype.toString = function(encoding, start, end) {
|
|
encoding = String(encoding || 'utf8').toLowerCase();
|
|
start = +start || 0;
|
|
if (typeof end == 'undefined') end = this.length;
|
|
|
|
// Fastpath empty strings
|
|
if (+end == start) {
|
|
return '';
|
|
}
|
|
|
|
switch (encoding) {
|
|
case 'hex':
|
|
return this.hexSlice(start, end);
|
|
|
|
case 'utf8':
|
|
case 'utf-8':
|
|
return this.utf8Slice(start, end);
|
|
|
|
case 'ascii':
|
|
return this.asciiSlice(start, end);
|
|
|
|
case 'binary':
|
|
return this.binarySlice(start, end);
|
|
|
|
case 'base64':
|
|
return this.base64Slice(start, end);
|
|
|
|
case 'ucs2':
|
|
case 'ucs-2':
|
|
return this.ucs2Slice(start, end);
|
|
|
|
default:
|
|
throw new Error('Unknown encoding');
|
|
}
|
|
};
|
|
|
|
|
|
Buffer.prototype.hexWrite = function(string, offset, length) {
|
|
offset = +offset || 0;
|
|
var remaining = this.length - offset;
|
|
if (!length) {
|
|
length = remaining;
|
|
} else {
|
|
length = +length;
|
|
if (length > remaining) {
|
|
length = remaining;
|
|
}
|
|
}
|
|
|
|
// must be an even number of digits
|
|
var strLen = string.length;
|
|
if (strLen % 2) {
|
|
throw new Error('Invalid hex string');
|
|
}
|
|
if (length > strLen / 2) {
|
|
length = strLen / 2;
|
|
}
|
|
for (var i = 0; i < length; i++) {
|
|
var byte = parseInt(string.substr(i * 2, 2), 16);
|
|
if (isNaN(byte)) throw new Error('Invalid hex string');
|
|
this[offset + i] = byte;
|
|
}
|
|
Buffer._charsWritten = i * 2;
|
|
return i;
|
|
};
|
|
|
|
|
|
Buffer.prototype.write = function(string, offset, length, encoding) {
|
|
// Support both (string, offset, length, encoding)
|
|
// and the legacy (string, encoding, offset, length)
|
|
if (isFinite(offset)) {
|
|
if (!isFinite(length)) {
|
|
encoding = length;
|
|
length = undefined;
|
|
}
|
|
} else { // legacy
|
|
var swap = encoding;
|
|
encoding = offset;
|
|
offset = length;
|
|
length = swap;
|
|
}
|
|
|
|
offset = +offset || 0;
|
|
var remaining = this.length - offset;
|
|
if (!length) {
|
|
length = remaining;
|
|
} else {
|
|
length = +length;
|
|
if (length > remaining) {
|
|
length = remaining;
|
|
}
|
|
}
|
|
encoding = String(encoding || 'utf8').toLowerCase();
|
|
|
|
switch (encoding) {
|
|
case 'hex':
|
|
return this.hexWrite(string, offset, length);
|
|
|
|
case 'utf8':
|
|
case 'utf-8':
|
|
return this.utf8Write(string, offset, length);
|
|
|
|
case 'ascii':
|
|
return this.asciiWrite(string, offset, length);
|
|
|
|
case 'binary':
|
|
return this.binaryWrite(string, offset, length);
|
|
|
|
case 'base64':
|
|
return this.base64Write(string, offset, length);
|
|
|
|
case 'ucs2':
|
|
case 'ucs-2':
|
|
return this.ucs2Write(string, offset, length);
|
|
|
|
default:
|
|
throw new Error('Unknown encoding');
|
|
}
|
|
};
|
|
|
|
// slice(start, end)
|
|
function clamp(index, len, defaultValue) {
|
|
if (typeof index !== 'number') return defaultValue;
|
|
index = ~~index; // Coerce to integer.
|
|
if (index >= len) return len;
|
|
if (index >= 0) return index;
|
|
index += len;
|
|
if (index >= 0) return index;
|
|
return 0;
|
|
}
|
|
|
|
Buffer.prototype.slice = function(start, end) {
|
|
var len = this.length;
|
|
start = clamp(start, len, 0);
|
|
end = clamp(end, len, len);
|
|
return new Buffer(this, end - start, +start);
|
|
};
|
|
|
|
// copy(targetBuffer, targetStart=0, sourceStart=0, sourceEnd=buffer.length)
|
|
Buffer.prototype.copy = function(target, target_start, start, end) {
|
|
var source = this;
|
|
start || (start = 0);
|
|
if (end === undefined || isNaN(end)) {
|
|
end = this.length;
|
|
}
|
|
target_start || (target_start = 0);
|
|
|
|
if (end < start) throw new Error('sourceEnd < sourceStart');
|
|
|
|
// Copy 0 bytes; we're done
|
|
if (end === start) return 0;
|
|
if (target.length == 0 || source.length == 0) return 0;
|
|
|
|
if (target_start < 0 || target_start >= target.length) {
|
|
throw new Error('targetStart out of bounds');
|
|
}
|
|
|
|
if (start < 0 || start >= source.length) {
|
|
throw new Error('sourceStart out of bounds');
|
|
}
|
|
|
|
if (end < 0 || end > source.length) {
|
|
throw new Error('sourceEnd out of bounds');
|
|
}
|
|
|
|
// Are we oob?
|
|
if (end > this.length) {
|
|
end = this.length;
|
|
}
|
|
|
|
if (target.length - target_start < end - start) {
|
|
end = target.length - target_start + start;
|
|
}
|
|
|
|
var temp = [];
|
|
for (var i=start; i<end; i++) {
|
|
assert.ok(typeof this[i] !== 'undefined', "copying undefined buffer bytes!");
|
|
temp.push(this[i]);
|
|
}
|
|
|
|
for (var i=target_start; i<target_start+temp.length; i++) {
|
|
target[i] = temp[i-target_start];
|
|
}
|
|
};
|
|
|
|
// fill(value, start=0, end=buffer.length)
|
|
Buffer.prototype.fill = function fill(value, start, end) {
|
|
value || (value = 0);
|
|
start || (start = 0);
|
|
end || (end = this.length);
|
|
|
|
if (typeof value === 'string') {
|
|
value = value.charCodeAt(0);
|
|
}
|
|
if (!(typeof value === 'number') || isNaN(value)) {
|
|
throw new Error('value is not a number');
|
|
}
|
|
|
|
if (end < start) throw new Error('end < start');
|
|
|
|
// Fill 0 bytes; we're done
|
|
if (end === start) return 0;
|
|
if (this.length == 0) return 0;
|
|
|
|
if (start < 0 || start >= this.length) {
|
|
throw new Error('start out of bounds');
|
|
}
|
|
|
|
if (end < 0 || end > this.length) {
|
|
throw new Error('end out of bounds');
|
|
}
|
|
|
|
for (var i = start; i < end; i++) {
|
|
this[i] = value;
|
|
}
|
|
}
|
|
|
|
// Static methods
|
|
Buffer.isBuffer = function isBuffer(b) {
|
|
return b instanceof Buffer || b instanceof Buffer;
|
|
};
|
|
|
|
Buffer.concat = function (list, totalLength) {
|
|
if (!isArray(list)) {
|
|
throw new Error("Usage: Buffer.concat(list, [totalLength])\n \
|
|
list should be an Array.");
|
|
}
|
|
|
|
if (list.length === 0) {
|
|
return new Buffer(0);
|
|
} else if (list.length === 1) {
|
|
return list[0];
|
|
}
|
|
|
|
if (typeof totalLength !== 'number') {
|
|
totalLength = 0;
|
|
for (var i = 0; i < list.length; i++) {
|
|
var buf = list[i];
|
|
totalLength += buf.length;
|
|
}
|
|
}
|
|
|
|
var buffer = new Buffer(totalLength);
|
|
var pos = 0;
|
|
for (var i = 0; i < list.length; i++) {
|
|
var buf = list[i];
|
|
buf.copy(buffer, pos);
|
|
pos += buf.length;
|
|
}
|
|
return buffer;
|
|
};
|
|
|
|
Buffer.isEncoding = function(encoding) {
|
|
switch ((encoding + '').toLowerCase()) {
|
|
case 'hex':
|
|
case 'utf8':
|
|
case 'utf-8':
|
|
case 'ascii':
|
|
case 'binary':
|
|
case 'base64':
|
|
case 'ucs2':
|
|
case 'ucs-2':
|
|
case 'utf16le':
|
|
case 'utf-16le':
|
|
case 'raw':
|
|
return true;
|
|
|
|
default:
|
|
return false;
|
|
}
|
|
};
|
|
|
|
// helpers
|
|
|
|
function coerce(length) {
|
|
// Coerce length to a number (possibly NaN), round up
|
|
// in case it's fractional (e.g. 123.456) then do a
|
|
// double negate to coerce a NaN to 0. Easy, right?
|
|
length = ~~Math.ceil(+length);
|
|
return length < 0 ? 0 : length;
|
|
}
|
|
|
|
function isArray(subject) {
|
|
return (Array.isArray ||
|
|
function(subject){
|
|
return {}.toString.apply(subject) == '[object Array]'
|
|
})
|
|
(subject)
|
|
}
|
|
|
|
function isArrayIsh(subject) {
|
|
return isArray(subject) || Buffer.isBuffer(subject) ||
|
|
subject && typeof subject === 'object' &&
|
|
typeof subject.length === 'number';
|
|
}
|
|
|
|
function toHex(n) {
|
|
if (n < 16) return '0' + n.toString(16);
|
|
return n.toString(16);
|
|
}
|
|
|
|
function utf8ToBytes(str) {
|
|
var byteArray = [];
|
|
for (var i = 0; i < str.length; i++)
|
|
if (str.charCodeAt(i) <= 0x7F)
|
|
byteArray.push(str.charCodeAt(i));
|
|
else {
|
|
var h = encodeURIComponent(str.charAt(i)).substr(1).split('%');
|
|
for (var j = 0; j < h.length; j++)
|
|
byteArray.push(parseInt(h[j], 16));
|
|
}
|
|
|
|
return byteArray;
|
|
}
|
|
|
|
function asciiToBytes(str) {
|
|
var byteArray = []
|
|
for (var i = 0; i < str.length; i++ )
|
|
// Node's code seems to be doing this and not & 0x7F..
|
|
byteArray.push( str.charCodeAt(i) & 0xFF );
|
|
|
|
return byteArray;
|
|
}
|
|
|
|
function base64ToBytes(str) {
|
|
return require("base64-js").toByteArray(str);
|
|
}
|
|
|
|
function blitBuffer(src, dst, offset, length) {
|
|
var pos, i = 0;
|
|
while (i < length) {
|
|
if ((i+offset >= dst.length) || (i >= src.length))
|
|
break;
|
|
|
|
dst[i + offset] = src[i];
|
|
i++;
|
|
}
|
|
return i;
|
|
}
|
|
|
|
function decodeUtf8Char(str) {
|
|
try {
|
|
return decodeURIComponent(str);
|
|
} catch (err) {
|
|
return String.fromCharCode(0xFFFD); // UTF 8 invalid char
|
|
}
|
|
}
|
|
|
|
// read/write bit-twiddling
|
|
|
|
Buffer.prototype.readUInt8 = function(offset, noAssert) {
|
|
var buffer = this;
|
|
|
|
if (!noAssert) {
|
|
assert.ok(offset !== undefined && offset !== null,
|
|
'missing offset');
|
|
|
|
assert.ok(offset < buffer.length,
|
|
'Trying to read beyond buffer length');
|
|
}
|
|
|
|
if (offset >= buffer.length) return;
|
|
|
|
return buffer[offset];
|
|
};
|
|
|
|
function readUInt16(buffer, offset, isBigEndian, noAssert) {
|
|
var val = 0;
|
|
|
|
|
|
if (!noAssert) {
|
|
assert.ok(typeof (isBigEndian) === 'boolean',
|
|
'missing or invalid endian');
|
|
|
|
assert.ok(offset !== undefined && offset !== null,
|
|
'missing offset');
|
|
|
|
assert.ok(offset + 1 < buffer.length,
|
|
'Trying to read beyond buffer length');
|
|
}
|
|
|
|
if (offset >= buffer.length) return 0;
|
|
|
|
if (isBigEndian) {
|
|
val = buffer[offset] << 8;
|
|
if (offset + 1 < buffer.length) {
|
|
val |= buffer[offset + 1];
|
|
}
|
|
} else {
|
|
val = buffer[offset];
|
|
if (offset + 1 < buffer.length) {
|
|
val |= buffer[offset + 1] << 8;
|
|
}
|
|
}
|
|
|
|
return val;
|
|
}
|
|
|
|
Buffer.prototype.readUInt16LE = function(offset, noAssert) {
|
|
return readUInt16(this, offset, false, noAssert);
|
|
};
|
|
|
|
Buffer.prototype.readUInt16BE = function(offset, noAssert) {
|
|
return readUInt16(this, offset, true, noAssert);
|
|
};
|
|
|
|
function readUInt32(buffer, offset, isBigEndian, noAssert) {
|
|
var val = 0;
|
|
|
|
if (!noAssert) {
|
|
assert.ok(typeof (isBigEndian) === 'boolean',
|
|
'missing or invalid endian');
|
|
|
|
assert.ok(offset !== undefined && offset !== null,
|
|
'missing offset');
|
|
|
|
assert.ok(offset + 3 < buffer.length,
|
|
'Trying to read beyond buffer length');
|
|
}
|
|
|
|
if (offset >= buffer.length) return 0;
|
|
|
|
if (isBigEndian) {
|
|
if (offset + 1 < buffer.length)
|
|
val = buffer[offset + 1] << 16;
|
|
if (offset + 2 < buffer.length)
|
|
val |= buffer[offset + 2] << 8;
|
|
if (offset + 3 < buffer.length)
|
|
val |= buffer[offset + 3];
|
|
val = val + (buffer[offset] << 24 >>> 0);
|
|
} else {
|
|
if (offset + 2 < buffer.length)
|
|
val = buffer[offset + 2] << 16;
|
|
if (offset + 1 < buffer.length)
|
|
val |= buffer[offset + 1] << 8;
|
|
val |= buffer[offset];
|
|
if (offset + 3 < buffer.length)
|
|
val = val + (buffer[offset + 3] << 24 >>> 0);
|
|
}
|
|
|
|
return val;
|
|
}
|
|
|
|
Buffer.prototype.readUInt32LE = function(offset, noAssert) {
|
|
return readUInt32(this, offset, false, noAssert);
|
|
};
|
|
|
|
Buffer.prototype.readUInt32BE = function(offset, noAssert) {
|
|
return readUInt32(this, offset, true, noAssert);
|
|
};
|
|
|
|
|
|
/*
|
|
* Signed integer types, yay team! A reminder on how two's complement actually
|
|
* works. The first bit is the signed bit, i.e. tells us whether or not the
|
|
* number should be positive or negative. If the two's complement value is
|
|
* positive, then we're done, as it's equivalent to the unsigned representation.
|
|
*
|
|
* Now if the number is positive, you're pretty much done, you can just leverage
|
|
* the unsigned translations and return those. Unfortunately, negative numbers
|
|
* aren't quite that straightforward.
|
|
*
|
|
* At first glance, one might be inclined to use the traditional formula to
|
|
* translate binary numbers between the positive and negative values in two's
|
|
* complement. (Though it doesn't quite work for the most negative value)
|
|
* Mainly:
|
|
* - invert all the bits
|
|
* - add one to the result
|
|
*
|
|
* Of course, this doesn't quite work in Javascript. Take for example the value
|
|
* of -128. This could be represented in 16 bits (big-endian) as 0xff80. But of
|
|
* course, Javascript will do the following:
|
|
*
|
|
* > ~0xff80
|
|
* -65409
|
|
*
|
|
* Whoh there, Javascript, that's not quite right. But wait, according to
|
|
* Javascript that's perfectly correct. When Javascript ends up seeing the
|
|
* constant 0xff80, it has no notion that it is actually a signed number. It
|
|
* assumes that we've input the unsigned value 0xff80. Thus, when it does the
|
|
* binary negation, it casts it into a signed value, (positive 0xff80). Then
|
|
* when you perform binary negation on that, it turns it into a negative number.
|
|
*
|
|
* Instead, we're going to have to use the following general formula, that works
|
|
* in a rather Javascript friendly way. I'm glad we don't support this kind of
|
|
* weird numbering scheme in the kernel.
|
|
*
|
|
* (BIT-MAX - (unsigned)val + 1) * -1
|
|
*
|
|
* The astute observer, may think that this doesn't make sense for 8-bit numbers
|
|
* (really it isn't necessary for them). However, when you get 16-bit numbers,
|
|
* you do. Let's go back to our prior example and see how this will look:
|
|
*
|
|
* (0xffff - 0xff80 + 1) * -1
|
|
* (0x007f + 1) * -1
|
|
* (0x0080) * -1
|
|
*/
|
|
Buffer.prototype.readInt8 = function(offset, noAssert) {
|
|
var buffer = this;
|
|
var neg;
|
|
|
|
if (!noAssert) {
|
|
assert.ok(offset !== undefined && offset !== null,
|
|
'missing offset');
|
|
|
|
assert.ok(offset < buffer.length,
|
|
'Trying to read beyond buffer length');
|
|
}
|
|
|
|
if (offset >= buffer.length) return;
|
|
|
|
neg = buffer[offset] & 0x80;
|
|
if (!neg) {
|
|
return (buffer[offset]);
|
|
}
|
|
|
|
return ((0xff - buffer[offset] + 1) * -1);
|
|
};
|
|
|
|
function readInt16(buffer, offset, isBigEndian, noAssert) {
|
|
var neg, val;
|
|
|
|
if (!noAssert) {
|
|
assert.ok(typeof (isBigEndian) === 'boolean',
|
|
'missing or invalid endian');
|
|
|
|
assert.ok(offset !== undefined && offset !== null,
|
|
'missing offset');
|
|
|
|
assert.ok(offset + 1 < buffer.length,
|
|
'Trying to read beyond buffer length');
|
|
}
|
|
|
|
val = readUInt16(buffer, offset, isBigEndian, noAssert);
|
|
neg = val & 0x8000;
|
|
if (!neg) {
|
|
return val;
|
|
}
|
|
|
|
return (0xffff - val + 1) * -1;
|
|
}
|
|
|
|
Buffer.prototype.readInt16LE = function(offset, noAssert) {
|
|
return readInt16(this, offset, false, noAssert);
|
|
};
|
|
|
|
Buffer.prototype.readInt16BE = function(offset, noAssert) {
|
|
return readInt16(this, offset, true, noAssert);
|
|
};
|
|
|
|
function readInt32(buffer, offset, isBigEndian, noAssert) {
|
|
var neg, val;
|
|
|
|
if (!noAssert) {
|
|
assert.ok(typeof (isBigEndian) === 'boolean',
|
|
'missing or invalid endian');
|
|
|
|
assert.ok(offset !== undefined && offset !== null,
|
|
'missing offset');
|
|
|
|
assert.ok(offset + 3 < buffer.length,
|
|
'Trying to read beyond buffer length');
|
|
}
|
|
|
|
val = readUInt32(buffer, offset, isBigEndian, noAssert);
|
|
neg = val & 0x80000000;
|
|
if (!neg) {
|
|
return (val);
|
|
}
|
|
|
|
return (0xffffffff - val + 1) * -1;
|
|
}
|
|
|
|
Buffer.prototype.readInt32LE = function(offset, noAssert) {
|
|
return readInt32(this, offset, false, noAssert);
|
|
};
|
|
|
|
Buffer.prototype.readInt32BE = function(offset, noAssert) {
|
|
return readInt32(this, offset, true, noAssert);
|
|
};
|
|
|
|
function readFloat(buffer, offset, isBigEndian, noAssert) {
|
|
if (!noAssert) {
|
|
assert.ok(typeof (isBigEndian) === 'boolean',
|
|
'missing or invalid endian');
|
|
|
|
assert.ok(offset + 3 < buffer.length,
|
|
'Trying to read beyond buffer length');
|
|
}
|
|
|
|
return require('./buffer_ieee754').readIEEE754(buffer, offset, isBigEndian,
|
|
23, 4);
|
|
}
|
|
|
|
Buffer.prototype.readFloatLE = function(offset, noAssert) {
|
|
return readFloat(this, offset, false, noAssert);
|
|
};
|
|
|
|
Buffer.prototype.readFloatBE = function(offset, noAssert) {
|
|
return readFloat(this, offset, true, noAssert);
|
|
};
|
|
|
|
function readDouble(buffer, offset, isBigEndian, noAssert) {
|
|
if (!noAssert) {
|
|
assert.ok(typeof (isBigEndian) === 'boolean',
|
|
'missing or invalid endian');
|
|
|
|
assert.ok(offset + 7 < buffer.length,
|
|
'Trying to read beyond buffer length');
|
|
}
|
|
|
|
return require('./buffer_ieee754').readIEEE754(buffer, offset, isBigEndian,
|
|
52, 8);
|
|
}
|
|
|
|
Buffer.prototype.readDoubleLE = function(offset, noAssert) {
|
|
return readDouble(this, offset, false, noAssert);
|
|
};
|
|
|
|
Buffer.prototype.readDoubleBE = function(offset, noAssert) {
|
|
return readDouble(this, offset, true, noAssert);
|
|
};
|
|
|
|
|
|
/*
|
|
* We have to make sure that the value is a valid integer. This means that it is
|
|
* non-negative. It has no fractional component and that it does not exceed the
|
|
* maximum allowed value.
|
|
*
|
|
* value The number to check for validity
|
|
*
|
|
* max The maximum value
|
|
*/
|
|
function verifuint(value, max) {
|
|
assert.ok(typeof (value) == 'number',
|
|
'cannot write a non-number as a number');
|
|
|
|
assert.ok(value >= 0,
|
|
'specified a negative value for writing an unsigned value');
|
|
|
|
assert.ok(value <= max, 'value is larger than maximum value for type');
|
|
|
|
assert.ok(Math.floor(value) === value, 'value has a fractional component');
|
|
}
|
|
|
|
Buffer.prototype.writeUInt8 = function(value, offset, noAssert) {
|
|
var buffer = this;
|
|
|
|
if (!noAssert) {
|
|
assert.ok(value !== undefined && value !== null,
|
|
'missing value');
|
|
|
|
assert.ok(offset !== undefined && offset !== null,
|
|
'missing offset');
|
|
|
|
assert.ok(offset < buffer.length,
|
|
'trying to write beyond buffer length');
|
|
|
|
verifuint(value, 0xff);
|
|
}
|
|
|
|
if (offset < buffer.length) {
|
|
buffer[offset] = value;
|
|
}
|
|
};
|
|
|
|
function writeUInt16(buffer, value, offset, isBigEndian, noAssert) {
|
|
if (!noAssert) {
|
|
assert.ok(value !== undefined && value !== null,
|
|
'missing value');
|
|
|
|
assert.ok(typeof (isBigEndian) === 'boolean',
|
|
'missing or invalid endian');
|
|
|
|
assert.ok(offset !== undefined && offset !== null,
|
|
'missing offset');
|
|
|
|
assert.ok(offset + 1 < buffer.length,
|
|
'trying to write beyond buffer length');
|
|
|
|
verifuint(value, 0xffff);
|
|
}
|
|
|
|
for (var i = 0; i < Math.min(buffer.length - offset, 2); i++) {
|
|
buffer[offset + i] =
|
|
(value & (0xff << (8 * (isBigEndian ? 1 - i : i)))) >>>
|
|
(isBigEndian ? 1 - i : i) * 8;
|
|
}
|
|
|
|
}
|
|
|
|
Buffer.prototype.writeUInt16LE = function(value, offset, noAssert) {
|
|
writeUInt16(this, value, offset, false, noAssert);
|
|
};
|
|
|
|
Buffer.prototype.writeUInt16BE = function(value, offset, noAssert) {
|
|
writeUInt16(this, value, offset, true, noAssert);
|
|
};
|
|
|
|
function writeUInt32(buffer, value, offset, isBigEndian, noAssert) {
|
|
if (!noAssert) {
|
|
assert.ok(value !== undefined && value !== null,
|
|
'missing value');
|
|
|
|
assert.ok(typeof (isBigEndian) === 'boolean',
|
|
'missing or invalid endian');
|
|
|
|
assert.ok(offset !== undefined && offset !== null,
|
|
'missing offset');
|
|
|
|
assert.ok(offset + 3 < buffer.length,
|
|
'trying to write beyond buffer length');
|
|
|
|
verifuint(value, 0xffffffff);
|
|
}
|
|
|
|
for (var i = 0; i < Math.min(buffer.length - offset, 4); i++) {
|
|
buffer[offset + i] =
|
|
(value >>> (isBigEndian ? 3 - i : i) * 8) & 0xff;
|
|
}
|
|
}
|
|
|
|
Buffer.prototype.writeUInt32LE = function(value, offset, noAssert) {
|
|
writeUInt32(this, value, offset, false, noAssert);
|
|
};
|
|
|
|
Buffer.prototype.writeUInt32BE = function(value, offset, noAssert) {
|
|
writeUInt32(this, value, offset, true, noAssert);
|
|
};
|
|
|
|
|
|
/*
|
|
* We now move onto our friends in the signed number category. Unlike unsigned
|
|
* numbers, we're going to have to worry a bit more about how we put values into
|
|
* arrays. Since we are only worrying about signed 32-bit values, we're in
|
|
* slightly better shape. Unfortunately, we really can't do our favorite binary
|
|
* & in this system. It really seems to do the wrong thing. For example:
|
|
*
|
|
* > -32 & 0xff
|
|
* 224
|
|
*
|
|
* What's happening above is really: 0xe0 & 0xff = 0xe0. However, the results of
|
|
* this aren't treated as a signed number. Ultimately a bad thing.
|
|
*
|
|
* What we're going to want to do is basically create the unsigned equivalent of
|
|
* our representation and pass that off to the wuint* functions. To do that
|
|
* we're going to do the following:
|
|
*
|
|
* - if the value is positive
|
|
* we can pass it directly off to the equivalent wuint
|
|
* - if the value is negative
|
|
* we do the following computation:
|
|
* mb + val + 1, where
|
|
* mb is the maximum unsigned value in that byte size
|
|
* val is the Javascript negative integer
|
|
*
|
|
*
|
|
* As a concrete value, take -128. In signed 16 bits this would be 0xff80. If
|
|
* you do out the computations:
|
|
*
|
|
* 0xffff - 128 + 1
|
|
* 0xffff - 127
|
|
* 0xff80
|
|
*
|
|
* You can then encode this value as the signed version. This is really rather
|
|
* hacky, but it should work and get the job done which is our goal here.
|
|
*/
|
|
|
|
/*
|
|
* A series of checks to make sure we actually have a signed 32-bit number
|
|
*/
|
|
function verifsint(value, max, min) {
|
|
assert.ok(typeof (value) == 'number',
|
|
'cannot write a non-number as a number');
|
|
|
|
assert.ok(value <= max, 'value larger than maximum allowed value');
|
|
|
|
assert.ok(value >= min, 'value smaller than minimum allowed value');
|
|
|
|
assert.ok(Math.floor(value) === value, 'value has a fractional component');
|
|
}
|
|
|
|
function verifIEEE754(value, max, min) {
|
|
assert.ok(typeof (value) == 'number',
|
|
'cannot write a non-number as a number');
|
|
|
|
assert.ok(value <= max, 'value larger than maximum allowed value');
|
|
|
|
assert.ok(value >= min, 'value smaller than minimum allowed value');
|
|
}
|
|
|
|
Buffer.prototype.writeInt8 = function(value, offset, noAssert) {
|
|
var buffer = this;
|
|
|
|
if (!noAssert) {
|
|
assert.ok(value !== undefined && value !== null,
|
|
'missing value');
|
|
|
|
assert.ok(offset !== undefined && offset !== null,
|
|
'missing offset');
|
|
|
|
assert.ok(offset < buffer.length,
|
|
'Trying to write beyond buffer length');
|
|
|
|
verifsint(value, 0x7f, -0x80);
|
|
}
|
|
|
|
if (value >= 0) {
|
|
buffer.writeUInt8(value, offset, noAssert);
|
|
} else {
|
|
buffer.writeUInt8(0xff + value + 1, offset, noAssert);
|
|
}
|
|
};
|
|
|
|
function writeInt16(buffer, value, offset, isBigEndian, noAssert) {
|
|
if (!noAssert) {
|
|
assert.ok(value !== undefined && value !== null,
|
|
'missing value');
|
|
|
|
assert.ok(typeof (isBigEndian) === 'boolean',
|
|
'missing or invalid endian');
|
|
|
|
assert.ok(offset !== undefined && offset !== null,
|
|
'missing offset');
|
|
|
|
assert.ok(offset + 1 < buffer.length,
|
|
'Trying to write beyond buffer length');
|
|
|
|
verifsint(value, 0x7fff, -0x8000);
|
|
}
|
|
|
|
if (value >= 0) {
|
|
writeUInt16(buffer, value, offset, isBigEndian, noAssert);
|
|
} else {
|
|
writeUInt16(buffer, 0xffff + value + 1, offset, isBigEndian, noAssert);
|
|
}
|
|
}
|
|
|
|
Buffer.prototype.writeInt16LE = function(value, offset, noAssert) {
|
|
writeInt16(this, value, offset, false, noAssert);
|
|
};
|
|
|
|
Buffer.prototype.writeInt16BE = function(value, offset, noAssert) {
|
|
writeInt16(this, value, offset, true, noAssert);
|
|
};
|
|
|
|
function writeInt32(buffer, value, offset, isBigEndian, noAssert) {
|
|
if (!noAssert) {
|
|
assert.ok(value !== undefined && value !== null,
|
|
'missing value');
|
|
|
|
assert.ok(typeof (isBigEndian) === 'boolean',
|
|
'missing or invalid endian');
|
|
|
|
assert.ok(offset !== undefined && offset !== null,
|
|
'missing offset');
|
|
|
|
assert.ok(offset + 3 < buffer.length,
|
|
'Trying to write beyond buffer length');
|
|
|
|
verifsint(value, 0x7fffffff, -0x80000000);
|
|
}
|
|
|
|
if (value >= 0) {
|
|
writeUInt32(buffer, value, offset, isBigEndian, noAssert);
|
|
} else {
|
|
writeUInt32(buffer, 0xffffffff + value + 1, offset, isBigEndian, noAssert);
|
|
}
|
|
}
|
|
|
|
Buffer.prototype.writeInt32LE = function(value, offset, noAssert) {
|
|
writeInt32(this, value, offset, false, noAssert);
|
|
};
|
|
|
|
Buffer.prototype.writeInt32BE = function(value, offset, noAssert) {
|
|
writeInt32(this, value, offset, true, noAssert);
|
|
};
|
|
|
|
function writeFloat(buffer, value, offset, isBigEndian, noAssert) {
|
|
if (!noAssert) {
|
|
assert.ok(value !== undefined && value !== null,
|
|
'missing value');
|
|
|
|
assert.ok(typeof (isBigEndian) === 'boolean',
|
|
'missing or invalid endian');
|
|
|
|
assert.ok(offset !== undefined && offset !== null,
|
|
'missing offset');
|
|
|
|
assert.ok(offset + 3 < buffer.length,
|
|
'Trying to write beyond buffer length');
|
|
|
|
verifIEEE754(value, 3.4028234663852886e+38, -3.4028234663852886e+38);
|
|
}
|
|
|
|
require('./buffer_ieee754').writeIEEE754(buffer, value, offset, isBigEndian,
|
|
23, 4);
|
|
}
|
|
|
|
Buffer.prototype.writeFloatLE = function(value, offset, noAssert) {
|
|
writeFloat(this, value, offset, false, noAssert);
|
|
};
|
|
|
|
Buffer.prototype.writeFloatBE = function(value, offset, noAssert) {
|
|
writeFloat(this, value, offset, true, noAssert);
|
|
};
|
|
|
|
function writeDouble(buffer, value, offset, isBigEndian, noAssert) {
|
|
if (!noAssert) {
|
|
assert.ok(value !== undefined && value !== null,
|
|
'missing value');
|
|
|
|
assert.ok(typeof (isBigEndian) === 'boolean',
|
|
'missing or invalid endian');
|
|
|
|
assert.ok(offset !== undefined && offset !== null,
|
|
'missing offset');
|
|
|
|
assert.ok(offset + 7 < buffer.length,
|
|
'Trying to write beyond buffer length');
|
|
|
|
verifIEEE754(value, 1.7976931348623157E+308, -1.7976931348623157E+308);
|
|
}
|
|
|
|
require('./buffer_ieee754').writeIEEE754(buffer, value, offset, isBigEndian,
|
|
52, 8);
|
|
}
|
|
|
|
Buffer.prototype.writeDoubleLE = function(value, offset, noAssert) {
|
|
writeDouble(this, value, offset, false, noAssert);
|
|
};
|
|
|
|
Buffer.prototype.writeDoubleBE = function(value, offset, noAssert) {
|
|
writeDouble(this, value, offset, true, noAssert);
|
|
};
|
|
|
|
},{"./buffer_ieee754":1,"assert":6,"base64-js":4}],"buffer-browserify":[function(require,module,exports){
|
|
module.exports=require('q9TxCC');
|
|
},{}],4:[function(require,module,exports){
|
|
(function (exports) {
|
|
'use strict';
|
|
|
|
var lookup = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/';
|
|
|
|
function b64ToByteArray(b64) {
|
|
var i, j, l, tmp, placeHolders, arr;
|
|
|
|
if (b64.length % 4 > 0) {
|
|
throw 'Invalid string. Length must be a multiple of 4';
|
|
}
|
|
|
|
// the number of equal signs (place holders)
|
|
// if there are two placeholders, than the two characters before it
|
|
// represent one byte
|
|
// if there is only one, then the three characters before it represent 2 bytes
|
|
// this is just a cheap hack to not do indexOf twice
|
|
placeHolders = b64.indexOf('=');
|
|
placeHolders = placeHolders > 0 ? b64.length - placeHolders : 0;
|
|
|
|
// base64 is 4/3 + up to two characters of the original data
|
|
arr = [];//new Uint8Array(b64.length * 3 / 4 - placeHolders);
|
|
|
|
// if there are placeholders, only get up to the last complete 4 chars
|
|
l = placeHolders > 0 ? b64.length - 4 : b64.length;
|
|
|
|
for (i = 0, j = 0; i < l; i += 4, j += 3) {
|
|
tmp = (lookup.indexOf(b64[i]) << 18) | (lookup.indexOf(b64[i + 1]) << 12) | (lookup.indexOf(b64[i + 2]) << 6) | lookup.indexOf(b64[i + 3]);
|
|
arr.push((tmp & 0xFF0000) >> 16);
|
|
arr.push((tmp & 0xFF00) >> 8);
|
|
arr.push(tmp & 0xFF);
|
|
}
|
|
|
|
if (placeHolders === 2) {
|
|
tmp = (lookup.indexOf(b64[i]) << 2) | (lookup.indexOf(b64[i + 1]) >> 4);
|
|
arr.push(tmp & 0xFF);
|
|
} else if (placeHolders === 1) {
|
|
tmp = (lookup.indexOf(b64[i]) << 10) | (lookup.indexOf(b64[i + 1]) << 4) | (lookup.indexOf(b64[i + 2]) >> 2);
|
|
arr.push((tmp >> 8) & 0xFF);
|
|
arr.push(tmp & 0xFF);
|
|
}
|
|
|
|
return arr;
|
|
}
|
|
|
|
function uint8ToBase64(uint8) {
|
|
var i,
|
|
extraBytes = uint8.length % 3, // if we have 1 byte left, pad 2 bytes
|
|
output = "",
|
|
temp, length;
|
|
|
|
function tripletToBase64 (num) {
|
|
return lookup[num >> 18 & 0x3F] + lookup[num >> 12 & 0x3F] + lookup[num >> 6 & 0x3F] + lookup[num & 0x3F];
|
|
};
|
|
|
|
// go through the array every three bytes, we'll deal with trailing stuff later
|
|
for (i = 0, length = uint8.length - extraBytes; i < length; i += 3) {
|
|
temp = (uint8[i] << 16) + (uint8[i + 1] << 8) + (uint8[i + 2]);
|
|
output += tripletToBase64(temp);
|
|
}
|
|
|
|
// pad the end with zeros, but make sure to not forget the extra bytes
|
|
switch (extraBytes) {
|
|
case 1:
|
|
temp = uint8[uint8.length - 1];
|
|
output += lookup[temp >> 2];
|
|
output += lookup[(temp << 4) & 0x3F];
|
|
output += '==';
|
|
break;
|
|
case 2:
|
|
temp = (uint8[uint8.length - 2] << 8) + (uint8[uint8.length - 1]);
|
|
output += lookup[temp >> 10];
|
|
output += lookup[(temp >> 4) & 0x3F];
|
|
output += lookup[(temp << 2) & 0x3F];
|
|
output += '=';
|
|
break;
|
|
}
|
|
|
|
return output;
|
|
}
|
|
|
|
module.exports.toByteArray = b64ToByteArray;
|
|
module.exports.fromByteArray = uint8ToBase64;
|
|
}());
|
|
|
|
},{}],5:[function(require,module,exports){
|
|
|
|
|
|
//
|
|
// The shims in this file are not fully implemented shims for the ES5
|
|
// features, but do work for the particular usecases there is in
|
|
// the other modules.
|
|
//
|
|
|
|
var toString = Object.prototype.toString;
|
|
var hasOwnProperty = Object.prototype.hasOwnProperty;
|
|
|
|
// Array.isArray is supported in IE9
|
|
function isArray(xs) {
|
|
return toString.call(xs) === '[object Array]';
|
|
}
|
|
exports.isArray = typeof Array.isArray === 'function' ? Array.isArray : isArray;
|
|
|
|
// Array.prototype.indexOf is supported in IE9
|
|
exports.indexOf = function indexOf(xs, x) {
|
|
if (xs.indexOf) return xs.indexOf(x);
|
|
for (var i = 0; i < xs.length; i++) {
|
|
if (x === xs[i]) return i;
|
|
}
|
|
return -1;
|
|
};
|
|
|
|
// Array.prototype.filter is supported in IE9
|
|
exports.filter = function filter(xs, fn) {
|
|
if (xs.filter) return xs.filter(fn);
|
|
var res = [];
|
|
for (var i = 0; i < xs.length; i++) {
|
|
if (fn(xs[i], i, xs)) res.push(xs[i]);
|
|
}
|
|
return res;
|
|
};
|
|
|
|
// Array.prototype.forEach is supported in IE9
|
|
exports.forEach = function forEach(xs, fn, self) {
|
|
if (xs.forEach) return xs.forEach(fn, self);
|
|
for (var i = 0; i < xs.length; i++) {
|
|
fn.call(self, xs[i], i, xs);
|
|
}
|
|
};
|
|
|
|
// Array.prototype.map is supported in IE9
|
|
exports.map = function map(xs, fn) {
|
|
if (xs.map) return xs.map(fn);
|
|
var out = new Array(xs.length);
|
|
for (var i = 0; i < xs.length; i++) {
|
|
out[i] = fn(xs[i], i, xs);
|
|
}
|
|
return out;
|
|
};
|
|
|
|
// Array.prototype.reduce is supported in IE9
|
|
exports.reduce = function reduce(array, callback, opt_initialValue) {
|
|
if (array.reduce) return array.reduce(callback, opt_initialValue);
|
|
var value, isValueSet = false;
|
|
|
|
if (2 < arguments.length) {
|
|
value = opt_initialValue;
|
|
isValueSet = true;
|
|
}
|
|
for (var i = 0, l = array.length; l > i; ++i) {
|
|
if (array.hasOwnProperty(i)) {
|
|
if (isValueSet) {
|
|
value = callback(value, array[i], i, array);
|
|
}
|
|
else {
|
|
value = array[i];
|
|
isValueSet = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
return value;
|
|
};
|
|
|
|
// String.prototype.substr - negative index don't work in IE8
|
|
if ('ab'.substr(-1) !== 'b') {
|
|
exports.substr = function (str, start, length) {
|
|
// did we get a negative start, calculate how much it is from the beginning of the string
|
|
if (start < 0) start = str.length + start;
|
|
|
|
// call the original function
|
|
return str.substr(start, length);
|
|
};
|
|
} else {
|
|
exports.substr = function (str, start, length) {
|
|
return str.substr(start, length);
|
|
};
|
|
}
|
|
|
|
// String.prototype.trim is supported in IE9
|
|
exports.trim = function (str) {
|
|
if (str.trim) return str.trim();
|
|
return str.replace(/^\s+|\s+$/g, '');
|
|
};
|
|
|
|
// Function.prototype.bind is supported in IE9
|
|
exports.bind = function () {
|
|
var args = Array.prototype.slice.call(arguments);
|
|
var fn = args.shift();
|
|
if (fn.bind) return fn.bind.apply(fn, args);
|
|
var self = args.shift();
|
|
return function () {
|
|
fn.apply(self, args.concat([Array.prototype.slice.call(arguments)]));
|
|
};
|
|
};
|
|
|
|
// Object.create is supported in IE9
|
|
function create(prototype, properties) {
|
|
var object;
|
|
if (prototype === null) {
|
|
object = { '__proto__' : null };
|
|
}
|
|
else {
|
|
if (typeof prototype !== 'object') {
|
|
throw new TypeError(
|
|
'typeof prototype[' + (typeof prototype) + '] != \'object\''
|
|
);
|
|
}
|
|
var Type = function () {};
|
|
Type.prototype = prototype;
|
|
object = new Type();
|
|
object.__proto__ = prototype;
|
|
}
|
|
if (typeof properties !== 'undefined' && Object.defineProperties) {
|
|
Object.defineProperties(object, properties);
|
|
}
|
|
return object;
|
|
}
|
|
exports.create = typeof Object.create === 'function' ? Object.create : create;
|
|
|
|
// Object.keys and Object.getOwnPropertyNames is supported in IE9 however
|
|
// they do show a description and number property on Error objects
|
|
function notObject(object) {
|
|
return ((typeof object != "object" && typeof object != "function") || object === null);
|
|
}
|
|
|
|
function keysShim(object) {
|
|
if (notObject(object)) {
|
|
throw new TypeError("Object.keys called on a non-object");
|
|
}
|
|
|
|
var result = [];
|
|
for (var name in object) {
|
|
if (hasOwnProperty.call(object, name)) {
|
|
result.push(name);
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
// getOwnPropertyNames is almost the same as Object.keys one key feature
|
|
// is that it returns hidden properties, since that can't be implemented,
|
|
// this feature gets reduced so it just shows the length property on arrays
|
|
function propertyShim(object) {
|
|
if (notObject(object)) {
|
|
throw new TypeError("Object.getOwnPropertyNames called on a non-object");
|
|
}
|
|
|
|
var result = keysShim(object);
|
|
if (exports.isArray(object) && exports.indexOf(object, 'length') === -1) {
|
|
result.push('length');
|
|
}
|
|
return result;
|
|
}
|
|
|
|
var keys = typeof Object.keys === 'function' ? Object.keys : keysShim;
|
|
var getOwnPropertyNames = typeof Object.getOwnPropertyNames === 'function' ?
|
|
Object.getOwnPropertyNames : propertyShim;
|
|
|
|
if (new Error().hasOwnProperty('description')) {
|
|
var ERROR_PROPERTY_FILTER = function (obj, array) {
|
|
if (toString.call(obj) === '[object Error]') {
|
|
array = exports.filter(array, function (name) {
|
|
return name !== 'description' && name !== 'number' && name !== 'message';
|
|
});
|
|
}
|
|
return array;
|
|
};
|
|
|
|
exports.keys = function (object) {
|
|
return ERROR_PROPERTY_FILTER(object, keys(object));
|
|
};
|
|
exports.getOwnPropertyNames = function (object) {
|
|
return ERROR_PROPERTY_FILTER(object, getOwnPropertyNames(object));
|
|
};
|
|
} else {
|
|
exports.keys = keys;
|
|
exports.getOwnPropertyNames = getOwnPropertyNames;
|
|
}
|
|
|
|
// Object.getOwnPropertyDescriptor - supported in IE8 but only on dom elements
|
|
function valueObject(value, key) {
|
|
return { value: value[key] };
|
|
}
|
|
|
|
if (typeof Object.getOwnPropertyDescriptor === 'function') {
|
|
try {
|
|
Object.getOwnPropertyDescriptor({'a': 1}, 'a');
|
|
exports.getOwnPropertyDescriptor = Object.getOwnPropertyDescriptor;
|
|
} catch (e) {
|
|
// IE8 dom element issue - use a try catch and default to valueObject
|
|
exports.getOwnPropertyDescriptor = function (value, key) {
|
|
try {
|
|
return Object.getOwnPropertyDescriptor(value, key);
|
|
} catch (e) {
|
|
return valueObject(value, key);
|
|
}
|
|
};
|
|
}
|
|
} else {
|
|
exports.getOwnPropertyDescriptor = valueObject;
|
|
}
|
|
|
|
},{}],6:[function(require,module,exports){
|
|
// Copyright Joyent, Inc. and other Node contributors.
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining a
|
|
// copy of this software and associated documentation files (the
|
|
// "Software"), to deal in the Software without restriction, including
|
|
// without limitation the rights to use, copy, modify, merge, publish,
|
|
// distribute, sublicense, and/or sell copies of the Software, and to permit
|
|
// persons to whom the Software is furnished to do so, subject to the
|
|
// following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be included
|
|
// in all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
|
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
|
|
// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
|
|
// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
|
|
// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
|
|
// USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
// UTILITY
|
|
var util = require('util');
|
|
var shims = require('_shims');
|
|
var pSlice = Array.prototype.slice;
|
|
|
|
// 1. The assert module provides functions that throw
|
|
// AssertionError's when particular conditions are not met. The
|
|
// assert module must conform to the following interface.
|
|
|
|
var assert = module.exports = ok;
|
|
|
|
// 2. The AssertionError is defined in assert.
|
|
// new assert.AssertionError({ message: message,
|
|
// actual: actual,
|
|
// expected: expected })
|
|
|
|
assert.AssertionError = function AssertionError(options) {
|
|
this.name = 'AssertionError';
|
|
this.actual = options.actual;
|
|
this.expected = options.expected;
|
|
this.operator = options.operator;
|
|
this.message = options.message || getMessage(this);
|
|
};
|
|
|
|
// assert.AssertionError instanceof Error
|
|
util.inherits(assert.AssertionError, Error);
|
|
|
|
function replacer(key, value) {
|
|
if (util.isUndefined(value)) {
|
|
return '' + value;
|
|
}
|
|
if (util.isNumber(value) && (isNaN(value) || !isFinite(value))) {
|
|
return value.toString();
|
|
}
|
|
if (util.isFunction(value) || util.isRegExp(value)) {
|
|
return value.toString();
|
|
}
|
|
return value;
|
|
}
|
|
|
|
function truncate(s, n) {
|
|
if (util.isString(s)) {
|
|
return s.length < n ? s : s.slice(0, n);
|
|
} else {
|
|
return s;
|
|
}
|
|
}
|
|
|
|
function getMessage(self) {
|
|
return truncate(JSON.stringify(self.actual, replacer), 128) + ' ' +
|
|
self.operator + ' ' +
|
|
truncate(JSON.stringify(self.expected, replacer), 128);
|
|
}
|
|
|
|
// At present only the three keys mentioned above are used and
|
|
// understood by the spec. Implementations or sub modules can pass
|
|
// other keys to the AssertionError's constructor - they will be
|
|
// ignored.
|
|
|
|
// 3. All of the following functions must throw an AssertionError
|
|
// when a corresponding condition is not met, with a message that
|
|
// may be undefined if not provided. All assertion methods provide
|
|
// both the actual and expected values to the assertion error for
|
|
// display purposes.
|
|
|
|
function fail(actual, expected, message, operator, stackStartFunction) {
|
|
throw new assert.AssertionError({
|
|
message: message,
|
|
actual: actual,
|
|
expected: expected,
|
|
operator: operator,
|
|
stackStartFunction: stackStartFunction
|
|
});
|
|
}
|
|
|
|
// EXTENSION! allows for well behaved errors defined elsewhere.
|
|
assert.fail = fail;
|
|
|
|
// 4. Pure assertion tests whether a value is truthy, as determined
|
|
// by !!guard.
|
|
// assert.ok(guard, message_opt);
|
|
// This statement is equivalent to assert.equal(true, !!guard,
|
|
// message_opt);. To test strictly for the value true, use
|
|
// assert.strictEqual(true, guard, message_opt);.
|
|
|
|
function ok(value, message) {
|
|
if (!value) fail(value, true, message, '==', assert.ok);
|
|
}
|
|
assert.ok = ok;
|
|
|
|
// 5. The equality assertion tests shallow, coercive equality with
|
|
// ==.
|
|
// assert.equal(actual, expected, message_opt);
|
|
|
|
assert.equal = function equal(actual, expected, message) {
|
|
if (actual != expected) fail(actual, expected, message, '==', assert.equal);
|
|
};
|
|
|
|
// 6. The non-equality assertion tests for whether two objects are not equal
|
|
// with != assert.notEqual(actual, expected, message_opt);
|
|
|
|
assert.notEqual = function notEqual(actual, expected, message) {
|
|
if (actual == expected) {
|
|
fail(actual, expected, message, '!=', assert.notEqual);
|
|
}
|
|
};
|
|
|
|
// 7. The equivalence assertion tests a deep equality relation.
|
|
// assert.deepEqual(actual, expected, message_opt);
|
|
|
|
assert.deepEqual = function deepEqual(actual, expected, message) {
|
|
if (!_deepEqual(actual, expected)) {
|
|
fail(actual, expected, message, 'deepEqual', assert.deepEqual);
|
|
}
|
|
};
|
|
|
|
function _deepEqual(actual, expected) {
|
|
// 7.1. All identical values are equivalent, as determined by ===.
|
|
if (actual === expected) {
|
|
return true;
|
|
|
|
} else if (util.isBuffer(actual) && util.isBuffer(expected)) {
|
|
if (actual.length != expected.length) return false;
|
|
|
|
for (var i = 0; i < actual.length; i++) {
|
|
if (actual[i] !== expected[i]) return false;
|
|
}
|
|
|
|
return true;
|
|
|
|
// 7.2. If the expected value is a Date object, the actual value is
|
|
// equivalent if it is also a Date object that refers to the same time.
|
|
} else if (util.isDate(actual) && util.isDate(expected)) {
|
|
return actual.getTime() === expected.getTime();
|
|
|
|
// 7.3 If the expected value is a RegExp object, the actual value is
|
|
// equivalent if it is also a RegExp object with the same source and
|
|
// properties (`global`, `multiline`, `lastIndex`, `ignoreCase`).
|
|
} else if (util.isRegExp(actual) && util.isRegExp(expected)) {
|
|
return actual.source === expected.source &&
|
|
actual.global === expected.global &&
|
|
actual.multiline === expected.multiline &&
|
|
actual.lastIndex === expected.lastIndex &&
|
|
actual.ignoreCase === expected.ignoreCase;
|
|
|
|
// 7.4. Other pairs that do not both pass typeof value == 'object',
|
|
// equivalence is determined by ==.
|
|
} else if (!util.isObject(actual) && !util.isObject(expected)) {
|
|
return actual == expected;
|
|
|
|
// 7.5 For all other Object pairs, including Array objects, equivalence is
|
|
// determined by having the same number of owned properties (as verified
|
|
// with Object.prototype.hasOwnProperty.call), the same set of keys
|
|
// (although not necessarily the same order), equivalent values for every
|
|
// corresponding key, and an identical 'prototype' property. Note: this
|
|
// accounts for both named and indexed properties on Arrays.
|
|
} else {
|
|
return objEquiv(actual, expected);
|
|
}
|
|
}
|
|
|
|
function isArguments(object) {
|
|
return Object.prototype.toString.call(object) == '[object Arguments]';
|
|
}
|
|
|
|
function objEquiv(a, b) {
|
|
if (util.isNullOrUndefined(a) || util.isNullOrUndefined(b))
|
|
return false;
|
|
// an identical 'prototype' property.
|
|
if (a.prototype !== b.prototype) return false;
|
|
//~~~I've managed to break Object.keys through screwy arguments passing.
|
|
// Converting to array solves the problem.
|
|
if (isArguments(a)) {
|
|
if (!isArguments(b)) {
|
|
return false;
|
|
}
|
|
a = pSlice.call(a);
|
|
b = pSlice.call(b);
|
|
return _deepEqual(a, b);
|
|
}
|
|
try {
|
|
var ka = shims.keys(a),
|
|
kb = shims.keys(b),
|
|
key, i;
|
|
} catch (e) {//happens when one is a string literal and the other isn't
|
|
return false;
|
|
}
|
|
// having the same number of owned properties (keys incorporates
|
|
// hasOwnProperty)
|
|
if (ka.length != kb.length)
|
|
return false;
|
|
//the same set of keys (although not necessarily the same order),
|
|
ka.sort();
|
|
kb.sort();
|
|
//~~~cheap key test
|
|
for (i = ka.length - 1; i >= 0; i--) {
|
|
if (ka[i] != kb[i])
|
|
return false;
|
|
}
|
|
//equivalent values for every corresponding key, and
|
|
//~~~possibly expensive deep test
|
|
for (i = ka.length - 1; i >= 0; i--) {
|
|
key = ka[i];
|
|
if (!_deepEqual(a[key], b[key])) return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
// 8. The non-equivalence assertion tests for any deep inequality.
|
|
// assert.notDeepEqual(actual, expected, message_opt);
|
|
|
|
assert.notDeepEqual = function notDeepEqual(actual, expected, message) {
|
|
if (_deepEqual(actual, expected)) {
|
|
fail(actual, expected, message, 'notDeepEqual', assert.notDeepEqual);
|
|
}
|
|
};
|
|
|
|
// 9. The strict equality assertion tests strict equality, as determined by ===.
|
|
// assert.strictEqual(actual, expected, message_opt);
|
|
|
|
assert.strictEqual = function strictEqual(actual, expected, message) {
|
|
if (actual !== expected) {
|
|
fail(actual, expected, message, '===', assert.strictEqual);
|
|
}
|
|
};
|
|
|
|
// 10. The strict non-equality assertion tests for strict inequality, as
|
|
// determined by !==. assert.notStrictEqual(actual, expected, message_opt);
|
|
|
|
assert.notStrictEqual = function notStrictEqual(actual, expected, message) {
|
|
if (actual === expected) {
|
|
fail(actual, expected, message, '!==', assert.notStrictEqual);
|
|
}
|
|
};
|
|
|
|
function expectedException(actual, expected) {
|
|
if (!actual || !expected) {
|
|
return false;
|
|
}
|
|
|
|
if (Object.prototype.toString.call(expected) == '[object RegExp]') {
|
|
return expected.test(actual);
|
|
} else if (actual instanceof expected) {
|
|
return true;
|
|
} else if (expected.call({}, actual) === true) {
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
function _throws(shouldThrow, block, expected, message) {
|
|
var actual;
|
|
|
|
if (util.isString(expected)) {
|
|
message = expected;
|
|
expected = null;
|
|
}
|
|
|
|
try {
|
|
block();
|
|
} catch (e) {
|
|
actual = e;
|
|
}
|
|
|
|
message = (expected && expected.name ? ' (' + expected.name + ').' : '.') +
|
|
(message ? ' ' + message : '.');
|
|
|
|
if (shouldThrow && !actual) {
|
|
fail(actual, expected, 'Missing expected exception' + message);
|
|
}
|
|
|
|
if (!shouldThrow && expectedException(actual, expected)) {
|
|
fail(actual, expected, 'Got unwanted exception' + message);
|
|
}
|
|
|
|
if ((shouldThrow && actual && expected &&
|
|
!expectedException(actual, expected)) || (!shouldThrow && actual)) {
|
|
throw actual;
|
|
}
|
|
}
|
|
|
|
// 11. Expected to throw an error:
|
|
// assert.throws(block, Error_opt, message_opt);
|
|
|
|
assert.throws = function(block, /*optional*/error, /*optional*/message) {
|
|
_throws.apply(this, [true].concat(pSlice.call(arguments)));
|
|
};
|
|
|
|
// EXTENSION! This is annoying to write outside this module.
|
|
assert.doesNotThrow = function(block, /*optional*/message) {
|
|
_throws.apply(this, [false].concat(pSlice.call(arguments)));
|
|
};
|
|
|
|
assert.ifError = function(err) { if (err) {throw err;}};
|
|
},{"_shims":5,"util":7}],7:[function(require,module,exports){
|
|
// Copyright Joyent, Inc. and other Node contributors.
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining a
|
|
// copy of this software and associated documentation files (the
|
|
// "Software"), to deal in the Software without restriction, including
|
|
// without limitation the rights to use, copy, modify, merge, publish,
|
|
// distribute, sublicense, and/or sell copies of the Software, and to permit
|
|
// persons to whom the Software is furnished to do so, subject to the
|
|
// following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be included
|
|
// in all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
|
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
|
|
// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
|
|
// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
|
|
// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
|
|
// USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
var shims = require('_shims');
|
|
|
|
var formatRegExp = /%[sdj%]/g;
|
|
exports.format = function(f) {
|
|
if (!isString(f)) {
|
|
var objects = [];
|
|
for (var i = 0; i < arguments.length; i++) {
|
|
objects.push(inspect(arguments[i]));
|
|
}
|
|
return objects.join(' ');
|
|
}
|
|
|
|
var i = 1;
|
|
var args = arguments;
|
|
var len = args.length;
|
|
var str = String(f).replace(formatRegExp, function(x) {
|
|
if (x === '%%') return '%';
|
|
if (i >= len) return x;
|
|
switch (x) {
|
|
case '%s': return String(args[i++]);
|
|
case '%d': return Number(args[i++]);
|
|
case '%j':
|
|
try {
|
|
return JSON.stringify(args[i++]);
|
|
} catch (_) {
|
|
return '[Circular]';
|
|
}
|
|
default:
|
|
return x;
|
|
}
|
|
});
|
|
for (var x = args[i]; i < len; x = args[++i]) {
|
|
if (isNull(x) || !isObject(x)) {
|
|
str += ' ' + x;
|
|
} else {
|
|
str += ' ' + inspect(x);
|
|
}
|
|
}
|
|
return str;
|
|
};
|
|
|
|
/**
|
|
* Echos the value of a value. Trys to print the value out
|
|
* in the best way possible given the different types.
|
|
*
|
|
* @param {Object} obj The object to print out.
|
|
* @param {Object} opts Optional options object that alters the output.
|
|
*/
|
|
/* legacy: obj, showHidden, depth, colors*/
|
|
function inspect(obj, opts) {
|
|
// default options
|
|
var ctx = {
|
|
seen: [],
|
|
stylize: stylizeNoColor
|
|
};
|
|
// legacy...
|
|
if (arguments.length >= 3) ctx.depth = arguments[2];
|
|
if (arguments.length >= 4) ctx.colors = arguments[3];
|
|
if (isBoolean(opts)) {
|
|
// legacy...
|
|
ctx.showHidden = opts;
|
|
} else if (opts) {
|
|
// got an "options" object
|
|
exports._extend(ctx, opts);
|
|
}
|
|
// set default options
|
|
if (isUndefined(ctx.showHidden)) ctx.showHidden = false;
|
|
if (isUndefined(ctx.depth)) ctx.depth = 2;
|
|
if (isUndefined(ctx.colors)) ctx.colors = false;
|
|
if (isUndefined(ctx.customInspect)) ctx.customInspect = true;
|
|
if (ctx.colors) ctx.stylize = stylizeWithColor;
|
|
return formatValue(ctx, obj, ctx.depth);
|
|
}
|
|
exports.inspect = inspect;
|
|
|
|
|
|
// http://en.wikipedia.org/wiki/ANSI_escape_code#graphics
|
|
inspect.colors = {
|
|
'bold' : [1, 22],
|
|
'italic' : [3, 23],
|
|
'underline' : [4, 24],
|
|
'inverse' : [7, 27],
|
|
'white' : [37, 39],
|
|
'grey' : [90, 39],
|
|
'black' : [30, 39],
|
|
'blue' : [34, 39],
|
|
'cyan' : [36, 39],
|
|
'green' : [32, 39],
|
|
'magenta' : [35, 39],
|
|
'red' : [31, 39],
|
|
'yellow' : [33, 39]
|
|
};
|
|
|
|
// Don't use 'blue' not visible on cmd.exe
|
|
inspect.styles = {
|
|
'special': 'cyan',
|
|
'number': 'yellow',
|
|
'boolean': 'yellow',
|
|
'undefined': 'grey',
|
|
'null': 'bold',
|
|
'string': 'green',
|
|
'date': 'magenta',
|
|
// "name": intentionally not styling
|
|
'regexp': 'red'
|
|
};
|
|
|
|
|
|
function stylizeWithColor(str, styleType) {
|
|
var style = inspect.styles[styleType];
|
|
|
|
if (style) {
|
|
return '\u001b[' + inspect.colors[style][0] + 'm' + str +
|
|
'\u001b[' + inspect.colors[style][1] + 'm';
|
|
} else {
|
|
return str;
|
|
}
|
|
}
|
|
|
|
|
|
function stylizeNoColor(str, styleType) {
|
|
return str;
|
|
}
|
|
|
|
|
|
function arrayToHash(array) {
|
|
var hash = {};
|
|
|
|
shims.forEach(array, function(val, idx) {
|
|
hash[val] = true;
|
|
});
|
|
|
|
return hash;
|
|
}
|
|
|
|
|
|
function formatValue(ctx, value, recurseTimes) {
|
|
// Provide a hook for user-specified inspect functions.
|
|
// Check that value is an object with an inspect function on it
|
|
if (ctx.customInspect &&
|
|
value &&
|
|
isFunction(value.inspect) &&
|
|
// Filter out the util module, it's inspect function is special
|
|
value.inspect !== exports.inspect &&
|
|
// Also filter out any prototype objects using the circular check.
|
|
!(value.constructor && value.constructor.prototype === value)) {
|
|
var ret = value.inspect(recurseTimes);
|
|
if (!isString(ret)) {
|
|
ret = formatValue(ctx, ret, recurseTimes);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
// Primitive types cannot have properties
|
|
var primitive = formatPrimitive(ctx, value);
|
|
if (primitive) {
|
|
return primitive;
|
|
}
|
|
|
|
// Look up the keys of the object.
|
|
var keys = shims.keys(value);
|
|
var visibleKeys = arrayToHash(keys);
|
|
|
|
if (ctx.showHidden) {
|
|
keys = shims.getOwnPropertyNames(value);
|
|
}
|
|
|
|
// Some type of object without properties can be shortcutted.
|
|
if (keys.length === 0) {
|
|
if (isFunction(value)) {
|
|
var name = value.name ? ': ' + value.name : '';
|
|
return ctx.stylize('[Function' + name + ']', 'special');
|
|
}
|
|
if (isRegExp(value)) {
|
|
return ctx.stylize(RegExp.prototype.toString.call(value), 'regexp');
|
|
}
|
|
if (isDate(value)) {
|
|
return ctx.stylize(Date.prototype.toString.call(value), 'date');
|
|
}
|
|
if (isError(value)) {
|
|
return formatError(value);
|
|
}
|
|
}
|
|
|
|
var base = '', array = false, braces = ['{', '}'];
|
|
|
|
// Make Array say that they are Array
|
|
if (isArray(value)) {
|
|
array = true;
|
|
braces = ['[', ']'];
|
|
}
|
|
|
|
// Make functions say that they are functions
|
|
if (isFunction(value)) {
|
|
var n = value.name ? ': ' + value.name : '';
|
|
base = ' [Function' + n + ']';
|
|
}
|
|
|
|
// Make RegExps say that they are RegExps
|
|
if (isRegExp(value)) {
|
|
base = ' ' + RegExp.prototype.toString.call(value);
|
|
}
|
|
|
|
// Make dates with properties first say the date
|
|
if (isDate(value)) {
|
|
base = ' ' + Date.prototype.toUTCString.call(value);
|
|
}
|
|
|
|
// Make error with message first say the error
|
|
if (isError(value)) {
|
|
base = ' ' + formatError(value);
|
|
}
|
|
|
|
if (keys.length === 0 && (!array || value.length == 0)) {
|
|
return braces[0] + base + braces[1];
|
|
}
|
|
|
|
if (recurseTimes < 0) {
|
|
if (isRegExp(value)) {
|
|
return ctx.stylize(RegExp.prototype.toString.call(value), 'regexp');
|
|
} else {
|
|
return ctx.stylize('[Object]', 'special');
|
|
}
|
|
}
|
|
|
|
ctx.seen.push(value);
|
|
|
|
var output;
|
|
if (array) {
|
|
output = formatArray(ctx, value, recurseTimes, visibleKeys, keys);
|
|
} else {
|
|
output = keys.map(function(key) {
|
|
return formatProperty(ctx, value, recurseTimes, visibleKeys, key, array);
|
|
});
|
|
}
|
|
|
|
ctx.seen.pop();
|
|
|
|
return reduceToSingleString(output, base, braces);
|
|
}
|
|
|
|
|
|
function formatPrimitive(ctx, value) {
|
|
if (isUndefined(value))
|
|
return ctx.stylize('undefined', 'undefined');
|
|
if (isString(value)) {
|
|
var simple = '\'' + JSON.stringify(value).replace(/^"|"$/g, '')
|
|
.replace(/'/g, "\\'")
|
|
.replace(/\\"/g, '"') + '\'';
|
|
return ctx.stylize(simple, 'string');
|
|
}
|
|
if (isNumber(value))
|
|
return ctx.stylize('' + value, 'number');
|
|
if (isBoolean(value))
|
|
return ctx.stylize('' + value, 'boolean');
|
|
// For some reason typeof null is "object", so special case here.
|
|
if (isNull(value))
|
|
return ctx.stylize('null', 'null');
|
|
}
|
|
|
|
|
|
function formatError(value) {
|
|
return '[' + Error.prototype.toString.call(value) + ']';
|
|
}
|
|
|
|
|
|
function formatArray(ctx, value, recurseTimes, visibleKeys, keys) {
|
|
var output = [];
|
|
for (var i = 0, l = value.length; i < l; ++i) {
|
|
if (hasOwnProperty(value, String(i))) {
|
|
output.push(formatProperty(ctx, value, recurseTimes, visibleKeys,
|
|
String(i), true));
|
|
} else {
|
|
output.push('');
|
|
}
|
|
}
|
|
|
|
shims.forEach(keys, function(key) {
|
|
if (!key.match(/^\d+$/)) {
|
|
output.push(formatProperty(ctx, value, recurseTimes, visibleKeys,
|
|
key, true));
|
|
}
|
|
});
|
|
return output;
|
|
}
|
|
|
|
|
|
function formatProperty(ctx, value, recurseTimes, visibleKeys, key, array) {
|
|
var name, str, desc;
|
|
desc = shims.getOwnPropertyDescriptor(value, key) || { value: value[key] };
|
|
if (desc.get) {
|
|
if (desc.set) {
|
|
str = ctx.stylize('[Getter/Setter]', 'special');
|
|
} else {
|
|
str = ctx.stylize('[Getter]', 'special');
|
|
}
|
|
} else {
|
|
if (desc.set) {
|
|
str = ctx.stylize('[Setter]', 'special');
|
|
}
|
|
}
|
|
|
|
if (!hasOwnProperty(visibleKeys, key)) {
|
|
name = '[' + key + ']';
|
|
}
|
|
if (!str) {
|
|
if (shims.indexOf(ctx.seen, desc.value) < 0) {
|
|
if (isNull(recurseTimes)) {
|
|
str = formatValue(ctx, desc.value, null);
|
|
} else {
|
|
str = formatValue(ctx, desc.value, recurseTimes - 1);
|
|
}
|
|
if (str.indexOf('\n') > -1) {
|
|
if (array) {
|
|
str = str.split('\n').map(function(line) {
|
|
return ' ' + line;
|
|
}).join('\n').substr(2);
|
|
} else {
|
|
str = '\n' + str.split('\n').map(function(line) {
|
|
return ' ' + line;
|
|
}).join('\n');
|
|
}
|
|
}
|
|
} else {
|
|
str = ctx.stylize('[Circular]', 'special');
|
|
}
|
|
}
|
|
if (isUndefined(name)) {
|
|
if (array && key.match(/^\d+$/)) {
|
|
return str;
|
|
}
|
|
name = JSON.stringify('' + key);
|
|
if (name.match(/^"([a-zA-Z_][a-zA-Z_0-9]*)"$/)) {
|
|
name = name.substr(1, name.length - 2);
|
|
name = ctx.stylize(name, 'name');
|
|
} else {
|
|
name = name.replace(/'/g, "\\'")
|
|
.replace(/\\"/g, '"')
|
|
.replace(/(^"|"$)/g, "'");
|
|
name = ctx.stylize(name, 'string');
|
|
}
|
|
}
|
|
|
|
return name + ': ' + str;
|
|
}
|
|
|
|
|
|
function reduceToSingleString(output, base, braces) {
|
|
var numLinesEst = 0;
|
|
var length = shims.reduce(output, function(prev, cur) {
|
|
numLinesEst++;
|
|
if (cur.indexOf('\n') >= 0) numLinesEst++;
|
|
return prev + cur.replace(/\u001b\[\d\d?m/g, '').length + 1;
|
|
}, 0);
|
|
|
|
if (length > 60) {
|
|
return braces[0] +
|
|
(base === '' ? '' : base + '\n ') +
|
|
' ' +
|
|
output.join(',\n ') +
|
|
' ' +
|
|
braces[1];
|
|
}
|
|
|
|
return braces[0] + base + ' ' + output.join(', ') + ' ' + braces[1];
|
|
}
|
|
|
|
|
|
// NOTE: These type checking functions intentionally don't use `instanceof`
|
|
// because it is fragile and can be easily faked with `Object.create()`.
|
|
function isArray(ar) {
|
|
return shims.isArray(ar);
|
|
}
|
|
exports.isArray = isArray;
|
|
|
|
function isBoolean(arg) {
|
|
return typeof arg === 'boolean';
|
|
}
|
|
exports.isBoolean = isBoolean;
|
|
|
|
function isNull(arg) {
|
|
return arg === null;
|
|
}
|
|
exports.isNull = isNull;
|
|
|
|
function isNullOrUndefined(arg) {
|
|
return arg == null;
|
|
}
|
|
exports.isNullOrUndefined = isNullOrUndefined;
|
|
|
|
function isNumber(arg) {
|
|
return typeof arg === 'number';
|
|
}
|
|
exports.isNumber = isNumber;
|
|
|
|
function isString(arg) {
|
|
return typeof arg === 'string';
|
|
}
|
|
exports.isString = isString;
|
|
|
|
function isSymbol(arg) {
|
|
return typeof arg === 'symbol';
|
|
}
|
|
exports.isSymbol = isSymbol;
|
|
|
|
function isUndefined(arg) {
|
|
return arg === void 0;
|
|
}
|
|
exports.isUndefined = isUndefined;
|
|
|
|
function isRegExp(re) {
|
|
return isObject(re) && objectToString(re) === '[object RegExp]';
|
|
}
|
|
exports.isRegExp = isRegExp;
|
|
|
|
function isObject(arg) {
|
|
return typeof arg === 'object' && arg;
|
|
}
|
|
exports.isObject = isObject;
|
|
|
|
function isDate(d) {
|
|
return isObject(d) && objectToString(d) === '[object Date]';
|
|
}
|
|
exports.isDate = isDate;
|
|
|
|
function isError(e) {
|
|
return isObject(e) && objectToString(e) === '[object Error]';
|
|
}
|
|
exports.isError = isError;
|
|
|
|
function isFunction(arg) {
|
|
return typeof arg === 'function';
|
|
}
|
|
exports.isFunction = isFunction;
|
|
|
|
function isPrimitive(arg) {
|
|
return arg === null ||
|
|
typeof arg === 'boolean' ||
|
|
typeof arg === 'number' ||
|
|
typeof arg === 'string' ||
|
|
typeof arg === 'symbol' || // ES6 symbol
|
|
typeof arg === 'undefined';
|
|
}
|
|
exports.isPrimitive = isPrimitive;
|
|
|
|
function isBuffer(arg) {
|
|
return arg instanceof Buffer;
|
|
}
|
|
exports.isBuffer = isBuffer;
|
|
|
|
function objectToString(o) {
|
|
return Object.prototype.toString.call(o);
|
|
}
|
|
|
|
|
|
function pad(n) {
|
|
return n < 10 ? '0' + n.toString(10) : n.toString(10);
|
|
}
|
|
|
|
|
|
var months = ['Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug', 'Sep',
|
|
'Oct', 'Nov', 'Dec'];
|
|
|
|
// 26 Feb 16:19:34
|
|
function timestamp() {
|
|
var d = new Date();
|
|
var time = [pad(d.getHours()),
|
|
pad(d.getMinutes()),
|
|
pad(d.getSeconds())].join(':');
|
|
return [d.getDate(), months[d.getMonth()], time].join(' ');
|
|
}
|
|
|
|
|
|
// log is just a thin wrapper to console.log that prepends a timestamp
|
|
exports.log = function() {
|
|
console.log('%s - %s', timestamp(), exports.format.apply(exports, arguments));
|
|
};
|
|
|
|
|
|
/**
|
|
* Inherit the prototype methods from one constructor into another.
|
|
*
|
|
* The Function.prototype.inherits from lang.js rewritten as a standalone
|
|
* function (not on Function.prototype). NOTE: If this file is to be loaded
|
|
* during bootstrapping this function needs to be rewritten using some native
|
|
* functions as prototype setup using normal JavaScript does not work as
|
|
* expected during bootstrapping (see mirror.js in r114903).
|
|
*
|
|
* @param {function} ctor Constructor function which needs to inherit the
|
|
* prototype.
|
|
* @param {function} superCtor Constructor function to inherit prototype from.
|
|
*/
|
|
exports.inherits = function(ctor, superCtor) {
|
|
ctor.super_ = superCtor;
|
|
ctor.prototype = shims.create(superCtor.prototype, {
|
|
constructor: {
|
|
value: ctor,
|
|
enumerable: false,
|
|
writable: true,
|
|
configurable: true
|
|
}
|
|
});
|
|
};
|
|
|
|
exports._extend = function(origin, add) {
|
|
// Don't do anything if add isn't an object
|
|
if (!add || !isObject(add)) return origin;
|
|
|
|
var keys = shims.keys(add);
|
|
var i = keys.length;
|
|
while (i--) {
|
|
origin[keys[i]] = add[keys[i]];
|
|
}
|
|
return origin;
|
|
};
|
|
|
|
function hasOwnProperty(obj, prop) {
|
|
return Object.prototype.hasOwnProperty.call(obj, prop);
|
|
}
|
|
|
|
},{"_shims":5}]},{},[])
|
|
;;module.exports=require("buffer-browserify")
|
|
|
|
},{}],15:[function(require,module,exports){
|
|
// shim for using process in browser
|
|
|
|
var process = module.exports = {};
|
|
|
|
process.nextTick = (function () {
|
|
var canSetImmediate = typeof window !== 'undefined'
|
|
&& window.setImmediate;
|
|
var canPost = typeof window !== 'undefined'
|
|
&& window.postMessage && window.addEventListener
|
|
;
|
|
|
|
if (canSetImmediate) {
|
|
return function (f) { return window.setImmediate(f) };
|
|
}
|
|
|
|
if (canPost) {
|
|
var queue = [];
|
|
window.addEventListener('message', function (ev) {
|
|
if (ev.source === window && ev.data === 'process-tick') {
|
|
ev.stopPropagation();
|
|
if (queue.length > 0) {
|
|
var fn = queue.shift();
|
|
fn();
|
|
}
|
|
}
|
|
}, true);
|
|
|
|
return function nextTick(fn) {
|
|
queue.push(fn);
|
|
window.postMessage('process-tick', '*');
|
|
};
|
|
}
|
|
|
|
return function nextTick(fn) {
|
|
setTimeout(fn, 0);
|
|
};
|
|
})();
|
|
|
|
process.title = 'browser';
|
|
process.browser = true;
|
|
process.env = {};
|
|
process.argv = [];
|
|
|
|
process.binding = function (name) {
|
|
throw new Error('process.binding is not supported');
|
|
}
|
|
|
|
// TODO(shtylman)
|
|
process.cwd = function () { return '/' };
|
|
process.chdir = function (dir) {
|
|
throw new Error('process.chdir is not supported');
|
|
};
|
|
|
|
},{}],16:[function(require,module,exports){
|
|
var Buffer=require("__browserify_Buffer").Buffer;
|
|
(function (global, module) {
|
|
|
|
if ('undefined' == typeof module) {
|
|
var module = { exports: {} }
|
|
, exports = module.exports
|
|
}
|
|
|
|
/**
|
|
* Exports.
|
|
*/
|
|
|
|
module.exports = expect;
|
|
expect.Assertion = Assertion;
|
|
|
|
/**
|
|
* Exports version.
|
|
*/
|
|
|
|
expect.version = '0.1.2';
|
|
|
|
/**
|
|
* Possible assertion flags.
|
|
*/
|
|
|
|
var flags = {
|
|
not: ['to', 'be', 'have', 'include', 'only']
|
|
, to: ['be', 'have', 'include', 'only', 'not']
|
|
, only: ['have']
|
|
, have: ['own']
|
|
, be: ['an']
|
|
};
|
|
|
|
function expect (obj) {
|
|
return new Assertion(obj);
|
|
}
|
|
|
|
/**
|
|
* Constructor
|
|
*
|
|
* @api private
|
|
*/
|
|
|
|
function Assertion (obj, flag, parent) {
|
|
this.obj = obj;
|
|
this.flags = {};
|
|
|
|
if (undefined != parent) {
|
|
this.flags[flag] = true;
|
|
|
|
for (var i in parent.flags) {
|
|
if (parent.flags.hasOwnProperty(i)) {
|
|
this.flags[i] = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
var $flags = flag ? flags[flag] : keys(flags)
|
|
, self = this
|
|
|
|
if ($flags) {
|
|
for (var i = 0, l = $flags.length; i < l; i++) {
|
|
// avoid recursion
|
|
if (this.flags[$flags[i]]) continue;
|
|
|
|
var name = $flags[i]
|
|
, assertion = new Assertion(this.obj, name, this)
|
|
|
|
if ('function' == typeof Assertion.prototype[name]) {
|
|
// clone the function, make sure we dont touch the prot reference
|
|
var old = this[name];
|
|
this[name] = function () {
|
|
return old.apply(self, arguments);
|
|
}
|
|
|
|
for (var fn in Assertion.prototype) {
|
|
if (Assertion.prototype.hasOwnProperty(fn) && fn != name) {
|
|
this[name][fn] = bind(assertion[fn], assertion);
|
|
}
|
|
}
|
|
} else {
|
|
this[name] = assertion;
|
|
}
|
|
}
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Performs an assertion
|
|
*
|
|
* @api private
|
|
*/
|
|
|
|
Assertion.prototype.assert = function (truth, msg, error) {
|
|
var msg = this.flags.not ? error : msg
|
|
, ok = this.flags.not ? !truth : truth;
|
|
|
|
if (!ok) {
|
|
throw new Error(msg.call(this));
|
|
}
|
|
|
|
this.and = new Assertion(this.obj);
|
|
};
|
|
|
|
/**
|
|
* Check if the value is truthy
|
|
*
|
|
* @api public
|
|
*/
|
|
|
|
Assertion.prototype.ok = function () {
|
|
this.assert(
|
|
!!this.obj
|
|
, function(){ return 'expected ' + i(this.obj) + ' to be truthy' }
|
|
, function(){ return 'expected ' + i(this.obj) + ' to be falsy' });
|
|
};
|
|
|
|
/**
|
|
* Assert that the function throws.
|
|
*
|
|
* @param {Function|RegExp} callback, or regexp to match error string against
|
|
* @api public
|
|
*/
|
|
|
|
Assertion.prototype.throwError =
|
|
Assertion.prototype.throwException = function (fn) {
|
|
expect(this.obj).to.be.a('function');
|
|
|
|
var thrown = false
|
|
, not = this.flags.not
|
|
|
|
try {
|
|
this.obj();
|
|
} catch (e) {
|
|
if ('function' == typeof fn) {
|
|
fn(e);
|
|
} else if ('object' == typeof fn) {
|
|
var subject = 'string' == typeof e ? e : e.message;
|
|
if (not) {
|
|
expect(subject).to.not.match(fn);
|
|
} else {
|
|
expect(subject).to.match(fn);
|
|
}
|
|
}
|
|
thrown = true;
|
|
}
|
|
|
|
if ('object' == typeof fn && not) {
|
|
// in the presence of a matcher, ensure the `not` only applies to
|
|
// the matching.
|
|
this.flags.not = false;
|
|
}
|
|
|
|
var name = this.obj.name || 'fn';
|
|
this.assert(
|
|
thrown
|
|
, function(){ return 'expected ' + name + ' to throw an exception' }
|
|
, function(){ return 'expected ' + name + ' not to throw an exception' });
|
|
};
|
|
|
|
/**
|
|
* Checks if the array is empty.
|
|
*
|
|
* @api public
|
|
*/
|
|
|
|
Assertion.prototype.empty = function () {
|
|
var expectation;
|
|
|
|
if ('object' == typeof this.obj && null !== this.obj && !isArray(this.obj)) {
|
|
if ('number' == typeof this.obj.length) {
|
|
expectation = !this.obj.length;
|
|
} else {
|
|
expectation = !keys(this.obj).length;
|
|
}
|
|
} else {
|
|
if ('string' != typeof this.obj) {
|
|
expect(this.obj).to.be.an('object');
|
|
}
|
|
|
|
expect(this.obj).to.have.property('length');
|
|
expectation = !this.obj.length;
|
|
}
|
|
|
|
this.assert(
|
|
expectation
|
|
, function(){ return 'expected ' + i(this.obj) + ' to be empty' }
|
|
, function(){ return 'expected ' + i(this.obj) + ' to not be empty' });
|
|
return this;
|
|
};
|
|
|
|
/**
|
|
* Checks if the obj exactly equals another.
|
|
*
|
|
* @api public
|
|
*/
|
|
|
|
Assertion.prototype.be =
|
|
Assertion.prototype.equal = function (obj) {
|
|
this.assert(
|
|
obj === this.obj
|
|
, function(){ return 'expected ' + i(this.obj) + ' to equal ' + i(obj) }
|
|
, function(){ return 'expected ' + i(this.obj) + ' to not equal ' + i(obj) });
|
|
return this;
|
|
};
|
|
|
|
/**
|
|
* Checks if the obj sortof equals another.
|
|
*
|
|
* @api public
|
|
*/
|
|
|
|
Assertion.prototype.eql = function (obj) {
|
|
this.assert(
|
|
expect.eql(obj, this.obj)
|
|
, function(){ return 'expected ' + i(this.obj) + ' to sort of equal ' + i(obj) }
|
|
, function(){ return 'expected ' + i(this.obj) + ' to sort of not equal ' + i(obj) });
|
|
return this;
|
|
};
|
|
|
|
/**
|
|
* Assert within start to finish (inclusive).
|
|
*
|
|
* @param {Number} start
|
|
* @param {Number} finish
|
|
* @api public
|
|
*/
|
|
|
|
Assertion.prototype.within = function (start, finish) {
|
|
var range = start + '..' + finish;
|
|
this.assert(
|
|
this.obj >= start && this.obj <= finish
|
|
, function(){ return 'expected ' + i(this.obj) + ' to be within ' + range }
|
|
, function(){ return 'expected ' + i(this.obj) + ' to not be within ' + range });
|
|
return this;
|
|
};
|
|
|
|
/**
|
|
* Assert typeof / instance of
|
|
*
|
|
* @api public
|
|
*/
|
|
|
|
Assertion.prototype.a =
|
|
Assertion.prototype.an = function (type) {
|
|
if ('string' == typeof type) {
|
|
// proper english in error msg
|
|
var n = /^[aeiou]/.test(type) ? 'n' : '';
|
|
|
|
// typeof with support for 'array'
|
|
this.assert(
|
|
'array' == type ? isArray(this.obj) :
|
|
'object' == type
|
|
? 'object' == typeof this.obj && null !== this.obj
|
|
: type == typeof this.obj
|
|
, function(){ return 'expected ' + i(this.obj) + ' to be a' + n + ' ' + type }
|
|
, function(){ return 'expected ' + i(this.obj) + ' not to be a' + n + ' ' + type });
|
|
} else {
|
|
// instanceof
|
|
var name = type.name || 'supplied constructor';
|
|
this.assert(
|
|
this.obj instanceof type
|
|
, function(){ return 'expected ' + i(this.obj) + ' to be an instance of ' + name }
|
|
, function(){ return 'expected ' + i(this.obj) + ' not to be an instance of ' + name });
|
|
}
|
|
|
|
return this;
|
|
};
|
|
|
|
/**
|
|
* Assert numeric value above _n_.
|
|
*
|
|
* @param {Number} n
|
|
* @api public
|
|
*/
|
|
|
|
Assertion.prototype.greaterThan =
|
|
Assertion.prototype.above = function (n) {
|
|
this.assert(
|
|
this.obj > n
|
|
, function(){ return 'expected ' + i(this.obj) + ' to be above ' + n }
|
|
, function(){ return 'expected ' + i(this.obj) + ' to be below ' + n });
|
|
return this;
|
|
};
|
|
|
|
/**
|
|
* Assert numeric value below _n_.
|
|
*
|
|
* @param {Number} n
|
|
* @api public
|
|
*/
|
|
|
|
Assertion.prototype.lessThan =
|
|
Assertion.prototype.below = function (n) {
|
|
this.assert(
|
|
this.obj < n
|
|
, function(){ return 'expected ' + i(this.obj) + ' to be below ' + n }
|
|
, function(){ return 'expected ' + i(this.obj) + ' to be above ' + n });
|
|
return this;
|
|
};
|
|
|
|
/**
|
|
* Assert string value matches _regexp_.
|
|
*
|
|
* @param {RegExp} regexp
|
|
* @api public
|
|
*/
|
|
|
|
Assertion.prototype.match = function (regexp) {
|
|
this.assert(
|
|
regexp.exec(this.obj)
|
|
, function(){ return 'expected ' + i(this.obj) + ' to match ' + regexp }
|
|
, function(){ return 'expected ' + i(this.obj) + ' not to match ' + regexp });
|
|
return this;
|
|
};
|
|
|
|
/**
|
|
* Assert property "length" exists and has value of _n_.
|
|
*
|
|
* @param {Number} n
|
|
* @api public
|
|
*/
|
|
|
|
Assertion.prototype.length = function (n) {
|
|
expect(this.obj).to.have.property('length');
|
|
var len = this.obj.length;
|
|
this.assert(
|
|
n == len
|
|
, function(){ return 'expected ' + i(this.obj) + ' to have a length of ' + n + ' but got ' + len }
|
|
, function(){ return 'expected ' + i(this.obj) + ' to not have a length of ' + len });
|
|
return this;
|
|
};
|
|
|
|
/**
|
|
* Assert property _name_ exists, with optional _val_.
|
|
*
|
|
* @param {String} name
|
|
* @param {Mixed} val
|
|
* @api public
|
|
*/
|
|
|
|
Assertion.prototype.property = function (name, val) {
|
|
if (this.flags.own) {
|
|
this.assert(
|
|
Object.prototype.hasOwnProperty.call(this.obj, name)
|
|
, function(){ return 'expected ' + i(this.obj) + ' to have own property ' + i(name) }
|
|
, function(){ return 'expected ' + i(this.obj) + ' to not have own property ' + i(name) });
|
|
return this;
|
|
}
|
|
|
|
if (this.flags.not && undefined !== val) {
|
|
if (undefined === this.obj[name]) {
|
|
throw new Error(i(this.obj) + ' has no property ' + i(name));
|
|
}
|
|
} else {
|
|
var hasProp;
|
|
try {
|
|
hasProp = name in this.obj
|
|
} catch (e) {
|
|
hasProp = undefined !== this.obj[name]
|
|
}
|
|
|
|
this.assert(
|
|
hasProp
|
|
, function(){ return 'expected ' + i(this.obj) + ' to have a property ' + i(name) }
|
|
, function(){ return 'expected ' + i(this.obj) + ' to not have a property ' + i(name) });
|
|
}
|
|
|
|
if (undefined !== val) {
|
|
this.assert(
|
|
val === this.obj[name]
|
|
, function(){ return 'expected ' + i(this.obj) + ' to have a property ' + i(name)
|
|
+ ' of ' + i(val) + ', but got ' + i(this.obj[name]) }
|
|
, function(){ return 'expected ' + i(this.obj) + ' to not have a property ' + i(name)
|
|
+ ' of ' + i(val) });
|
|
}
|
|
|
|
this.obj = this.obj[name];
|
|
return this;
|
|
};
|
|
|
|
/**
|
|
* Assert that the array contains _obj_ or string contains _obj_.
|
|
*
|
|
* @param {Mixed} obj|string
|
|
* @api public
|
|
*/
|
|
|
|
Assertion.prototype.string =
|
|
Assertion.prototype.contain = function (obj) {
|
|
if ('string' == typeof this.obj) {
|
|
this.assert(
|
|
~this.obj.indexOf(obj)
|
|
, function(){ return 'expected ' + i(this.obj) + ' to contain ' + i(obj) }
|
|
, function(){ return 'expected ' + i(this.obj) + ' to not contain ' + i(obj) });
|
|
} else {
|
|
this.assert(
|
|
~indexOf(this.obj, obj)
|
|
, function(){ return 'expected ' + i(this.obj) + ' to contain ' + i(obj) }
|
|
, function(){ return 'expected ' + i(this.obj) + ' to not contain ' + i(obj) });
|
|
}
|
|
return this;
|
|
};
|
|
|
|
/**
|
|
* Assert exact keys or inclusion of keys by using
|
|
* the `.own` modifier.
|
|
*
|
|
* @param {Array|String ...} keys
|
|
* @api public
|
|
*/
|
|
|
|
Assertion.prototype.key =
|
|
Assertion.prototype.keys = function ($keys) {
|
|
var str
|
|
, ok = true;
|
|
|
|
$keys = isArray($keys)
|
|
? $keys
|
|
: Array.prototype.slice.call(arguments);
|
|
|
|
if (!$keys.length) throw new Error('keys required');
|
|
|
|
var actual = keys(this.obj)
|
|
, len = $keys.length;
|
|
|
|
// Inclusion
|
|
ok = every($keys, function (key) {
|
|
return ~indexOf(actual, key);
|
|
});
|
|
|
|
// Strict
|
|
if (!this.flags.not && this.flags.only) {
|
|
ok = ok && $keys.length == actual.length;
|
|
}
|
|
|
|
// Key string
|
|
if (len > 1) {
|
|
$keys = map($keys, function (key) {
|
|
return i(key);
|
|
});
|
|
var last = $keys.pop();
|
|
str = $keys.join(', ') + ', and ' + last;
|
|
} else {
|
|
str = i($keys[0]);
|
|
}
|
|
|
|
// Form
|
|
str = (len > 1 ? 'keys ' : 'key ') + str;
|
|
|
|
// Have / include
|
|
str = (!this.flags.only ? 'include ' : 'only have ') + str;
|
|
|
|
// Assertion
|
|
this.assert(
|
|
ok
|
|
, function(){ return 'expected ' + i(this.obj) + ' to ' + str }
|
|
, function(){ return 'expected ' + i(this.obj) + ' to not ' + str });
|
|
|
|
return this;
|
|
};
|
|
/**
|
|
* Assert a failure.
|
|
*
|
|
* @param {String ...} custom message
|
|
* @api public
|
|
*/
|
|
Assertion.prototype.fail = function (msg) {
|
|
msg = msg || "explicit failure";
|
|
this.assert(false, msg, msg);
|
|
return this;
|
|
};
|
|
|
|
/**
|
|
* Function bind implementation.
|
|
*/
|
|
|
|
function bind (fn, scope) {
|
|
return function () {
|
|
return fn.apply(scope, arguments);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Array every compatibility
|
|
*
|
|
* @see bit.ly/5Fq1N2
|
|
* @api public
|
|
*/
|
|
|
|
function every (arr, fn, thisObj) {
|
|
var scope = thisObj || global;
|
|
for (var i = 0, j = arr.length; i < j; ++i) {
|
|
if (!fn.call(scope, arr[i], i, arr)) {
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
};
|
|
|
|
/**
|
|
* Array indexOf compatibility.
|
|
*
|
|
* @see bit.ly/a5Dxa2
|
|
* @api public
|
|
*/
|
|
|
|
function indexOf (arr, o, i) {
|
|
if (Array.prototype.indexOf) {
|
|
return Array.prototype.indexOf.call(arr, o, i);
|
|
}
|
|
|
|
if (arr.length === undefined) {
|
|
return -1;
|
|
}
|
|
|
|
for (var j = arr.length, i = i < 0 ? i + j < 0 ? 0 : i + j : i || 0
|
|
; i < j && arr[i] !== o; i++);
|
|
|
|
return j <= i ? -1 : i;
|
|
};
|
|
|
|
// https://gist.github.com/1044128/
|
|
var getOuterHTML = function(element) {
|
|
if ('outerHTML' in element) return element.outerHTML;
|
|
var ns = "http://www.w3.org/1999/xhtml";
|
|
var container = document.createElementNS(ns, '_');
|
|
var elemProto = (window.HTMLElement || window.Element).prototype;
|
|
var xmlSerializer = new XMLSerializer();
|
|
var html;
|
|
if (document.xmlVersion) {
|
|
return xmlSerializer.serializeToString(element);
|
|
} else {
|
|
container.appendChild(element.cloneNode(false));
|
|
html = container.innerHTML.replace('><', '>' + element.innerHTML + '<');
|
|
container.innerHTML = '';
|
|
return html;
|
|
}
|
|
};
|
|
|
|
// Returns true if object is a DOM element.
|
|
var isDOMElement = function (object) {
|
|
if (typeof HTMLElement === 'object') {
|
|
return object instanceof HTMLElement;
|
|
} else {
|
|
return object &&
|
|
typeof object === 'object' &&
|
|
object.nodeType === 1 &&
|
|
typeof object.nodeName === 'string';
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Inspects an object.
|
|
*
|
|
* @see taken from node.js `util` module (copyright Joyent, MIT license)
|
|
* @api private
|
|
*/
|
|
|
|
function i (obj, showHidden, depth) {
|
|
var seen = [];
|
|
|
|
function stylize (str) {
|
|
return str;
|
|
};
|
|
|
|
function format (value, recurseTimes) {
|
|
// Provide a hook for user-specified inspect functions.
|
|
// Check that value is an object with an inspect function on it
|
|
if (value && typeof value.inspect === 'function' &&
|
|
// Filter out the util module, it's inspect function is special
|
|
value !== exports &&
|
|
// Also filter out any prototype objects using the circular check.
|
|
!(value.constructor && value.constructor.prototype === value)) {
|
|
return value.inspect(recurseTimes);
|
|
}
|
|
|
|
// Primitive types cannot have properties
|
|
switch (typeof value) {
|
|
case 'undefined':
|
|
return stylize('undefined', 'undefined');
|
|
|
|
case 'string':
|
|
var simple = '\'' + json.stringify(value).replace(/^"|"$/g, '')
|
|
.replace(/'/g, "\\'")
|
|
.replace(/\\"/g, '"') + '\'';
|
|
return stylize(simple, 'string');
|
|
|
|
case 'number':
|
|
return stylize('' + value, 'number');
|
|
|
|
case 'boolean':
|
|
return stylize('' + value, 'boolean');
|
|
}
|
|
// For some reason typeof null is "object", so special case here.
|
|
if (value === null) {
|
|
return stylize('null', 'null');
|
|
}
|
|
|
|
if (isDOMElement(value)) {
|
|
return getOuterHTML(value);
|
|
}
|
|
|
|
// Look up the keys of the object.
|
|
var visible_keys = keys(value);
|
|
var $keys = showHidden ? Object.getOwnPropertyNames(value) : visible_keys;
|
|
|
|
// Functions without properties can be shortcutted.
|
|
if (typeof value === 'function' && $keys.length === 0) {
|
|
if (isRegExp(value)) {
|
|
return stylize('' + value, 'regexp');
|
|
} else {
|
|
var name = value.name ? ': ' + value.name : '';
|
|
return stylize('[Function' + name + ']', 'special');
|
|
}
|
|
}
|
|
|
|
// Dates without properties can be shortcutted
|
|
if (isDate(value) && $keys.length === 0) {
|
|
return stylize(value.toUTCString(), 'date');
|
|
}
|
|
|
|
var base, type, braces;
|
|
// Determine the object type
|
|
if (isArray(value)) {
|
|
type = 'Array';
|
|
braces = ['[', ']'];
|
|
} else {
|
|
type = 'Object';
|
|
braces = ['{', '}'];
|
|
}
|
|
|
|
// Make functions say that they are functions
|
|
if (typeof value === 'function') {
|
|
var n = value.name ? ': ' + value.name : '';
|
|
base = (isRegExp(value)) ? ' ' + value : ' [Function' + n + ']';
|
|
} else {
|
|
base = '';
|
|
}
|
|
|
|
// Make dates with properties first say the date
|
|
if (isDate(value)) {
|
|
base = ' ' + value.toUTCString();
|
|
}
|
|
|
|
if ($keys.length === 0) {
|
|
return braces[0] + base + braces[1];
|
|
}
|
|
|
|
if (recurseTimes < 0) {
|
|
if (isRegExp(value)) {
|
|
return stylize('' + value, 'regexp');
|
|
} else {
|
|
return stylize('[Object]', 'special');
|
|
}
|
|
}
|
|
|
|
seen.push(value);
|
|
|
|
var output = map($keys, function (key) {
|
|
var name, str;
|
|
if (value.__lookupGetter__) {
|
|
if (value.__lookupGetter__(key)) {
|
|
if (value.__lookupSetter__(key)) {
|
|
str = stylize('[Getter/Setter]', 'special');
|
|
} else {
|
|
str = stylize('[Getter]', 'special');
|
|
}
|
|
} else {
|
|
if (value.__lookupSetter__(key)) {
|
|
str = stylize('[Setter]', 'special');
|
|
}
|
|
}
|
|
}
|
|
if (indexOf(visible_keys, key) < 0) {
|
|
name = '[' + key + ']';
|
|
}
|
|
if (!str) {
|
|
if (indexOf(seen, value[key]) < 0) {
|
|
if (recurseTimes === null) {
|
|
str = format(value[key]);
|
|
} else {
|
|
str = format(value[key], recurseTimes - 1);
|
|
}
|
|
if (str.indexOf('\n') > -1) {
|
|
if (isArray(value)) {
|
|
str = map(str.split('\n'), function (line) {
|
|
return ' ' + line;
|
|
}).join('\n').substr(2);
|
|
} else {
|
|
str = '\n' + map(str.split('\n'), function (line) {
|
|
return ' ' + line;
|
|
}).join('\n');
|
|
}
|
|
}
|
|
} else {
|
|
str = stylize('[Circular]', 'special');
|
|
}
|
|
}
|
|
if (typeof name === 'undefined') {
|
|
if (type === 'Array' && key.match(/^\d+$/)) {
|
|
return str;
|
|
}
|
|
name = json.stringify('' + key);
|
|
if (name.match(/^"([a-zA-Z_][a-zA-Z_0-9]*)"$/)) {
|
|
name = name.substr(1, name.length - 2);
|
|
name = stylize(name, 'name');
|
|
} else {
|
|
name = name.replace(/'/g, "\\'")
|
|
.replace(/\\"/g, '"')
|
|
.replace(/(^"|"$)/g, "'");
|
|
name = stylize(name, 'string');
|
|
}
|
|
}
|
|
|
|
return name + ': ' + str;
|
|
});
|
|
|
|
seen.pop();
|
|
|
|
var numLinesEst = 0;
|
|
var length = reduce(output, function (prev, cur) {
|
|
numLinesEst++;
|
|
if (indexOf(cur, '\n') >= 0) numLinesEst++;
|
|
return prev + cur.length + 1;
|
|
}, 0);
|
|
|
|
if (length > 50) {
|
|
output = braces[0] +
|
|
(base === '' ? '' : base + '\n ') +
|
|
' ' +
|
|
output.join(',\n ') +
|
|
' ' +
|
|
braces[1];
|
|
|
|
} else {
|
|
output = braces[0] + base + ' ' + output.join(', ') + ' ' + braces[1];
|
|
}
|
|
|
|
return output;
|
|
}
|
|
return format(obj, (typeof depth === 'undefined' ? 2 : depth));
|
|
};
|
|
|
|
function isArray (ar) {
|
|
return Object.prototype.toString.call(ar) == '[object Array]';
|
|
};
|
|
|
|
function isRegExp(re) {
|
|
var s;
|
|
try {
|
|
s = '' + re;
|
|
} catch (e) {
|
|
return false;
|
|
}
|
|
|
|
return re instanceof RegExp || // easy case
|
|
// duck-type for context-switching evalcx case
|
|
typeof(re) === 'function' &&
|
|
re.constructor.name === 'RegExp' &&
|
|
re.compile &&
|
|
re.test &&
|
|
re.exec &&
|
|
s.match(/^\/.*\/[gim]{0,3}$/);
|
|
};
|
|
|
|
function isDate(d) {
|
|
if (d instanceof Date) return true;
|
|
return false;
|
|
};
|
|
|
|
function keys (obj) {
|
|
if (Object.keys) {
|
|
return Object.keys(obj);
|
|
}
|
|
|
|
var keys = [];
|
|
|
|
for (var i in obj) {
|
|
if (Object.prototype.hasOwnProperty.call(obj, i)) {
|
|
keys.push(i);
|
|
}
|
|
}
|
|
|
|
return keys;
|
|
}
|
|
|
|
function map (arr, mapper, that) {
|
|
if (Array.prototype.map) {
|
|
return Array.prototype.map.call(arr, mapper, that);
|
|
}
|
|
|
|
var other= new Array(arr.length);
|
|
|
|
for (var i= 0, n = arr.length; i<n; i++)
|
|
if (i in arr)
|
|
other[i] = mapper.call(that, arr[i], i, arr);
|
|
|
|
return other;
|
|
};
|
|
|
|
function reduce (arr, fun) {
|
|
if (Array.prototype.reduce) {
|
|
return Array.prototype.reduce.apply(
|
|
arr
|
|
, Array.prototype.slice.call(arguments, 1)
|
|
);
|
|
}
|
|
|
|
var len = +this.length;
|
|
|
|
if (typeof fun !== "function")
|
|
throw new TypeError();
|
|
|
|
// no value to return if no initial value and an empty array
|
|
if (len === 0 && arguments.length === 1)
|
|
throw new TypeError();
|
|
|
|
var i = 0;
|
|
if (arguments.length >= 2) {
|
|
var rv = arguments[1];
|
|
} else {
|
|
do {
|
|
if (i in this) {
|
|
rv = this[i++];
|
|
break;
|
|
}
|
|
|
|
// if array contains no values, no initial value to return
|
|
if (++i >= len)
|
|
throw new TypeError();
|
|
} while (true);
|
|
}
|
|
|
|
for (; i < len; i++) {
|
|
if (i in this)
|
|
rv = fun.call(null, rv, this[i], i, this);
|
|
}
|
|
|
|
return rv;
|
|
};
|
|
|
|
/**
|
|
* Asserts deep equality
|
|
*
|
|
* @see taken from node.js `assert` module (copyright Joyent, MIT license)
|
|
* @api private
|
|
*/
|
|
|
|
expect.eql = function eql (actual, expected) {
|
|
// 7.1. All identical values are equivalent, as determined by ===.
|
|
if (actual === expected) {
|
|
return true;
|
|
} else if ('undefined' != typeof Buffer
|
|
&& Buffer.isBuffer(actual) && Buffer.isBuffer(expected)) {
|
|
if (actual.length != expected.length) return false;
|
|
|
|
for (var i = 0; i < actual.length; i++) {
|
|
if (actual[i] !== expected[i]) return false;
|
|
}
|
|
|
|
return true;
|
|
|
|
// 7.2. If the expected value is a Date object, the actual value is
|
|
// equivalent if it is also a Date object that refers to the same time.
|
|
} else if (actual instanceof Date && expected instanceof Date) {
|
|
return actual.getTime() === expected.getTime();
|
|
|
|
// 7.3. Other pairs that do not both pass typeof value == "object",
|
|
// equivalence is determined by ==.
|
|
} else if (typeof actual != 'object' && typeof expected != 'object') {
|
|
return actual == expected;
|
|
|
|
// 7.4. For all other Object pairs, including Array objects, equivalence is
|
|
// determined by having the same number of owned properties (as verified
|
|
// with Object.prototype.hasOwnProperty.call), the same set of keys
|
|
// (although not necessarily the same order), equivalent values for every
|
|
// corresponding key, and an identical "prototype" property. Note: this
|
|
// accounts for both named and indexed properties on Arrays.
|
|
} else {
|
|
return objEquiv(actual, expected);
|
|
}
|
|
}
|
|
|
|
function isUndefinedOrNull (value) {
|
|
return value === null || value === undefined;
|
|
}
|
|
|
|
function isArguments (object) {
|
|
return Object.prototype.toString.call(object) == '[object Arguments]';
|
|
}
|
|
|
|
function objEquiv (a, b) {
|
|
if (isUndefinedOrNull(a) || isUndefinedOrNull(b))
|
|
return false;
|
|
// an identical "prototype" property.
|
|
if (a.prototype !== b.prototype) return false;
|
|
//~~~I've managed to break Object.keys through screwy arguments passing.
|
|
// Converting to array solves the problem.
|
|
if (isArguments(a)) {
|
|
if (!isArguments(b)) {
|
|
return false;
|
|
}
|
|
a = pSlice.call(a);
|
|
b = pSlice.call(b);
|
|
return expect.eql(a, b);
|
|
}
|
|
try{
|
|
var ka = keys(a),
|
|
kb = keys(b),
|
|
key, i;
|
|
} catch (e) {//happens when one is a string literal and the other isn't
|
|
return false;
|
|
}
|
|
// having the same number of owned properties (keys incorporates hasOwnProperty)
|
|
if (ka.length != kb.length)
|
|
return false;
|
|
//the same set of keys (although not necessarily the same order),
|
|
ka.sort();
|
|
kb.sort();
|
|
//~~~cheap key test
|
|
for (i = ka.length - 1; i >= 0; i--) {
|
|
if (ka[i] != kb[i])
|
|
return false;
|
|
}
|
|
//equivalent values for every corresponding key, and
|
|
//~~~possibly expensive deep test
|
|
for (i = ka.length - 1; i >= 0; i--) {
|
|
key = ka[i];
|
|
if (!expect.eql(a[key], b[key]))
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
var json = (function () {
|
|
"use strict";
|
|
|
|
if ('object' == typeof JSON && JSON.parse && JSON.stringify) {
|
|
return {
|
|
parse: nativeJSON.parse
|
|
, stringify: nativeJSON.stringify
|
|
}
|
|
}
|
|
|
|
var JSON = {};
|
|
|
|
function f(n) {
|
|
// Format integers to have at least two digits.
|
|
return n < 10 ? '0' + n : n;
|
|
}
|
|
|
|
function date(d, key) {
|
|
return isFinite(d.valueOf()) ?
|
|
d.getUTCFullYear() + '-' +
|
|
f(d.getUTCMonth() + 1) + '-' +
|
|
f(d.getUTCDate()) + 'T' +
|
|
f(d.getUTCHours()) + ':' +
|
|
f(d.getUTCMinutes()) + ':' +
|
|
f(d.getUTCSeconds()) + 'Z' : null;
|
|
};
|
|
|
|
var cx = /[\u0000\u00ad\u0600-\u0604\u070f\u17b4\u17b5\u200c-\u200f\u2028-\u202f\u2060-\u206f\ufeff\ufff0-\uffff]/g,
|
|
escapable = /[\\\"\x00-\x1f\x7f-\x9f\u00ad\u0600-\u0604\u070f\u17b4\u17b5\u200c-\u200f\u2028-\u202f\u2060-\u206f\ufeff\ufff0-\uffff]/g,
|
|
gap,
|
|
indent,
|
|
meta = { // table of character substitutions
|
|
'\b': '\\b',
|
|
'\t': '\\t',
|
|
'\n': '\\n',
|
|
'\f': '\\f',
|
|
'\r': '\\r',
|
|
'"' : '\\"',
|
|
'\\': '\\\\'
|
|
},
|
|
rep;
|
|
|
|
|
|
function quote(string) {
|
|
|
|
// If the string contains no control characters, no quote characters, and no
|
|
// backslash characters, then we can safely slap some quotes around it.
|
|
// Otherwise we must also replace the offending characters with safe escape
|
|
// sequences.
|
|
|
|
escapable.lastIndex = 0;
|
|
return escapable.test(string) ? '"' + string.replace(escapable, function (a) {
|
|
var c = meta[a];
|
|
return typeof c === 'string' ? c :
|
|
'\\u' + ('0000' + a.charCodeAt(0).toString(16)).slice(-4);
|
|
}) + '"' : '"' + string + '"';
|
|
}
|
|
|
|
|
|
function str(key, holder) {
|
|
|
|
// Produce a string from holder[key].
|
|
|
|
var i, // The loop counter.
|
|
k, // The member key.
|
|
v, // The member value.
|
|
length,
|
|
mind = gap,
|
|
partial,
|
|
value = holder[key];
|
|
|
|
// If the value has a toJSON method, call it to obtain a replacement value.
|
|
|
|
if (value instanceof Date) {
|
|
value = date(key);
|
|
}
|
|
|
|
// If we were called with a replacer function, then call the replacer to
|
|
// obtain a replacement value.
|
|
|
|
if (typeof rep === 'function') {
|
|
value = rep.call(holder, key, value);
|
|
}
|
|
|
|
// What happens next depends on the value's type.
|
|
|
|
switch (typeof value) {
|
|
case 'string':
|
|
return quote(value);
|
|
|
|
case 'number':
|
|
|
|
// JSON numbers must be finite. Encode non-finite numbers as null.
|
|
|
|
return isFinite(value) ? String(value) : 'null';
|
|
|
|
case 'boolean':
|
|
case 'null':
|
|
|
|
// If the value is a boolean or null, convert it to a string. Note:
|
|
// typeof null does not produce 'null'. The case is included here in
|
|
// the remote chance that this gets fixed someday.
|
|
|
|
return String(value);
|
|
|
|
// If the type is 'object', we might be dealing with an object or an array or
|
|
// null.
|
|
|
|
case 'object':
|
|
|
|
// Due to a specification blunder in ECMAScript, typeof null is 'object',
|
|
// so watch out for that case.
|
|
|
|
if (!value) {
|
|
return 'null';
|
|
}
|
|
|
|
// Make an array to hold the partial results of stringifying this object value.
|
|
|
|
gap += indent;
|
|
partial = [];
|
|
|
|
// Is the value an array?
|
|
|
|
if (Object.prototype.toString.apply(value) === '[object Array]') {
|
|
|
|
// The value is an array. Stringify every element. Use null as a placeholder
|
|
// for non-JSON values.
|
|
|
|
length = value.length;
|
|
for (i = 0; i < length; i += 1) {
|
|
partial[i] = str(i, value) || 'null';
|
|
}
|
|
|
|
// Join all of the elements together, separated with commas, and wrap them in
|
|
// brackets.
|
|
|
|
v = partial.length === 0 ? '[]' : gap ?
|
|
'[\n' + gap + partial.join(',\n' + gap) + '\n' + mind + ']' :
|
|
'[' + partial.join(',') + ']';
|
|
gap = mind;
|
|
return v;
|
|
}
|
|
|
|
// If the replacer is an array, use it to select the members to be stringified.
|
|
|
|
if (rep && typeof rep === 'object') {
|
|
length = rep.length;
|
|
for (i = 0; i < length; i += 1) {
|
|
if (typeof rep[i] === 'string') {
|
|
k = rep[i];
|
|
v = str(k, value);
|
|
if (v) {
|
|
partial.push(quote(k) + (gap ? ': ' : ':') + v);
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
|
|
// Otherwise, iterate through all of the keys in the object.
|
|
|
|
for (k in value) {
|
|
if (Object.prototype.hasOwnProperty.call(value, k)) {
|
|
v = str(k, value);
|
|
if (v) {
|
|
partial.push(quote(k) + (gap ? ': ' : ':') + v);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Join all of the member texts together, separated with commas,
|
|
// and wrap them in braces.
|
|
|
|
v = partial.length === 0 ? '{}' : gap ?
|
|
'{\n' + gap + partial.join(',\n' + gap) + '\n' + mind + '}' :
|
|
'{' + partial.join(',') + '}';
|
|
gap = mind;
|
|
return v;
|
|
}
|
|
}
|
|
|
|
// If the JSON object does not yet have a stringify method, give it one.
|
|
|
|
JSON.stringify = function (value, replacer, space) {
|
|
|
|
// The stringify method takes a value and an optional replacer, and an optional
|
|
// space parameter, and returns a JSON text. The replacer can be a function
|
|
// that can replace values, or an array of strings that will select the keys.
|
|
// A default replacer method can be provided. Use of the space parameter can
|
|
// produce text that is more easily readable.
|
|
|
|
var i;
|
|
gap = '';
|
|
indent = '';
|
|
|
|
// If the space parameter is a number, make an indent string containing that
|
|
// many spaces.
|
|
|
|
if (typeof space === 'number') {
|
|
for (i = 0; i < space; i += 1) {
|
|
indent += ' ';
|
|
}
|
|
|
|
// If the space parameter is a string, it will be used as the indent string.
|
|
|
|
} else if (typeof space === 'string') {
|
|
indent = space;
|
|
}
|
|
|
|
// If there is a replacer, it must be a function or an array.
|
|
// Otherwise, throw an error.
|
|
|
|
rep = replacer;
|
|
if (replacer && typeof replacer !== 'function' &&
|
|
(typeof replacer !== 'object' ||
|
|
typeof replacer.length !== 'number')) {
|
|
throw new Error('JSON.stringify');
|
|
}
|
|
|
|
// Make a fake root object containing our value under the key of ''.
|
|
// Return the result of stringifying the value.
|
|
|
|
return str('', {'': value});
|
|
};
|
|
|
|
// If the JSON object does not yet have a parse method, give it one.
|
|
|
|
JSON.parse = function (text, reviver) {
|
|
// The parse method takes a text and an optional reviver function, and returns
|
|
// a JavaScript value if the text is a valid JSON text.
|
|
|
|
var j;
|
|
|
|
function walk(holder, key) {
|
|
|
|
// The walk method is used to recursively walk the resulting structure so
|
|
// that modifications can be made.
|
|
|
|
var k, v, value = holder[key];
|
|
if (value && typeof value === 'object') {
|
|
for (k in value) {
|
|
if (Object.prototype.hasOwnProperty.call(value, k)) {
|
|
v = walk(value, k);
|
|
if (v !== undefined) {
|
|
value[k] = v;
|
|
} else {
|
|
delete value[k];
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return reviver.call(holder, key, value);
|
|
}
|
|
|
|
|
|
// Parsing happens in four stages. In the first stage, we replace certain
|
|
// Unicode characters with escape sequences. JavaScript handles many characters
|
|
// incorrectly, either silently deleting them, or treating them as line endings.
|
|
|
|
text = String(text);
|
|
cx.lastIndex = 0;
|
|
if (cx.test(text)) {
|
|
text = text.replace(cx, function (a) {
|
|
return '\\u' +
|
|
('0000' + a.charCodeAt(0).toString(16)).slice(-4);
|
|
});
|
|
}
|
|
|
|
// In the second stage, we run the text against regular expressions that look
|
|
// for non-JSON patterns. We are especially concerned with '()' and 'new'
|
|
// because they can cause invocation, and '=' because it can cause mutation.
|
|
// But just to be safe, we want to reject all unexpected forms.
|
|
|
|
// We split the second stage into 4 regexp operations in order to work around
|
|
// crippling inefficiencies in IE's and Safari's regexp engines. First we
|
|
// replace the JSON backslash pairs with '@' (a non-JSON character). Second, we
|
|
// replace all simple value tokens with ']' characters. Third, we delete all
|
|
// open brackets that follow a colon or comma or that begin the text. Finally,
|
|
// we look to see that the remaining characters are only whitespace or ']' or
|
|
// ',' or ':' or '{' or '}'. If that is so, then the text is safe for eval.
|
|
|
|
if (/^[\],:{}\s]*$/
|
|
.test(text.replace(/\\(?:["\\\/bfnrt]|u[0-9a-fA-F]{4})/g, '@')
|
|
.replace(/"[^"\\\n\r]*"|true|false|null|-?\d+(?:\.\d*)?(?:[eE][+\-]?\d+)?/g, ']')
|
|
.replace(/(?:^|:|,)(?:\s*\[)+/g, ''))) {
|
|
|
|
// In the third stage we use the eval function to compile the text into a
|
|
// JavaScript structure. The '{' operator is subject to a syntactic ambiguity
|
|
// in JavaScript: it can begin a block or an object literal. We wrap the text
|
|
// in parens to eliminate the ambiguity.
|
|
|
|
j = eval('(' + text + ')');
|
|
|
|
// In the optional fourth stage, we recursively walk the new structure, passing
|
|
// each name/value pair to a reviver function for possible transformation.
|
|
|
|
return typeof reviver === 'function' ?
|
|
walk({'': j}, '') : j;
|
|
}
|
|
|
|
// If the text is not JSON parseable, then a SyntaxError is thrown.
|
|
|
|
throw new SyntaxError('JSON.parse');
|
|
};
|
|
|
|
return JSON;
|
|
})();
|
|
|
|
if ('undefined' != typeof window) {
|
|
window.expect = module.exports;
|
|
}
|
|
|
|
})(
|
|
this
|
|
, 'undefined' != typeof module ? module : {}
|
|
, 'undefined' != typeof exports ? exports : {}
|
|
);
|
|
|
|
},{"__browserify_Buffer":14}],17:[function(require,module,exports){
|
|
if (typeof Object.create === 'function') {
|
|
// implementation from standard node.js 'util' module
|
|
module.exports = function inherits(ctor, superCtor) {
|
|
ctor.super_ = superCtor
|
|
ctor.prototype = Object.create(superCtor.prototype, {
|
|
constructor: {
|
|
value: ctor,
|
|
enumerable: false,
|
|
writable: true,
|
|
configurable: true
|
|
}
|
|
});
|
|
};
|
|
} else {
|
|
// old school shim for old browsers
|
|
module.exports = function inherits(ctor, superCtor) {
|
|
ctor.super_ = superCtor
|
|
var TempCtor = function () {}
|
|
TempCtor.prototype = superCtor.prototype
|
|
ctor.prototype = new TempCtor()
|
|
ctor.prototype.constructor = ctor
|
|
}
|
|
}
|
|
|
|
},{}],18:[function(require,module,exports){
|
|
module.exports = require('./lib/js-yaml.js');
|
|
|
|
},{"./lib/js-yaml.js":19}],19:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
|
|
var loader = require('./js-yaml/loader');
|
|
var dumper = require('./js-yaml/dumper');
|
|
|
|
|
|
function deprecated(name) {
|
|
return function () {
|
|
throw new Error('Function ' + name + ' is deprecated and cannot be used.');
|
|
};
|
|
}
|
|
|
|
|
|
module.exports.NIL = require('./js-yaml/common').NIL;
|
|
module.exports.Type = require('./js-yaml/type');
|
|
module.exports.Schema = require('./js-yaml/schema');
|
|
module.exports.FAILSAFE_SCHEMA = require('./js-yaml/schema/failsafe');
|
|
module.exports.JSON_SCHEMA = require('./js-yaml/schema/json');
|
|
module.exports.CORE_SCHEMA = require('./js-yaml/schema/core');
|
|
module.exports.DEFAULT_SAFE_SCHEMA = require('./js-yaml/schema/default_safe');
|
|
module.exports.DEFAULT_FULL_SCHEMA = require('./js-yaml/schema/default_full');
|
|
module.exports.load = loader.load;
|
|
module.exports.loadAll = loader.loadAll;
|
|
module.exports.safeLoad = loader.safeLoad;
|
|
module.exports.safeLoadAll = loader.safeLoadAll;
|
|
module.exports.dump = dumper.dump;
|
|
module.exports.safeDump = dumper.safeDump;
|
|
module.exports.YAMLException = require('./js-yaml/exception');
|
|
|
|
// Deprecared schema names from JS-YAML 2.0.x
|
|
module.exports.MINIMAL_SCHEMA = require('./js-yaml/schema/failsafe');
|
|
module.exports.SAFE_SCHEMA = require('./js-yaml/schema/default_safe');
|
|
module.exports.DEFAULT_SCHEMA = require('./js-yaml/schema/default_full');
|
|
|
|
// Deprecated functions from JS-YAML 1.x.x
|
|
module.exports.scan = deprecated('scan');
|
|
module.exports.parse = deprecated('parse');
|
|
module.exports.compose = deprecated('compose');
|
|
module.exports.addConstructor = deprecated('addConstructor');
|
|
|
|
|
|
require('./js-yaml/require');
|
|
|
|
},{"./js-yaml/common":20,"./js-yaml/dumper":21,"./js-yaml/exception":22,"./js-yaml/loader":23,"./js-yaml/require":25,"./js-yaml/schema":26,"./js-yaml/schema/core":27,"./js-yaml/schema/default_full":28,"./js-yaml/schema/default_safe":29,"./js-yaml/schema/failsafe":30,"./js-yaml/schema/json":31,"./js-yaml/type":32}],20:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
|
|
var NIL = {};
|
|
|
|
|
|
function isNothing(subject) {
|
|
return (undefined === subject) || (null === subject);
|
|
}
|
|
|
|
|
|
function isObject(subject) {
|
|
return ('object' === typeof subject) && (null !== subject);
|
|
}
|
|
|
|
|
|
function toArray(sequence) {
|
|
if (Array.isArray(sequence)) {
|
|
return sequence;
|
|
} else if (isNothing(sequence)) {
|
|
return [];
|
|
} else {
|
|
return [ sequence ];
|
|
}
|
|
}
|
|
|
|
|
|
function extend(target, source) {
|
|
var index, length, key, sourceKeys;
|
|
|
|
if (source) {
|
|
sourceKeys = Object.keys(source);
|
|
|
|
for (index = 0, length = sourceKeys.length; index < length; index += 1) {
|
|
key = sourceKeys[index];
|
|
target[key] = source[key];
|
|
}
|
|
}
|
|
|
|
return target;
|
|
}
|
|
|
|
|
|
function repeat(string, count) {
|
|
var result = '', cycle;
|
|
|
|
for (cycle = 0; cycle < count; cycle += 1) {
|
|
result += string;
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
|
|
module.exports.NIL = NIL;
|
|
module.exports.isNothing = isNothing;
|
|
module.exports.isObject = isObject;
|
|
module.exports.toArray = toArray;
|
|
module.exports.repeat = repeat;
|
|
module.exports.extend = extend;
|
|
|
|
},{}],21:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
|
|
var common = require('./common');
|
|
var NIL = common.NIL;
|
|
var YAMLException = require('./exception');
|
|
var DEFAULT_FULL_SCHEMA = require('./schema/default_full');
|
|
var DEFAULT_SAFE_SCHEMA = require('./schema/default_safe');
|
|
|
|
|
|
var _hasOwnProperty = Object.prototype.hasOwnProperty;
|
|
|
|
|
|
var CHAR_TAB = 0x09; /* Tab */
|
|
var CHAR_LINE_FEED = 0x0A; /* LF */
|
|
var CHAR_CARRIAGE_RETURN = 0x0D; /* CR */
|
|
var CHAR_SPACE = 0x20; /* Space */
|
|
var CHAR_EXCLAMATION = 0x21; /* ! */
|
|
var CHAR_DOUBLE_QUOTE = 0x22; /* " */
|
|
var CHAR_SHARP = 0x23; /* # */
|
|
var CHAR_PERCENT = 0x25; /* % */
|
|
var CHAR_AMPERSAND = 0x26; /* & */
|
|
var CHAR_SINGLE_QUOTE = 0x27; /* ' */
|
|
var CHAR_ASTERISK = 0x2A; /* * */
|
|
var CHAR_COMMA = 0x2C; /* , */
|
|
var CHAR_MINUS = 0x2D; /* - */
|
|
var CHAR_COLON = 0x3A; /* : */
|
|
var CHAR_GREATER_THAN = 0x3E; /* > */
|
|
var CHAR_QUESTION = 0x3F; /* ? */
|
|
var CHAR_COMMERCIAL_AT = 0x40; /* @ */
|
|
var CHAR_LEFT_SQUARE_BRACKET = 0x5B; /* [ */
|
|
var CHAR_RIGHT_SQUARE_BRACKET = 0x5D; /* ] */
|
|
var CHAR_GRAVE_ACCENT = 0x60; /* ` */
|
|
var CHAR_LEFT_CURLY_BRACKET = 0x7B; /* { */
|
|
var CHAR_VERTICAL_LINE = 0x7C; /* | */
|
|
var CHAR_RIGHT_CURLY_BRACKET = 0x7D; /* } */
|
|
|
|
|
|
var ESCAPE_SEQUENCES = {};
|
|
|
|
ESCAPE_SEQUENCES[0x00] = '\\0';
|
|
ESCAPE_SEQUENCES[0x07] = '\\a';
|
|
ESCAPE_SEQUENCES[0x08] = '\\b';
|
|
ESCAPE_SEQUENCES[0x09] = '\\t';
|
|
ESCAPE_SEQUENCES[0x0A] = '\\n';
|
|
ESCAPE_SEQUENCES[0x0B] = '\\v';
|
|
ESCAPE_SEQUENCES[0x0C] = '\\f';
|
|
ESCAPE_SEQUENCES[0x0D] = '\\r';
|
|
ESCAPE_SEQUENCES[0x1B] = '\\e';
|
|
ESCAPE_SEQUENCES[0x22] = '\\"';
|
|
ESCAPE_SEQUENCES[0x5C] = '\\\\';
|
|
ESCAPE_SEQUENCES[0x85] = '\\N';
|
|
ESCAPE_SEQUENCES[0xA0] = '\\_';
|
|
ESCAPE_SEQUENCES[0x2028] = '\\L';
|
|
ESCAPE_SEQUENCES[0x2029] = '\\P';
|
|
|
|
|
|
var DEPRECATED_BOOLEANS_SYNTAX = [
|
|
'y', 'Y', 'yes', 'Yes', 'YES', 'on', 'On', 'ON',
|
|
'n', 'N', 'no', 'No', 'NO', 'off', 'Off', 'OFF'
|
|
];
|
|
|
|
|
|
function kindOf(object) {
|
|
var kind = typeof object;
|
|
|
|
if (null === object) {
|
|
return 'null';
|
|
} else if ('number' === kind) {
|
|
return 0 === object % 1 ? 'integer' : 'float';
|
|
} else if ('object' === kind && Array.isArray(object)) {
|
|
return 'array';
|
|
} else {
|
|
return kind;
|
|
}
|
|
}
|
|
|
|
|
|
function compileStyleMap(schema, map) {
|
|
var result, keys, index, length, tag, style, type;
|
|
|
|
if (null === map) {
|
|
return {};
|
|
}
|
|
|
|
result = {};
|
|
keys = Object.keys(map);
|
|
|
|
for (index = 0, length = keys.length; index < length; index += 1) {
|
|
tag = keys[index];
|
|
style = String(map[tag]);
|
|
|
|
if ('!!' === tag.slice(0, 2)) {
|
|
tag = 'tag:yaml.org,2002:' + tag.slice(2);
|
|
}
|
|
|
|
type = schema.compiledTypeMap[tag];
|
|
|
|
if (type && type.dumper) {
|
|
if (_hasOwnProperty.call(type.dumper.styleAliases, style)) {
|
|
style = type.dumper.styleAliases[style];
|
|
}
|
|
}
|
|
|
|
result[tag] = style;
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
|
|
function encodeHex(character) {
|
|
var string, handle, length;
|
|
|
|
string = character.toString(16).toUpperCase();
|
|
|
|
if (character <= 0xFF) {
|
|
handle = 'x';
|
|
length = 2;
|
|
} else if (character <= 0xFFFF) {
|
|
handle = 'u';
|
|
length = 4;
|
|
} else if (character <= 0xFFFFFFFF) {
|
|
handle = 'U';
|
|
length = 8;
|
|
} else {
|
|
throw new YAMLException('code point within a string may not be greater than 0xFFFFFFFF');
|
|
}
|
|
|
|
return '\\' + handle + common.repeat('0', length - string.length) + string;
|
|
}
|
|
|
|
|
|
function dump(input, options) {
|
|
options = options || {};
|
|
|
|
var schema = options['schema'] || DEFAULT_FULL_SCHEMA,
|
|
indent = Math.max(1, (options['indent'] || 2)),
|
|
skipInvalid = options['skipInvalid'] || false,
|
|
flowLevel = (common.isNothing(options['flowLevel']) ? -1 : options['flowLevel']),
|
|
styleMap = compileStyleMap(schema, options['styles'] || null),
|
|
|
|
implicitTypes = schema.compiledImplicit,
|
|
explicitTypes = schema.compiledExplicit,
|
|
|
|
kind,
|
|
tag,
|
|
result;
|
|
|
|
function generateNextLine(level) {
|
|
return '\n' + common.repeat(' ', indent * level);
|
|
}
|
|
|
|
function testImplicitResolving(object) {
|
|
var index, length, type;
|
|
|
|
for (index = 0, length = implicitTypes.length; index < length; index += 1) {
|
|
type = implicitTypes[index];
|
|
|
|
if (null !== type.loader &&
|
|
NIL !== type.loader.resolver(object, false)) {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
function writeScalar(object) {
|
|
var isQuoted, checkpoint, position, length, character;
|
|
|
|
result = '';
|
|
isQuoted = false;
|
|
checkpoint = 0;
|
|
|
|
// Ensure compatibility with YAML 1.0/1.1 loaders.
|
|
// TODO: This check should be moved into boolean type definition.
|
|
if (-1 !== DEPRECATED_BOOLEANS_SYNTAX.indexOf(object)) {
|
|
isQuoted = true;
|
|
}
|
|
|
|
if (0 === object.length ||
|
|
CHAR_SPACE === object.charCodeAt(0) ||
|
|
CHAR_SPACE === object.charCodeAt(object.length - 1)) {
|
|
isQuoted = true;
|
|
}
|
|
|
|
for (position = 0, length = object.length; position < length; position += 1) {
|
|
character = object.charCodeAt(position);
|
|
|
|
if (!isQuoted) {
|
|
if (CHAR_TAB === character ||
|
|
CHAR_LINE_FEED === character ||
|
|
CHAR_CARRIAGE_RETURN === character ||
|
|
CHAR_COMMA === character ||
|
|
CHAR_LEFT_SQUARE_BRACKET === character ||
|
|
CHAR_RIGHT_SQUARE_BRACKET === character ||
|
|
CHAR_LEFT_CURLY_BRACKET === character ||
|
|
CHAR_RIGHT_CURLY_BRACKET === character ||
|
|
CHAR_SHARP === character ||
|
|
CHAR_AMPERSAND === character ||
|
|
CHAR_ASTERISK === character ||
|
|
CHAR_EXCLAMATION === character ||
|
|
CHAR_VERTICAL_LINE === character ||
|
|
CHAR_GREATER_THAN === character ||
|
|
CHAR_SINGLE_QUOTE === character ||
|
|
CHAR_DOUBLE_QUOTE === character ||
|
|
CHAR_PERCENT === character ||
|
|
CHAR_COMMERCIAL_AT === character ||
|
|
CHAR_GRAVE_ACCENT === character ||
|
|
CHAR_QUESTION === character ||
|
|
CHAR_COLON === character ||
|
|
CHAR_MINUS === character) {
|
|
isQuoted = true;
|
|
}
|
|
}
|
|
|
|
if (ESCAPE_SEQUENCES[character] ||
|
|
!((0x00020 <= character && character <= 0x00007E) ||
|
|
(0x00085 === character) ||
|
|
(0x000A0 <= character && character <= 0x00D7FF) ||
|
|
(0x0E000 <= character && character <= 0x00FFFD) ||
|
|
(0x10000 <= character && character <= 0x10FFFF))) {
|
|
result += object.slice(checkpoint, position);
|
|
result += ESCAPE_SEQUENCES[character] || encodeHex(character);
|
|
checkpoint = position + 1;
|
|
isQuoted = true;
|
|
}
|
|
}
|
|
|
|
if (checkpoint < position) {
|
|
result += object.slice(checkpoint, position);
|
|
}
|
|
|
|
if (!isQuoted && testImplicitResolving(result)) {
|
|
isQuoted = true;
|
|
}
|
|
|
|
if (isQuoted) {
|
|
result = '"' + result + '"';
|
|
}
|
|
}
|
|
|
|
function writeFlowSequence(level, object) {
|
|
var _result = '',
|
|
_tag = tag,
|
|
index,
|
|
length;
|
|
|
|
for (index = 0, length = object.length; index < length; index += 1) {
|
|
// Write only valid elements.
|
|
if (writeNode(level, object[index], false, false)) {
|
|
if (0 !== index) {
|
|
_result += ', ';
|
|
}
|
|
_result += result;
|
|
}
|
|
}
|
|
|
|
tag = _tag;
|
|
result = '[' + _result + ']';
|
|
}
|
|
|
|
function writeBlockSequence(level, object, compact) {
|
|
var _result = '',
|
|
_tag = tag,
|
|
index,
|
|
length;
|
|
|
|
for (index = 0, length = object.length; index < length; index += 1) {
|
|
// Write only valid elements.
|
|
if (writeNode(level + 1, object[index], true, true)) {
|
|
if (!compact || 0 !== index) {
|
|
_result += generateNextLine(level);
|
|
}
|
|
_result += '- ' + result;
|
|
}
|
|
}
|
|
|
|
tag = _tag;
|
|
result = _result || '[]'; // Empty sequence if no valid values.
|
|
}
|
|
|
|
function writeFlowMapping(level, object) {
|
|
var _result = '',
|
|
_tag = tag,
|
|
objectKeyList = Object.keys(object),
|
|
index,
|
|
length,
|
|
objectKey,
|
|
objectValue,
|
|
pairBuffer;
|
|
|
|
for (index = 0, length = objectKeyList.length; index < length; index += 1) {
|
|
pairBuffer = '';
|
|
|
|
if (0 !== index) {
|
|
pairBuffer += ', ';
|
|
}
|
|
|
|
objectKey = objectKeyList[index];
|
|
objectValue = object[objectKey];
|
|
|
|
if (!writeNode(level, objectKey, false, false)) {
|
|
continue; // Skip this pair because of invalid key;
|
|
}
|
|
|
|
if (result.length > 1024) {
|
|
pairBuffer += '? ';
|
|
}
|
|
|
|
pairBuffer += result + ': ';
|
|
|
|
if (!writeNode(level, objectValue, false, false)) {
|
|
continue; // Skip this pair because of invalid value.
|
|
}
|
|
|
|
pairBuffer += result;
|
|
|
|
// Both key and value are valid.
|
|
_result += pairBuffer;
|
|
}
|
|
|
|
tag = _tag;
|
|
result = '{' + _result + '}';
|
|
}
|
|
|
|
function writeBlockMapping(level, object, compact) {
|
|
var _result = '',
|
|
_tag = tag,
|
|
objectKeyList = Object.keys(object),
|
|
index,
|
|
length,
|
|
objectKey,
|
|
objectValue,
|
|
explicitPair,
|
|
pairBuffer;
|
|
|
|
for (index = 0, length = objectKeyList.length; index < length; index += 1) {
|
|
pairBuffer = '';
|
|
|
|
if (!compact || 0 !== index) {
|
|
pairBuffer += generateNextLine(level);
|
|
}
|
|
|
|
objectKey = objectKeyList[index];
|
|
objectValue = object[objectKey];
|
|
|
|
if (!writeNode(level + 1, objectKey, true, true)) {
|
|
continue; // Skip this pair because of invalid key.
|
|
}
|
|
|
|
explicitPair = (null !== tag && '?' !== tag && result.length <= 1024);
|
|
|
|
if (explicitPair) {
|
|
pairBuffer += '? ';
|
|
}
|
|
|
|
pairBuffer += result;
|
|
|
|
if (explicitPair) {
|
|
pairBuffer += generateNextLine(level);
|
|
}
|
|
|
|
if (!writeNode(level + 1, objectValue, true, explicitPair)) {
|
|
continue; // Skip this pair because of invalid value.
|
|
}
|
|
|
|
pairBuffer += ': ' + result;
|
|
|
|
// Both key and value are valid.
|
|
_result += pairBuffer;
|
|
}
|
|
|
|
tag = _tag;
|
|
result = _result || '{}'; // Empty mapping if no valid pairs.
|
|
}
|
|
|
|
function detectType(object, explicit) {
|
|
var _result, typeList, index, length, type, style;
|
|
|
|
typeList = explicit ? explicitTypes : implicitTypes;
|
|
kind = kindOf(object);
|
|
|
|
for (index = 0, length = typeList.length; index < length; index += 1) {
|
|
type = typeList[index];
|
|
|
|
if ((null !== type.dumper) &&
|
|
(null === type.dumper.kind || kind === type.dumper.kind) &&
|
|
(null === type.dumper.instanceOf || object instanceof type.dumper.instanceOf) &&
|
|
(null === type.dumper.predicate || type.dumper.predicate(object))) {
|
|
tag = explicit ? type.tag : '?';
|
|
|
|
if (null !== type.dumper.representer) {
|
|
style = styleMap[type.tag] || type.dumper.defaultStyle;
|
|
|
|
if ('function' === typeof type.dumper.representer) {
|
|
_result = type.dumper.representer(object, style);
|
|
} else if (_hasOwnProperty.call(type.dumper.representer, style)) {
|
|
_result = type.dumper.representer[style](object, style);
|
|
} else {
|
|
throw new YAMLException('!<' + type.tag + '> tag resolver accepts not "' + style + '" style');
|
|
}
|
|
|
|
if (NIL !== _result) {
|
|
kind = kindOf(_result);
|
|
result = _result;
|
|
} else {
|
|
if (explicit) {
|
|
throw new YAMLException('cannot represent an object of !<' + type.tag + '> type');
|
|
} else {
|
|
continue;
|
|
}
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
// Serializes `object` and writes it to global `result`.
|
|
// Returns true on success, or false on invalid object.
|
|
//
|
|
function writeNode(level, object, block, compact) {
|
|
tag = null;
|
|
result = object;
|
|
|
|
if (!detectType(object, false)) {
|
|
detectType(object, true);
|
|
}
|
|
|
|
if (block) {
|
|
block = (0 > flowLevel || flowLevel > level);
|
|
}
|
|
|
|
if ((null !== tag && '?' !== tag) || (2 !== indent && level > 0)) {
|
|
compact = false;
|
|
}
|
|
|
|
if ('object' === kind) {
|
|
if (block && (0 !== Object.keys(result).length)) {
|
|
writeBlockMapping(level, result, compact);
|
|
} else {
|
|
writeFlowMapping(level, result);
|
|
}
|
|
} else if ('array' === kind) {
|
|
if (block && (0 !== result.length)) {
|
|
writeBlockSequence(level, result, compact);
|
|
} else {
|
|
writeFlowSequence(level, result);
|
|
}
|
|
} else if ('string' === kind) {
|
|
if ('?' !== tag) {
|
|
writeScalar(result);
|
|
}
|
|
} else if (skipInvalid) {
|
|
return false;
|
|
} else {
|
|
throw new YAMLException('unacceptabe kind of an object to dump (' + kind + ')');
|
|
}
|
|
|
|
if (null !== tag && '?' !== tag) {
|
|
result = '!<' + tag + '> ' + result;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
if (writeNode(0, input, true, true)) {
|
|
return result + '\n';
|
|
} else {
|
|
return '';
|
|
}
|
|
}
|
|
|
|
|
|
function safeDump(input, options) {
|
|
return dump(input, common.extend({ schema: DEFAULT_SAFE_SCHEMA }, options));
|
|
}
|
|
|
|
|
|
module.exports.dump = dump;
|
|
module.exports.safeDump = safeDump;
|
|
|
|
},{"./common":20,"./exception":22,"./schema/default_full":28,"./schema/default_safe":29}],22:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
|
|
function YAMLException(reason, mark) {
|
|
this.name = 'YAMLException';
|
|
this.reason = reason;
|
|
this.mark = mark;
|
|
this.message = this.toString(false);
|
|
}
|
|
|
|
|
|
YAMLException.prototype.toString = function toString(compact) {
|
|
var result;
|
|
|
|
result = 'JS-YAML: ' + (this.reason || '(unknown reason)');
|
|
|
|
if (!compact && this.mark) {
|
|
result += ' ' + this.mark.toString();
|
|
}
|
|
|
|
return result;
|
|
};
|
|
|
|
|
|
module.exports = YAMLException;
|
|
|
|
},{}],23:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
|
|
var common = require('./common');
|
|
var YAMLException = require('./exception');
|
|
var Mark = require('./mark');
|
|
var NIL = common.NIL;
|
|
var DEFAULT_SAFE_SCHEMA = require('./schema/default_safe');
|
|
var DEFAULT_FULL_SCHEMA = require('./schema/default_full');
|
|
|
|
|
|
var _hasOwnProperty = Object.prototype.hasOwnProperty;
|
|
|
|
|
|
var KIND_STRING = 'string';
|
|
var KIND_ARRAY = 'array';
|
|
var KIND_OBJECT = 'object';
|
|
|
|
|
|
var CONTEXT_FLOW_IN = 1;
|
|
var CONTEXT_FLOW_OUT = 2;
|
|
var CONTEXT_BLOCK_IN = 3;
|
|
var CONTEXT_BLOCK_OUT = 4;
|
|
|
|
|
|
var CHOMPING_CLIP = 1;
|
|
var CHOMPING_STRIP = 2;
|
|
var CHOMPING_KEEP = 3;
|
|
|
|
|
|
var CHAR_TAB = 0x09; /* Tab */
|
|
var CHAR_LINE_FEED = 0x0A; /* LF */
|
|
var CHAR_CARRIAGE_RETURN = 0x0D; /* CR */
|
|
var CHAR_SPACE = 0x20; /* Space */
|
|
var CHAR_EXCLAMATION = 0x21; /* ! */
|
|
var CHAR_DOUBLE_QUOTE = 0x22; /* " */
|
|
var CHAR_SHARP = 0x23; /* # */
|
|
var CHAR_PERCENT = 0x25; /* % */
|
|
var CHAR_AMPERSAND = 0x26; /* & */
|
|
var CHAR_SINGLE_QUOTE = 0x27; /* ' */
|
|
var CHAR_ASTERISK = 0x2A; /* * */
|
|
var CHAR_PLUS = 0x2B; /* + */
|
|
var CHAR_COMMA = 0x2C; /* , */
|
|
var CHAR_MINUS = 0x2D; /* - */
|
|
var CHAR_DOT = 0x2E; /* . */
|
|
var CHAR_SLASH = 0x2F; /* / */
|
|
var CHAR_DIGIT_ZERO = 0x30; /* 0 */
|
|
var CHAR_DIGIT_ONE = 0x31; /* 1 */
|
|
var CHAR_DIGIT_NINE = 0x39; /* 9 */
|
|
var CHAR_COLON = 0x3A; /* : */
|
|
var CHAR_LESS_THAN = 0x3C; /* < */
|
|
var CHAR_GREATER_THAN = 0x3E; /* > */
|
|
var CHAR_QUESTION = 0x3F; /* ? */
|
|
var CHAR_COMMERCIAL_AT = 0x40; /* @ */
|
|
var CHAR_CAPITAL_A = 0x41; /* A */
|
|
var CHAR_CAPITAL_F = 0x46; /* F */
|
|
var CHAR_CAPITAL_L = 0x4C; /* L */
|
|
var CHAR_CAPITAL_N = 0x4E; /* N */
|
|
var CHAR_CAPITAL_P = 0x50; /* P */
|
|
var CHAR_CAPITAL_U = 0x55; /* U */
|
|
var CHAR_LEFT_SQUARE_BRACKET = 0x5B; /* [ */
|
|
var CHAR_BACKSLASH = 0x5C; /* \ */
|
|
var CHAR_RIGHT_SQUARE_BRACKET = 0x5D; /* ] */
|
|
var CHAR_UNDERSCORE = 0x5F; /* _ */
|
|
var CHAR_GRAVE_ACCENT = 0x60; /* ` */
|
|
var CHAR_SMALL_A = 0x61; /* a */
|
|
var CHAR_SMALL_B = 0x62; /* b */
|
|
var CHAR_SMALL_E = 0x65; /* e */
|
|
var CHAR_SMALL_F = 0x66; /* f */
|
|
var CHAR_SMALL_N = 0x6E; /* n */
|
|
var CHAR_SMALL_R = 0x72; /* r */
|
|
var CHAR_SMALL_T = 0x74; /* t */
|
|
var CHAR_SMALL_U = 0x75; /* u */
|
|
var CHAR_SMALL_V = 0x76; /* v */
|
|
var CHAR_SMALL_X = 0x78; /* x */
|
|
var CHAR_LEFT_CURLY_BRACKET = 0x7B; /* { */
|
|
var CHAR_VERTICAL_LINE = 0x7C; /* | */
|
|
var CHAR_RIGHT_CURLY_BRACKET = 0x7D; /* } */
|
|
|
|
|
|
var SIMPLE_ESCAPE_SEQUENCES = {};
|
|
|
|
SIMPLE_ESCAPE_SEQUENCES[CHAR_DIGIT_ZERO] = '\x00';
|
|
SIMPLE_ESCAPE_SEQUENCES[CHAR_SMALL_A] = '\x07';
|
|
SIMPLE_ESCAPE_SEQUENCES[CHAR_SMALL_B] = '\x08';
|
|
SIMPLE_ESCAPE_SEQUENCES[CHAR_SMALL_T] = '\x09';
|
|
SIMPLE_ESCAPE_SEQUENCES[CHAR_TAB] = '\x09';
|
|
SIMPLE_ESCAPE_SEQUENCES[CHAR_SMALL_N] = '\x0A';
|
|
SIMPLE_ESCAPE_SEQUENCES[CHAR_SMALL_V] = '\x0B';
|
|
SIMPLE_ESCAPE_SEQUENCES[CHAR_SMALL_F] = '\x0C';
|
|
SIMPLE_ESCAPE_SEQUENCES[CHAR_SMALL_R] = '\x0D';
|
|
SIMPLE_ESCAPE_SEQUENCES[CHAR_SMALL_E] = '\x1B';
|
|
SIMPLE_ESCAPE_SEQUENCES[CHAR_SPACE] = ' ';
|
|
SIMPLE_ESCAPE_SEQUENCES[CHAR_DOUBLE_QUOTE] = '\x22';
|
|
SIMPLE_ESCAPE_SEQUENCES[CHAR_SLASH] = '/';
|
|
SIMPLE_ESCAPE_SEQUENCES[CHAR_BACKSLASH] = '\x5C';
|
|
SIMPLE_ESCAPE_SEQUENCES[CHAR_CAPITAL_N] = '\x85';
|
|
SIMPLE_ESCAPE_SEQUENCES[CHAR_UNDERSCORE] = '\xA0';
|
|
SIMPLE_ESCAPE_SEQUENCES[CHAR_CAPITAL_L] = '\u2028';
|
|
SIMPLE_ESCAPE_SEQUENCES[CHAR_CAPITAL_P] = '\u2029';
|
|
|
|
|
|
var HEXADECIMAL_ESCAPE_SEQUENCES = {};
|
|
|
|
HEXADECIMAL_ESCAPE_SEQUENCES[CHAR_SMALL_X] = 2;
|
|
HEXADECIMAL_ESCAPE_SEQUENCES[CHAR_SMALL_U] = 4;
|
|
HEXADECIMAL_ESCAPE_SEQUENCES[CHAR_CAPITAL_U] = 8;
|
|
|
|
|
|
var PATTERN_NON_PRINTABLE = /[\x00-\x08\x0B\x0C\x0E-\x1F\x7F-\x84\x86-\x9F\uD800-\uDFFF\uFFFE\uFFFF]/;
|
|
var PATTERN_NON_ASCII_LINE_BREAKS = /[\x85\u2028\u2029]/;
|
|
var PATTERN_FLOW_INDICATORS = /[,\[\]\{\}]/;
|
|
var PATTERN_TAG_HANDLE = /^(?:!|!!|![a-z\-]+!)$/i;
|
|
var PATTERN_TAG_URI = /^(?:!|[^,\[\]\{\}])(?:%[0-9a-f]{2}|[0-9a-z\-#;\/\?:@&=\+\$,_\.!~\*'\(\)\[\]])*$/i;
|
|
|
|
|
|
function loadAll(input, output, options) {
|
|
options = options || {};
|
|
|
|
var filename = options['filename'] || null,
|
|
schema = options['schema'] || DEFAULT_FULL_SCHEMA,
|
|
resolve = options['resolve'] || true,
|
|
validate = options['validate'] || true,
|
|
strict = options['strict'] || false,
|
|
legacy = options['legacy'] || false,
|
|
|
|
directiveHandlers = {},
|
|
implicitTypes = schema.compiledImplicit,
|
|
typeMap = schema.compiledTypeMap,
|
|
|
|
length = input.length,
|
|
position = 0,
|
|
line = 0,
|
|
lineStart = 0,
|
|
lineIndent = 0,
|
|
character = input.charCodeAt(position),
|
|
|
|
version,
|
|
checkLineBreaks,
|
|
tagMap,
|
|
anchorMap,
|
|
tag,
|
|
anchor,
|
|
kind,
|
|
result;
|
|
|
|
function generateError(message) {
|
|
return new YAMLException(
|
|
message,
|
|
new Mark(filename, input, position, line, (position - lineStart)));
|
|
}
|
|
|
|
function throwError(message) {
|
|
throw generateError(message);
|
|
}
|
|
|
|
function throwWarning(message) {
|
|
var error = generateError(message);
|
|
|
|
if (strict) {
|
|
throw error;
|
|
} else {
|
|
console.warn(error.toString());
|
|
}
|
|
}
|
|
|
|
directiveHandlers['YAML'] = function handleYamlDirective(name, args) {
|
|
var match, major, minor;
|
|
|
|
if (null !== version) {
|
|
throwError('duplication of %YAML directive');
|
|
}
|
|
|
|
if (1 !== args.length) {
|
|
throwError('YAML directive accepts exactly one argument');
|
|
}
|
|
|
|
match = /^([0-9]+)\.([0-9]+)$/.exec(args[0]);
|
|
|
|
if (null === match) {
|
|
throwError('ill-formed argument of the YAML directive');
|
|
}
|
|
|
|
major = parseInt(match[1], 10);
|
|
minor = parseInt(match[2], 10);
|
|
|
|
if (1 !== major) {
|
|
throwError('unacceptable YAML version of the document');
|
|
}
|
|
|
|
version = args[0];
|
|
checkLineBreaks = (minor < 2);
|
|
|
|
if (1 !== minor && 2 !== minor) {
|
|
throwWarning('unsupported YAML version of the document');
|
|
}
|
|
};
|
|
|
|
directiveHandlers['TAG'] = function handleTagDirective(name, args) {
|
|
var handle, prefix;
|
|
|
|
if (2 !== args.length) {
|
|
throwError('TAG directive accepts exactly two arguments');
|
|
}
|
|
|
|
handle = args[0];
|
|
prefix = args[1];
|
|
|
|
if (!PATTERN_TAG_HANDLE.test(handle)) {
|
|
throwError('ill-formed tag handle (first argument) of the TAG directive');
|
|
}
|
|
|
|
if (_hasOwnProperty.call(tagMap, handle)) {
|
|
throwError('there is a previously declared suffix for "' + handle + '" tag handle');
|
|
}
|
|
|
|
if (!PATTERN_TAG_URI.test(prefix)) {
|
|
throwError('ill-formed tag prefix (second argument) of the TAG directive');
|
|
}
|
|
|
|
tagMap[handle] = prefix;
|
|
};
|
|
|
|
function captureSegment(start, end, checkJson) {
|
|
var _position, _length, _character, _result;
|
|
|
|
if (start < end) {
|
|
_result = input.slice(start, end);
|
|
|
|
if (checkJson && validate) {
|
|
for (_position = 0, _length = _result.length;
|
|
_position < _length;
|
|
_position += 1) {
|
|
_character = _result.charCodeAt(_position);
|
|
if (!(0x09 === _character ||
|
|
0x20 <= _character && _character <= 0x10FFFF)) {
|
|
throwError('expected valid JSON character');
|
|
}
|
|
}
|
|
}
|
|
|
|
result += _result;
|
|
}
|
|
}
|
|
|
|
function mergeMappings(destination, source) {
|
|
var sourceKeys, key, index, quantity;
|
|
|
|
if (!common.isObject(source)) {
|
|
throwError('cannot merge mappings; the provided source object is unacceptable');
|
|
}
|
|
|
|
sourceKeys = Object.keys(source);
|
|
|
|
for (index = 0, quantity = sourceKeys.length; index < quantity; index += 1) {
|
|
key = sourceKeys[index];
|
|
|
|
if (!_hasOwnProperty.call(destination, key)) {
|
|
destination[key] = source[key];
|
|
}
|
|
}
|
|
}
|
|
|
|
function storeMappingPair(_result, keyTag, keyNode, valueNode) {
|
|
var index, quantity;
|
|
|
|
keyNode = String(keyNode);
|
|
|
|
if (null === _result) {
|
|
_result = {};
|
|
}
|
|
|
|
if ('tag:yaml.org,2002:merge' === keyTag) {
|
|
if (Array.isArray(valueNode)) {
|
|
for (index = 0, quantity = valueNode.length; index < quantity; index += 1) {
|
|
mergeMappings(_result, valueNode[index]);
|
|
}
|
|
} else {
|
|
mergeMappings(_result, valueNode);
|
|
}
|
|
} else {
|
|
_result[keyNode] = valueNode;
|
|
}
|
|
|
|
return _result;
|
|
}
|
|
|
|
function readLineBreak() {
|
|
if (CHAR_LINE_FEED === character) {
|
|
position += 1;
|
|
} else if (CHAR_CARRIAGE_RETURN === character) {
|
|
if (CHAR_LINE_FEED === input.charCodeAt(position + 1)) {
|
|
position += 2;
|
|
} else {
|
|
position += 1;
|
|
}
|
|
} else {
|
|
throwError('a line break is expected');
|
|
}
|
|
|
|
line += 1;
|
|
lineStart = position;
|
|
character = input.charCodeAt(position);
|
|
}
|
|
|
|
function skipSeparationSpace(allowComments, checkIndent) {
|
|
var lineBreaks = 0;
|
|
|
|
while (position < length) {
|
|
while (CHAR_SPACE === character || CHAR_TAB === character) {
|
|
character = input.charCodeAt(++position);
|
|
}
|
|
|
|
if (allowComments && CHAR_SHARP === character) {
|
|
do { character = input.charCodeAt(++position); }
|
|
while (position < length &&
|
|
CHAR_LINE_FEED !== character &&
|
|
CHAR_CARRIAGE_RETURN !== character);
|
|
}
|
|
|
|
if (CHAR_LINE_FEED === character || CHAR_CARRIAGE_RETURN === character) {
|
|
readLineBreak();
|
|
lineBreaks += 1;
|
|
lineIndent = 0;
|
|
|
|
while (CHAR_SPACE === character) {
|
|
lineIndent += 1;
|
|
character = input.charCodeAt(++position);
|
|
}
|
|
|
|
if (lineIndent < checkIndent) {
|
|
throwWarning('deficient indentation');
|
|
}
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
|
|
return lineBreaks;
|
|
}
|
|
|
|
function testDocumentSeparator() {
|
|
var _position, _character;
|
|
|
|
if (position === lineStart &&
|
|
(CHAR_MINUS === character || CHAR_DOT === character) &&
|
|
input.charCodeAt(position + 1) === character &&
|
|
input.charCodeAt(position + 2) === character) {
|
|
|
|
_position = position + 3;
|
|
_character = input.charCodeAt(_position);
|
|
|
|
if (_position >= length ||
|
|
CHAR_SPACE === _character ||
|
|
CHAR_TAB === _character ||
|
|
CHAR_LINE_FEED === _character ||
|
|
CHAR_CARRIAGE_RETURN === _character) {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
function writeFoldedLines(count) {
|
|
if (1 === count) {
|
|
result += ' ';
|
|
} else if (count > 1) {
|
|
result += common.repeat('\n', count - 1);
|
|
}
|
|
}
|
|
|
|
function readPlainScalar(nodeIndent, withinFlowCollection) {
|
|
var preceding,
|
|
following,
|
|
captureStart,
|
|
captureEnd,
|
|
hasPendingContent,
|
|
_line,
|
|
_lineStart,
|
|
_lineIndent,
|
|
_kind = kind,
|
|
_result = result;
|
|
|
|
if (CHAR_SPACE === character ||
|
|
CHAR_TAB === character ||
|
|
CHAR_LINE_FEED === character ||
|
|
CHAR_CARRIAGE_RETURN === character ||
|
|
CHAR_COMMA === character ||
|
|
CHAR_LEFT_SQUARE_BRACKET === character ||
|
|
CHAR_RIGHT_SQUARE_BRACKET === character ||
|
|
CHAR_LEFT_CURLY_BRACKET === character ||
|
|
CHAR_RIGHT_CURLY_BRACKET === character ||
|
|
CHAR_SHARP === character ||
|
|
CHAR_AMPERSAND === character ||
|
|
CHAR_ASTERISK === character ||
|
|
CHAR_EXCLAMATION === character ||
|
|
CHAR_VERTICAL_LINE === character ||
|
|
CHAR_GREATER_THAN === character ||
|
|
CHAR_SINGLE_QUOTE === character ||
|
|
CHAR_DOUBLE_QUOTE === character ||
|
|
CHAR_PERCENT === character ||
|
|
CHAR_COMMERCIAL_AT === character ||
|
|
CHAR_GRAVE_ACCENT === character) {
|
|
return false;
|
|
}
|
|
|
|
if (CHAR_QUESTION === character ||
|
|
CHAR_MINUS === character) {
|
|
following = input.charCodeAt(position + 1);
|
|
|
|
if (CHAR_SPACE === following ||
|
|
CHAR_TAB === following ||
|
|
CHAR_LINE_FEED === following ||
|
|
CHAR_CARRIAGE_RETURN === following ||
|
|
withinFlowCollection &&
|
|
(CHAR_COMMA === following ||
|
|
CHAR_LEFT_SQUARE_BRACKET === following ||
|
|
CHAR_RIGHT_SQUARE_BRACKET === following ||
|
|
CHAR_LEFT_CURLY_BRACKET === following ||
|
|
CHAR_RIGHT_CURLY_BRACKET === following)) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
kind = KIND_STRING;
|
|
result = '';
|
|
captureStart = captureEnd = position;
|
|
hasPendingContent = false;
|
|
|
|
while (position < length) {
|
|
if (CHAR_COLON === character) {
|
|
following = input.charCodeAt(position + 1);
|
|
|
|
if (CHAR_SPACE === following ||
|
|
CHAR_TAB === following ||
|
|
CHAR_LINE_FEED === following ||
|
|
CHAR_CARRIAGE_RETURN === following ||
|
|
withinFlowCollection &&
|
|
(CHAR_COMMA === following ||
|
|
CHAR_LEFT_SQUARE_BRACKET === following ||
|
|
CHAR_RIGHT_SQUARE_BRACKET === following ||
|
|
CHAR_LEFT_CURLY_BRACKET === following ||
|
|
CHAR_RIGHT_CURLY_BRACKET === following)) {
|
|
break;
|
|
}
|
|
|
|
} else if (CHAR_SHARP === character) {
|
|
preceding = input.charCodeAt(position - 1);
|
|
|
|
if (CHAR_SPACE === preceding ||
|
|
CHAR_TAB === preceding ||
|
|
CHAR_LINE_FEED === preceding ||
|
|
CHAR_CARRIAGE_RETURN === preceding) {
|
|
break;
|
|
}
|
|
|
|
} else if ((position === lineStart && testDocumentSeparator()) ||
|
|
withinFlowCollection &&
|
|
(CHAR_COMMA === character ||
|
|
CHAR_LEFT_SQUARE_BRACKET === character ||
|
|
CHAR_RIGHT_SQUARE_BRACKET === character ||
|
|
CHAR_LEFT_CURLY_BRACKET === character ||
|
|
CHAR_RIGHT_CURLY_BRACKET === character)) {
|
|
break;
|
|
|
|
} else if (CHAR_LINE_FEED === character ||
|
|
CHAR_CARRIAGE_RETURN === character) {
|
|
_line = line;
|
|
_lineStart = lineStart;
|
|
_lineIndent = lineIndent;
|
|
skipSeparationSpace(false, -1);
|
|
|
|
if (lineIndent >= nodeIndent) {
|
|
hasPendingContent = true;
|
|
continue;
|
|
} else {
|
|
position = captureEnd;
|
|
line = _line;
|
|
lineStart = _lineStart;
|
|
lineIndent = _lineIndent;
|
|
character = input.charCodeAt(position);
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (hasPendingContent) {
|
|
captureSegment(captureStart, captureEnd, false);
|
|
writeFoldedLines(line - _line);
|
|
captureStart = captureEnd = position;
|
|
hasPendingContent = false;
|
|
}
|
|
|
|
if (CHAR_SPACE !== character && CHAR_TAB !== character) {
|
|
captureEnd = position + 1;
|
|
}
|
|
|
|
character = input.charCodeAt(++position);
|
|
}
|
|
|
|
captureSegment(captureStart, captureEnd, false);
|
|
|
|
if (result) {
|
|
return true;
|
|
} else {
|
|
kind = _kind;
|
|
result = _result;
|
|
return false;
|
|
}
|
|
}
|
|
|
|
function readSingleQuotedScalar(nodeIndent) {
|
|
var captureStart, captureEnd;
|
|
|
|
if (CHAR_SINGLE_QUOTE !== character) {
|
|
return false;
|
|
}
|
|
|
|
kind = KIND_STRING;
|
|
result = '';
|
|
character = input.charCodeAt(++position);
|
|
captureStart = captureEnd = position;
|
|
|
|
while (position < length) {
|
|
if (CHAR_SINGLE_QUOTE === character) {
|
|
captureSegment(captureStart, position, true);
|
|
character = input.charCodeAt(++position);
|
|
|
|
if (CHAR_SINGLE_QUOTE === character) {
|
|
captureStart = captureEnd = position;
|
|
character = input.charCodeAt(++position);
|
|
} else {
|
|
return true;
|
|
}
|
|
|
|
} else if (CHAR_LINE_FEED === character ||
|
|
CHAR_CARRIAGE_RETURN === character) {
|
|
captureSegment(captureStart, captureEnd, true);
|
|
writeFoldedLines(skipSeparationSpace(false, nodeIndent));
|
|
captureStart = captureEnd = position;
|
|
character = input.charCodeAt(position);
|
|
|
|
} else if (position === lineStart && testDocumentSeparator()) {
|
|
throwError('unexpected end of the document within a single quoted scalar');
|
|
|
|
} else {
|
|
character = input.charCodeAt(++position);
|
|
captureEnd = position;
|
|
}
|
|
}
|
|
|
|
throwError('unexpected end of the stream within a single quoted scalar');
|
|
}
|
|
|
|
function readDoubleQuotedScalar(nodeIndent) {
|
|
var captureStart,
|
|
captureEnd,
|
|
hexLength,
|
|
hexIndex,
|
|
hexOffset,
|
|
hexResult;
|
|
|
|
if (CHAR_DOUBLE_QUOTE !== character) {
|
|
return false;
|
|
}
|
|
|
|
kind = KIND_STRING;
|
|
result = '';
|
|
character = input.charCodeAt(++position);
|
|
captureStart = captureEnd = position;
|
|
|
|
while (position < length) {
|
|
if (CHAR_DOUBLE_QUOTE === character) {
|
|
captureSegment(captureStart, position, true);
|
|
character = input.charCodeAt(++position);
|
|
return true;
|
|
|
|
} else if (CHAR_BACKSLASH === character) {
|
|
captureSegment(captureStart, position, true);
|
|
character = input.charCodeAt(++position);
|
|
|
|
if (CHAR_LINE_FEED === character ||
|
|
CHAR_CARRIAGE_RETURN === character) {
|
|
skipSeparationSpace(false, nodeIndent);
|
|
|
|
} else if (SIMPLE_ESCAPE_SEQUENCES[character]) {
|
|
result += SIMPLE_ESCAPE_SEQUENCES[character];
|
|
character = input.charCodeAt(++position);
|
|
|
|
} else if (HEXADECIMAL_ESCAPE_SEQUENCES[character]) {
|
|
hexLength = HEXADECIMAL_ESCAPE_SEQUENCES[character];
|
|
hexResult = 0;
|
|
|
|
for (hexIndex = 1; hexIndex <= hexLength; hexIndex += 1) {
|
|
hexOffset = (hexLength - hexIndex) * 4;
|
|
character = input.charCodeAt(++position);
|
|
|
|
if (CHAR_DIGIT_ZERO <= character && character <= CHAR_DIGIT_NINE) {
|
|
hexResult |= (character - CHAR_DIGIT_ZERO) << hexOffset;
|
|
|
|
} else if (CHAR_CAPITAL_A <= character && character <= CHAR_CAPITAL_F) {
|
|
hexResult |= (character - CHAR_CAPITAL_A + 10) << hexOffset;
|
|
|
|
} else if (CHAR_SMALL_A <= character && character <= CHAR_SMALL_F) {
|
|
hexResult |= (character - CHAR_SMALL_A + 10) << hexOffset;
|
|
|
|
} else {
|
|
throwError('expected hexadecimal character');
|
|
}
|
|
}
|
|
|
|
result += String.fromCharCode(hexResult);
|
|
character = input.charCodeAt(++position);
|
|
|
|
} else {
|
|
throwError('unknown escape sequence');
|
|
}
|
|
|
|
captureStart = captureEnd = position;
|
|
|
|
} else if (CHAR_LINE_FEED === character ||
|
|
CHAR_CARRIAGE_RETURN === character) {
|
|
captureSegment(captureStart, captureEnd, true);
|
|
writeFoldedLines(skipSeparationSpace(false, nodeIndent));
|
|
captureStart = captureEnd = position;
|
|
character = input.charCodeAt(position);
|
|
|
|
} else if (position === lineStart && testDocumentSeparator()) {
|
|
throwError('unexpected end of the document within a double quoted scalar');
|
|
|
|
} else {
|
|
character = input.charCodeAt(++position);
|
|
captureEnd = position;
|
|
}
|
|
}
|
|
|
|
throwError('unexpected end of the stream within a double quoted scalar');
|
|
}
|
|
|
|
function readFlowCollection(nodeIndent) {
|
|
var readNext = true,
|
|
_line,
|
|
_tag = tag,
|
|
_result,
|
|
following,
|
|
terminator,
|
|
isPair,
|
|
isExplicitPair,
|
|
isMapping,
|
|
keyNode,
|
|
keyTag,
|
|
valueNode;
|
|
|
|
switch (character) {
|
|
case CHAR_LEFT_SQUARE_BRACKET:
|
|
terminator = CHAR_RIGHT_SQUARE_BRACKET;
|
|
isMapping = false;
|
|
_result = [];
|
|
break;
|
|
|
|
case CHAR_LEFT_CURLY_BRACKET:
|
|
terminator = CHAR_RIGHT_CURLY_BRACKET;
|
|
isMapping = true;
|
|
_result = {};
|
|
break;
|
|
|
|
default:
|
|
return false;
|
|
}
|
|
|
|
if (null !== anchor) {
|
|
anchorMap[anchor] = _result;
|
|
}
|
|
|
|
character = input.charCodeAt(++position);
|
|
|
|
while (position < length) {
|
|
skipSeparationSpace(true, nodeIndent);
|
|
|
|
if (character === terminator) {
|
|
character = input.charCodeAt(++position);
|
|
tag = _tag;
|
|
kind = isMapping ? KIND_OBJECT : KIND_ARRAY;
|
|
result = _result;
|
|
return true;
|
|
} else if (!readNext) {
|
|
throwError('missed comma between flow collection entries');
|
|
}
|
|
|
|
keyTag = keyNode = valueNode = null;
|
|
isPair = isExplicitPair = false;
|
|
|
|
if (CHAR_QUESTION === character) {
|
|
following = input.charCodeAt(position + 1);
|
|
|
|
if (CHAR_SPACE === following ||
|
|
CHAR_TAB === following ||
|
|
CHAR_LINE_FEED === following ||
|
|
CHAR_CARRIAGE_RETURN === following) {
|
|
isPair = isExplicitPair = true;
|
|
position += 1;
|
|
character = following;
|
|
skipSeparationSpace(true, nodeIndent);
|
|
}
|
|
}
|
|
|
|
_line = line;
|
|
composeNode(nodeIndent, CONTEXT_FLOW_IN, false, true);
|
|
keyTag = tag;
|
|
keyNode = result;
|
|
skipSeparationSpace(true, nodeIndent);
|
|
|
|
if ((isExplicitPair || line === _line) && CHAR_COLON === character) {
|
|
isPair = true;
|
|
character = input.charCodeAt(++position);
|
|
skipSeparationSpace(true, nodeIndent);
|
|
composeNode(nodeIndent, CONTEXT_FLOW_IN, false, true);
|
|
valueNode = result;
|
|
}
|
|
|
|
if (isMapping) {
|
|
storeMappingPair(_result, keyTag, keyNode, valueNode);
|
|
} else if (isPair) {
|
|
_result.push(storeMappingPair(null, keyTag, keyNode, valueNode));
|
|
} else {
|
|
_result.push(keyNode);
|
|
}
|
|
|
|
skipSeparationSpace(true, nodeIndent);
|
|
|
|
if (CHAR_COMMA === character) {
|
|
readNext = true;
|
|
character = input.charCodeAt(++position);
|
|
} else {
|
|
readNext = false;
|
|
}
|
|
}
|
|
|
|
throwError('unexpected end of the stream within a flow collection');
|
|
}
|
|
|
|
function readBlockScalar(nodeIndent) {
|
|
var captureStart,
|
|
folding,
|
|
chomping = CHOMPING_CLIP,
|
|
detectedIndent = false,
|
|
textIndent = nodeIndent,
|
|
emptyLines = 0,
|
|
atMoreIndented = false;
|
|
|
|
switch (character) {
|
|
case CHAR_VERTICAL_LINE:
|
|
folding = false;
|
|
break;
|
|
|
|
case CHAR_GREATER_THAN:
|
|
folding = true;
|
|
break;
|
|
|
|
default:
|
|
return false;
|
|
}
|
|
|
|
kind = KIND_STRING;
|
|
result = '';
|
|
|
|
while (position < length) {
|
|
character = input.charCodeAt(++position);
|
|
|
|
if (CHAR_PLUS === character || CHAR_MINUS === character) {
|
|
if (CHOMPING_CLIP === chomping) {
|
|
chomping = (CHAR_PLUS === character) ? CHOMPING_KEEP : CHOMPING_STRIP;
|
|
} else {
|
|
throwError('repeat of a chomping mode identifier');
|
|
}
|
|
|
|
} else if (CHAR_DIGIT_ZERO <= character && character <= CHAR_DIGIT_NINE) {
|
|
if (CHAR_DIGIT_ZERO === character) {
|
|
throwError('bad explicit indentation width of a block scalar; it cannot be less than one');
|
|
} else if (!detectedIndent) {
|
|
textIndent = nodeIndent + (character - CHAR_DIGIT_ONE);
|
|
detectedIndent = true;
|
|
} else {
|
|
throwError('repeat of an indentation width identifier');
|
|
}
|
|
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (CHAR_SPACE === character || CHAR_TAB === character) {
|
|
do { character = input.charCodeAt(++position); }
|
|
while (CHAR_SPACE === character || CHAR_TAB === character);
|
|
|
|
if (CHAR_SHARP === character) {
|
|
do { character = input.charCodeAt(++position); }
|
|
while (position < length &&
|
|
CHAR_LINE_FEED !== character &&
|
|
CHAR_CARRIAGE_RETURN !== character);
|
|
}
|
|
}
|
|
|
|
while (position < length) {
|
|
readLineBreak();
|
|
lineIndent = 0;
|
|
|
|
while ((!detectedIndent || lineIndent < textIndent) &&
|
|
(CHAR_SPACE === character)) {
|
|
lineIndent += 1;
|
|
character = input.charCodeAt(++position);
|
|
}
|
|
|
|
if (!detectedIndent && lineIndent > textIndent) {
|
|
textIndent = lineIndent;
|
|
}
|
|
|
|
if (CHAR_LINE_FEED === character || CHAR_CARRIAGE_RETURN === character) {
|
|
emptyLines += 1;
|
|
continue;
|
|
}
|
|
|
|
// End of the scalar.
|
|
if (lineIndent < textIndent) {
|
|
|
|
// Perform the chomping.
|
|
switch (chomping) {
|
|
case CHOMPING_KEEP:
|
|
result += common.repeat('\n', emptyLines);
|
|
break;
|
|
|
|
case CHOMPING_CLIP:
|
|
if (detectedIndent) { // i.e. only if the scalar is not empty.
|
|
result += '\n';
|
|
}
|
|
break;
|
|
}
|
|
|
|
// Break this `while` cycle and go to the funciton's epilogue.
|
|
break;
|
|
}
|
|
|
|
// Folded style: use fancy rules to handle line breaks.
|
|
if (folding) {
|
|
|
|
// Lines starting with white space characters (more-indented lines) are not folded.
|
|
if (CHAR_SPACE === character || CHAR_TAB === character) {
|
|
atMoreIndented = true;
|
|
result += common.repeat('\n', emptyLines + 1);
|
|
|
|
// End of more-indented block.
|
|
} else if (atMoreIndented) {
|
|
atMoreIndented = false;
|
|
result += common.repeat('\n', emptyLines + 1);
|
|
|
|
// Just one line break - perceive as the same line.
|
|
} else if (0 === emptyLines) {
|
|
if (detectedIndent) { // i.e. only if we have already read some scalar content.
|
|
result += ' ';
|
|
}
|
|
|
|
// Several line breaks - perceive as different lines.
|
|
} else {
|
|
result += common.repeat('\n', emptyLines);
|
|
}
|
|
|
|
// Literal style: just add exact number of line breaks between content lines.
|
|
} else {
|
|
result += common.repeat('\n', emptyLines + 1);
|
|
}
|
|
|
|
detectedIndent = true;
|
|
emptyLines = 0;
|
|
captureStart = position;
|
|
|
|
do { character = input.charCodeAt(++position); }
|
|
while (position < length &&
|
|
CHAR_LINE_FEED !== character &&
|
|
CHAR_CARRIAGE_RETURN !== character);
|
|
|
|
captureSegment(captureStart, position, false);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
function readBlockSequence(nodeIndent) {
|
|
var _line,
|
|
_tag = tag,
|
|
_result = [],
|
|
following,
|
|
detected = false;
|
|
|
|
if (null !== anchor) {
|
|
anchorMap[anchor] = _result;
|
|
}
|
|
|
|
while (position < length) {
|
|
if (CHAR_MINUS !== character) {
|
|
break;
|
|
}
|
|
|
|
following = input.charCodeAt(position + 1);
|
|
|
|
if (CHAR_SPACE !== following &&
|
|
CHAR_TAB !== following &&
|
|
CHAR_LINE_FEED !== following &&
|
|
CHAR_CARRIAGE_RETURN !== following) {
|
|
break;
|
|
}
|
|
|
|
detected = true;
|
|
position += 1;
|
|
character = following;
|
|
|
|
if (skipSeparationSpace(true, -1)) {
|
|
if (lineIndent <= nodeIndent) {
|
|
_result.push(null);
|
|
continue;
|
|
}
|
|
}
|
|
|
|
_line = line;
|
|
composeNode(nodeIndent, CONTEXT_BLOCK_IN, false, true);
|
|
_result.push(result);
|
|
skipSeparationSpace(true, -1);
|
|
|
|
if ((line === _line || lineIndent > nodeIndent) && position < length) {
|
|
throwError('bad indentation of a sequence entry');
|
|
} else if (lineIndent < nodeIndent) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (detected) {
|
|
tag = _tag;
|
|
kind = KIND_ARRAY;
|
|
result = _result;
|
|
return true;
|
|
} else {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
function readBlockMapping(nodeIndent) {
|
|
var following,
|
|
allowCompact,
|
|
_line,
|
|
_tag = tag,
|
|
_result = {},
|
|
keyTag = null,
|
|
keyNode = null,
|
|
valueNode = null,
|
|
atExplicitKey = false,
|
|
detected = false;
|
|
|
|
if (null !== anchor) {
|
|
anchorMap[anchor] = _result;
|
|
}
|
|
|
|
while (position < length) {
|
|
following = input.charCodeAt(position + 1);
|
|
_line = line; // Save the current line.
|
|
|
|
//
|
|
// Explicit notation case. There are two separate blocks:
|
|
// first for the key (denoted by "?") and second for the value (denoted by ":")
|
|
//
|
|
if ((CHAR_QUESTION === character ||
|
|
CHAR_COLON === character) &&
|
|
(CHAR_SPACE === following ||
|
|
CHAR_TAB === following ||
|
|
CHAR_LINE_FEED === following ||
|
|
CHAR_CARRIAGE_RETURN === following)) {
|
|
|
|
if (CHAR_QUESTION === character) {
|
|
if (atExplicitKey) {
|
|
storeMappingPair(_result, keyTag, keyNode, null);
|
|
keyTag = keyNode = valueNode = null;
|
|
}
|
|
|
|
detected = true;
|
|
atExplicitKey = true;
|
|
allowCompact = true;
|
|
|
|
} else if (atExplicitKey) {
|
|
// i.e. CHAR_COLON === character after the explicit key.
|
|
atExplicitKey = false;
|
|
allowCompact = true;
|
|
|
|
} else {
|
|
throwError('incomplete explicit mapping pair; a key node is missed');
|
|
}
|
|
|
|
position += 1;
|
|
character = following;
|
|
|
|
//
|
|
// Implicit notation case. Flow-style node as the key first, then ":", and the value.
|
|
//
|
|
} else if (composeNode(nodeIndent, CONTEXT_FLOW_OUT, false, true)) {
|
|
if (line === _line) {
|
|
while (CHAR_SPACE === character ||
|
|
CHAR_TAB === character) {
|
|
character = input.charCodeAt(++position);
|
|
}
|
|
|
|
if (CHAR_COLON === character) {
|
|
character = input.charCodeAt(++position);
|
|
|
|
if (CHAR_SPACE !== character &&
|
|
CHAR_TAB !== character &&
|
|
CHAR_LINE_FEED !== character &&
|
|
CHAR_CARRIAGE_RETURN !== character) {
|
|
throwError('a whitespace character is expected after the key-value separator within a block mapping');
|
|
}
|
|
|
|
if (atExplicitKey) {
|
|
storeMappingPair(_result, keyTag, keyNode, null);
|
|
keyTag = keyNode = valueNode = null;
|
|
}
|
|
|
|
detected = true;
|
|
atExplicitKey = false;
|
|
allowCompact = false;
|
|
keyTag = tag;
|
|
keyNode = result;
|
|
|
|
} else if (detected) {
|
|
throwError('can not read an implicit mapping pair; a colon is missed');
|
|
|
|
} else {
|
|
tag = _tag;
|
|
return true; // Keep the result of `composeNode`.
|
|
}
|
|
|
|
} else if (detected) {
|
|
throwError('can not read a block mapping entry; a multiline key may not be an implicit key');
|
|
|
|
} else {
|
|
tag = _tag;
|
|
return true; // Keep the result of `composeNode`.
|
|
}
|
|
|
|
} else {
|
|
break; // Reading is done. Go to the epilogue.
|
|
}
|
|
|
|
//
|
|
// Common reading code for both explicit and implicit notations.
|
|
//
|
|
if (line === _line || lineIndent > nodeIndent) {
|
|
if (composeNode(nodeIndent, CONTEXT_BLOCK_OUT, true, allowCompact)) {
|
|
if (atExplicitKey) {
|
|
keyNode = result;
|
|
} else {
|
|
valueNode = result;
|
|
}
|
|
}
|
|
|
|
if (!atExplicitKey) {
|
|
storeMappingPair(_result, keyTag, keyNode, valueNode);
|
|
keyTag = keyNode = valueNode = null;
|
|
}
|
|
|
|
skipSeparationSpace(true, -1);
|
|
}
|
|
|
|
if (lineIndent > nodeIndent && position < length) {
|
|
throwError('bad indentation of a mapping entry');
|
|
} else if (lineIndent < nodeIndent) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Epilogue.
|
|
//
|
|
|
|
// Special case: last mapping's node contains only the key in explicit notation.
|
|
if (atExplicitKey) {
|
|
storeMappingPair(_result, keyTag, keyNode, null);
|
|
}
|
|
|
|
// Expose the resulting mapping.
|
|
if (detected) {
|
|
tag = _tag;
|
|
kind = KIND_OBJECT;
|
|
result = _result;
|
|
}
|
|
|
|
return detected;
|
|
}
|
|
|
|
function readTagProperty() {
|
|
var _position,
|
|
isVerbatim = false,
|
|
isNamed = false,
|
|
tagHandle,
|
|
tagName;
|
|
|
|
if (CHAR_EXCLAMATION !== character) {
|
|
return false;
|
|
}
|
|
|
|
if (null !== tag) {
|
|
throwError('duplication of a tag property');
|
|
}
|
|
|
|
character = input.charCodeAt(++position);
|
|
|
|
if (CHAR_LESS_THAN === character) {
|
|
isVerbatim = true;
|
|
character = input.charCodeAt(++position);
|
|
|
|
} else if (CHAR_EXCLAMATION === character) {
|
|
isNamed = true;
|
|
tagHandle = '!!';
|
|
character = input.charCodeAt(++position);
|
|
|
|
} else {
|
|
tagHandle = '!';
|
|
}
|
|
|
|
_position = position;
|
|
|
|
if (isVerbatim) {
|
|
do { character = input.charCodeAt(++position); }
|
|
while (position < length && CHAR_GREATER_THAN !== character);
|
|
|
|
if (position < length) {
|
|
tagName = input.slice(_position, position);
|
|
character = input.charCodeAt(++position);
|
|
} else {
|
|
throwError('unexpected end of the stream within a verbatim tag');
|
|
}
|
|
} else {
|
|
while (position < length &&
|
|
CHAR_SPACE !== character &&
|
|
CHAR_TAB !== character &&
|
|
CHAR_LINE_FEED !== character &&
|
|
CHAR_CARRIAGE_RETURN !== character) {
|
|
|
|
if (CHAR_EXCLAMATION === character) {
|
|
if (!isNamed) {
|
|
tagHandle = input.slice(_position - 1, position + 1);
|
|
|
|
if (validate && !PATTERN_TAG_HANDLE.test(tagHandle)) {
|
|
throwError('named tag handle cannot contain such characters');
|
|
}
|
|
|
|
isNamed = true;
|
|
_position = position + 1;
|
|
} else {
|
|
throwError('tag suffix cannot contain exclamation marks');
|
|
}
|
|
}
|
|
|
|
character = input.charCodeAt(++position);
|
|
}
|
|
|
|
tagName = input.slice(_position, position);
|
|
|
|
if (validate && PATTERN_FLOW_INDICATORS.test(tagName)) {
|
|
throwError('tag suffix cannot contain flow indicator characters');
|
|
}
|
|
}
|
|
|
|
if (validate && tagName && !PATTERN_TAG_URI.test(tagName)) {
|
|
throwError('tag name cannot contain such characters: ' + tagName);
|
|
}
|
|
|
|
if (isVerbatim) {
|
|
tag = tagName;
|
|
|
|
} else if (_hasOwnProperty.call(tagMap, tagHandle)) {
|
|
tag = tagMap[tagHandle] + tagName;
|
|
|
|
} else if ('!' === tagHandle) {
|
|
tag = '!' + tagName;
|
|
|
|
} else if ('!!' === tagHandle) {
|
|
tag = 'tag:yaml.org,2002:' + tagName;
|
|
|
|
} else {
|
|
throwError('undeclared tag handle "' + tagHandle + '"');
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
function readAnchorProperty() {
|
|
var _position;
|
|
|
|
if (CHAR_AMPERSAND !== character) {
|
|
return false;
|
|
}
|
|
|
|
if (null !== anchor) {
|
|
throwError('duplication of an anchor property');
|
|
}
|
|
|
|
character = input.charCodeAt(++position);
|
|
_position = position;
|
|
|
|
while (position < length &&
|
|
CHAR_SPACE !== character &&
|
|
CHAR_TAB !== character &&
|
|
CHAR_LINE_FEED !== character &&
|
|
CHAR_CARRIAGE_RETURN !== character &&
|
|
CHAR_COMMA !== character &&
|
|
CHAR_LEFT_SQUARE_BRACKET !== character &&
|
|
CHAR_RIGHT_SQUARE_BRACKET !== character &&
|
|
CHAR_LEFT_CURLY_BRACKET !== character &&
|
|
CHAR_RIGHT_CURLY_BRACKET !== character) {
|
|
character = input.charCodeAt(++position);
|
|
}
|
|
|
|
if (position === _position) {
|
|
throwError('name of an anchor node must contain at least one character');
|
|
}
|
|
|
|
anchor = input.slice(_position, position);
|
|
return true;
|
|
}
|
|
|
|
function readAlias() {
|
|
var _position, alias;
|
|
|
|
if (CHAR_ASTERISK !== character) {
|
|
return false;
|
|
}
|
|
|
|
character = input.charCodeAt(++position);
|
|
_position = position;
|
|
|
|
while (position < length &&
|
|
CHAR_SPACE !== character &&
|
|
CHAR_TAB !== character &&
|
|
CHAR_LINE_FEED !== character &&
|
|
CHAR_CARRIAGE_RETURN !== character &&
|
|
CHAR_COMMA !== character &&
|
|
CHAR_LEFT_SQUARE_BRACKET !== character &&
|
|
CHAR_RIGHT_SQUARE_BRACKET !== character &&
|
|
CHAR_LEFT_CURLY_BRACKET !== character &&
|
|
CHAR_RIGHT_CURLY_BRACKET !== character) {
|
|
character = input.charCodeAt(++position);
|
|
}
|
|
|
|
if (position === _position) {
|
|
throwError('name of an alias node must contain at least one character');
|
|
}
|
|
|
|
alias = input.slice(_position, position);
|
|
|
|
if (!anchorMap.hasOwnProperty(alias)) {
|
|
throwError('unidentified alias "' + alias + '"');
|
|
}
|
|
|
|
result = anchorMap[alias];
|
|
skipSeparationSpace(true, -1);
|
|
return true;
|
|
}
|
|
|
|
function composeNode(parentIndent, nodeContext, allowToSeek, allowCompact) {
|
|
var allowBlockStyles,
|
|
allowBlockScalars,
|
|
allowBlockCollections,
|
|
atNewLine = false,
|
|
isIndented = true,
|
|
hasContent = false,
|
|
typeIndex,
|
|
typeQuantity,
|
|
type,
|
|
typeLoader,
|
|
flowIndent,
|
|
blockIndent,
|
|
_result;
|
|
|
|
tag = null;
|
|
anchor = null;
|
|
kind = null;
|
|
result = null;
|
|
|
|
allowBlockStyles = allowBlockScalars = allowBlockCollections =
|
|
CONTEXT_BLOCK_OUT === nodeContext ||
|
|
CONTEXT_BLOCK_IN === nodeContext;
|
|
|
|
if (allowToSeek) {
|
|
if (skipSeparationSpace(true, -1)) {
|
|
atNewLine = true;
|
|
|
|
if (lineIndent === parentIndent) {
|
|
isIndented = false;
|
|
|
|
} else if (lineIndent > parentIndent) {
|
|
isIndented = true;
|
|
|
|
} else {
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (isIndented) {
|
|
while (readTagProperty() || readAnchorProperty()) {
|
|
if (skipSeparationSpace(true, -1)) {
|
|
atNewLine = true;
|
|
|
|
if (lineIndent > parentIndent) {
|
|
isIndented = true;
|
|
allowBlockCollections = allowBlockStyles;
|
|
|
|
} else if (lineIndent === parentIndent) {
|
|
isIndented = false;
|
|
allowBlockCollections = allowBlockStyles;
|
|
|
|
} else {
|
|
return true;
|
|
}
|
|
} else {
|
|
allowBlockCollections = false;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (allowBlockCollections) {
|
|
allowBlockCollections = atNewLine || allowCompact;
|
|
}
|
|
|
|
if (isIndented || CONTEXT_BLOCK_OUT === nodeContext) {
|
|
if (CONTEXT_FLOW_IN === nodeContext || CONTEXT_FLOW_OUT === nodeContext) {
|
|
flowIndent = parentIndent;
|
|
} else {
|
|
flowIndent = parentIndent + 1;
|
|
}
|
|
|
|
blockIndent = position - lineStart;
|
|
|
|
if (isIndented) {
|
|
if (allowBlockCollections &&
|
|
(readBlockSequence(blockIndent) ||
|
|
readBlockMapping(blockIndent)) ||
|
|
readFlowCollection(flowIndent)) {
|
|
hasContent = true;
|
|
} else {
|
|
if ((allowBlockScalars && readBlockScalar(flowIndent)) ||
|
|
readSingleQuotedScalar(flowIndent) ||
|
|
readDoubleQuotedScalar(flowIndent)) {
|
|
hasContent = true;
|
|
|
|
} else if (readAlias()) {
|
|
hasContent = true;
|
|
|
|
if (null !== tag || null !== anchor) {
|
|
throwError('alias node should not have any properties');
|
|
}
|
|
|
|
} else if (readPlainScalar(flowIndent, CONTEXT_FLOW_IN === nodeContext)) {
|
|
hasContent = true;
|
|
|
|
if (null === tag) {
|
|
tag = '?';
|
|
}
|
|
}
|
|
|
|
if (null !== anchor) {
|
|
anchorMap[anchor] = result;
|
|
}
|
|
}
|
|
} else {
|
|
hasContent = allowBlockCollections && readBlockSequence(blockIndent);
|
|
}
|
|
}
|
|
|
|
if (null !== tag && '!' !== tag) {
|
|
if ('?' === tag) {
|
|
if (resolve) {
|
|
for (typeIndex = 0, typeQuantity = implicitTypes.length;
|
|
typeIndex < typeQuantity;
|
|
typeIndex += 1) {
|
|
type = implicitTypes[typeIndex];
|
|
|
|
// Implicit resolving is not allowed for non-scalar types, and '?'
|
|
// non-specific tag is only assigned to plain scalars. So, it isn't
|
|
// needed to check for 'kind' conformity.
|
|
_result = type.loader.resolver(result, false);
|
|
|
|
if (NIL !== _result) {
|
|
tag = type.tag;
|
|
result = _result;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
} else if (_hasOwnProperty.call(typeMap, tag)) {
|
|
typeLoader = typeMap[tag].loader;
|
|
|
|
if (null !== result && typeLoader.kind !== kind) {
|
|
throwError('unacceptable node kind for !<' + tag + '> tag; it should be "' + typeLoader.kind + '", not "' + kind + '"');
|
|
}
|
|
|
|
if (typeLoader.resolver) {
|
|
_result = typeLoader.resolver(result, true);
|
|
|
|
if (NIL !== _result) {
|
|
result = _result;
|
|
} else {
|
|
throwError('cannot resolve a node with !<' + tag + '> explicit tag');
|
|
}
|
|
}
|
|
} else {
|
|
throwWarning('unknown tag !<' + tag + '>');
|
|
}
|
|
}
|
|
|
|
return null !== tag || null !== anchor || hasContent;
|
|
}
|
|
|
|
function readDocument() {
|
|
var documentStart = position,
|
|
_position,
|
|
directiveName,
|
|
directiveArgs,
|
|
hasDirectives = false;
|
|
|
|
version = null;
|
|
checkLineBreaks = legacy;
|
|
tagMap = {};
|
|
anchorMap = {};
|
|
|
|
while (position < length) {
|
|
skipSeparationSpace(true, -1);
|
|
|
|
if (lineIndent > 0 || CHAR_PERCENT !== character) {
|
|
break;
|
|
}
|
|
|
|
hasDirectives = true;
|
|
character = input.charCodeAt(++position);
|
|
_position = position;
|
|
|
|
while (position < length &&
|
|
CHAR_SPACE !== character &&
|
|
CHAR_TAB !== character &&
|
|
CHAR_LINE_FEED !== character &&
|
|
CHAR_CARRIAGE_RETURN !== character) {
|
|
character = input.charCodeAt(++position);
|
|
}
|
|
|
|
directiveName = input.slice(_position, position);
|
|
directiveArgs = [];
|
|
|
|
if (directiveName.length < 1) {
|
|
throwError('directive name must not be less than one character in length');
|
|
}
|
|
|
|
while (position < length) {
|
|
while (CHAR_SPACE === character || CHAR_TAB === character) {
|
|
character = input.charCodeAt(++position);
|
|
}
|
|
|
|
if (CHAR_SHARP === character) {
|
|
do { character = input.charCodeAt(++position); }
|
|
while (position < length &&
|
|
CHAR_LINE_FEED !== character &&
|
|
CHAR_CARRIAGE_RETURN !== character);
|
|
break;
|
|
}
|
|
|
|
if (CHAR_LINE_FEED === character || CHAR_CARRIAGE_RETURN === character) {
|
|
break;
|
|
}
|
|
|
|
_position = position;
|
|
|
|
while (position < length &&
|
|
CHAR_SPACE !== character &&
|
|
CHAR_TAB !== character &&
|
|
CHAR_LINE_FEED !== character &&
|
|
CHAR_CARRIAGE_RETURN !== character) {
|
|
character = input.charCodeAt(++position);
|
|
}
|
|
|
|
directiveArgs.push(input.slice(_position, position));
|
|
}
|
|
|
|
if (position < length) {
|
|
readLineBreak();
|
|
}
|
|
|
|
if (_hasOwnProperty.call(directiveHandlers, directiveName)) {
|
|
directiveHandlers[directiveName](directiveName, directiveArgs);
|
|
} else {
|
|
throwWarning('unknown document directive "' + directiveName + '"');
|
|
}
|
|
}
|
|
|
|
skipSeparationSpace(true, -1);
|
|
|
|
if (0 === lineIndent &&
|
|
CHAR_MINUS === character &&
|
|
CHAR_MINUS === input.charCodeAt(position + 1) &&
|
|
CHAR_MINUS === input.charCodeAt(position + 2)) {
|
|
position += 3;
|
|
character = input.charCodeAt(position);
|
|
skipSeparationSpace(true, -1);
|
|
|
|
} else if (hasDirectives) {
|
|
throwError('directives end mark is expected');
|
|
}
|
|
|
|
composeNode(lineIndent - 1, CONTEXT_BLOCK_OUT, false, true);
|
|
skipSeparationSpace(true, -1);
|
|
|
|
if (validate && checkLineBreaks &&
|
|
PATTERN_NON_ASCII_LINE_BREAKS.test(input.slice(documentStart, position))) {
|
|
throwWarning('non-ASCII line breaks are interpreted as content');
|
|
}
|
|
|
|
output(result);
|
|
|
|
if (position === lineStart && testDocumentSeparator()) {
|
|
if (CHAR_DOT === character) {
|
|
position += 3;
|
|
character = input.charCodeAt(position);
|
|
skipSeparationSpace(true, -1);
|
|
}
|
|
return;
|
|
}
|
|
|
|
if (position < length) {
|
|
throwError('end of the stream or a document separator is expected');
|
|
} else {
|
|
return;
|
|
}
|
|
}
|
|
|
|
if (validate && PATTERN_NON_PRINTABLE.test(input)) {
|
|
throwError('the stream contains non-printable characters');
|
|
}
|
|
|
|
while (CHAR_SPACE === character) {
|
|
lineIndent += 1;
|
|
character = input.charCodeAt(++position);
|
|
}
|
|
|
|
while (position < length) {
|
|
readDocument();
|
|
}
|
|
}
|
|
|
|
|
|
function load(input, options) {
|
|
var result = null, received = false;
|
|
|
|
function callback(data) {
|
|
if (!received) {
|
|
result = data;
|
|
received = true;
|
|
} else {
|
|
throw new YAMLException('expected a single document in the stream, but found more');
|
|
}
|
|
}
|
|
|
|
loadAll(input, callback, options);
|
|
|
|
return result;
|
|
}
|
|
|
|
|
|
function safeLoadAll(input, output, options) {
|
|
loadAll(input, output, common.extend({ schema: DEFAULT_SAFE_SCHEMA }, options));
|
|
}
|
|
|
|
|
|
function safeLoad(input, options) {
|
|
return load(input, common.extend({ schema: DEFAULT_SAFE_SCHEMA }, options));
|
|
}
|
|
|
|
|
|
module.exports.loadAll = loadAll;
|
|
module.exports.load = load;
|
|
module.exports.safeLoadAll = safeLoadAll;
|
|
module.exports.safeLoad = safeLoad;
|
|
|
|
},{"./common":20,"./exception":22,"./mark":24,"./schema/default_full":28,"./schema/default_safe":29}],24:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
|
|
var common = require('./common');
|
|
|
|
|
|
function Mark(name, buffer, position, line, column) {
|
|
this.name = name;
|
|
this.buffer = buffer;
|
|
this.position = position;
|
|
this.line = line;
|
|
this.column = column;
|
|
}
|
|
|
|
|
|
Mark.prototype.getSnippet = function getSnippet(indent, maxLength) {
|
|
var head, start, tail, end, snippet;
|
|
|
|
if (!this.buffer) {
|
|
return null;
|
|
}
|
|
|
|
indent = indent || 4;
|
|
maxLength = maxLength || 75;
|
|
|
|
head = '';
|
|
start = this.position;
|
|
|
|
while (start > 0 && -1 === '\x00\r\n\x85\u2028\u2029'.indexOf(this.buffer.charAt(start - 1))) {
|
|
start -= 1;
|
|
if (this.position - start > (maxLength / 2 - 1)) {
|
|
head = ' ... ';
|
|
start += 5;
|
|
break;
|
|
}
|
|
}
|
|
|
|
tail = '';
|
|
end = this.position;
|
|
|
|
while (end < this.buffer.length && -1 === '\x00\r\n\x85\u2028\u2029'.indexOf(this.buffer.charAt(end))) {
|
|
end += 1;
|
|
if (end - this.position > (maxLength / 2 - 1)) {
|
|
tail = ' ... ';
|
|
end -= 5;
|
|
break;
|
|
}
|
|
}
|
|
|
|
snippet = this.buffer.slice(start, end);
|
|
|
|
return common.repeat(' ', indent) + head + snippet + tail + '\n' +
|
|
common.repeat(' ', indent + this.position - start + head.length) + '^';
|
|
};
|
|
|
|
|
|
Mark.prototype.toString = function toString(compact) {
|
|
var snippet, where = '';
|
|
|
|
if (this.name) {
|
|
where += 'in "' + this.name + '" ';
|
|
}
|
|
|
|
where += 'at line ' + (this.line + 1) + ', column ' + (this.column + 1);
|
|
|
|
if (!compact) {
|
|
snippet = this.getSnippet();
|
|
|
|
if (snippet) {
|
|
where += ':\n' + snippet;
|
|
}
|
|
}
|
|
|
|
return where;
|
|
};
|
|
|
|
|
|
module.exports = Mark;
|
|
|
|
},{"./common":20}],25:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
|
|
var fs = require('fs');
|
|
var loader = require('./loader');
|
|
|
|
|
|
function yamlRequireHandler(module, filename) {
|
|
var content = fs.readFileSync(filename, 'utf8');
|
|
|
|
// fill in documents
|
|
module.exports = loader.safeLoad(content, { filename: filename });
|
|
}
|
|
|
|
// register require extensions only if we're on node.js
|
|
// hack for browserify
|
|
if (undefined !== require.extensions) {
|
|
require.extensions['.yml'] = yamlRequireHandler;
|
|
require.extensions['.yaml'] = yamlRequireHandler;
|
|
}
|
|
|
|
|
|
module.exports = require;
|
|
|
|
},{"./loader":23,"fs":6}],26:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
|
|
var common = require('./common');
|
|
var YAMLException = require('./exception');
|
|
var Type = require('./type');
|
|
|
|
|
|
function compileList(schema, name, result) {
|
|
var exclude = [];
|
|
|
|
schema.include.forEach(function (includedSchema) {
|
|
result = compileList(includedSchema, name, result);
|
|
});
|
|
|
|
schema[name].forEach(function (currentType) {
|
|
result.forEach(function (previousType, previousIndex) {
|
|
if (previousType.tag === currentType.tag) {
|
|
exclude.push(previousIndex);
|
|
}
|
|
});
|
|
|
|
result.push(currentType);
|
|
});
|
|
|
|
return result.filter(function (type, index) {
|
|
return -1 === exclude.indexOf(index);
|
|
});
|
|
}
|
|
|
|
|
|
function compileMap(/* lists... */) {
|
|
var result = {}, index, length;
|
|
|
|
function collectType(type) {
|
|
result[type.tag] = type;
|
|
}
|
|
|
|
for (index = 0, length = arguments.length; index < length; index += 1) {
|
|
arguments[index].forEach(collectType);
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
|
|
function Schema(definition) {
|
|
this.include = definition.include || [];
|
|
this.implicit = definition.implicit || [];
|
|
this.explicit = definition.explicit || [];
|
|
|
|
this.implicit.forEach(function (type) {
|
|
if (null !== type.loader && 'string' !== type.loader.kind) {
|
|
throw new YAMLException('There is a non-scalar type in the implicit list of a schema. Implicit resolving of such types is not supported.');
|
|
}
|
|
});
|
|
|
|
this.compiledImplicit = compileList(this, 'implicit', []);
|
|
this.compiledExplicit = compileList(this, 'explicit', []);
|
|
this.compiledTypeMap = compileMap(this.compiledImplicit, this.compiledExplicit);
|
|
}
|
|
|
|
|
|
Schema.DEFAULT = null;
|
|
|
|
|
|
Schema.create = function createSchema() {
|
|
var schemas, types;
|
|
|
|
switch (arguments.length) {
|
|
case 1:
|
|
schemas = Schema.DEFAULT;
|
|
types = arguments[0];
|
|
break;
|
|
|
|
case 2:
|
|
schemas = arguments[0];
|
|
types = arguments[1];
|
|
break;
|
|
|
|
default:
|
|
throw new YAMLException('Wrong number of arguments for Schema.create function');
|
|
}
|
|
|
|
schemas = common.toArray(schemas);
|
|
types = common.toArray(types);
|
|
|
|
if (!schemas.every(function (schema) { return schema instanceof Schema; })) {
|
|
throw new YAMLException('Specified list of super schemas (or a single Schema object) contains a non-Schema object.');
|
|
}
|
|
|
|
if (!types.every(function (type) { return type instanceof Type; })) {
|
|
throw new YAMLException('Specified list of YAML types (or a single Type object) contains a non-Type object.');
|
|
}
|
|
|
|
return new Schema({
|
|
include: schemas,
|
|
explicit: types
|
|
});
|
|
};
|
|
|
|
|
|
module.exports = Schema;
|
|
|
|
},{"./common":20,"./exception":22,"./type":32}],27:[function(require,module,exports){
|
|
// Standard YAML's Core schema.
|
|
// http://www.yaml.org/spec/1.2/spec.html#id2804923
|
|
//
|
|
// NOTE: JS-YAML does not support schema-specific tag resolution restrictions.
|
|
// So, Core schema has no distinctions from JSON schema is JS-YAML.
|
|
|
|
|
|
'use strict';
|
|
|
|
|
|
var Schema = require('../schema');
|
|
|
|
|
|
module.exports = new Schema({
|
|
include: [
|
|
require('./json')
|
|
]
|
|
});
|
|
|
|
},{"../schema":26,"./json":31}],28:[function(require,module,exports){
|
|
// JS-YAML's default schema for `load` function.
|
|
// It is not described in the YAML specification.
|
|
//
|
|
// This schema is based on JS-YAML's default safe schema and includes
|
|
// JavaScript-specific types: !!js/undefined, !!js/regexp and !!js/function.
|
|
//
|
|
// Also this schema is used as default base schema at `Schema.create` function.
|
|
|
|
|
|
'use strict';
|
|
|
|
|
|
var Schema = require('../schema');
|
|
|
|
|
|
module.exports = Schema.DEFAULT = new Schema({
|
|
include: [
|
|
require('./default_safe')
|
|
],
|
|
explicit: [
|
|
require('../type/js/undefined'),
|
|
require('../type/js/regexp'),
|
|
require('../type/js/function')
|
|
]
|
|
});
|
|
|
|
},{"../schema":26,"../type/js/function":37,"../type/js/regexp":38,"../type/js/undefined":39,"./default_safe":29}],29:[function(require,module,exports){
|
|
// JS-YAML's default schema for `safeLoad` function.
|
|
// It is not described in the YAML specification.
|
|
//
|
|
// This schema is based on standard YAML's Core schema and includes most of
|
|
// extra types described at YAML tag repository. (http://yaml.org/type/)
|
|
|
|
|
|
'use strict';
|
|
|
|
|
|
var Schema = require('../schema');
|
|
|
|
|
|
module.exports = new Schema({
|
|
include: [
|
|
require('./core')
|
|
],
|
|
implicit: [
|
|
require('../type/timestamp'),
|
|
require('../type/merge')
|
|
],
|
|
explicit: [
|
|
require('../type/binary'),
|
|
require('../type/omap'),
|
|
require('../type/pairs'),
|
|
require('../type/set')
|
|
]
|
|
});
|
|
|
|
},{"../schema":26,"../type/binary":33,"../type/merge":41,"../type/omap":43,"../type/pairs":44,"../type/set":46,"../type/timestamp":48,"./core":27}],30:[function(require,module,exports){
|
|
// Standard YAML's Failsafe schema.
|
|
// http://www.yaml.org/spec/1.2/spec.html#id2802346
|
|
|
|
|
|
'use strict';
|
|
|
|
|
|
var Schema = require('../schema');
|
|
|
|
|
|
module.exports = new Schema({
|
|
explicit: [
|
|
require('../type/str'),
|
|
require('../type/seq'),
|
|
require('../type/map')
|
|
]
|
|
});
|
|
|
|
},{"../schema":26,"../type/map":40,"../type/seq":45,"../type/str":47}],31:[function(require,module,exports){
|
|
// Standard YAML's JSON schema.
|
|
// http://www.yaml.org/spec/1.2/spec.html#id2803231
|
|
//
|
|
// NOTE: JS-YAML does not support schema-specific tag resolution restrictions.
|
|
// So, this schema is not such strict as defined in the YAML specification.
|
|
// It allows numbers in binary notaion, use `Null` and `NULL` as `null`, etc.
|
|
|
|
|
|
'use strict';
|
|
|
|
|
|
var Schema = require('../schema');
|
|
|
|
|
|
module.exports = new Schema({
|
|
include: [
|
|
require('./failsafe')
|
|
],
|
|
implicit: [
|
|
require('../type/null'),
|
|
require('../type/bool'),
|
|
require('../type/int'),
|
|
require('../type/float')
|
|
]
|
|
});
|
|
|
|
},{"../schema":26,"../type/bool":34,"../type/float":35,"../type/int":36,"../type/null":42,"./failsafe":30}],32:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
|
|
var YAMLException = require('./exception');
|
|
|
|
|
|
// TODO: Add tag format check.
|
|
function Type(tag, options) {
|
|
options = options || {};
|
|
|
|
this.tag = tag;
|
|
this.loader = options['loader'] || null;
|
|
this.dumper = options['dumper'] || null;
|
|
|
|
if (null === this.loader && null === this.dumper) {
|
|
throw new YAMLException('Incomplete YAML type definition. "loader" or "dumper" setting must be specified.');
|
|
}
|
|
|
|
if (null !== this.loader) {
|
|
this.loader = new Type.Loader(this.loader);
|
|
}
|
|
|
|
if (null !== this.dumper) {
|
|
this.dumper = new Type.Dumper(this.dumper);
|
|
}
|
|
}
|
|
|
|
|
|
Type.Loader = function TypeLoader(options) {
|
|
options = options || {};
|
|
|
|
this.kind = options['kind'] || null;
|
|
this.resolver = options['resolver'] || null;
|
|
|
|
if ('string' !== this.kind &&
|
|
'array' !== this.kind &&
|
|
'object' !== this.kind) {
|
|
throw new YAMLException('Unacceptable "kind" setting of a type loader.');
|
|
}
|
|
};
|
|
|
|
|
|
function compileAliases(map) {
|
|
var result = {};
|
|
|
|
if (null !== map) {
|
|
Object.keys(map).forEach(function (style) {
|
|
map[style].forEach(function (alias) {
|
|
result[String(alias)] = style;
|
|
});
|
|
});
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
|
|
Type.Dumper = function TypeDumper(options) {
|
|
options = options || {};
|
|
|
|
this.kind = options['kind'] || null;
|
|
this.defaultStyle = options['defaultStyle'] || null;
|
|
this.instanceOf = options['instanceOf'] || null;
|
|
this.predicate = options['predicate'] || null;
|
|
this.representer = options['representer'] || null;
|
|
this.styleAliases = compileAliases(options['styleAliases'] || null);
|
|
|
|
if ('undefined' !== this.kind &&
|
|
'null' !== this.kind &&
|
|
'boolean' !== this.kind &&
|
|
'integer' !== this.kind &&
|
|
'float' !== this.kind &&
|
|
'string' !== this.kind &&
|
|
'array' !== this.kind &&
|
|
'object' !== this.kind &&
|
|
'function' !== this.kind) {
|
|
throw new YAMLException('Unacceptable "kind" setting of a type dumper.');
|
|
}
|
|
};
|
|
|
|
|
|
module.exports = Type;
|
|
|
|
},{"./exception":22}],33:[function(require,module,exports){
|
|
// Modified from:
|
|
// https://raw.github.com/kanaka/noVNC/d890e8640f20fba3215ba7be8e0ff145aeb8c17c/include/base64.js
|
|
|
|
'use strict';
|
|
|
|
|
|
var NodeBuffer = require('buffer').Buffer; // A trick for browserified version.
|
|
var common = require('../common');
|
|
var NIL = common.NIL;
|
|
var Type = require('../type');
|
|
|
|
|
|
|
|
var BASE64_PADDING = '=';
|
|
|
|
var BASE64_BINTABLE = [
|
|
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
|
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
|
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, 62, -1, -1, -1, 63,
|
|
52, 53, 54, 55, 56, 57, 58, 59, 60, 61, -1, -1, -1, 0, -1, -1,
|
|
-1, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14,
|
|
15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, -1, -1, -1, -1, -1,
|
|
-1, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35, 36, 37, 38, 39, 40,
|
|
41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 51, -1, -1, -1, -1, -1
|
|
];
|
|
|
|
var BASE64_CHARTABLE =
|
|
'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/'.split('');
|
|
|
|
|
|
function resolveYamlBinary(object /*, explicit*/) {
|
|
var value, code, idx = 0, result = [], leftbits, leftdata;
|
|
|
|
leftbits = 0; // number of bits decoded, but yet to be appended
|
|
leftdata = 0; // bits decoded, but yet to be appended
|
|
|
|
// Convert one by one.
|
|
for (idx = 0; idx < object.length; idx += 1) {
|
|
code = object.charCodeAt(idx);
|
|
value = BASE64_BINTABLE[code & 0x7F];
|
|
|
|
// Skip LF(NL) || CR
|
|
if (0x0A !== code && 0x0D !== code) {
|
|
// Fail on illegal characters
|
|
if (-1 === value) {
|
|
return NIL;
|
|
}
|
|
|
|
// Collect data into leftdata, update bitcount
|
|
leftdata = (leftdata << 6) | value;
|
|
leftbits += 6;
|
|
|
|
// If we have 8 or more bits, append 8 bits to the result
|
|
if (leftbits >= 8) {
|
|
leftbits -= 8;
|
|
|
|
// Append if not padding.
|
|
if (BASE64_PADDING !== object.charAt(idx)) {
|
|
result.push((leftdata >> leftbits) & 0xFF);
|
|
}
|
|
|
|
leftdata &= (1 << leftbits) - 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
// If there are any bits left, the base64 string was corrupted
|
|
if (leftbits) {
|
|
return NIL;
|
|
} else {
|
|
return new NodeBuffer(result);
|
|
}
|
|
}
|
|
|
|
|
|
function representYamlBinary(object /*, style*/) {
|
|
var result = '', index, length, rest;
|
|
|
|
// Convert every three bytes to 4 ASCII characters.
|
|
for (index = 0, length = object.length - 2; index < length; index += 3) {
|
|
result += BASE64_CHARTABLE[object[index + 0] >> 2];
|
|
result += BASE64_CHARTABLE[((object[index + 0] & 0x03) << 4) + (object[index + 1] >> 4)];
|
|
result += BASE64_CHARTABLE[((object[index + 1] & 0x0F) << 2) + (object[index + 2] >> 6)];
|
|
result += BASE64_CHARTABLE[object[index + 2] & 0x3F];
|
|
}
|
|
|
|
rest = object.length % 3;
|
|
|
|
// Convert the remaining 1 or 2 bytes, padding out to 4 characters.
|
|
if (0 !== rest) {
|
|
index = object.length - rest;
|
|
result += BASE64_CHARTABLE[object[index + 0] >> 2];
|
|
|
|
if (2 === rest) {
|
|
result += BASE64_CHARTABLE[((object[index + 0] & 0x03) << 4) + (object[index + 1] >> 4)];
|
|
result += BASE64_CHARTABLE[(object[index + 1] & 0x0F) << 2];
|
|
result += BASE64_PADDING;
|
|
} else {
|
|
result += BASE64_CHARTABLE[(object[index + 0] & 0x03) << 4];
|
|
result += BASE64_PADDING + BASE64_PADDING;
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
|
|
module.exports = new Type('tag:yaml.org,2002:binary', {
|
|
loader: {
|
|
kind: 'string',
|
|
resolver: resolveYamlBinary
|
|
},
|
|
dumper: {
|
|
kind: 'object',
|
|
instanceOf: NodeBuffer,
|
|
representer: representYamlBinary
|
|
}
|
|
});
|
|
|
|
},{"../common":20,"../type":32,"buffer":12}],34:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
|
|
var NIL = require('../common').NIL;
|
|
var Type = require('../type');
|
|
|
|
|
|
var YAML_IMPLICIT_BOOLEAN_MAP = {
|
|
'true' : true,
|
|
'True' : true,
|
|
'TRUE' : true,
|
|
'false' : false,
|
|
'False' : false,
|
|
'FALSE' : false
|
|
};
|
|
|
|
var YAML_EXPLICIT_BOOLEAN_MAP = {
|
|
'true' : true,
|
|
'True' : true,
|
|
'TRUE' : true,
|
|
'false' : false,
|
|
'False' : false,
|
|
'FALSE' : false,
|
|
'y' : true,
|
|
'Y' : true,
|
|
'yes' : true,
|
|
'Yes' : true,
|
|
'YES' : true,
|
|
'n' : false,
|
|
'N' : false,
|
|
'no' : false,
|
|
'No' : false,
|
|
'NO' : false,
|
|
'on' : true,
|
|
'On' : true,
|
|
'ON' : true,
|
|
'off' : false,
|
|
'Off' : false,
|
|
'OFF' : false
|
|
};
|
|
|
|
|
|
function resolveYamlBoolean(object, explicit) {
|
|
if (explicit) {
|
|
if (YAML_EXPLICIT_BOOLEAN_MAP.hasOwnProperty(object)) {
|
|
return YAML_EXPLICIT_BOOLEAN_MAP[object];
|
|
} else {
|
|
return NIL;
|
|
}
|
|
} else {
|
|
if (YAML_IMPLICIT_BOOLEAN_MAP.hasOwnProperty(object)) {
|
|
return YAML_IMPLICIT_BOOLEAN_MAP[object];
|
|
} else {
|
|
return NIL;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
module.exports = new Type('tag:yaml.org,2002:bool', {
|
|
loader: {
|
|
kind: 'string',
|
|
resolver: resolveYamlBoolean
|
|
},
|
|
dumper: {
|
|
kind: 'boolean',
|
|
defaultStyle: 'lowercase',
|
|
representer: {
|
|
lowercase: function (object) { return object ? 'true' : 'false'; },
|
|
uppercase: function (object) { return object ? 'TRUE' : 'FALSE'; },
|
|
camelcase: function (object) { return object ? 'True' : 'False'; }
|
|
}
|
|
}
|
|
});
|
|
|
|
},{"../common":20,"../type":32}],35:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
|
|
var NIL = require('../common').NIL;
|
|
var Type = require('../type');
|
|
|
|
|
|
var YAML_FLOAT_PATTERN = new RegExp(
|
|
'^(?:[-+]?(?:[0-9][0-9_]*)\\.[0-9_]*(?:[eE][-+][0-9]+)?' +
|
|
'|\\.[0-9_]+(?:[eE][-+][0-9]+)?' +
|
|
'|[-+]?[0-9][0-9_]*(?::[0-5]?[0-9])+\\.[0-9_]*' +
|
|
'|[-+]?\\.(?:inf|Inf|INF)' +
|
|
'|\\.(?:nan|NaN|NAN))$');
|
|
|
|
|
|
function resolveYamlFloat(object /*, explicit*/) {
|
|
var value, sign, base, digits;
|
|
|
|
if (!YAML_FLOAT_PATTERN.test(object)) {
|
|
return NIL;
|
|
}
|
|
|
|
value = object.replace(/_/g, '').toLowerCase();
|
|
sign = '-' === value[0] ? -1 : 1;
|
|
digits = [];
|
|
|
|
if (0 <= '+-'.indexOf(value[0])) {
|
|
value = value.slice(1);
|
|
}
|
|
|
|
if ('.inf' === value) {
|
|
return (1 === sign) ? Number.POSITIVE_INFINITY : Number.NEGATIVE_INFINITY;
|
|
|
|
} else if ('.nan' === value) {
|
|
return NaN;
|
|
|
|
} else if (0 <= value.indexOf(':')) {
|
|
value.split(':').forEach(function (v) {
|
|
digits.unshift(parseFloat(v, 10));
|
|
});
|
|
|
|
value = 0.0;
|
|
base = 1;
|
|
|
|
digits.forEach(function (d) {
|
|
value += d * base;
|
|
base *= 60;
|
|
});
|
|
|
|
return sign * value;
|
|
|
|
} else {
|
|
return sign * parseFloat(value, 10);
|
|
}
|
|
}
|
|
|
|
|
|
function representYamlFloat(object, style) {
|
|
if (isNaN(object)) {
|
|
switch (style) {
|
|
case 'lowercase':
|
|
return '.nan';
|
|
case 'uppercase':
|
|
return '.NAN';
|
|
case 'camelcase':
|
|
return '.NaN';
|
|
}
|
|
} else if (Number.POSITIVE_INFINITY === object) {
|
|
switch (style) {
|
|
case 'lowercase':
|
|
return '.inf';
|
|
case 'uppercase':
|
|
return '.INF';
|
|
case 'camelcase':
|
|
return '.Inf';
|
|
}
|
|
} else if (Number.NEGATIVE_INFINITY === object) {
|
|
switch (style) {
|
|
case 'lowercase':
|
|
return '-.inf';
|
|
case 'uppercase':
|
|
return '-.INF';
|
|
case 'camelcase':
|
|
return '-.Inf';
|
|
}
|
|
} else {
|
|
return object.toString(10);
|
|
}
|
|
}
|
|
|
|
|
|
module.exports = new Type('tag:yaml.org,2002:float', {
|
|
loader: {
|
|
kind: 'string',
|
|
resolver: resolveYamlFloat
|
|
},
|
|
dumper: {
|
|
kind: 'float',
|
|
defaultStyle: 'lowercase',
|
|
representer: representYamlFloat
|
|
}
|
|
});
|
|
|
|
},{"../common":20,"../type":32}],36:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
|
|
var NIL = require('../common').NIL;
|
|
var Type = require('../type');
|
|
|
|
|
|
var YAML_INTEGER_PATTERN = new RegExp(
|
|
'^(?:[-+]?0b[0-1_]+' +
|
|
'|[-+]?0[0-7_]+' +
|
|
'|[-+]?(?:0|[1-9][0-9_]*)' +
|
|
'|[-+]?0x[0-9a-fA-F_]+' +
|
|
'|[-+]?[1-9][0-9_]*(?::[0-5]?[0-9])+)$');
|
|
|
|
|
|
function resolveYamlInteger(object /*, explicit*/) {
|
|
var value, sign, base, digits;
|
|
|
|
if (!YAML_INTEGER_PATTERN.test(object)) {
|
|
return NIL;
|
|
}
|
|
|
|
value = object.replace(/_/g, '');
|
|
sign = '-' === value[0] ? -1 : 1;
|
|
digits = [];
|
|
|
|
if (0 <= '+-'.indexOf(value[0])) {
|
|
value = value.slice(1);
|
|
}
|
|
|
|
if ('0' === value) {
|
|
return 0;
|
|
|
|
} else if (/^0b/.test(value)) {
|
|
return sign * parseInt(value.slice(2), 2);
|
|
|
|
} else if (/^0x/.test(value)) {
|
|
return sign * parseInt(value, 16);
|
|
|
|
} else if ('0' === value[0]) {
|
|
return sign * parseInt(value, 8);
|
|
|
|
} else if (0 <= value.indexOf(':')) {
|
|
value.split(':').forEach(function (v) {
|
|
digits.unshift(parseInt(v, 10));
|
|
});
|
|
|
|
value = 0;
|
|
base = 1;
|
|
|
|
digits.forEach(function (d) {
|
|
value += (d * base);
|
|
base *= 60;
|
|
});
|
|
|
|
return sign * value;
|
|
|
|
} else {
|
|
return sign * parseInt(value, 10);
|
|
}
|
|
}
|
|
|
|
|
|
module.exports = new Type('tag:yaml.org,2002:int', {
|
|
loader: {
|
|
kind: 'string',
|
|
resolver: resolveYamlInteger
|
|
},
|
|
dumper: {
|
|
kind: 'integer',
|
|
defaultStyle: 'decimal',
|
|
representer: {
|
|
binary: function (object) { return '0b' + object.toString(2); },
|
|
octal: function (object) { return '0' + object.toString(8); },
|
|
decimal: function (object) { return object.toString(10); },
|
|
hexadecimal: function (object) { return '0x' + object.toString(16).toUpperCase(); }
|
|
},
|
|
styleAliases: {
|
|
binary: [ 2, 'bin' ],
|
|
octal: [ 8, 'oct' ],
|
|
decimal: [ 10, 'dec' ],
|
|
hexadecimal: [ 16, 'hex' ]
|
|
}
|
|
}
|
|
});
|
|
|
|
},{"../common":20,"../type":32}],37:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
|
|
var esprima = require('esprima');
|
|
|
|
|
|
var NIL = require('../../common').NIL;
|
|
var Type = require('../../type');
|
|
|
|
|
|
function resolveJavascriptFunction(object /*, explicit*/) {
|
|
/*jslint evil:true*/
|
|
|
|
try {
|
|
var source = '(' + object + ')',
|
|
ast = esprima.parse(source, { range: true }),
|
|
params = [],
|
|
body;
|
|
|
|
if ('Program' !== ast.type ||
|
|
1 !== ast.body.length ||
|
|
'ExpressionStatement' !== ast.body[0].type ||
|
|
'FunctionExpression' !== ast.body[0].expression.type) {
|
|
return NIL;
|
|
}
|
|
|
|
ast.body[0].expression.params.forEach(function (param) {
|
|
params.push(param.name);
|
|
});
|
|
|
|
body = ast.body[0].expression.body.range;
|
|
|
|
// Esprima's ranges include the first '{' and the last '}' characters on
|
|
// function expressions. So cut them out.
|
|
return new Function(params, source.slice(body[0]+1, body[1]-1));
|
|
} catch (err) {
|
|
return NIL;
|
|
}
|
|
}
|
|
|
|
|
|
function representJavascriptFunction(object /*, style*/) {
|
|
return object.toString();
|
|
}
|
|
|
|
|
|
module.exports = new Type('tag:yaml.org,2002:js/function', {
|
|
loader: {
|
|
kind: 'string',
|
|
resolver: resolveJavascriptFunction
|
|
},
|
|
dumper: {
|
|
kind: 'function',
|
|
representer: representJavascriptFunction
|
|
}
|
|
});
|
|
|
|
},{"../../common":20,"../../type":32,"esprima":49}],38:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
|
|
var NIL = require('../../common').NIL;
|
|
var Type = require('../../type');
|
|
|
|
|
|
function resolveJavascriptRegExp(object /*, explicit*/) {
|
|
var regexp = object,
|
|
tail = /\/([gim]*)$/.exec(object),
|
|
modifiers;
|
|
|
|
// `/foo/gim` - tail can be maximum 4 chars
|
|
if ('/' === regexp[0] && tail && 4 >= tail[0].length) {
|
|
regexp = regexp.slice(1, regexp.length - tail[0].length);
|
|
modifiers = tail[1];
|
|
}
|
|
|
|
try {
|
|
return new RegExp(regexp, modifiers);
|
|
} catch (error) {
|
|
return NIL;
|
|
}
|
|
}
|
|
|
|
|
|
function representJavascriptRegExp(object /*, style*/) {
|
|
var result = '/' + object.source + '/';
|
|
|
|
if (object.global) {
|
|
result += 'g';
|
|
}
|
|
|
|
if (object.multiline) {
|
|
result += 'm';
|
|
}
|
|
|
|
if (object.ignoreCase) {
|
|
result += 'i';
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
|
|
module.exports = new Type('tag:yaml.org,2002:js/regexp', {
|
|
loader: {
|
|
kind: 'string',
|
|
resolver: resolveJavascriptRegExp
|
|
},
|
|
dumper: {
|
|
kind: 'object',
|
|
instanceOf: RegExp,
|
|
representer: representJavascriptRegExp
|
|
}
|
|
});
|
|
|
|
},{"../../common":20,"../../type":32}],39:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
|
|
var Type = require('../../type');
|
|
|
|
|
|
function resolveJavascriptUndefined(/*object, explicit*/) {
|
|
var undef;
|
|
|
|
return undef;
|
|
}
|
|
|
|
|
|
function representJavascriptUndefined(/*object, explicit*/) {
|
|
return '';
|
|
}
|
|
|
|
|
|
module.exports = new Type('tag:yaml.org,2002:js/undefined', {
|
|
loader: {
|
|
kind: 'string',
|
|
resolver: resolveJavascriptUndefined
|
|
},
|
|
dumper: {
|
|
kind: 'undefined',
|
|
representer: representJavascriptUndefined
|
|
}
|
|
});
|
|
|
|
},{"../../type":32}],40:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
|
|
var Type = require('../type');
|
|
|
|
|
|
module.exports = new Type('tag:yaml.org,2002:map', {
|
|
loader: {
|
|
kind: 'object'
|
|
}
|
|
});
|
|
|
|
},{"../type":32}],41:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
|
|
var NIL = require('../common').NIL;
|
|
var Type = require('../type');
|
|
|
|
|
|
function resolveYamlMerge(object /*, explicit*/) {
|
|
return '<<' === object ? object : NIL;
|
|
}
|
|
|
|
|
|
module.exports = new Type('tag:yaml.org,2002:merge', {
|
|
loader: {
|
|
kind: 'string',
|
|
resolver: resolveYamlMerge
|
|
}
|
|
});
|
|
|
|
},{"../common":20,"../type":32}],42:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
|
|
var NIL = require('../common').NIL;
|
|
var Type = require('../type');
|
|
|
|
|
|
var YAML_NULL_MAP = {
|
|
'~' : true,
|
|
'null' : true,
|
|
'Null' : true,
|
|
'NULL' : true
|
|
};
|
|
|
|
|
|
function resolveYamlNull(object /*, explicit*/) {
|
|
return YAML_NULL_MAP[object] ? null : NIL;
|
|
}
|
|
|
|
|
|
module.exports = new Type('tag:yaml.org,2002:null', {
|
|
loader: {
|
|
kind: 'string',
|
|
resolver: resolveYamlNull
|
|
},
|
|
dumper: {
|
|
kind: 'null',
|
|
defaultStyle: 'lowercase',
|
|
representer: {
|
|
canonical: function () { return '~'; },
|
|
lowercase: function () { return 'null'; },
|
|
uppercase: function () { return 'NULL'; },
|
|
camelcase: function () { return 'Null'; }
|
|
}
|
|
}
|
|
});
|
|
|
|
},{"../common":20,"../type":32}],43:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
|
|
var NIL = require('../common').NIL;
|
|
var Type = require('../type');
|
|
|
|
|
|
var _hasOwnProperty = Object.prototype.hasOwnProperty;
|
|
var _toString = Object.prototype.toString;
|
|
|
|
|
|
function resolveYamlOmap(object /*, explicit*/) {
|
|
var objectKeys = [], index, length, pair, pairKey, pairHasKey;
|
|
|
|
for (index = 0, length = object.length; index < length; index += 1) {
|
|
pair = object[index];
|
|
pairHasKey = false;
|
|
|
|
if ('[object Object]' !== _toString.call(pair)) {
|
|
return NIL;
|
|
}
|
|
|
|
for (pairKey in pair) {
|
|
if (_hasOwnProperty.call(pair, pairKey)) {
|
|
if (!pairHasKey) {
|
|
pairHasKey = true;
|
|
} else {
|
|
return NIL;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!pairHasKey) {
|
|
return NIL;
|
|
}
|
|
|
|
if (-1 === objectKeys.indexOf(pairKey)) {
|
|
objectKeys.push(pairKey);
|
|
} else {
|
|
return NIL;
|
|
}
|
|
}
|
|
|
|
return object;
|
|
}
|
|
|
|
|
|
module.exports = new Type('tag:yaml.org,2002:omap', {
|
|
loader: {
|
|
kind: 'array',
|
|
resolver: resolveYamlOmap
|
|
}
|
|
});
|
|
|
|
},{"../common":20,"../type":32}],44:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
|
|
var NIL = require('../common').NIL;
|
|
var Type = require('../type');
|
|
|
|
|
|
var _toString = Object.prototype.toString;
|
|
|
|
|
|
function resolveYamlPairs(object /*, explicit*/) {
|
|
var index, length, pair, keys, result;
|
|
|
|
result = new Array(object.length);
|
|
|
|
for (index = 0, length = object.length; index < length; index += 1) {
|
|
pair = object[index];
|
|
|
|
if ('[object Object]' !== _toString.call(pair)) {
|
|
return NIL;
|
|
}
|
|
|
|
keys = Object.keys(pair);
|
|
|
|
if (1 !== keys.length) {
|
|
return NIL;
|
|
}
|
|
|
|
result[index] = [ keys[0], pair[keys[0]] ];
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
|
|
module.exports = new Type('tag:yaml.org,2002:pairs', {
|
|
loader: {
|
|
kind: 'array',
|
|
resolver: resolveYamlPairs
|
|
}
|
|
});
|
|
|
|
},{"../common":20,"../type":32}],45:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
|
|
var Type = require('../type');
|
|
|
|
|
|
module.exports = new Type('tag:yaml.org,2002:seq', {
|
|
loader: {
|
|
kind: 'array'
|
|
}
|
|
});
|
|
|
|
},{"../type":32}],46:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
|
|
var NIL = require('../common').NIL;
|
|
var Type = require('../type');
|
|
|
|
|
|
var _hasOwnProperty = Object.prototype.hasOwnProperty;
|
|
|
|
|
|
function resolveYamlSet(object /*, explicit*/) {
|
|
var key;
|
|
|
|
for (key in object) {
|
|
if (_hasOwnProperty.call(object, key)) {
|
|
if (null !== object[key]) {
|
|
return NIL;
|
|
}
|
|
}
|
|
}
|
|
|
|
return object;
|
|
}
|
|
|
|
|
|
module.exports = new Type('tag:yaml.org,2002:set', {
|
|
loader: {
|
|
kind: 'object',
|
|
resolver: resolveYamlSet
|
|
}
|
|
});
|
|
|
|
},{"../common":20,"../type":32}],47:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
|
|
var Type = require('../type');
|
|
|
|
|
|
module.exports = new Type('tag:yaml.org,2002:str', {
|
|
loader: {
|
|
kind: 'string'
|
|
}
|
|
});
|
|
|
|
},{"../type":32}],48:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
|
|
var NIL = require('../common').NIL;
|
|
var Type = require('../type');
|
|
|
|
|
|
var YAML_TIMESTAMP_REGEXP = new RegExp(
|
|
'^([0-9][0-9][0-9][0-9])' + // [1] year
|
|
'-([0-9][0-9]?)' + // [2] month
|
|
'-([0-9][0-9]?)' + // [3] day
|
|
'(?:(?:[Tt]|[ \\t]+)' + // ...
|
|
'([0-9][0-9]?)' + // [4] hour
|
|
':([0-9][0-9])' + // [5] minute
|
|
':([0-9][0-9])' + // [6] second
|
|
'(?:\\.([0-9]*))?' + // [7] fraction
|
|
'(?:[ \\t]*(Z|([-+])([0-9][0-9]?)' + // [8] tz [9] tz_sign [10] tz_hour
|
|
'(?::([0-9][0-9]))?))?)?$'); // [11] tz_minute
|
|
|
|
|
|
function resolveYamlTimestamp(object /*, explicit*/) {
|
|
var match, year, month, day, hour, minute, second, fraction = 0,
|
|
delta = null, tz_hour, tz_minute, data;
|
|
|
|
match = YAML_TIMESTAMP_REGEXP.exec(object);
|
|
|
|
if (null === match) {
|
|
return NIL;
|
|
}
|
|
|
|
// match: [1] year [2] month [3] day
|
|
|
|
year = +(match[1]);
|
|
month = +(match[2]) - 1; // JS month starts with 0
|
|
day = +(match[3]);
|
|
|
|
if (!match[4]) { // no hour
|
|
return new Date(Date.UTC(year, month, day));
|
|
}
|
|
|
|
// match: [4] hour [5] minute [6] second [7] fraction
|
|
|
|
hour = +(match[4]);
|
|
minute = +(match[5]);
|
|
second = +(match[6]);
|
|
|
|
if (match[7]) {
|
|
fraction = match[7].slice(0, 3);
|
|
while (fraction.length < 3) { // milli-seconds
|
|
fraction += '0';
|
|
}
|
|
fraction = +fraction;
|
|
}
|
|
|
|
// match: [8] tz [9] tz_sign [10] tz_hour [11] tz_minute
|
|
|
|
if (match[9]) {
|
|
tz_hour = +(match[10]);
|
|
tz_minute = +(match[11] || 0);
|
|
delta = (tz_hour * 60 + tz_minute) * 60000; // delta in mili-seconds
|
|
if ('-' === match[9]) {
|
|
delta = -delta;
|
|
}
|
|
}
|
|
|
|
data = new Date(Date.UTC(year, month, day, hour, minute, second, fraction));
|
|
|
|
if (delta) {
|
|
data.setTime(data.getTime() - delta);
|
|
}
|
|
|
|
return data;
|
|
}
|
|
|
|
|
|
function representYamlTimestamp(object /*, style*/) {
|
|
return object.toISOString();
|
|
}
|
|
|
|
|
|
module.exports = new Type('tag:yaml.org,2002:timestamp', {
|
|
loader: {
|
|
kind: 'string',
|
|
resolver: resolveYamlTimestamp
|
|
},
|
|
dumper: {
|
|
kind: 'object',
|
|
instanceOf: Date,
|
|
representer: representYamlTimestamp
|
|
}
|
|
});
|
|
|
|
},{"../common":20,"../type":32}],49:[function(require,module,exports){
|
|
/*
|
|
Copyright (C) 2012 Ariya Hidayat <ariya.hidayat@gmail.com>
|
|
Copyright (C) 2012 Mathias Bynens <mathias@qiwi.be>
|
|
Copyright (C) 2012 Joost-Wim Boekesteijn <joost-wim@boekesteijn.nl>
|
|
Copyright (C) 2012 Kris Kowal <kris.kowal@cixar.com>
|
|
Copyright (C) 2012 Yusuke Suzuki <utatane.tea@gmail.com>
|
|
Copyright (C) 2012 Arpad Borsos <arpad.borsos@googlemail.com>
|
|
Copyright (C) 2011 Ariya Hidayat <ariya.hidayat@gmail.com>
|
|
|
|
Redistribution and use in source and binary forms, with or without
|
|
modification, are permitted provided that the following conditions are met:
|
|
|
|
* Redistributions of source code must retain the above copyright
|
|
notice, this list of conditions and the following disclaimer.
|
|
* Redistributions in binary form must reproduce the above copyright
|
|
notice, this list of conditions and the following disclaimer in the
|
|
documentation and/or other materials provided with the distribution.
|
|
|
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
|
AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
ARE DISCLAIMED. IN NO EVENT SHALL <COPYRIGHT HOLDER> BE LIABLE FOR ANY
|
|
DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
|
ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
|
|
THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
*/
|
|
|
|
/*jslint bitwise:true plusplus:true */
|
|
/*global esprima:true, define:true, exports:true, window: true,
|
|
throwError: true, createLiteral: true, generateStatement: true,
|
|
parseAssignmentExpression: true, parseBlock: true, parseExpression: true,
|
|
parseFunctionDeclaration: true, parseFunctionExpression: true,
|
|
parseFunctionSourceElements: true, parseVariableIdentifier: true,
|
|
parseLeftHandSideExpression: true,
|
|
parseStatement: true, parseSourceElement: true */
|
|
|
|
(function (root, factory) {
|
|
'use strict';
|
|
|
|
// Universal Module Definition (UMD) to support AMD, CommonJS/Node.js,
|
|
// Rhino, and plain browser loading.
|
|
if (typeof define === 'function' && define.amd) {
|
|
define(['exports'], factory);
|
|
} else if (typeof exports !== 'undefined') {
|
|
factory(exports);
|
|
} else {
|
|
factory((root.esprima = {}));
|
|
}
|
|
}(this, function (exports) {
|
|
'use strict';
|
|
|
|
var Token,
|
|
TokenName,
|
|
Syntax,
|
|
PropertyKind,
|
|
Messages,
|
|
Regex,
|
|
source,
|
|
strict,
|
|
index,
|
|
lineNumber,
|
|
lineStart,
|
|
length,
|
|
buffer,
|
|
state,
|
|
extra;
|
|
|
|
Token = {
|
|
BooleanLiteral: 1,
|
|
EOF: 2,
|
|
Identifier: 3,
|
|
Keyword: 4,
|
|
NullLiteral: 5,
|
|
NumericLiteral: 6,
|
|
Punctuator: 7,
|
|
StringLiteral: 8
|
|
};
|
|
|
|
TokenName = {};
|
|
TokenName[Token.BooleanLiteral] = 'Boolean';
|
|
TokenName[Token.EOF] = '<end>';
|
|
TokenName[Token.Identifier] = 'Identifier';
|
|
TokenName[Token.Keyword] = 'Keyword';
|
|
TokenName[Token.NullLiteral] = 'Null';
|
|
TokenName[Token.NumericLiteral] = 'Numeric';
|
|
TokenName[Token.Punctuator] = 'Punctuator';
|
|
TokenName[Token.StringLiteral] = 'String';
|
|
|
|
Syntax = {
|
|
AssignmentExpression: 'AssignmentExpression',
|
|
ArrayExpression: 'ArrayExpression',
|
|
BlockStatement: 'BlockStatement',
|
|
BinaryExpression: 'BinaryExpression',
|
|
BreakStatement: 'BreakStatement',
|
|
CallExpression: 'CallExpression',
|
|
CatchClause: 'CatchClause',
|
|
ConditionalExpression: 'ConditionalExpression',
|
|
ContinueStatement: 'ContinueStatement',
|
|
DoWhileStatement: 'DoWhileStatement',
|
|
DebuggerStatement: 'DebuggerStatement',
|
|
EmptyStatement: 'EmptyStatement',
|
|
ExpressionStatement: 'ExpressionStatement',
|
|
ForStatement: 'ForStatement',
|
|
ForInStatement: 'ForInStatement',
|
|
FunctionDeclaration: 'FunctionDeclaration',
|
|
FunctionExpression: 'FunctionExpression',
|
|
Identifier: 'Identifier',
|
|
IfStatement: 'IfStatement',
|
|
Literal: 'Literal',
|
|
LabeledStatement: 'LabeledStatement',
|
|
LogicalExpression: 'LogicalExpression',
|
|
MemberExpression: 'MemberExpression',
|
|
NewExpression: 'NewExpression',
|
|
ObjectExpression: 'ObjectExpression',
|
|
Program: 'Program',
|
|
Property: 'Property',
|
|
ReturnStatement: 'ReturnStatement',
|
|
SequenceExpression: 'SequenceExpression',
|
|
SwitchStatement: 'SwitchStatement',
|
|
SwitchCase: 'SwitchCase',
|
|
ThisExpression: 'ThisExpression',
|
|
ThrowStatement: 'ThrowStatement',
|
|
TryStatement: 'TryStatement',
|
|
UnaryExpression: 'UnaryExpression',
|
|
UpdateExpression: 'UpdateExpression',
|
|
VariableDeclaration: 'VariableDeclaration',
|
|
VariableDeclarator: 'VariableDeclarator',
|
|
WhileStatement: 'WhileStatement',
|
|
WithStatement: 'WithStatement'
|
|
};
|
|
|
|
PropertyKind = {
|
|
Data: 1,
|
|
Get: 2,
|
|
Set: 4
|
|
};
|
|
|
|
// Error messages should be identical to V8.
|
|
Messages = {
|
|
UnexpectedToken: 'Unexpected token %0',
|
|
UnexpectedNumber: 'Unexpected number',
|
|
UnexpectedString: 'Unexpected string',
|
|
UnexpectedIdentifier: 'Unexpected identifier',
|
|
UnexpectedReserved: 'Unexpected reserved word',
|
|
UnexpectedEOS: 'Unexpected end of input',
|
|
NewlineAfterThrow: 'Illegal newline after throw',
|
|
InvalidRegExp: 'Invalid regular expression',
|
|
UnterminatedRegExp: 'Invalid regular expression: missing /',
|
|
InvalidLHSInAssignment: 'Invalid left-hand side in assignment',
|
|
InvalidLHSInForIn: 'Invalid left-hand side in for-in',
|
|
MultipleDefaultsInSwitch: 'More than one default clause in switch statement',
|
|
NoCatchOrFinally: 'Missing catch or finally after try',
|
|
UnknownLabel: 'Undefined label \'%0\'',
|
|
Redeclaration: '%0 \'%1\' has already been declared',
|
|
IllegalContinue: 'Illegal continue statement',
|
|
IllegalBreak: 'Illegal break statement',
|
|
IllegalReturn: 'Illegal return statement',
|
|
StrictModeWith: 'Strict mode code may not include a with statement',
|
|
StrictCatchVariable: 'Catch variable may not be eval or arguments in strict mode',
|
|
StrictVarName: 'Variable name may not be eval or arguments in strict mode',
|
|
StrictParamName: 'Parameter name eval or arguments is not allowed in strict mode',
|
|
StrictParamDupe: 'Strict mode function may not have duplicate parameter names',
|
|
StrictFunctionName: 'Function name may not be eval or arguments in strict mode',
|
|
StrictOctalLiteral: 'Octal literals are not allowed in strict mode.',
|
|
StrictDelete: 'Delete of an unqualified identifier in strict mode.',
|
|
StrictDuplicateProperty: 'Duplicate data property in object literal not allowed in strict mode',
|
|
AccessorDataProperty: 'Object literal may not have data and accessor property with the same name',
|
|
AccessorGetSet: 'Object literal may not have multiple get/set accessors with the same name',
|
|
StrictLHSAssignment: 'Assignment to eval or arguments is not allowed in strict mode',
|
|
StrictLHSPostfix: 'Postfix increment/decrement may not have eval or arguments operand in strict mode',
|
|
StrictLHSPrefix: 'Prefix increment/decrement may not have eval or arguments operand in strict mode',
|
|
StrictReservedWord: 'Use of future reserved word in strict mode'
|
|
};
|
|
|
|
// See also tools/generate-unicode-regex.py.
|
|
Regex = {
|
|
NonAsciiIdentifierStart: new RegExp('[\xaa\xb5\xba\xc0-\xd6\xd8-\xf6\xf8-\u02c1\u02c6-\u02d1\u02e0-\u02e4\u02ec\u02ee\u0370-\u0374\u0376\u0377\u037a-\u037d\u0386\u0388-\u038a\u038c\u038e-\u03a1\u03a3-\u03f5\u03f7-\u0481\u048a-\u0527\u0531-\u0556\u0559\u0561-\u0587\u05d0-\u05ea\u05f0-\u05f2\u0620-\u064a\u066e\u066f\u0671-\u06d3\u06d5\u06e5\u06e6\u06ee\u06ef\u06fa-\u06fc\u06ff\u0710\u0712-\u072f\u074d-\u07a5\u07b1\u07ca-\u07ea\u07f4\u07f5\u07fa\u0800-\u0815\u081a\u0824\u0828\u0840-\u0858\u08a0\u08a2-\u08ac\u0904-\u0939\u093d\u0950\u0958-\u0961\u0971-\u0977\u0979-\u097f\u0985-\u098c\u098f\u0990\u0993-\u09a8\u09aa-\u09b0\u09b2\u09b6-\u09b9\u09bd\u09ce\u09dc\u09dd\u09df-\u09e1\u09f0\u09f1\u0a05-\u0a0a\u0a0f\u0a10\u0a13-\u0a28\u0a2a-\u0a30\u0a32\u0a33\u0a35\u0a36\u0a38\u0a39\u0a59-\u0a5c\u0a5e\u0a72-\u0a74\u0a85-\u0a8d\u0a8f-\u0a91\u0a93-\u0aa8\u0aaa-\u0ab0\u0ab2\u0ab3\u0ab5-\u0ab9\u0abd\u0ad0\u0ae0\u0ae1\u0b05-\u0b0c\u0b0f\u0b10\u0b13-\u0b28\u0b2a-\u0b30\u0b32\u0b33\u0b35-\u0b39\u0b3d\u0b5c\u0b5d\u0b5f-\u0b61\u0b71\u0b83\u0b85-\u0b8a\u0b8e-\u0b90\u0b92-\u0b95\u0b99\u0b9a\u0b9c\u0b9e\u0b9f\u0ba3\u0ba4\u0ba8-\u0baa\u0bae-\u0bb9\u0bd0\u0c05-\u0c0c\u0c0e-\u0c10\u0c12-\u0c28\u0c2a-\u0c33\u0c35-\u0c39\u0c3d\u0c58\u0c59\u0c60\u0c61\u0c85-\u0c8c\u0c8e-\u0c90\u0c92-\u0ca8\u0caa-\u0cb3\u0cb5-\u0cb9\u0cbd\u0cde\u0ce0\u0ce1\u0cf1\u0cf2\u0d05-\u0d0c\u0d0e-\u0d10\u0d12-\u0d3a\u0d3d\u0d4e\u0d60\u0d61\u0d7a-\u0d7f\u0d85-\u0d96\u0d9a-\u0db1\u0db3-\u0dbb\u0dbd\u0dc0-\u0dc6\u0e01-\u0e30\u0e32\u0e33\u0e40-\u0e46\u0e81\u0e82\u0e84\u0e87\u0e88\u0e8a\u0e8d\u0e94-\u0e97\u0e99-\u0e9f\u0ea1-\u0ea3\u0ea5\u0ea7\u0eaa\u0eab\u0ead-\u0eb0\u0eb2\u0eb3\u0ebd\u0ec0-\u0ec4\u0ec6\u0edc-\u0edf\u0f00\u0f40-\u0f47\u0f49-\u0f6c\u0f88-\u0f8c\u1000-\u102a\u103f\u1050-\u1055\u105a-\u105d\u1061\u1065\u1066\u106e-\u1070\u1075-\u1081\u108e\u10a0-\u10c5\u10c7\u10cd\u10d0-\u10fa\u10fc-\u1248\u124a-\u124d\u1250-\u1256\u1258\u125a-\u125d\u1260-\u1288\u128a-\u128d\u1290-\u12b0\u12b2-\u12b5\u12b8-\u12be\u12c0\u12c2-\u12c5\u12c8-\u12d6\u12d8-\u1310\u1312-\u1315\u1318-\u135a\u1380-\u138f\u13a0-\u13f4\u1401-\u166c\u166f-\u167f\u1681-\u169a\u16a0-\u16ea\u16ee-\u16f0\u1700-\u170c\u170e-\u1711\u1720-\u1731\u1740-\u1751\u1760-\u176c\u176e-\u1770\u1780-\u17b3\u17d7\u17dc\u1820-\u1877\u1880-\u18a8\u18aa\u18b0-\u18f5\u1900-\u191c\u1950-\u196d\u1970-\u1974\u1980-\u19ab\u19c1-\u19c7\u1a00-\u1a16\u1a20-\u1a54\u1aa7\u1b05-\u1b33\u1b45-\u1b4b\u1b83-\u1ba0\u1bae\u1baf\u1bba-\u1be5\u1c00-\u1c23\u1c4d-\u1c4f\u1c5a-\u1c7d\u1ce9-\u1cec\u1cee-\u1cf1\u1cf5\u1cf6\u1d00-\u1dbf\u1e00-\u1f15\u1f18-\u1f1d\u1f20-\u1f45\u1f48-\u1f4d\u1f50-\u1f57\u1f59\u1f5b\u1f5d\u1f5f-\u1f7d\u1f80-\u1fb4\u1fb6-\u1fbc\u1fbe\u1fc2-\u1fc4\u1fc6-\u1fcc\u1fd0-\u1fd3\u1fd6-\u1fdb\u1fe0-\u1fec\u1ff2-\u1ff4\u1ff6-\u1ffc\u2071\u207f\u2090-\u209c\u2102\u2107\u210a-\u2113\u2115\u2119-\u211d\u2124\u2126\u2128\u212a-\u212d\u212f-\u2139\u213c-\u213f\u2145-\u2149\u214e\u2160-\u2188\u2c00-\u2c2e\u2c30-\u2c5e\u2c60-\u2ce4\u2ceb-\u2cee\u2cf2\u2cf3\u2d00-\u2d25\u2d27\u2d2d\u2d30-\u2d67\u2d6f\u2d80-\u2d96\u2da0-\u2da6\u2da8-\u2dae\u2db0-\u2db6\u2db8-\u2dbe\u2dc0-\u2dc6\u2dc8-\u2dce\u2dd0-\u2dd6\u2dd8-\u2dde\u2e2f\u3005-\u3007\u3021-\u3029\u3031-\u3035\u3038-\u303c\u3041-\u3096\u309d-\u309f\u30a1-\u30fa\u30fc-\u30ff\u3105-\u312d\u3131-\u318e\u31a0-\u31ba\u31f0-\u31ff\u3400-\u4db5\u4e00-\u9fcc\ua000-\ua48c\ua4d0-\ua4fd\ua500-\ua60c\ua610-\ua61f\ua62a\ua62b\ua640-\ua66e\ua67f-\ua697\ua6a0-\ua6ef\ua717-\ua71f\ua722-\ua788\ua78b-\ua78e\ua790-\ua793\ua7a0-\ua7aa\ua7f8-\ua801\ua803-\ua805\ua807-\ua80a\ua80c-\ua822\ua840-\ua873\ua882-\ua8b3\ua8f2-\ua8f7\ua8fb\ua90a-\ua925\ua930-\ua946\ua960-\ua97c\ua984-\ua9b2\ua9cf\uaa00-\uaa28\uaa40-\uaa42\uaa44-\uaa4b\uaa60-\uaa76\uaa7a\uaa80-\uaaaf\uaab1\uaab5\uaab6\uaab9-\uaabd\uaac0\uaac2\uaadb-\uaadd\uaae0-\uaaea\uaaf2-\uaaf4\uab01-\uab06\uab09-\uab0e\uab11-\uab16\uab20-\uab26\uab28-\uab2e\uabc0-\uabe2\uac00-\ud7a3\ud7b0-\ud7c6\ud7cb-\ud7fb\uf900-\ufa6d\ufa70-\ufad9\ufb00-\ufb06\ufb13-\ufb17\ufb1d\ufb1f-\ufb28\ufb2a-\ufb36\ufb38-\ufb3c\ufb3e\ufb40\ufb41\ufb43\ufb44\ufb46-\ufbb1\ufbd3-\ufd3d\ufd50-\ufd8f\ufd92-\ufdc7\ufdf0-\ufdfb\ufe70-\ufe74\ufe76-\ufefc\uff21-\uff3a\uff41-\uff5a\uff66-\uffbe\uffc2-\uffc7\uffca-\uffcf\uffd2-\uffd7\uffda-\uffdc]'),
|
|
NonAsciiIdentifierPart: new RegExp('[\xaa\xb5\xba\xc0-\xd6\xd8-\xf6\xf8-\u02c1\u02c6-\u02d1\u02e0-\u02e4\u02ec\u02ee\u0300-\u0374\u0376\u0377\u037a-\u037d\u0386\u0388-\u038a\u038c\u038e-\u03a1\u03a3-\u03f5\u03f7-\u0481\u0483-\u0487\u048a-\u0527\u0531-\u0556\u0559\u0561-\u0587\u0591-\u05bd\u05bf\u05c1\u05c2\u05c4\u05c5\u05c7\u05d0-\u05ea\u05f0-\u05f2\u0610-\u061a\u0620-\u0669\u066e-\u06d3\u06d5-\u06dc\u06df-\u06e8\u06ea-\u06fc\u06ff\u0710-\u074a\u074d-\u07b1\u07c0-\u07f5\u07fa\u0800-\u082d\u0840-\u085b\u08a0\u08a2-\u08ac\u08e4-\u08fe\u0900-\u0963\u0966-\u096f\u0971-\u0977\u0979-\u097f\u0981-\u0983\u0985-\u098c\u098f\u0990\u0993-\u09a8\u09aa-\u09b0\u09b2\u09b6-\u09b9\u09bc-\u09c4\u09c7\u09c8\u09cb-\u09ce\u09d7\u09dc\u09dd\u09df-\u09e3\u09e6-\u09f1\u0a01-\u0a03\u0a05-\u0a0a\u0a0f\u0a10\u0a13-\u0a28\u0a2a-\u0a30\u0a32\u0a33\u0a35\u0a36\u0a38\u0a39\u0a3c\u0a3e-\u0a42\u0a47\u0a48\u0a4b-\u0a4d\u0a51\u0a59-\u0a5c\u0a5e\u0a66-\u0a75\u0a81-\u0a83\u0a85-\u0a8d\u0a8f-\u0a91\u0a93-\u0aa8\u0aaa-\u0ab0\u0ab2\u0ab3\u0ab5-\u0ab9\u0abc-\u0ac5\u0ac7-\u0ac9\u0acb-\u0acd\u0ad0\u0ae0-\u0ae3\u0ae6-\u0aef\u0b01-\u0b03\u0b05-\u0b0c\u0b0f\u0b10\u0b13-\u0b28\u0b2a-\u0b30\u0b32\u0b33\u0b35-\u0b39\u0b3c-\u0b44\u0b47\u0b48\u0b4b-\u0b4d\u0b56\u0b57\u0b5c\u0b5d\u0b5f-\u0b63\u0b66-\u0b6f\u0b71\u0b82\u0b83\u0b85-\u0b8a\u0b8e-\u0b90\u0b92-\u0b95\u0b99\u0b9a\u0b9c\u0b9e\u0b9f\u0ba3\u0ba4\u0ba8-\u0baa\u0bae-\u0bb9\u0bbe-\u0bc2\u0bc6-\u0bc8\u0bca-\u0bcd\u0bd0\u0bd7\u0be6-\u0bef\u0c01-\u0c03\u0c05-\u0c0c\u0c0e-\u0c10\u0c12-\u0c28\u0c2a-\u0c33\u0c35-\u0c39\u0c3d-\u0c44\u0c46-\u0c48\u0c4a-\u0c4d\u0c55\u0c56\u0c58\u0c59\u0c60-\u0c63\u0c66-\u0c6f\u0c82\u0c83\u0c85-\u0c8c\u0c8e-\u0c90\u0c92-\u0ca8\u0caa-\u0cb3\u0cb5-\u0cb9\u0cbc-\u0cc4\u0cc6-\u0cc8\u0cca-\u0ccd\u0cd5\u0cd6\u0cde\u0ce0-\u0ce3\u0ce6-\u0cef\u0cf1\u0cf2\u0d02\u0d03\u0d05-\u0d0c\u0d0e-\u0d10\u0d12-\u0d3a\u0d3d-\u0d44\u0d46-\u0d48\u0d4a-\u0d4e\u0d57\u0d60-\u0d63\u0d66-\u0d6f\u0d7a-\u0d7f\u0d82\u0d83\u0d85-\u0d96\u0d9a-\u0db1\u0db3-\u0dbb\u0dbd\u0dc0-\u0dc6\u0dca\u0dcf-\u0dd4\u0dd6\u0dd8-\u0ddf\u0df2\u0df3\u0e01-\u0e3a\u0e40-\u0e4e\u0e50-\u0e59\u0e81\u0e82\u0e84\u0e87\u0e88\u0e8a\u0e8d\u0e94-\u0e97\u0e99-\u0e9f\u0ea1-\u0ea3\u0ea5\u0ea7\u0eaa\u0eab\u0ead-\u0eb9\u0ebb-\u0ebd\u0ec0-\u0ec4\u0ec6\u0ec8-\u0ecd\u0ed0-\u0ed9\u0edc-\u0edf\u0f00\u0f18\u0f19\u0f20-\u0f29\u0f35\u0f37\u0f39\u0f3e-\u0f47\u0f49-\u0f6c\u0f71-\u0f84\u0f86-\u0f97\u0f99-\u0fbc\u0fc6\u1000-\u1049\u1050-\u109d\u10a0-\u10c5\u10c7\u10cd\u10d0-\u10fa\u10fc-\u1248\u124a-\u124d\u1250-\u1256\u1258\u125a-\u125d\u1260-\u1288\u128a-\u128d\u1290-\u12b0\u12b2-\u12b5\u12b8-\u12be\u12c0\u12c2-\u12c5\u12c8-\u12d6\u12d8-\u1310\u1312-\u1315\u1318-\u135a\u135d-\u135f\u1380-\u138f\u13a0-\u13f4\u1401-\u166c\u166f-\u167f\u1681-\u169a\u16a0-\u16ea\u16ee-\u16f0\u1700-\u170c\u170e-\u1714\u1720-\u1734\u1740-\u1753\u1760-\u176c\u176e-\u1770\u1772\u1773\u1780-\u17d3\u17d7\u17dc\u17dd\u17e0-\u17e9\u180b-\u180d\u1810-\u1819\u1820-\u1877\u1880-\u18aa\u18b0-\u18f5\u1900-\u191c\u1920-\u192b\u1930-\u193b\u1946-\u196d\u1970-\u1974\u1980-\u19ab\u19b0-\u19c9\u19d0-\u19d9\u1a00-\u1a1b\u1a20-\u1a5e\u1a60-\u1a7c\u1a7f-\u1a89\u1a90-\u1a99\u1aa7\u1b00-\u1b4b\u1b50-\u1b59\u1b6b-\u1b73\u1b80-\u1bf3\u1c00-\u1c37\u1c40-\u1c49\u1c4d-\u1c7d\u1cd0-\u1cd2\u1cd4-\u1cf6\u1d00-\u1de6\u1dfc-\u1f15\u1f18-\u1f1d\u1f20-\u1f45\u1f48-\u1f4d\u1f50-\u1f57\u1f59\u1f5b\u1f5d\u1f5f-\u1f7d\u1f80-\u1fb4\u1fb6-\u1fbc\u1fbe\u1fc2-\u1fc4\u1fc6-\u1fcc\u1fd0-\u1fd3\u1fd6-\u1fdb\u1fe0-\u1fec\u1ff2-\u1ff4\u1ff6-\u1ffc\u200c\u200d\u203f\u2040\u2054\u2071\u207f\u2090-\u209c\u20d0-\u20dc\u20e1\u20e5-\u20f0\u2102\u2107\u210a-\u2113\u2115\u2119-\u211d\u2124\u2126\u2128\u212a-\u212d\u212f-\u2139\u213c-\u213f\u2145-\u2149\u214e\u2160-\u2188\u2c00-\u2c2e\u2c30-\u2c5e\u2c60-\u2ce4\u2ceb-\u2cf3\u2d00-\u2d25\u2d27\u2d2d\u2d30-\u2d67\u2d6f\u2d7f-\u2d96\u2da0-\u2da6\u2da8-\u2dae\u2db0-\u2db6\u2db8-\u2dbe\u2dc0-\u2dc6\u2dc8-\u2dce\u2dd0-\u2dd6\u2dd8-\u2dde\u2de0-\u2dff\u2e2f\u3005-\u3007\u3021-\u302f\u3031-\u3035\u3038-\u303c\u3041-\u3096\u3099\u309a\u309d-\u309f\u30a1-\u30fa\u30fc-\u30ff\u3105-\u312d\u3131-\u318e\u31a0-\u31ba\u31f0-\u31ff\u3400-\u4db5\u4e00-\u9fcc\ua000-\ua48c\ua4d0-\ua4fd\ua500-\ua60c\ua610-\ua62b\ua640-\ua66f\ua674-\ua67d\ua67f-\ua697\ua69f-\ua6f1\ua717-\ua71f\ua722-\ua788\ua78b-\ua78e\ua790-\ua793\ua7a0-\ua7aa\ua7f8-\ua827\ua840-\ua873\ua880-\ua8c4\ua8d0-\ua8d9\ua8e0-\ua8f7\ua8fb\ua900-\ua92d\ua930-\ua953\ua960-\ua97c\ua980-\ua9c0\ua9cf-\ua9d9\uaa00-\uaa36\uaa40-\uaa4d\uaa50-\uaa59\uaa60-\uaa76\uaa7a\uaa7b\uaa80-\uaac2\uaadb-\uaadd\uaae0-\uaaef\uaaf2-\uaaf6\uab01-\uab06\uab09-\uab0e\uab11-\uab16\uab20-\uab26\uab28-\uab2e\uabc0-\uabea\uabec\uabed\uabf0-\uabf9\uac00-\ud7a3\ud7b0-\ud7c6\ud7cb-\ud7fb\uf900-\ufa6d\ufa70-\ufad9\ufb00-\ufb06\ufb13-\ufb17\ufb1d-\ufb28\ufb2a-\ufb36\ufb38-\ufb3c\ufb3e\ufb40\ufb41\ufb43\ufb44\ufb46-\ufbb1\ufbd3-\ufd3d\ufd50-\ufd8f\ufd92-\ufdc7\ufdf0-\ufdfb\ufe00-\ufe0f\ufe20-\ufe26\ufe33\ufe34\ufe4d-\ufe4f\ufe70-\ufe74\ufe76-\ufefc\uff10-\uff19\uff21-\uff3a\uff3f\uff41-\uff5a\uff66-\uffbe\uffc2-\uffc7\uffca-\uffcf\uffd2-\uffd7\uffda-\uffdc]')
|
|
};
|
|
|
|
// Ensure the condition is true, otherwise throw an error.
|
|
// This is only to have a better contract semantic, i.e. another safety net
|
|
// to catch a logic error. The condition shall be fulfilled in normal case.
|
|
// Do NOT use this to enforce a certain condition on any user input.
|
|
|
|
function assert(condition, message) {
|
|
if (!condition) {
|
|
throw new Error('ASSERT: ' + message);
|
|
}
|
|
}
|
|
|
|
function sliceSource(from, to) {
|
|
return source.slice(from, to);
|
|
}
|
|
|
|
if (typeof 'esprima'[0] === 'undefined') {
|
|
sliceSource = function sliceArraySource(from, to) {
|
|
return source.slice(from, to).join('');
|
|
};
|
|
}
|
|
|
|
function isDecimalDigit(ch) {
|
|
return '0123456789'.indexOf(ch) >= 0;
|
|
}
|
|
|
|
function isHexDigit(ch) {
|
|
return '0123456789abcdefABCDEF'.indexOf(ch) >= 0;
|
|
}
|
|
|
|
function isOctalDigit(ch) {
|
|
return '01234567'.indexOf(ch) >= 0;
|
|
}
|
|
|
|
|
|
// 7.2 White Space
|
|
|
|
function isWhiteSpace(ch) {
|
|
return (ch === ' ') || (ch === '\u0009') || (ch === '\u000B') ||
|
|
(ch === '\u000C') || (ch === '\u00A0') ||
|
|
(ch.charCodeAt(0) >= 0x1680 &&
|
|
'\u1680\u180E\u2000\u2001\u2002\u2003\u2004\u2005\u2006\u2007\u2008\u2009\u200A\u202F\u205F\u3000\uFEFF'.indexOf(ch) >= 0);
|
|
}
|
|
|
|
// 7.3 Line Terminators
|
|
|
|
function isLineTerminator(ch) {
|
|
return (ch === '\n' || ch === '\r' || ch === '\u2028' || ch === '\u2029');
|
|
}
|
|
|
|
// 7.6 Identifier Names and Identifiers
|
|
|
|
function isIdentifierStart(ch) {
|
|
return (ch === '$') || (ch === '_') || (ch === '\\') ||
|
|
(ch >= 'a' && ch <= 'z') || (ch >= 'A' && ch <= 'Z') ||
|
|
((ch.charCodeAt(0) >= 0x80) && Regex.NonAsciiIdentifierStart.test(ch));
|
|
}
|
|
|
|
function isIdentifierPart(ch) {
|
|
return (ch === '$') || (ch === '_') || (ch === '\\') ||
|
|
(ch >= 'a' && ch <= 'z') || (ch >= 'A' && ch <= 'Z') ||
|
|
((ch >= '0') && (ch <= '9')) ||
|
|
((ch.charCodeAt(0) >= 0x80) && Regex.NonAsciiIdentifierPart.test(ch));
|
|
}
|
|
|
|
// 7.6.1.2 Future Reserved Words
|
|
|
|
function isFutureReservedWord(id) {
|
|
switch (id) {
|
|
|
|
// Future reserved words.
|
|
case 'class':
|
|
case 'enum':
|
|
case 'export':
|
|
case 'extends':
|
|
case 'import':
|
|
case 'super':
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
function isStrictModeReservedWord(id) {
|
|
switch (id) {
|
|
|
|
// Strict Mode reserved words.
|
|
case 'implements':
|
|
case 'interface':
|
|
case 'package':
|
|
case 'private':
|
|
case 'protected':
|
|
case 'public':
|
|
case 'static':
|
|
case 'yield':
|
|
case 'let':
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
function isRestrictedWord(id) {
|
|
return id === 'eval' || id === 'arguments';
|
|
}
|
|
|
|
// 7.6.1.1 Keywords
|
|
|
|
function isKeyword(id) {
|
|
var keyword = false;
|
|
switch (id.length) {
|
|
case 2:
|
|
keyword = (id === 'if') || (id === 'in') || (id === 'do');
|
|
break;
|
|
case 3:
|
|
keyword = (id === 'var') || (id === 'for') || (id === 'new') || (id === 'try');
|
|
break;
|
|
case 4:
|
|
keyword = (id === 'this') || (id === 'else') || (id === 'case') || (id === 'void') || (id === 'with');
|
|
break;
|
|
case 5:
|
|
keyword = (id === 'while') || (id === 'break') || (id === 'catch') || (id === 'throw');
|
|
break;
|
|
case 6:
|
|
keyword = (id === 'return') || (id === 'typeof') || (id === 'delete') || (id === 'switch');
|
|
break;
|
|
case 7:
|
|
keyword = (id === 'default') || (id === 'finally');
|
|
break;
|
|
case 8:
|
|
keyword = (id === 'function') || (id === 'continue') || (id === 'debugger');
|
|
break;
|
|
case 10:
|
|
keyword = (id === 'instanceof');
|
|
break;
|
|
}
|
|
|
|
if (keyword) {
|
|
return true;
|
|
}
|
|
|
|
switch (id) {
|
|
// Future reserved words.
|
|
// 'const' is specialized as Keyword in V8.
|
|
case 'const':
|
|
return true;
|
|
|
|
// For compatiblity to SpiderMonkey and ES.next
|
|
case 'yield':
|
|
case 'let':
|
|
return true;
|
|
}
|
|
|
|
if (strict && isStrictModeReservedWord(id)) {
|
|
return true;
|
|
}
|
|
|
|
return isFutureReservedWord(id);
|
|
}
|
|
|
|
// 7.4 Comments
|
|
|
|
function skipComment() {
|
|
var ch, blockComment, lineComment;
|
|
|
|
blockComment = false;
|
|
lineComment = false;
|
|
|
|
while (index < length) {
|
|
ch = source[index];
|
|
|
|
if (lineComment) {
|
|
ch = source[index++];
|
|
if (isLineTerminator(ch)) {
|
|
lineComment = false;
|
|
if (ch === '\r' && source[index] === '\n') {
|
|
++index;
|
|
}
|
|
++lineNumber;
|
|
lineStart = index;
|
|
}
|
|
} else if (blockComment) {
|
|
if (isLineTerminator(ch)) {
|
|
if (ch === '\r' && source[index + 1] === '\n') {
|
|
++index;
|
|
}
|
|
++lineNumber;
|
|
++index;
|
|
lineStart = index;
|
|
if (index >= length) {
|
|
throwError({}, Messages.UnexpectedToken, 'ILLEGAL');
|
|
}
|
|
} else {
|
|
ch = source[index++];
|
|
if (index >= length) {
|
|
throwError({}, Messages.UnexpectedToken, 'ILLEGAL');
|
|
}
|
|
if (ch === '*') {
|
|
ch = source[index];
|
|
if (ch === '/') {
|
|
++index;
|
|
blockComment = false;
|
|
}
|
|
}
|
|
}
|
|
} else if (ch === '/') {
|
|
ch = source[index + 1];
|
|
if (ch === '/') {
|
|
index += 2;
|
|
lineComment = true;
|
|
} else if (ch === '*') {
|
|
index += 2;
|
|
blockComment = true;
|
|
if (index >= length) {
|
|
throwError({}, Messages.UnexpectedToken, 'ILLEGAL');
|
|
}
|
|
} else {
|
|
break;
|
|
}
|
|
} else if (isWhiteSpace(ch)) {
|
|
++index;
|
|
} else if (isLineTerminator(ch)) {
|
|
++index;
|
|
if (ch === '\r' && source[index] === '\n') {
|
|
++index;
|
|
}
|
|
++lineNumber;
|
|
lineStart = index;
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
function scanHexEscape(prefix) {
|
|
var i, len, ch, code = 0;
|
|
|
|
len = (prefix === 'u') ? 4 : 2;
|
|
for (i = 0; i < len; ++i) {
|
|
if (index < length && isHexDigit(source[index])) {
|
|
ch = source[index++];
|
|
code = code * 16 + '0123456789abcdef'.indexOf(ch.toLowerCase());
|
|
} else {
|
|
return '';
|
|
}
|
|
}
|
|
return String.fromCharCode(code);
|
|
}
|
|
|
|
function scanIdentifier() {
|
|
var ch, start, id, restore;
|
|
|
|
ch = source[index];
|
|
if (!isIdentifierStart(ch)) {
|
|
return;
|
|
}
|
|
|
|
start = index;
|
|
if (ch === '\\') {
|
|
++index;
|
|
if (source[index] !== 'u') {
|
|
return;
|
|
}
|
|
++index;
|
|
restore = index;
|
|
ch = scanHexEscape('u');
|
|
if (ch) {
|
|
if (ch === '\\' || !isIdentifierStart(ch)) {
|
|
return;
|
|
}
|
|
id = ch;
|
|
} else {
|
|
index = restore;
|
|
id = 'u';
|
|
}
|
|
} else {
|
|
id = source[index++];
|
|
}
|
|
|
|
while (index < length) {
|
|
ch = source[index];
|
|
if (!isIdentifierPart(ch)) {
|
|
break;
|
|
}
|
|
if (ch === '\\') {
|
|
++index;
|
|
if (source[index] !== 'u') {
|
|
return;
|
|
}
|
|
++index;
|
|
restore = index;
|
|
ch = scanHexEscape('u');
|
|
if (ch) {
|
|
if (ch === '\\' || !isIdentifierPart(ch)) {
|
|
return;
|
|
}
|
|
id += ch;
|
|
} else {
|
|
index = restore;
|
|
id += 'u';
|
|
}
|
|
} else {
|
|
id += source[index++];
|
|
}
|
|
}
|
|
|
|
// There is no keyword or literal with only one character.
|
|
// Thus, it must be an identifier.
|
|
if (id.length === 1) {
|
|
return {
|
|
type: Token.Identifier,
|
|
value: id,
|
|
lineNumber: lineNumber,
|
|
lineStart: lineStart,
|
|
range: [start, index]
|
|
};
|
|
}
|
|
|
|
if (isKeyword(id)) {
|
|
return {
|
|
type: Token.Keyword,
|
|
value: id,
|
|
lineNumber: lineNumber,
|
|
lineStart: lineStart,
|
|
range: [start, index]
|
|
};
|
|
}
|
|
|
|
// 7.8.1 Null Literals
|
|
|
|
if (id === 'null') {
|
|
return {
|
|
type: Token.NullLiteral,
|
|
value: id,
|
|
lineNumber: lineNumber,
|
|
lineStart: lineStart,
|
|
range: [start, index]
|
|
};
|
|
}
|
|
|
|
// 7.8.2 Boolean Literals
|
|
|
|
if (id === 'true' || id === 'false') {
|
|
return {
|
|
type: Token.BooleanLiteral,
|
|
value: id,
|
|
lineNumber: lineNumber,
|
|
lineStart: lineStart,
|
|
range: [start, index]
|
|
};
|
|
}
|
|
|
|
return {
|
|
type: Token.Identifier,
|
|
value: id,
|
|
lineNumber: lineNumber,
|
|
lineStart: lineStart,
|
|
range: [start, index]
|
|
};
|
|
}
|
|
|
|
// 7.7 Punctuators
|
|
|
|
function scanPunctuator() {
|
|
var start = index,
|
|
ch1 = source[index],
|
|
ch2,
|
|
ch3,
|
|
ch4;
|
|
|
|
// Check for most common single-character punctuators.
|
|
|
|
if (ch1 === ';' || ch1 === '{' || ch1 === '}') {
|
|
++index;
|
|
return {
|
|
type: Token.Punctuator,
|
|
value: ch1,
|
|
lineNumber: lineNumber,
|
|
lineStart: lineStart,
|
|
range: [start, index]
|
|
};
|
|
}
|
|
|
|
if (ch1 === ',' || ch1 === '(' || ch1 === ')') {
|
|
++index;
|
|
return {
|
|
type: Token.Punctuator,
|
|
value: ch1,
|
|
lineNumber: lineNumber,
|
|
lineStart: lineStart,
|
|
range: [start, index]
|
|
};
|
|
}
|
|
|
|
// Dot (.) can also start a floating-point number, hence the need
|
|
// to check the next character.
|
|
|
|
ch2 = source[index + 1];
|
|
if (ch1 === '.' && !isDecimalDigit(ch2)) {
|
|
return {
|
|
type: Token.Punctuator,
|
|
value: source[index++],
|
|
lineNumber: lineNumber,
|
|
lineStart: lineStart,
|
|
range: [start, index]
|
|
};
|
|
}
|
|
|
|
// Peek more characters.
|
|
|
|
ch3 = source[index + 2];
|
|
ch4 = source[index + 3];
|
|
|
|
// 4-character punctuator: >>>=
|
|
|
|
if (ch1 === '>' && ch2 === '>' && ch3 === '>') {
|
|
if (ch4 === '=') {
|
|
index += 4;
|
|
return {
|
|
type: Token.Punctuator,
|
|
value: '>>>=',
|
|
lineNumber: lineNumber,
|
|
lineStart: lineStart,
|
|
range: [start, index]
|
|
};
|
|
}
|
|
}
|
|
|
|
// 3-character punctuators: === !== >>> <<= >>=
|
|
|
|
if (ch1 === '=' && ch2 === '=' && ch3 === '=') {
|
|
index += 3;
|
|
return {
|
|
type: Token.Punctuator,
|
|
value: '===',
|
|
lineNumber: lineNumber,
|
|
lineStart: lineStart,
|
|
range: [start, index]
|
|
};
|
|
}
|
|
|
|
if (ch1 === '!' && ch2 === '=' && ch3 === '=') {
|
|
index += 3;
|
|
return {
|
|
type: Token.Punctuator,
|
|
value: '!==',
|
|
lineNumber: lineNumber,
|
|
lineStart: lineStart,
|
|
range: [start, index]
|
|
};
|
|
}
|
|
|
|
if (ch1 === '>' && ch2 === '>' && ch3 === '>') {
|
|
index += 3;
|
|
return {
|
|
type: Token.Punctuator,
|
|
value: '>>>',
|
|
lineNumber: lineNumber,
|
|
lineStart: lineStart,
|
|
range: [start, index]
|
|
};
|
|
}
|
|
|
|
if (ch1 === '<' && ch2 === '<' && ch3 === '=') {
|
|
index += 3;
|
|
return {
|
|
type: Token.Punctuator,
|
|
value: '<<=',
|
|
lineNumber: lineNumber,
|
|
lineStart: lineStart,
|
|
range: [start, index]
|
|
};
|
|
}
|
|
|
|
if (ch1 === '>' && ch2 === '>' && ch3 === '=') {
|
|
index += 3;
|
|
return {
|
|
type: Token.Punctuator,
|
|
value: '>>=',
|
|
lineNumber: lineNumber,
|
|
lineStart: lineStart,
|
|
range: [start, index]
|
|
};
|
|
}
|
|
|
|
// 2-character punctuators: <= >= == != ++ -- << >> && ||
|
|
// += -= *= %= &= |= ^= /=
|
|
|
|
if (ch2 === '=') {
|
|
if ('<>=!+-*%&|^/'.indexOf(ch1) >= 0) {
|
|
index += 2;
|
|
return {
|
|
type: Token.Punctuator,
|
|
value: ch1 + ch2,
|
|
lineNumber: lineNumber,
|
|
lineStart: lineStart,
|
|
range: [start, index]
|
|
};
|
|
}
|
|
}
|
|
|
|
if (ch1 === ch2 && ('+-<>&|'.indexOf(ch1) >= 0)) {
|
|
if ('+-<>&|'.indexOf(ch2) >= 0) {
|
|
index += 2;
|
|
return {
|
|
type: Token.Punctuator,
|
|
value: ch1 + ch2,
|
|
lineNumber: lineNumber,
|
|
lineStart: lineStart,
|
|
range: [start, index]
|
|
};
|
|
}
|
|
}
|
|
|
|
// The remaining 1-character punctuators.
|
|
|
|
if ('[]<>+-*%&|^!~?:=/'.indexOf(ch1) >= 0) {
|
|
return {
|
|
type: Token.Punctuator,
|
|
value: source[index++],
|
|
lineNumber: lineNumber,
|
|
lineStart: lineStart,
|
|
range: [start, index]
|
|
};
|
|
}
|
|
}
|
|
|
|
// 7.8.3 Numeric Literals
|
|
|
|
function scanNumericLiteral() {
|
|
var number, start, ch;
|
|
|
|
ch = source[index];
|
|
assert(isDecimalDigit(ch) || (ch === '.'),
|
|
'Numeric literal must start with a decimal digit or a decimal point');
|
|
|
|
start = index;
|
|
number = '';
|
|
if (ch !== '.') {
|
|
number = source[index++];
|
|
ch = source[index];
|
|
|
|
// Hex number starts with '0x'.
|
|
// Octal number starts with '0'.
|
|
if (number === '0') {
|
|
if (ch === 'x' || ch === 'X') {
|
|
number += source[index++];
|
|
while (index < length) {
|
|
ch = source[index];
|
|
if (!isHexDigit(ch)) {
|
|
break;
|
|
}
|
|
number += source[index++];
|
|
}
|
|
|
|
if (number.length <= 2) {
|
|
// only 0x
|
|
throwError({}, Messages.UnexpectedToken, 'ILLEGAL');
|
|
}
|
|
|
|
if (index < length) {
|
|
ch = source[index];
|
|
if (isIdentifierStart(ch)) {
|
|
throwError({}, Messages.UnexpectedToken, 'ILLEGAL');
|
|
}
|
|
}
|
|
return {
|
|
type: Token.NumericLiteral,
|
|
value: parseInt(number, 16),
|
|
lineNumber: lineNumber,
|
|
lineStart: lineStart,
|
|
range: [start, index]
|
|
};
|
|
} else if (isOctalDigit(ch)) {
|
|
number += source[index++];
|
|
while (index < length) {
|
|
ch = source[index];
|
|
if (!isOctalDigit(ch)) {
|
|
break;
|
|
}
|
|
number += source[index++];
|
|
}
|
|
|
|
if (index < length) {
|
|
ch = source[index];
|
|
if (isIdentifierStart(ch) || isDecimalDigit(ch)) {
|
|
throwError({}, Messages.UnexpectedToken, 'ILLEGAL');
|
|
}
|
|
}
|
|
return {
|
|
type: Token.NumericLiteral,
|
|
value: parseInt(number, 8),
|
|
octal: true,
|
|
lineNumber: lineNumber,
|
|
lineStart: lineStart,
|
|
range: [start, index]
|
|
};
|
|
}
|
|
|
|
// decimal number starts with '0' such as '09' is illegal.
|
|
if (isDecimalDigit(ch)) {
|
|
throwError({}, Messages.UnexpectedToken, 'ILLEGAL');
|
|
}
|
|
}
|
|
|
|
while (index < length) {
|
|
ch = source[index];
|
|
if (!isDecimalDigit(ch)) {
|
|
break;
|
|
}
|
|
number += source[index++];
|
|
}
|
|
}
|
|
|
|
if (ch === '.') {
|
|
number += source[index++];
|
|
while (index < length) {
|
|
ch = source[index];
|
|
if (!isDecimalDigit(ch)) {
|
|
break;
|
|
}
|
|
number += source[index++];
|
|
}
|
|
}
|
|
|
|
if (ch === 'e' || ch === 'E') {
|
|
number += source[index++];
|
|
|
|
ch = source[index];
|
|
if (ch === '+' || ch === '-') {
|
|
number += source[index++];
|
|
}
|
|
|
|
ch = source[index];
|
|
if (isDecimalDigit(ch)) {
|
|
number += source[index++];
|
|
while (index < length) {
|
|
ch = source[index];
|
|
if (!isDecimalDigit(ch)) {
|
|
break;
|
|
}
|
|
number += source[index++];
|
|
}
|
|
} else {
|
|
ch = 'character ' + ch;
|
|
if (index >= length) {
|
|
ch = '<end>';
|
|
}
|
|
throwError({}, Messages.UnexpectedToken, 'ILLEGAL');
|
|
}
|
|
}
|
|
|
|
if (index < length) {
|
|
ch = source[index];
|
|
if (isIdentifierStart(ch)) {
|
|
throwError({}, Messages.UnexpectedToken, 'ILLEGAL');
|
|
}
|
|
}
|
|
|
|
return {
|
|
type: Token.NumericLiteral,
|
|
value: parseFloat(number),
|
|
lineNumber: lineNumber,
|
|
lineStart: lineStart,
|
|
range: [start, index]
|
|
};
|
|
}
|
|
|
|
// 7.8.4 String Literals
|
|
|
|
function scanStringLiteral() {
|
|
var str = '', quote, start, ch, code, unescaped, restore, octal = false;
|
|
|
|
quote = source[index];
|
|
assert((quote === '\'' || quote === '"'),
|
|
'String literal must starts with a quote');
|
|
|
|
start = index;
|
|
++index;
|
|
|
|
while (index < length) {
|
|
ch = source[index++];
|
|
|
|
if (ch === quote) {
|
|
quote = '';
|
|
break;
|
|
} else if (ch === '\\') {
|
|
ch = source[index++];
|
|
if (!isLineTerminator(ch)) {
|
|
switch (ch) {
|
|
case 'n':
|
|
str += '\n';
|
|
break;
|
|
case 'r':
|
|
str += '\r';
|
|
break;
|
|
case 't':
|
|
str += '\t';
|
|
break;
|
|
case 'u':
|
|
case 'x':
|
|
restore = index;
|
|
unescaped = scanHexEscape(ch);
|
|
if (unescaped) {
|
|
str += unescaped;
|
|
} else {
|
|
index = restore;
|
|
str += ch;
|
|
}
|
|
break;
|
|
case 'b':
|
|
str += '\b';
|
|
break;
|
|
case 'f':
|
|
str += '\f';
|
|
break;
|
|
case 'v':
|
|
str += '\x0B';
|
|
break;
|
|
|
|
default:
|
|
if (isOctalDigit(ch)) {
|
|
code = '01234567'.indexOf(ch);
|
|
|
|
// \0 is not octal escape sequence
|
|
if (code !== 0) {
|
|
octal = true;
|
|
}
|
|
|
|
if (index < length && isOctalDigit(source[index])) {
|
|
octal = true;
|
|
code = code * 8 + '01234567'.indexOf(source[index++]);
|
|
|
|
// 3 digits are only allowed when string starts
|
|
// with 0, 1, 2, 3
|
|
if ('0123'.indexOf(ch) >= 0 &&
|
|
index < length &&
|
|
isOctalDigit(source[index])) {
|
|
code = code * 8 + '01234567'.indexOf(source[index++]);
|
|
}
|
|
}
|
|
str += String.fromCharCode(code);
|
|
} else {
|
|
str += ch;
|
|
}
|
|
break;
|
|
}
|
|
} else {
|
|
++lineNumber;
|
|
if (ch === '\r' && source[index] === '\n') {
|
|
++index;
|
|
}
|
|
}
|
|
} else if (isLineTerminator(ch)) {
|
|
break;
|
|
} else {
|
|
str += ch;
|
|
}
|
|
}
|
|
|
|
if (quote !== '') {
|
|
throwError({}, Messages.UnexpectedToken, 'ILLEGAL');
|
|
}
|
|
|
|
return {
|
|
type: Token.StringLiteral,
|
|
value: str,
|
|
octal: octal,
|
|
lineNumber: lineNumber,
|
|
lineStart: lineStart,
|
|
range: [start, index]
|
|
};
|
|
}
|
|
|
|
function scanRegExp() {
|
|
var str, ch, start, pattern, flags, value, classMarker = false, restore, terminated = false;
|
|
|
|
buffer = null;
|
|
skipComment();
|
|
|
|
start = index;
|
|
ch = source[index];
|
|
assert(ch === '/', 'Regular expression literal must start with a slash');
|
|
str = source[index++];
|
|
|
|
while (index < length) {
|
|
ch = source[index++];
|
|
str += ch;
|
|
if (ch === '\\') {
|
|
ch = source[index++];
|
|
// ECMA-262 7.8.5
|
|
if (isLineTerminator(ch)) {
|
|
throwError({}, Messages.UnterminatedRegExp);
|
|
}
|
|
str += ch;
|
|
} else if (classMarker) {
|
|
if (ch === ']') {
|
|
classMarker = false;
|
|
}
|
|
} else {
|
|
if (ch === '/') {
|
|
terminated = true;
|
|
break;
|
|
} else if (ch === '[') {
|
|
classMarker = true;
|
|
} else if (isLineTerminator(ch)) {
|
|
throwError({}, Messages.UnterminatedRegExp);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!terminated) {
|
|
throwError({}, Messages.UnterminatedRegExp);
|
|
}
|
|
|
|
// Exclude leading and trailing slash.
|
|
pattern = str.substr(1, str.length - 2);
|
|
|
|
flags = '';
|
|
while (index < length) {
|
|
ch = source[index];
|
|
if (!isIdentifierPart(ch)) {
|
|
break;
|
|
}
|
|
|
|
++index;
|
|
if (ch === '\\' && index < length) {
|
|
ch = source[index];
|
|
if (ch === 'u') {
|
|
++index;
|
|
restore = index;
|
|
ch = scanHexEscape('u');
|
|
if (ch) {
|
|
flags += ch;
|
|
str += '\\u';
|
|
for (; restore < index; ++restore) {
|
|
str += source[restore];
|
|
}
|
|
} else {
|
|
index = restore;
|
|
flags += 'u';
|
|
str += '\\u';
|
|
}
|
|
} else {
|
|
str += '\\';
|
|
}
|
|
} else {
|
|
flags += ch;
|
|
str += ch;
|
|
}
|
|
}
|
|
|
|
try {
|
|
value = new RegExp(pattern, flags);
|
|
} catch (e) {
|
|
throwError({}, Messages.InvalidRegExp);
|
|
}
|
|
|
|
return {
|
|
literal: str,
|
|
value: value,
|
|
range: [start, index]
|
|
};
|
|
}
|
|
|
|
function isIdentifierName(token) {
|
|
return token.type === Token.Identifier ||
|
|
token.type === Token.Keyword ||
|
|
token.type === Token.BooleanLiteral ||
|
|
token.type === Token.NullLiteral;
|
|
}
|
|
|
|
function advance() {
|
|
var ch, token;
|
|
|
|
skipComment();
|
|
|
|
if (index >= length) {
|
|
return {
|
|
type: Token.EOF,
|
|
lineNumber: lineNumber,
|
|
lineStart: lineStart,
|
|
range: [index, index]
|
|
};
|
|
}
|
|
|
|
token = scanPunctuator();
|
|
if (typeof token !== 'undefined') {
|
|
return token;
|
|
}
|
|
|
|
ch = source[index];
|
|
|
|
if (ch === '\'' || ch === '"') {
|
|
return scanStringLiteral();
|
|
}
|
|
|
|
if (ch === '.' || isDecimalDigit(ch)) {
|
|
return scanNumericLiteral();
|
|
}
|
|
|
|
token = scanIdentifier();
|
|
if (typeof token !== 'undefined') {
|
|
return token;
|
|
}
|
|
|
|
throwError({}, Messages.UnexpectedToken, 'ILLEGAL');
|
|
}
|
|
|
|
function lex() {
|
|
var token;
|
|
|
|
if (buffer) {
|
|
index = buffer.range[1];
|
|
lineNumber = buffer.lineNumber;
|
|
lineStart = buffer.lineStart;
|
|
token = buffer;
|
|
buffer = null;
|
|
return token;
|
|
}
|
|
|
|
buffer = null;
|
|
return advance();
|
|
}
|
|
|
|
function lookahead() {
|
|
var pos, line, start;
|
|
|
|
if (buffer !== null) {
|
|
return buffer;
|
|
}
|
|
|
|
pos = index;
|
|
line = lineNumber;
|
|
start = lineStart;
|
|
buffer = advance();
|
|
index = pos;
|
|
lineNumber = line;
|
|
lineStart = start;
|
|
|
|
return buffer;
|
|
}
|
|
|
|
// Return true if there is a line terminator before the next token.
|
|
|
|
function peekLineTerminator() {
|
|
var pos, line, start, found;
|
|
|
|
pos = index;
|
|
line = lineNumber;
|
|
start = lineStart;
|
|
skipComment();
|
|
found = lineNumber !== line;
|
|
index = pos;
|
|
lineNumber = line;
|
|
lineStart = start;
|
|
|
|
return found;
|
|
}
|
|
|
|
// Throw an exception
|
|
|
|
function throwError(token, messageFormat) {
|
|
var error,
|
|
args = Array.prototype.slice.call(arguments, 2),
|
|
msg = messageFormat.replace(
|
|
/%(\d)/g,
|
|
function (whole, index) {
|
|
return args[index] || '';
|
|
}
|
|
);
|
|
|
|
if (typeof token.lineNumber === 'number') {
|
|
error = new Error('Line ' + token.lineNumber + ': ' + msg);
|
|
error.index = token.range[0];
|
|
error.lineNumber = token.lineNumber;
|
|
error.column = token.range[0] - lineStart + 1;
|
|
} else {
|
|
error = new Error('Line ' + lineNumber + ': ' + msg);
|
|
error.index = index;
|
|
error.lineNumber = lineNumber;
|
|
error.column = index - lineStart + 1;
|
|
}
|
|
|
|
throw error;
|
|
}
|
|
|
|
function throwErrorTolerant() {
|
|
try {
|
|
throwError.apply(null, arguments);
|
|
} catch (e) {
|
|
if (extra.errors) {
|
|
extra.errors.push(e);
|
|
} else {
|
|
throw e;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
// Throw an exception because of the token.
|
|
|
|
function throwUnexpected(token) {
|
|
if (token.type === Token.EOF) {
|
|
throwError(token, Messages.UnexpectedEOS);
|
|
}
|
|
|
|
if (token.type === Token.NumericLiteral) {
|
|
throwError(token, Messages.UnexpectedNumber);
|
|
}
|
|
|
|
if (token.type === Token.StringLiteral) {
|
|
throwError(token, Messages.UnexpectedString);
|
|
}
|
|
|
|
if (token.type === Token.Identifier) {
|
|
throwError(token, Messages.UnexpectedIdentifier);
|
|
}
|
|
|
|
if (token.type === Token.Keyword) {
|
|
if (isFutureReservedWord(token.value)) {
|
|
throwError(token, Messages.UnexpectedReserved);
|
|
} else if (strict && isStrictModeReservedWord(token.value)) {
|
|
throwErrorTolerant(token, Messages.StrictReservedWord);
|
|
return;
|
|
}
|
|
throwError(token, Messages.UnexpectedToken, token.value);
|
|
}
|
|
|
|
// BooleanLiteral, NullLiteral, or Punctuator.
|
|
throwError(token, Messages.UnexpectedToken, token.value);
|
|
}
|
|
|
|
// Expect the next token to match the specified punctuator.
|
|
// If not, an exception will be thrown.
|
|
|
|
function expect(value) {
|
|
var token = lex();
|
|
if (token.type !== Token.Punctuator || token.value !== value) {
|
|
throwUnexpected(token);
|
|
}
|
|
}
|
|
|
|
// Expect the next token to match the specified keyword.
|
|
// If not, an exception will be thrown.
|
|
|
|
function expectKeyword(keyword) {
|
|
var token = lex();
|
|
if (token.type !== Token.Keyword || token.value !== keyword) {
|
|
throwUnexpected(token);
|
|
}
|
|
}
|
|
|
|
// Return true if the next token matches the specified punctuator.
|
|
|
|
function match(value) {
|
|
var token = lookahead();
|
|
return token.type === Token.Punctuator && token.value === value;
|
|
}
|
|
|
|
// Return true if the next token matches the specified keyword
|
|
|
|
function matchKeyword(keyword) {
|
|
var token = lookahead();
|
|
return token.type === Token.Keyword && token.value === keyword;
|
|
}
|
|
|
|
// Return true if the next token is an assignment operator
|
|
|
|
function matchAssign() {
|
|
var token = lookahead(),
|
|
op = token.value;
|
|
|
|
if (token.type !== Token.Punctuator) {
|
|
return false;
|
|
}
|
|
return op === '=' ||
|
|
op === '*=' ||
|
|
op === '/=' ||
|
|
op === '%=' ||
|
|
op === '+=' ||
|
|
op === '-=' ||
|
|
op === '<<=' ||
|
|
op === '>>=' ||
|
|
op === '>>>=' ||
|
|
op === '&=' ||
|
|
op === '^=' ||
|
|
op === '|=';
|
|
}
|
|
|
|
function consumeSemicolon() {
|
|
var token, line;
|
|
|
|
// Catch the very common case first.
|
|
if (source[index] === ';') {
|
|
lex();
|
|
return;
|
|
}
|
|
|
|
line = lineNumber;
|
|
skipComment();
|
|
if (lineNumber !== line) {
|
|
return;
|
|
}
|
|
|
|
if (match(';')) {
|
|
lex();
|
|
return;
|
|
}
|
|
|
|
token = lookahead();
|
|
if (token.type !== Token.EOF && !match('}')) {
|
|
throwUnexpected(token);
|
|
}
|
|
}
|
|
|
|
// Return true if provided expression is LeftHandSideExpression
|
|
|
|
function isLeftHandSide(expr) {
|
|
return expr.type === Syntax.Identifier || expr.type === Syntax.MemberExpression;
|
|
}
|
|
|
|
// 11.1.4 Array Initialiser
|
|
|
|
function parseArrayInitialiser() {
|
|
var elements = [];
|
|
|
|
expect('[');
|
|
|
|
while (!match(']')) {
|
|
if (match(',')) {
|
|
lex();
|
|
elements.push(null);
|
|
} else {
|
|
elements.push(parseAssignmentExpression());
|
|
|
|
if (!match(']')) {
|
|
expect(',');
|
|
}
|
|
}
|
|
}
|
|
|
|
expect(']');
|
|
|
|
return {
|
|
type: Syntax.ArrayExpression,
|
|
elements: elements
|
|
};
|
|
}
|
|
|
|
// 11.1.5 Object Initialiser
|
|
|
|
function parsePropertyFunction(param, first) {
|
|
var previousStrict, body;
|
|
|
|
previousStrict = strict;
|
|
body = parseFunctionSourceElements();
|
|
if (first && strict && isRestrictedWord(param[0].name)) {
|
|
throwErrorTolerant(first, Messages.StrictParamName);
|
|
}
|
|
strict = previousStrict;
|
|
|
|
return {
|
|
type: Syntax.FunctionExpression,
|
|
id: null,
|
|
params: param,
|
|
defaults: [],
|
|
body: body,
|
|
rest: null,
|
|
generator: false,
|
|
expression: false
|
|
};
|
|
}
|
|
|
|
function parseObjectPropertyKey() {
|
|
var token = lex();
|
|
|
|
// Note: This function is called only from parseObjectProperty(), where
|
|
// EOF and Punctuator tokens are already filtered out.
|
|
|
|
if (token.type === Token.StringLiteral || token.type === Token.NumericLiteral) {
|
|
if (strict && token.octal) {
|
|
throwErrorTolerant(token, Messages.StrictOctalLiteral);
|
|
}
|
|
return createLiteral(token);
|
|
}
|
|
|
|
return {
|
|
type: Syntax.Identifier,
|
|
name: token.value
|
|
};
|
|
}
|
|
|
|
function parseObjectProperty() {
|
|
var token, key, id, param;
|
|
|
|
token = lookahead();
|
|
|
|
if (token.type === Token.Identifier) {
|
|
|
|
id = parseObjectPropertyKey();
|
|
|
|
// Property Assignment: Getter and Setter.
|
|
|
|
if (token.value === 'get' && !match(':')) {
|
|
key = parseObjectPropertyKey();
|
|
expect('(');
|
|
expect(')');
|
|
return {
|
|
type: Syntax.Property,
|
|
key: key,
|
|
value: parsePropertyFunction([]),
|
|
kind: 'get'
|
|
};
|
|
} else if (token.value === 'set' && !match(':')) {
|
|
key = parseObjectPropertyKey();
|
|
expect('(');
|
|
token = lookahead();
|
|
if (token.type !== Token.Identifier) {
|
|
expect(')');
|
|
throwErrorTolerant(token, Messages.UnexpectedToken, token.value);
|
|
return {
|
|
type: Syntax.Property,
|
|
key: key,
|
|
value: parsePropertyFunction([]),
|
|
kind: 'set'
|
|
};
|
|
} else {
|
|
param = [ parseVariableIdentifier() ];
|
|
expect(')');
|
|
return {
|
|
type: Syntax.Property,
|
|
key: key,
|
|
value: parsePropertyFunction(param, token),
|
|
kind: 'set'
|
|
};
|
|
}
|
|
} else {
|
|
expect(':');
|
|
return {
|
|
type: Syntax.Property,
|
|
key: id,
|
|
value: parseAssignmentExpression(),
|
|
kind: 'init'
|
|
};
|
|
}
|
|
} else if (token.type === Token.EOF || token.type === Token.Punctuator) {
|
|
throwUnexpected(token);
|
|
} else {
|
|
key = parseObjectPropertyKey();
|
|
expect(':');
|
|
return {
|
|
type: Syntax.Property,
|
|
key: key,
|
|
value: parseAssignmentExpression(),
|
|
kind: 'init'
|
|
};
|
|
}
|
|
}
|
|
|
|
function parseObjectInitialiser() {
|
|
var properties = [], property, name, kind, map = {}, toString = String;
|
|
|
|
expect('{');
|
|
|
|
while (!match('}')) {
|
|
property = parseObjectProperty();
|
|
|
|
if (property.key.type === Syntax.Identifier) {
|
|
name = property.key.name;
|
|
} else {
|
|
name = toString(property.key.value);
|
|
}
|
|
kind = (property.kind === 'init') ? PropertyKind.Data : (property.kind === 'get') ? PropertyKind.Get : PropertyKind.Set;
|
|
if (Object.prototype.hasOwnProperty.call(map, name)) {
|
|
if (map[name] === PropertyKind.Data) {
|
|
if (strict && kind === PropertyKind.Data) {
|
|
throwErrorTolerant({}, Messages.StrictDuplicateProperty);
|
|
} else if (kind !== PropertyKind.Data) {
|
|
throwErrorTolerant({}, Messages.AccessorDataProperty);
|
|
}
|
|
} else {
|
|
if (kind === PropertyKind.Data) {
|
|
throwErrorTolerant({}, Messages.AccessorDataProperty);
|
|
} else if (map[name] & kind) {
|
|
throwErrorTolerant({}, Messages.AccessorGetSet);
|
|
}
|
|
}
|
|
map[name] |= kind;
|
|
} else {
|
|
map[name] = kind;
|
|
}
|
|
|
|
properties.push(property);
|
|
|
|
if (!match('}')) {
|
|
expect(',');
|
|
}
|
|
}
|
|
|
|
expect('}');
|
|
|
|
return {
|
|
type: Syntax.ObjectExpression,
|
|
properties: properties
|
|
};
|
|
}
|
|
|
|
// 11.1.6 The Grouping Operator
|
|
|
|
function parseGroupExpression() {
|
|
var expr;
|
|
|
|
expect('(');
|
|
|
|
expr = parseExpression();
|
|
|
|
expect(')');
|
|
|
|
return expr;
|
|
}
|
|
|
|
|
|
// 11.1 Primary Expressions
|
|
|
|
function parsePrimaryExpression() {
|
|
var token = lookahead(),
|
|
type = token.type;
|
|
|
|
if (type === Token.Identifier) {
|
|
return {
|
|
type: Syntax.Identifier,
|
|
name: lex().value
|
|
};
|
|
}
|
|
|
|
if (type === Token.StringLiteral || type === Token.NumericLiteral) {
|
|
if (strict && token.octal) {
|
|
throwErrorTolerant(token, Messages.StrictOctalLiteral);
|
|
}
|
|
return createLiteral(lex());
|
|
}
|
|
|
|
if (type === Token.Keyword) {
|
|
if (matchKeyword('this')) {
|
|
lex();
|
|
return {
|
|
type: Syntax.ThisExpression
|
|
};
|
|
}
|
|
|
|
if (matchKeyword('function')) {
|
|
return parseFunctionExpression();
|
|
}
|
|
}
|
|
|
|
if (type === Token.BooleanLiteral) {
|
|
lex();
|
|
token.value = (token.value === 'true');
|
|
return createLiteral(token);
|
|
}
|
|
|
|
if (type === Token.NullLiteral) {
|
|
lex();
|
|
token.value = null;
|
|
return createLiteral(token);
|
|
}
|
|
|
|
if (match('[')) {
|
|
return parseArrayInitialiser();
|
|
}
|
|
|
|
if (match('{')) {
|
|
return parseObjectInitialiser();
|
|
}
|
|
|
|
if (match('(')) {
|
|
return parseGroupExpression();
|
|
}
|
|
|
|
if (match('/') || match('/=')) {
|
|
return createLiteral(scanRegExp());
|
|
}
|
|
|
|
return throwUnexpected(lex());
|
|
}
|
|
|
|
// 11.2 Left-Hand-Side Expressions
|
|
|
|
function parseArguments() {
|
|
var args = [];
|
|
|
|
expect('(');
|
|
|
|
if (!match(')')) {
|
|
while (index < length) {
|
|
args.push(parseAssignmentExpression());
|
|
if (match(')')) {
|
|
break;
|
|
}
|
|
expect(',');
|
|
}
|
|
}
|
|
|
|
expect(')');
|
|
|
|
return args;
|
|
}
|
|
|
|
function parseNonComputedProperty() {
|
|
var token = lex();
|
|
|
|
if (!isIdentifierName(token)) {
|
|
throwUnexpected(token);
|
|
}
|
|
|
|
return {
|
|
type: Syntax.Identifier,
|
|
name: token.value
|
|
};
|
|
}
|
|
|
|
function parseNonComputedMember() {
|
|
expect('.');
|
|
|
|
return parseNonComputedProperty();
|
|
}
|
|
|
|
function parseComputedMember() {
|
|
var expr;
|
|
|
|
expect('[');
|
|
|
|
expr = parseExpression();
|
|
|
|
expect(']');
|
|
|
|
return expr;
|
|
}
|
|
|
|
function parseNewExpression() {
|
|
var expr;
|
|
|
|
expectKeyword('new');
|
|
|
|
expr = {
|
|
type: Syntax.NewExpression,
|
|
callee: parseLeftHandSideExpression(),
|
|
'arguments': []
|
|
};
|
|
|
|
if (match('(')) {
|
|
expr['arguments'] = parseArguments();
|
|
}
|
|
|
|
return expr;
|
|
}
|
|
|
|
function parseLeftHandSideExpressionAllowCall() {
|
|
var expr;
|
|
|
|
expr = matchKeyword('new') ? parseNewExpression() : parsePrimaryExpression();
|
|
|
|
while (match('.') || match('[') || match('(')) {
|
|
if (match('(')) {
|
|
expr = {
|
|
type: Syntax.CallExpression,
|
|
callee: expr,
|
|
'arguments': parseArguments()
|
|
};
|
|
} else if (match('[')) {
|
|
expr = {
|
|
type: Syntax.MemberExpression,
|
|
computed: true,
|
|
object: expr,
|
|
property: parseComputedMember()
|
|
};
|
|
} else {
|
|
expr = {
|
|
type: Syntax.MemberExpression,
|
|
computed: false,
|
|
object: expr,
|
|
property: parseNonComputedMember()
|
|
};
|
|
}
|
|
}
|
|
|
|
return expr;
|
|
}
|
|
|
|
|
|
function parseLeftHandSideExpression() {
|
|
var expr;
|
|
|
|
expr = matchKeyword('new') ? parseNewExpression() : parsePrimaryExpression();
|
|
|
|
while (match('.') || match('[')) {
|
|
if (match('[')) {
|
|
expr = {
|
|
type: Syntax.MemberExpression,
|
|
computed: true,
|
|
object: expr,
|
|
property: parseComputedMember()
|
|
};
|
|
} else {
|
|
expr = {
|
|
type: Syntax.MemberExpression,
|
|
computed: false,
|
|
object: expr,
|
|
property: parseNonComputedMember()
|
|
};
|
|
}
|
|
}
|
|
|
|
return expr;
|
|
}
|
|
|
|
// 11.3 Postfix Expressions
|
|
|
|
function parsePostfixExpression() {
|
|
var expr = parseLeftHandSideExpressionAllowCall(), token;
|
|
|
|
token = lookahead();
|
|
if (token.type !== Token.Punctuator) {
|
|
return expr;
|
|
}
|
|
|
|
if ((match('++') || match('--')) && !peekLineTerminator()) {
|
|
// 11.3.1, 11.3.2
|
|
if (strict && expr.type === Syntax.Identifier && isRestrictedWord(expr.name)) {
|
|
throwErrorTolerant({}, Messages.StrictLHSPostfix);
|
|
}
|
|
if (!isLeftHandSide(expr)) {
|
|
throwErrorTolerant({}, Messages.InvalidLHSInAssignment);
|
|
}
|
|
|
|
expr = {
|
|
type: Syntax.UpdateExpression,
|
|
operator: lex().value,
|
|
argument: expr,
|
|
prefix: false
|
|
};
|
|
}
|
|
|
|
return expr;
|
|
}
|
|
|
|
// 11.4 Unary Operators
|
|
|
|
function parseUnaryExpression() {
|
|
var token, expr;
|
|
|
|
token = lookahead();
|
|
if (token.type !== Token.Punctuator && token.type !== Token.Keyword) {
|
|
return parsePostfixExpression();
|
|
}
|
|
|
|
if (match('++') || match('--')) {
|
|
token = lex();
|
|
expr = parseUnaryExpression();
|
|
// 11.4.4, 11.4.5
|
|
if (strict && expr.type === Syntax.Identifier && isRestrictedWord(expr.name)) {
|
|
throwErrorTolerant({}, Messages.StrictLHSPrefix);
|
|
}
|
|
|
|
if (!isLeftHandSide(expr)) {
|
|
throwErrorTolerant({}, Messages.InvalidLHSInAssignment);
|
|
}
|
|
|
|
expr = {
|
|
type: Syntax.UpdateExpression,
|
|
operator: token.value,
|
|
argument: expr,
|
|
prefix: true
|
|
};
|
|
return expr;
|
|
}
|
|
|
|
if (match('+') || match('-') || match('~') || match('!')) {
|
|
expr = {
|
|
type: Syntax.UnaryExpression,
|
|
operator: lex().value,
|
|
argument: parseUnaryExpression(),
|
|
prefix: true
|
|
};
|
|
return expr;
|
|
}
|
|
|
|
if (matchKeyword('delete') || matchKeyword('void') || matchKeyword('typeof')) {
|
|
expr = {
|
|
type: Syntax.UnaryExpression,
|
|
operator: lex().value,
|
|
argument: parseUnaryExpression(),
|
|
prefix: true
|
|
};
|
|
if (strict && expr.operator === 'delete' && expr.argument.type === Syntax.Identifier) {
|
|
throwErrorTolerant({}, Messages.StrictDelete);
|
|
}
|
|
return expr;
|
|
}
|
|
|
|
return parsePostfixExpression();
|
|
}
|
|
|
|
// 11.5 Multiplicative Operators
|
|
|
|
function parseMultiplicativeExpression() {
|
|
var expr = parseUnaryExpression();
|
|
|
|
while (match('*') || match('/') || match('%')) {
|
|
expr = {
|
|
type: Syntax.BinaryExpression,
|
|
operator: lex().value,
|
|
left: expr,
|
|
right: parseUnaryExpression()
|
|
};
|
|
}
|
|
|
|
return expr;
|
|
}
|
|
|
|
// 11.6 Additive Operators
|
|
|
|
function parseAdditiveExpression() {
|
|
var expr = parseMultiplicativeExpression();
|
|
|
|
while (match('+') || match('-')) {
|
|
expr = {
|
|
type: Syntax.BinaryExpression,
|
|
operator: lex().value,
|
|
left: expr,
|
|
right: parseMultiplicativeExpression()
|
|
};
|
|
}
|
|
|
|
return expr;
|
|
}
|
|
|
|
// 11.7 Bitwise Shift Operators
|
|
|
|
function parseShiftExpression() {
|
|
var expr = parseAdditiveExpression();
|
|
|
|
while (match('<<') || match('>>') || match('>>>')) {
|
|
expr = {
|
|
type: Syntax.BinaryExpression,
|
|
operator: lex().value,
|
|
left: expr,
|
|
right: parseAdditiveExpression()
|
|
};
|
|
}
|
|
|
|
return expr;
|
|
}
|
|
// 11.8 Relational Operators
|
|
|
|
function parseRelationalExpression() {
|
|
var expr, previousAllowIn;
|
|
|
|
previousAllowIn = state.allowIn;
|
|
state.allowIn = true;
|
|
|
|
expr = parseShiftExpression();
|
|
|
|
while (match('<') || match('>') || match('<=') || match('>=') || (previousAllowIn && matchKeyword('in')) || matchKeyword('instanceof')) {
|
|
expr = {
|
|
type: Syntax.BinaryExpression,
|
|
operator: lex().value,
|
|
left: expr,
|
|
right: parseShiftExpression()
|
|
};
|
|
}
|
|
|
|
state.allowIn = previousAllowIn;
|
|
return expr;
|
|
}
|
|
|
|
// 11.9 Equality Operators
|
|
|
|
function parseEqualityExpression() {
|
|
var expr = parseRelationalExpression();
|
|
|
|
while (match('==') || match('!=') || match('===') || match('!==')) {
|
|
expr = {
|
|
type: Syntax.BinaryExpression,
|
|
operator: lex().value,
|
|
left: expr,
|
|
right: parseRelationalExpression()
|
|
};
|
|
}
|
|
|
|
return expr;
|
|
}
|
|
|
|
// 11.10 Binary Bitwise Operators
|
|
|
|
function parseBitwiseANDExpression() {
|
|
var expr = parseEqualityExpression();
|
|
|
|
while (match('&')) {
|
|
lex();
|
|
expr = {
|
|
type: Syntax.BinaryExpression,
|
|
operator: '&',
|
|
left: expr,
|
|
right: parseEqualityExpression()
|
|
};
|
|
}
|
|
|
|
return expr;
|
|
}
|
|
|
|
function parseBitwiseXORExpression() {
|
|
var expr = parseBitwiseANDExpression();
|
|
|
|
while (match('^')) {
|
|
lex();
|
|
expr = {
|
|
type: Syntax.BinaryExpression,
|
|
operator: '^',
|
|
left: expr,
|
|
right: parseBitwiseANDExpression()
|
|
};
|
|
}
|
|
|
|
return expr;
|
|
}
|
|
|
|
function parseBitwiseORExpression() {
|
|
var expr = parseBitwiseXORExpression();
|
|
|
|
while (match('|')) {
|
|
lex();
|
|
expr = {
|
|
type: Syntax.BinaryExpression,
|
|
operator: '|',
|
|
left: expr,
|
|
right: parseBitwiseXORExpression()
|
|
};
|
|
}
|
|
|
|
return expr;
|
|
}
|
|
|
|
// 11.11 Binary Logical Operators
|
|
|
|
function parseLogicalANDExpression() {
|
|
var expr = parseBitwiseORExpression();
|
|
|
|
while (match('&&')) {
|
|
lex();
|
|
expr = {
|
|
type: Syntax.LogicalExpression,
|
|
operator: '&&',
|
|
left: expr,
|
|
right: parseBitwiseORExpression()
|
|
};
|
|
}
|
|
|
|
return expr;
|
|
}
|
|
|
|
function parseLogicalORExpression() {
|
|
var expr = parseLogicalANDExpression();
|
|
|
|
while (match('||')) {
|
|
lex();
|
|
expr = {
|
|
type: Syntax.LogicalExpression,
|
|
operator: '||',
|
|
left: expr,
|
|
right: parseLogicalANDExpression()
|
|
};
|
|
}
|
|
|
|
return expr;
|
|
}
|
|
|
|
// 11.12 Conditional Operator
|
|
|
|
function parseConditionalExpression() {
|
|
var expr, previousAllowIn, consequent;
|
|
|
|
expr = parseLogicalORExpression();
|
|
|
|
if (match('?')) {
|
|
lex();
|
|
previousAllowIn = state.allowIn;
|
|
state.allowIn = true;
|
|
consequent = parseAssignmentExpression();
|
|
state.allowIn = previousAllowIn;
|
|
expect(':');
|
|
|
|
expr = {
|
|
type: Syntax.ConditionalExpression,
|
|
test: expr,
|
|
consequent: consequent,
|
|
alternate: parseAssignmentExpression()
|
|
};
|
|
}
|
|
|
|
return expr;
|
|
}
|
|
|
|
// 11.13 Assignment Operators
|
|
|
|
function parseAssignmentExpression() {
|
|
var token, expr;
|
|
|
|
token = lookahead();
|
|
expr = parseConditionalExpression();
|
|
|
|
if (matchAssign()) {
|
|
// LeftHandSideExpression
|
|
if (!isLeftHandSide(expr)) {
|
|
throwErrorTolerant({}, Messages.InvalidLHSInAssignment);
|
|
}
|
|
|
|
// 11.13.1
|
|
if (strict && expr.type === Syntax.Identifier && isRestrictedWord(expr.name)) {
|
|
throwErrorTolerant(token, Messages.StrictLHSAssignment);
|
|
}
|
|
|
|
expr = {
|
|
type: Syntax.AssignmentExpression,
|
|
operator: lex().value,
|
|
left: expr,
|
|
right: parseAssignmentExpression()
|
|
};
|
|
}
|
|
|
|
return expr;
|
|
}
|
|
|
|
// 11.14 Comma Operator
|
|
|
|
function parseExpression() {
|
|
var expr = parseAssignmentExpression();
|
|
|
|
if (match(',')) {
|
|
expr = {
|
|
type: Syntax.SequenceExpression,
|
|
expressions: [ expr ]
|
|
};
|
|
|
|
while (index < length) {
|
|
if (!match(',')) {
|
|
break;
|
|
}
|
|
lex();
|
|
expr.expressions.push(parseAssignmentExpression());
|
|
}
|
|
|
|
}
|
|
return expr;
|
|
}
|
|
|
|
// 12.1 Block
|
|
|
|
function parseStatementList() {
|
|
var list = [],
|
|
statement;
|
|
|
|
while (index < length) {
|
|
if (match('}')) {
|
|
break;
|
|
}
|
|
statement = parseSourceElement();
|
|
if (typeof statement === 'undefined') {
|
|
break;
|
|
}
|
|
list.push(statement);
|
|
}
|
|
|
|
return list;
|
|
}
|
|
|
|
function parseBlock() {
|
|
var block;
|
|
|
|
expect('{');
|
|
|
|
block = parseStatementList();
|
|
|
|
expect('}');
|
|
|
|
return {
|
|
type: Syntax.BlockStatement,
|
|
body: block
|
|
};
|
|
}
|
|
|
|
// 12.2 Variable Statement
|
|
|
|
function parseVariableIdentifier() {
|
|
var token = lex();
|
|
|
|
if (token.type !== Token.Identifier) {
|
|
throwUnexpected(token);
|
|
}
|
|
|
|
return {
|
|
type: Syntax.Identifier,
|
|
name: token.value
|
|
};
|
|
}
|
|
|
|
function parseVariableDeclaration(kind) {
|
|
var id = parseVariableIdentifier(),
|
|
init = null;
|
|
|
|
// 12.2.1
|
|
if (strict && isRestrictedWord(id.name)) {
|
|
throwErrorTolerant({}, Messages.StrictVarName);
|
|
}
|
|
|
|
if (kind === 'const') {
|
|
expect('=');
|
|
init = parseAssignmentExpression();
|
|
} else if (match('=')) {
|
|
lex();
|
|
init = parseAssignmentExpression();
|
|
}
|
|
|
|
return {
|
|
type: Syntax.VariableDeclarator,
|
|
id: id,
|
|
init: init
|
|
};
|
|
}
|
|
|
|
function parseVariableDeclarationList(kind) {
|
|
var list = [];
|
|
|
|
do {
|
|
list.push(parseVariableDeclaration(kind));
|
|
if (!match(',')) {
|
|
break;
|
|
}
|
|
lex();
|
|
} while (index < length);
|
|
|
|
return list;
|
|
}
|
|
|
|
function parseVariableStatement() {
|
|
var declarations;
|
|
|
|
expectKeyword('var');
|
|
|
|
declarations = parseVariableDeclarationList();
|
|
|
|
consumeSemicolon();
|
|
|
|
return {
|
|
type: Syntax.VariableDeclaration,
|
|
declarations: declarations,
|
|
kind: 'var'
|
|
};
|
|
}
|
|
|
|
// kind may be `const` or `let`
|
|
// Both are experimental and not in the specification yet.
|
|
// see http://wiki.ecmascript.org/doku.php?id=harmony:const
|
|
// and http://wiki.ecmascript.org/doku.php?id=harmony:let
|
|
function parseConstLetDeclaration(kind) {
|
|
var declarations;
|
|
|
|
expectKeyword(kind);
|
|
|
|
declarations = parseVariableDeclarationList(kind);
|
|
|
|
consumeSemicolon();
|
|
|
|
return {
|
|
type: Syntax.VariableDeclaration,
|
|
declarations: declarations,
|
|
kind: kind
|
|
};
|
|
}
|
|
|
|
// 12.3 Empty Statement
|
|
|
|
function parseEmptyStatement() {
|
|
expect(';');
|
|
|
|
return {
|
|
type: Syntax.EmptyStatement
|
|
};
|
|
}
|
|
|
|
// 12.4 Expression Statement
|
|
|
|
function parseExpressionStatement() {
|
|
var expr = parseExpression();
|
|
|
|
consumeSemicolon();
|
|
|
|
return {
|
|
type: Syntax.ExpressionStatement,
|
|
expression: expr
|
|
};
|
|
}
|
|
|
|
// 12.5 If statement
|
|
|
|
function parseIfStatement() {
|
|
var test, consequent, alternate;
|
|
|
|
expectKeyword('if');
|
|
|
|
expect('(');
|
|
|
|
test = parseExpression();
|
|
|
|
expect(')');
|
|
|
|
consequent = parseStatement();
|
|
|
|
if (matchKeyword('else')) {
|
|
lex();
|
|
alternate = parseStatement();
|
|
} else {
|
|
alternate = null;
|
|
}
|
|
|
|
return {
|
|
type: Syntax.IfStatement,
|
|
test: test,
|
|
consequent: consequent,
|
|
alternate: alternate
|
|
};
|
|
}
|
|
|
|
// 12.6 Iteration Statements
|
|
|
|
function parseDoWhileStatement() {
|
|
var body, test, oldInIteration;
|
|
|
|
expectKeyword('do');
|
|
|
|
oldInIteration = state.inIteration;
|
|
state.inIteration = true;
|
|
|
|
body = parseStatement();
|
|
|
|
state.inIteration = oldInIteration;
|
|
|
|
expectKeyword('while');
|
|
|
|
expect('(');
|
|
|
|
test = parseExpression();
|
|
|
|
expect(')');
|
|
|
|
if (match(';')) {
|
|
lex();
|
|
}
|
|
|
|
return {
|
|
type: Syntax.DoWhileStatement,
|
|
body: body,
|
|
test: test
|
|
};
|
|
}
|
|
|
|
function parseWhileStatement() {
|
|
var test, body, oldInIteration;
|
|
|
|
expectKeyword('while');
|
|
|
|
expect('(');
|
|
|
|
test = parseExpression();
|
|
|
|
expect(')');
|
|
|
|
oldInIteration = state.inIteration;
|
|
state.inIteration = true;
|
|
|
|
body = parseStatement();
|
|
|
|
state.inIteration = oldInIteration;
|
|
|
|
return {
|
|
type: Syntax.WhileStatement,
|
|
test: test,
|
|
body: body
|
|
};
|
|
}
|
|
|
|
function parseForVariableDeclaration() {
|
|
var token = lex();
|
|
|
|
return {
|
|
type: Syntax.VariableDeclaration,
|
|
declarations: parseVariableDeclarationList(),
|
|
kind: token.value
|
|
};
|
|
}
|
|
|
|
function parseForStatement() {
|
|
var init, test, update, left, right, body, oldInIteration;
|
|
|
|
init = test = update = null;
|
|
|
|
expectKeyword('for');
|
|
|
|
expect('(');
|
|
|
|
if (match(';')) {
|
|
lex();
|
|
} else {
|
|
if (matchKeyword('var') || matchKeyword('let')) {
|
|
state.allowIn = false;
|
|
init = parseForVariableDeclaration();
|
|
state.allowIn = true;
|
|
|
|
if (init.declarations.length === 1 && matchKeyword('in')) {
|
|
lex();
|
|
left = init;
|
|
right = parseExpression();
|
|
init = null;
|
|
}
|
|
} else {
|
|
state.allowIn = false;
|
|
init = parseExpression();
|
|
state.allowIn = true;
|
|
|
|
if (matchKeyword('in')) {
|
|
// LeftHandSideExpression
|
|
if (!isLeftHandSide(init)) {
|
|
throwErrorTolerant({}, Messages.InvalidLHSInForIn);
|
|
}
|
|
|
|
lex();
|
|
left = init;
|
|
right = parseExpression();
|
|
init = null;
|
|
}
|
|
}
|
|
|
|
if (typeof left === 'undefined') {
|
|
expect(';');
|
|
}
|
|
}
|
|
|
|
if (typeof left === 'undefined') {
|
|
|
|
if (!match(';')) {
|
|
test = parseExpression();
|
|
}
|
|
expect(';');
|
|
|
|
if (!match(')')) {
|
|
update = parseExpression();
|
|
}
|
|
}
|
|
|
|
expect(')');
|
|
|
|
oldInIteration = state.inIteration;
|
|
state.inIteration = true;
|
|
|
|
body = parseStatement();
|
|
|
|
state.inIteration = oldInIteration;
|
|
|
|
if (typeof left === 'undefined') {
|
|
return {
|
|
type: Syntax.ForStatement,
|
|
init: init,
|
|
test: test,
|
|
update: update,
|
|
body: body
|
|
};
|
|
}
|
|
|
|
return {
|
|
type: Syntax.ForInStatement,
|
|
left: left,
|
|
right: right,
|
|
body: body,
|
|
each: false
|
|
};
|
|
}
|
|
|
|
// 12.7 The continue statement
|
|
|
|
function parseContinueStatement() {
|
|
var token, label = null;
|
|
|
|
expectKeyword('continue');
|
|
|
|
// Optimize the most common form: 'continue;'.
|
|
if (source[index] === ';') {
|
|
lex();
|
|
|
|
if (!state.inIteration) {
|
|
throwError({}, Messages.IllegalContinue);
|
|
}
|
|
|
|
return {
|
|
type: Syntax.ContinueStatement,
|
|
label: null
|
|
};
|
|
}
|
|
|
|
if (peekLineTerminator()) {
|
|
if (!state.inIteration) {
|
|
throwError({}, Messages.IllegalContinue);
|
|
}
|
|
|
|
return {
|
|
type: Syntax.ContinueStatement,
|
|
label: null
|
|
};
|
|
}
|
|
|
|
token = lookahead();
|
|
if (token.type === Token.Identifier) {
|
|
label = parseVariableIdentifier();
|
|
|
|
if (!Object.prototype.hasOwnProperty.call(state.labelSet, label.name)) {
|
|
throwError({}, Messages.UnknownLabel, label.name);
|
|
}
|
|
}
|
|
|
|
consumeSemicolon();
|
|
|
|
if (label === null && !state.inIteration) {
|
|
throwError({}, Messages.IllegalContinue);
|
|
}
|
|
|
|
return {
|
|
type: Syntax.ContinueStatement,
|
|
label: label
|
|
};
|
|
}
|
|
|
|
// 12.8 The break statement
|
|
|
|
function parseBreakStatement() {
|
|
var token, label = null;
|
|
|
|
expectKeyword('break');
|
|
|
|
// Optimize the most common form: 'break;'.
|
|
if (source[index] === ';') {
|
|
lex();
|
|
|
|
if (!(state.inIteration || state.inSwitch)) {
|
|
throwError({}, Messages.IllegalBreak);
|
|
}
|
|
|
|
return {
|
|
type: Syntax.BreakStatement,
|
|
label: null
|
|
};
|
|
}
|
|
|
|
if (peekLineTerminator()) {
|
|
if (!(state.inIteration || state.inSwitch)) {
|
|
throwError({}, Messages.IllegalBreak);
|
|
}
|
|
|
|
return {
|
|
type: Syntax.BreakStatement,
|
|
label: null
|
|
};
|
|
}
|
|
|
|
token = lookahead();
|
|
if (token.type === Token.Identifier) {
|
|
label = parseVariableIdentifier();
|
|
|
|
if (!Object.prototype.hasOwnProperty.call(state.labelSet, label.name)) {
|
|
throwError({}, Messages.UnknownLabel, label.name);
|
|
}
|
|
}
|
|
|
|
consumeSemicolon();
|
|
|
|
if (label === null && !(state.inIteration || state.inSwitch)) {
|
|
throwError({}, Messages.IllegalBreak);
|
|
}
|
|
|
|
return {
|
|
type: Syntax.BreakStatement,
|
|
label: label
|
|
};
|
|
}
|
|
|
|
// 12.9 The return statement
|
|
|
|
function parseReturnStatement() {
|
|
var token, argument = null;
|
|
|
|
expectKeyword('return');
|
|
|
|
if (!state.inFunctionBody) {
|
|
throwErrorTolerant({}, Messages.IllegalReturn);
|
|
}
|
|
|
|
// 'return' followed by a space and an identifier is very common.
|
|
if (source[index] === ' ') {
|
|
if (isIdentifierStart(source[index + 1])) {
|
|
argument = parseExpression();
|
|
consumeSemicolon();
|
|
return {
|
|
type: Syntax.ReturnStatement,
|
|
argument: argument
|
|
};
|
|
}
|
|
}
|
|
|
|
if (peekLineTerminator()) {
|
|
return {
|
|
type: Syntax.ReturnStatement,
|
|
argument: null
|
|
};
|
|
}
|
|
|
|
if (!match(';')) {
|
|
token = lookahead();
|
|
if (!match('}') && token.type !== Token.EOF) {
|
|
argument = parseExpression();
|
|
}
|
|
}
|
|
|
|
consumeSemicolon();
|
|
|
|
return {
|
|
type: Syntax.ReturnStatement,
|
|
argument: argument
|
|
};
|
|
}
|
|
|
|
// 12.10 The with statement
|
|
|
|
function parseWithStatement() {
|
|
var object, body;
|
|
|
|
if (strict) {
|
|
throwErrorTolerant({}, Messages.StrictModeWith);
|
|
}
|
|
|
|
expectKeyword('with');
|
|
|
|
expect('(');
|
|
|
|
object = parseExpression();
|
|
|
|
expect(')');
|
|
|
|
body = parseStatement();
|
|
|
|
return {
|
|
type: Syntax.WithStatement,
|
|
object: object,
|
|
body: body
|
|
};
|
|
}
|
|
|
|
// 12.10 The swith statement
|
|
|
|
function parseSwitchCase() {
|
|
var test,
|
|
consequent = [],
|
|
statement;
|
|
|
|
if (matchKeyword('default')) {
|
|
lex();
|
|
test = null;
|
|
} else {
|
|
expectKeyword('case');
|
|
test = parseExpression();
|
|
}
|
|
expect(':');
|
|
|
|
while (index < length) {
|
|
if (match('}') || matchKeyword('default') || matchKeyword('case')) {
|
|
break;
|
|
}
|
|
statement = parseStatement();
|
|
if (typeof statement === 'undefined') {
|
|
break;
|
|
}
|
|
consequent.push(statement);
|
|
}
|
|
|
|
return {
|
|
type: Syntax.SwitchCase,
|
|
test: test,
|
|
consequent: consequent
|
|
};
|
|
}
|
|
|
|
function parseSwitchStatement() {
|
|
var discriminant, cases, clause, oldInSwitch, defaultFound;
|
|
|
|
expectKeyword('switch');
|
|
|
|
expect('(');
|
|
|
|
discriminant = parseExpression();
|
|
|
|
expect(')');
|
|
|
|
expect('{');
|
|
|
|
cases = [];
|
|
|
|
if (match('}')) {
|
|
lex();
|
|
return {
|
|
type: Syntax.SwitchStatement,
|
|
discriminant: discriminant,
|
|
cases: cases
|
|
};
|
|
}
|
|
|
|
oldInSwitch = state.inSwitch;
|
|
state.inSwitch = true;
|
|
defaultFound = false;
|
|
|
|
while (index < length) {
|
|
if (match('}')) {
|
|
break;
|
|
}
|
|
clause = parseSwitchCase();
|
|
if (clause.test === null) {
|
|
if (defaultFound) {
|
|
throwError({}, Messages.MultipleDefaultsInSwitch);
|
|
}
|
|
defaultFound = true;
|
|
}
|
|
cases.push(clause);
|
|
}
|
|
|
|
state.inSwitch = oldInSwitch;
|
|
|
|
expect('}');
|
|
|
|
return {
|
|
type: Syntax.SwitchStatement,
|
|
discriminant: discriminant,
|
|
cases: cases
|
|
};
|
|
}
|
|
|
|
// 12.13 The throw statement
|
|
|
|
function parseThrowStatement() {
|
|
var argument;
|
|
|
|
expectKeyword('throw');
|
|
|
|
if (peekLineTerminator()) {
|
|
throwError({}, Messages.NewlineAfterThrow);
|
|
}
|
|
|
|
argument = parseExpression();
|
|
|
|
consumeSemicolon();
|
|
|
|
return {
|
|
type: Syntax.ThrowStatement,
|
|
argument: argument
|
|
};
|
|
}
|
|
|
|
// 12.14 The try statement
|
|
|
|
function parseCatchClause() {
|
|
var param;
|
|
|
|
expectKeyword('catch');
|
|
|
|
expect('(');
|
|
if (match(')')) {
|
|
throwUnexpected(lookahead());
|
|
}
|
|
|
|
param = parseVariableIdentifier();
|
|
// 12.14.1
|
|
if (strict && isRestrictedWord(param.name)) {
|
|
throwErrorTolerant({}, Messages.StrictCatchVariable);
|
|
}
|
|
|
|
expect(')');
|
|
|
|
return {
|
|
type: Syntax.CatchClause,
|
|
param: param,
|
|
body: parseBlock()
|
|
};
|
|
}
|
|
|
|
function parseTryStatement() {
|
|
var block, handlers = [], finalizer = null;
|
|
|
|
expectKeyword('try');
|
|
|
|
block = parseBlock();
|
|
|
|
if (matchKeyword('catch')) {
|
|
handlers.push(parseCatchClause());
|
|
}
|
|
|
|
if (matchKeyword('finally')) {
|
|
lex();
|
|
finalizer = parseBlock();
|
|
}
|
|
|
|
if (handlers.length === 0 && !finalizer) {
|
|
throwError({}, Messages.NoCatchOrFinally);
|
|
}
|
|
|
|
return {
|
|
type: Syntax.TryStatement,
|
|
block: block,
|
|
guardedHandlers: [],
|
|
handlers: handlers,
|
|
finalizer: finalizer
|
|
};
|
|
}
|
|
|
|
// 12.15 The debugger statement
|
|
|
|
function parseDebuggerStatement() {
|
|
expectKeyword('debugger');
|
|
|
|
consumeSemicolon();
|
|
|
|
return {
|
|
type: Syntax.DebuggerStatement
|
|
};
|
|
}
|
|
|
|
// 12 Statements
|
|
|
|
function parseStatement() {
|
|
var token = lookahead(),
|
|
expr,
|
|
labeledBody;
|
|
|
|
if (token.type === Token.EOF) {
|
|
throwUnexpected(token);
|
|
}
|
|
|
|
if (token.type === Token.Punctuator) {
|
|
switch (token.value) {
|
|
case ';':
|
|
return parseEmptyStatement();
|
|
case '{':
|
|
return parseBlock();
|
|
case '(':
|
|
return parseExpressionStatement();
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (token.type === Token.Keyword) {
|
|
switch (token.value) {
|
|
case 'break':
|
|
return parseBreakStatement();
|
|
case 'continue':
|
|
return parseContinueStatement();
|
|
case 'debugger':
|
|
return parseDebuggerStatement();
|
|
case 'do':
|
|
return parseDoWhileStatement();
|
|
case 'for':
|
|
return parseForStatement();
|
|
case 'function':
|
|
return parseFunctionDeclaration();
|
|
case 'if':
|
|
return parseIfStatement();
|
|
case 'return':
|
|
return parseReturnStatement();
|
|
case 'switch':
|
|
return parseSwitchStatement();
|
|
case 'throw':
|
|
return parseThrowStatement();
|
|
case 'try':
|
|
return parseTryStatement();
|
|
case 'var':
|
|
return parseVariableStatement();
|
|
case 'while':
|
|
return parseWhileStatement();
|
|
case 'with':
|
|
return parseWithStatement();
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
expr = parseExpression();
|
|
|
|
// 12.12 Labelled Statements
|
|
if ((expr.type === Syntax.Identifier) && match(':')) {
|
|
lex();
|
|
|
|
if (Object.prototype.hasOwnProperty.call(state.labelSet, expr.name)) {
|
|
throwError({}, Messages.Redeclaration, 'Label', expr.name);
|
|
}
|
|
|
|
state.labelSet[expr.name] = true;
|
|
labeledBody = parseStatement();
|
|
delete state.labelSet[expr.name];
|
|
|
|
return {
|
|
type: Syntax.LabeledStatement,
|
|
label: expr,
|
|
body: labeledBody
|
|
};
|
|
}
|
|
|
|
consumeSemicolon();
|
|
|
|
return {
|
|
type: Syntax.ExpressionStatement,
|
|
expression: expr
|
|
};
|
|
}
|
|
|
|
// 13 Function Definition
|
|
|
|
function parseFunctionSourceElements() {
|
|
var sourceElement, sourceElements = [], token, directive, firstRestricted,
|
|
oldLabelSet, oldInIteration, oldInSwitch, oldInFunctionBody;
|
|
|
|
expect('{');
|
|
|
|
while (index < length) {
|
|
token = lookahead();
|
|
if (token.type !== Token.StringLiteral) {
|
|
break;
|
|
}
|
|
|
|
sourceElement = parseSourceElement();
|
|
sourceElements.push(sourceElement);
|
|
if (sourceElement.expression.type !== Syntax.Literal) {
|
|
// this is not directive
|
|
break;
|
|
}
|
|
directive = sliceSource(token.range[0] + 1, token.range[1] - 1);
|
|
if (directive === 'use strict') {
|
|
strict = true;
|
|
if (firstRestricted) {
|
|
throwErrorTolerant(firstRestricted, Messages.StrictOctalLiteral);
|
|
}
|
|
} else {
|
|
if (!firstRestricted && token.octal) {
|
|
firstRestricted = token;
|
|
}
|
|
}
|
|
}
|
|
|
|
oldLabelSet = state.labelSet;
|
|
oldInIteration = state.inIteration;
|
|
oldInSwitch = state.inSwitch;
|
|
oldInFunctionBody = state.inFunctionBody;
|
|
|
|
state.labelSet = {};
|
|
state.inIteration = false;
|
|
state.inSwitch = false;
|
|
state.inFunctionBody = true;
|
|
|
|
while (index < length) {
|
|
if (match('}')) {
|
|
break;
|
|
}
|
|
sourceElement = parseSourceElement();
|
|
if (typeof sourceElement === 'undefined') {
|
|
break;
|
|
}
|
|
sourceElements.push(sourceElement);
|
|
}
|
|
|
|
expect('}');
|
|
|
|
state.labelSet = oldLabelSet;
|
|
state.inIteration = oldInIteration;
|
|
state.inSwitch = oldInSwitch;
|
|
state.inFunctionBody = oldInFunctionBody;
|
|
|
|
return {
|
|
type: Syntax.BlockStatement,
|
|
body: sourceElements
|
|
};
|
|
}
|
|
|
|
function parseFunctionDeclaration() {
|
|
var id, param, params = [], body, token, stricted, firstRestricted, message, previousStrict, paramSet;
|
|
|
|
expectKeyword('function');
|
|
token = lookahead();
|
|
id = parseVariableIdentifier();
|
|
if (strict) {
|
|
if (isRestrictedWord(token.value)) {
|
|
throwErrorTolerant(token, Messages.StrictFunctionName);
|
|
}
|
|
} else {
|
|
if (isRestrictedWord(token.value)) {
|
|
firstRestricted = token;
|
|
message = Messages.StrictFunctionName;
|
|
} else if (isStrictModeReservedWord(token.value)) {
|
|
firstRestricted = token;
|
|
message = Messages.StrictReservedWord;
|
|
}
|
|
}
|
|
|
|
expect('(');
|
|
|
|
if (!match(')')) {
|
|
paramSet = {};
|
|
while (index < length) {
|
|
token = lookahead();
|
|
param = parseVariableIdentifier();
|
|
if (strict) {
|
|
if (isRestrictedWord(token.value)) {
|
|
stricted = token;
|
|
message = Messages.StrictParamName;
|
|
}
|
|
if (Object.prototype.hasOwnProperty.call(paramSet, token.value)) {
|
|
stricted = token;
|
|
message = Messages.StrictParamDupe;
|
|
}
|
|
} else if (!firstRestricted) {
|
|
if (isRestrictedWord(token.value)) {
|
|
firstRestricted = token;
|
|
message = Messages.StrictParamName;
|
|
} else if (isStrictModeReservedWord(token.value)) {
|
|
firstRestricted = token;
|
|
message = Messages.StrictReservedWord;
|
|
} else if (Object.prototype.hasOwnProperty.call(paramSet, token.value)) {
|
|
firstRestricted = token;
|
|
message = Messages.StrictParamDupe;
|
|
}
|
|
}
|
|
params.push(param);
|
|
paramSet[param.name] = true;
|
|
if (match(')')) {
|
|
break;
|
|
}
|
|
expect(',');
|
|
}
|
|
}
|
|
|
|
expect(')');
|
|
|
|
previousStrict = strict;
|
|
body = parseFunctionSourceElements();
|
|
if (strict && firstRestricted) {
|
|
throwError(firstRestricted, message);
|
|
}
|
|
if (strict && stricted) {
|
|
throwErrorTolerant(stricted, message);
|
|
}
|
|
strict = previousStrict;
|
|
|
|
return {
|
|
type: Syntax.FunctionDeclaration,
|
|
id: id,
|
|
params: params,
|
|
defaults: [],
|
|
body: body,
|
|
rest: null,
|
|
generator: false,
|
|
expression: false
|
|
};
|
|
}
|
|
|
|
function parseFunctionExpression() {
|
|
var token, id = null, stricted, firstRestricted, message, param, params = [], body, previousStrict, paramSet;
|
|
|
|
expectKeyword('function');
|
|
|
|
if (!match('(')) {
|
|
token = lookahead();
|
|
id = parseVariableIdentifier();
|
|
if (strict) {
|
|
if (isRestrictedWord(token.value)) {
|
|
throwErrorTolerant(token, Messages.StrictFunctionName);
|
|
}
|
|
} else {
|
|
if (isRestrictedWord(token.value)) {
|
|
firstRestricted = token;
|
|
message = Messages.StrictFunctionName;
|
|
} else if (isStrictModeReservedWord(token.value)) {
|
|
firstRestricted = token;
|
|
message = Messages.StrictReservedWord;
|
|
}
|
|
}
|
|
}
|
|
|
|
expect('(');
|
|
|
|
if (!match(')')) {
|
|
paramSet = {};
|
|
while (index < length) {
|
|
token = lookahead();
|
|
param = parseVariableIdentifier();
|
|
if (strict) {
|
|
if (isRestrictedWord(token.value)) {
|
|
stricted = token;
|
|
message = Messages.StrictParamName;
|
|
}
|
|
if (Object.prototype.hasOwnProperty.call(paramSet, token.value)) {
|
|
stricted = token;
|
|
message = Messages.StrictParamDupe;
|
|
}
|
|
} else if (!firstRestricted) {
|
|
if (isRestrictedWord(token.value)) {
|
|
firstRestricted = token;
|
|
message = Messages.StrictParamName;
|
|
} else if (isStrictModeReservedWord(token.value)) {
|
|
firstRestricted = token;
|
|
message = Messages.StrictReservedWord;
|
|
} else if (Object.prototype.hasOwnProperty.call(paramSet, token.value)) {
|
|
firstRestricted = token;
|
|
message = Messages.StrictParamDupe;
|
|
}
|
|
}
|
|
params.push(param);
|
|
paramSet[param.name] = true;
|
|
if (match(')')) {
|
|
break;
|
|
}
|
|
expect(',');
|
|
}
|
|
}
|
|
|
|
expect(')');
|
|
|
|
previousStrict = strict;
|
|
body = parseFunctionSourceElements();
|
|
if (strict && firstRestricted) {
|
|
throwError(firstRestricted, message);
|
|
}
|
|
if (strict && stricted) {
|
|
throwErrorTolerant(stricted, message);
|
|
}
|
|
strict = previousStrict;
|
|
|
|
return {
|
|
type: Syntax.FunctionExpression,
|
|
id: id,
|
|
params: params,
|
|
defaults: [],
|
|
body: body,
|
|
rest: null,
|
|
generator: false,
|
|
expression: false
|
|
};
|
|
}
|
|
|
|
// 14 Program
|
|
|
|
function parseSourceElement() {
|
|
var token = lookahead();
|
|
|
|
if (token.type === Token.Keyword) {
|
|
switch (token.value) {
|
|
case 'const':
|
|
case 'let':
|
|
return parseConstLetDeclaration(token.value);
|
|
case 'function':
|
|
return parseFunctionDeclaration();
|
|
default:
|
|
return parseStatement();
|
|
}
|
|
}
|
|
|
|
if (token.type !== Token.EOF) {
|
|
return parseStatement();
|
|
}
|
|
}
|
|
|
|
function parseSourceElements() {
|
|
var sourceElement, sourceElements = [], token, directive, firstRestricted;
|
|
|
|
while (index < length) {
|
|
token = lookahead();
|
|
if (token.type !== Token.StringLiteral) {
|
|
break;
|
|
}
|
|
|
|
sourceElement = parseSourceElement();
|
|
sourceElements.push(sourceElement);
|
|
if (sourceElement.expression.type !== Syntax.Literal) {
|
|
// this is not directive
|
|
break;
|
|
}
|
|
directive = sliceSource(token.range[0] + 1, token.range[1] - 1);
|
|
if (directive === 'use strict') {
|
|
strict = true;
|
|
if (firstRestricted) {
|
|
throwErrorTolerant(firstRestricted, Messages.StrictOctalLiteral);
|
|
}
|
|
} else {
|
|
if (!firstRestricted && token.octal) {
|
|
firstRestricted = token;
|
|
}
|
|
}
|
|
}
|
|
|
|
while (index < length) {
|
|
sourceElement = parseSourceElement();
|
|
if (typeof sourceElement === 'undefined') {
|
|
break;
|
|
}
|
|
sourceElements.push(sourceElement);
|
|
}
|
|
return sourceElements;
|
|
}
|
|
|
|
function parseProgram() {
|
|
var program;
|
|
strict = false;
|
|
program = {
|
|
type: Syntax.Program,
|
|
body: parseSourceElements()
|
|
};
|
|
return program;
|
|
}
|
|
|
|
// The following functions are needed only when the option to preserve
|
|
// the comments is active.
|
|
|
|
function addComment(type, value, start, end, loc) {
|
|
assert(typeof start === 'number', 'Comment must have valid position');
|
|
|
|
// Because the way the actual token is scanned, often the comments
|
|
// (if any) are skipped twice during the lexical analysis.
|
|
// Thus, we need to skip adding a comment if the comment array already
|
|
// handled it.
|
|
if (extra.comments.length > 0) {
|
|
if (extra.comments[extra.comments.length - 1].range[1] > start) {
|
|
return;
|
|
}
|
|
}
|
|
|
|
extra.comments.push({
|
|
type: type,
|
|
value: value,
|
|
range: [start, end],
|
|
loc: loc
|
|
});
|
|
}
|
|
|
|
function scanComment() {
|
|
var comment, ch, loc, start, blockComment, lineComment;
|
|
|
|
comment = '';
|
|
blockComment = false;
|
|
lineComment = false;
|
|
|
|
while (index < length) {
|
|
ch = source[index];
|
|
|
|
if (lineComment) {
|
|
ch = source[index++];
|
|
if (isLineTerminator(ch)) {
|
|
loc.end = {
|
|
line: lineNumber,
|
|
column: index - lineStart - 1
|
|
};
|
|
lineComment = false;
|
|
addComment('Line', comment, start, index - 1, loc);
|
|
if (ch === '\r' && source[index] === '\n') {
|
|
++index;
|
|
}
|
|
++lineNumber;
|
|
lineStart = index;
|
|
comment = '';
|
|
} else if (index >= length) {
|
|
lineComment = false;
|
|
comment += ch;
|
|
loc.end = {
|
|
line: lineNumber,
|
|
column: length - lineStart
|
|
};
|
|
addComment('Line', comment, start, length, loc);
|
|
} else {
|
|
comment += ch;
|
|
}
|
|
} else if (blockComment) {
|
|
if (isLineTerminator(ch)) {
|
|
if (ch === '\r' && source[index + 1] === '\n') {
|
|
++index;
|
|
comment += '\r\n';
|
|
} else {
|
|
comment += ch;
|
|
}
|
|
++lineNumber;
|
|
++index;
|
|
lineStart = index;
|
|
if (index >= length) {
|
|
throwError({}, Messages.UnexpectedToken, 'ILLEGAL');
|
|
}
|
|
} else {
|
|
ch = source[index++];
|
|
if (index >= length) {
|
|
throwError({}, Messages.UnexpectedToken, 'ILLEGAL');
|
|
}
|
|
comment += ch;
|
|
if (ch === '*') {
|
|
ch = source[index];
|
|
if (ch === '/') {
|
|
comment = comment.substr(0, comment.length - 1);
|
|
blockComment = false;
|
|
++index;
|
|
loc.end = {
|
|
line: lineNumber,
|
|
column: index - lineStart
|
|
};
|
|
addComment('Block', comment, start, index, loc);
|
|
comment = '';
|
|
}
|
|
}
|
|
}
|
|
} else if (ch === '/') {
|
|
ch = source[index + 1];
|
|
if (ch === '/') {
|
|
loc = {
|
|
start: {
|
|
line: lineNumber,
|
|
column: index - lineStart
|
|
}
|
|
};
|
|
start = index;
|
|
index += 2;
|
|
lineComment = true;
|
|
if (index >= length) {
|
|
loc.end = {
|
|
line: lineNumber,
|
|
column: index - lineStart
|
|
};
|
|
lineComment = false;
|
|
addComment('Line', comment, start, index, loc);
|
|
}
|
|
} else if (ch === '*') {
|
|
start = index;
|
|
index += 2;
|
|
blockComment = true;
|
|
loc = {
|
|
start: {
|
|
line: lineNumber,
|
|
column: index - lineStart - 2
|
|
}
|
|
};
|
|
if (index >= length) {
|
|
throwError({}, Messages.UnexpectedToken, 'ILLEGAL');
|
|
}
|
|
} else {
|
|
break;
|
|
}
|
|
} else if (isWhiteSpace(ch)) {
|
|
++index;
|
|
} else if (isLineTerminator(ch)) {
|
|
++index;
|
|
if (ch === '\r' && source[index] === '\n') {
|
|
++index;
|
|
}
|
|
++lineNumber;
|
|
lineStart = index;
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
function filterCommentLocation() {
|
|
var i, entry, comment, comments = [];
|
|
|
|
for (i = 0; i < extra.comments.length; ++i) {
|
|
entry = extra.comments[i];
|
|
comment = {
|
|
type: entry.type,
|
|
value: entry.value
|
|
};
|
|
if (extra.range) {
|
|
comment.range = entry.range;
|
|
}
|
|
if (extra.loc) {
|
|
comment.loc = entry.loc;
|
|
}
|
|
comments.push(comment);
|
|
}
|
|
|
|
extra.comments = comments;
|
|
}
|
|
|
|
function collectToken() {
|
|
var start, loc, token, range, value;
|
|
|
|
skipComment();
|
|
start = index;
|
|
loc = {
|
|
start: {
|
|
line: lineNumber,
|
|
column: index - lineStart
|
|
}
|
|
};
|
|
|
|
token = extra.advance();
|
|
loc.end = {
|
|
line: lineNumber,
|
|
column: index - lineStart
|
|
};
|
|
|
|
if (token.type !== Token.EOF) {
|
|
range = [token.range[0], token.range[1]];
|
|
value = sliceSource(token.range[0], token.range[1]);
|
|
extra.tokens.push({
|
|
type: TokenName[token.type],
|
|
value: value,
|
|
range: range,
|
|
loc: loc
|
|
});
|
|
}
|
|
|
|
return token;
|
|
}
|
|
|
|
function collectRegex() {
|
|
var pos, loc, regex, token;
|
|
|
|
skipComment();
|
|
|
|
pos = index;
|
|
loc = {
|
|
start: {
|
|
line: lineNumber,
|
|
column: index - lineStart
|
|
}
|
|
};
|
|
|
|
regex = extra.scanRegExp();
|
|
loc.end = {
|
|
line: lineNumber,
|
|
column: index - lineStart
|
|
};
|
|
|
|
// Pop the previous token, which is likely '/' or '/='
|
|
if (extra.tokens.length > 0) {
|
|
token = extra.tokens[extra.tokens.length - 1];
|
|
if (token.range[0] === pos && token.type === 'Punctuator') {
|
|
if (token.value === '/' || token.value === '/=') {
|
|
extra.tokens.pop();
|
|
}
|
|
}
|
|
}
|
|
|
|
extra.tokens.push({
|
|
type: 'RegularExpression',
|
|
value: regex.literal,
|
|
range: [pos, index],
|
|
loc: loc
|
|
});
|
|
|
|
return regex;
|
|
}
|
|
|
|
function filterTokenLocation() {
|
|
var i, entry, token, tokens = [];
|
|
|
|
for (i = 0; i < extra.tokens.length; ++i) {
|
|
entry = extra.tokens[i];
|
|
token = {
|
|
type: entry.type,
|
|
value: entry.value
|
|
};
|
|
if (extra.range) {
|
|
token.range = entry.range;
|
|
}
|
|
if (extra.loc) {
|
|
token.loc = entry.loc;
|
|
}
|
|
tokens.push(token);
|
|
}
|
|
|
|
extra.tokens = tokens;
|
|
}
|
|
|
|
function createLiteral(token) {
|
|
return {
|
|
type: Syntax.Literal,
|
|
value: token.value
|
|
};
|
|
}
|
|
|
|
function createRawLiteral(token) {
|
|
return {
|
|
type: Syntax.Literal,
|
|
value: token.value,
|
|
raw: sliceSource(token.range[0], token.range[1])
|
|
};
|
|
}
|
|
|
|
function createLocationMarker() {
|
|
var marker = {};
|
|
|
|
marker.range = [index, index];
|
|
marker.loc = {
|
|
start: {
|
|
line: lineNumber,
|
|
column: index - lineStart
|
|
},
|
|
end: {
|
|
line: lineNumber,
|
|
column: index - lineStart
|
|
}
|
|
};
|
|
|
|
marker.end = function () {
|
|
this.range[1] = index;
|
|
this.loc.end.line = lineNumber;
|
|
this.loc.end.column = index - lineStart;
|
|
};
|
|
|
|
marker.applyGroup = function (node) {
|
|
if (extra.range) {
|
|
node.groupRange = [this.range[0], this.range[1]];
|
|
}
|
|
if (extra.loc) {
|
|
node.groupLoc = {
|
|
start: {
|
|
line: this.loc.start.line,
|
|
column: this.loc.start.column
|
|
},
|
|
end: {
|
|
line: this.loc.end.line,
|
|
column: this.loc.end.column
|
|
}
|
|
};
|
|
}
|
|
};
|
|
|
|
marker.apply = function (node) {
|
|
if (extra.range) {
|
|
node.range = [this.range[0], this.range[1]];
|
|
}
|
|
if (extra.loc) {
|
|
node.loc = {
|
|
start: {
|
|
line: this.loc.start.line,
|
|
column: this.loc.start.column
|
|
},
|
|
end: {
|
|
line: this.loc.end.line,
|
|
column: this.loc.end.column
|
|
}
|
|
};
|
|
}
|
|
};
|
|
|
|
return marker;
|
|
}
|
|
|
|
function trackGroupExpression() {
|
|
var marker, expr;
|
|
|
|
skipComment();
|
|
marker = createLocationMarker();
|
|
expect('(');
|
|
|
|
expr = parseExpression();
|
|
|
|
expect(')');
|
|
|
|
marker.end();
|
|
marker.applyGroup(expr);
|
|
|
|
return expr;
|
|
}
|
|
|
|
function trackLeftHandSideExpression() {
|
|
var marker, expr;
|
|
|
|
skipComment();
|
|
marker = createLocationMarker();
|
|
|
|
expr = matchKeyword('new') ? parseNewExpression() : parsePrimaryExpression();
|
|
|
|
while (match('.') || match('[')) {
|
|
if (match('[')) {
|
|
expr = {
|
|
type: Syntax.MemberExpression,
|
|
computed: true,
|
|
object: expr,
|
|
property: parseComputedMember()
|
|
};
|
|
marker.end();
|
|
marker.apply(expr);
|
|
} else {
|
|
expr = {
|
|
type: Syntax.MemberExpression,
|
|
computed: false,
|
|
object: expr,
|
|
property: parseNonComputedMember()
|
|
};
|
|
marker.end();
|
|
marker.apply(expr);
|
|
}
|
|
}
|
|
|
|
return expr;
|
|
}
|
|
|
|
function trackLeftHandSideExpressionAllowCall() {
|
|
var marker, expr;
|
|
|
|
skipComment();
|
|
marker = createLocationMarker();
|
|
|
|
expr = matchKeyword('new') ? parseNewExpression() : parsePrimaryExpression();
|
|
|
|
while (match('.') || match('[') || match('(')) {
|
|
if (match('(')) {
|
|
expr = {
|
|
type: Syntax.CallExpression,
|
|
callee: expr,
|
|
'arguments': parseArguments()
|
|
};
|
|
marker.end();
|
|
marker.apply(expr);
|
|
} else if (match('[')) {
|
|
expr = {
|
|
type: Syntax.MemberExpression,
|
|
computed: true,
|
|
object: expr,
|
|
property: parseComputedMember()
|
|
};
|
|
marker.end();
|
|
marker.apply(expr);
|
|
} else {
|
|
expr = {
|
|
type: Syntax.MemberExpression,
|
|
computed: false,
|
|
object: expr,
|
|
property: parseNonComputedMember()
|
|
};
|
|
marker.end();
|
|
marker.apply(expr);
|
|
}
|
|
}
|
|
|
|
return expr;
|
|
}
|
|
|
|
function filterGroup(node) {
|
|
var n, i, entry;
|
|
|
|
n = (Object.prototype.toString.apply(node) === '[object Array]') ? [] : {};
|
|
for (i in node) {
|
|
if (node.hasOwnProperty(i) && i !== 'groupRange' && i !== 'groupLoc') {
|
|
entry = node[i];
|
|
if (entry === null || typeof entry !== 'object' || entry instanceof RegExp) {
|
|
n[i] = entry;
|
|
} else {
|
|
n[i] = filterGroup(entry);
|
|
}
|
|
}
|
|
}
|
|
return n;
|
|
}
|
|
|
|
function wrapTrackingFunction(range, loc) {
|
|
|
|
return function (parseFunction) {
|
|
|
|
function isBinary(node) {
|
|
return node.type === Syntax.LogicalExpression ||
|
|
node.type === Syntax.BinaryExpression;
|
|
}
|
|
|
|
function visit(node) {
|
|
var start, end;
|
|
|
|
if (isBinary(node.left)) {
|
|
visit(node.left);
|
|
}
|
|
if (isBinary(node.right)) {
|
|
visit(node.right);
|
|
}
|
|
|
|
if (range) {
|
|
if (node.left.groupRange || node.right.groupRange) {
|
|
start = node.left.groupRange ? node.left.groupRange[0] : node.left.range[0];
|
|
end = node.right.groupRange ? node.right.groupRange[1] : node.right.range[1];
|
|
node.range = [start, end];
|
|
} else if (typeof node.range === 'undefined') {
|
|
start = node.left.range[0];
|
|
end = node.right.range[1];
|
|
node.range = [start, end];
|
|
}
|
|
}
|
|
if (loc) {
|
|
if (node.left.groupLoc || node.right.groupLoc) {
|
|
start = node.left.groupLoc ? node.left.groupLoc.start : node.left.loc.start;
|
|
end = node.right.groupLoc ? node.right.groupLoc.end : node.right.loc.end;
|
|
node.loc = {
|
|
start: start,
|
|
end: end
|
|
};
|
|
} else if (typeof node.loc === 'undefined') {
|
|
node.loc = {
|
|
start: node.left.loc.start,
|
|
end: node.right.loc.end
|
|
};
|
|
}
|
|
}
|
|
}
|
|
|
|
return function () {
|
|
var marker, node;
|
|
|
|
skipComment();
|
|
|
|
marker = createLocationMarker();
|
|
node = parseFunction.apply(null, arguments);
|
|
marker.end();
|
|
|
|
if (range && typeof node.range === 'undefined') {
|
|
marker.apply(node);
|
|
}
|
|
|
|
if (loc && typeof node.loc === 'undefined') {
|
|
marker.apply(node);
|
|
}
|
|
|
|
if (isBinary(node)) {
|
|
visit(node);
|
|
}
|
|
|
|
return node;
|
|
};
|
|
};
|
|
}
|
|
|
|
function patch() {
|
|
|
|
var wrapTracking;
|
|
|
|
if (extra.comments) {
|
|
extra.skipComment = skipComment;
|
|
skipComment = scanComment;
|
|
}
|
|
|
|
if (extra.raw) {
|
|
extra.createLiteral = createLiteral;
|
|
createLiteral = createRawLiteral;
|
|
}
|
|
|
|
if (extra.range || extra.loc) {
|
|
|
|
extra.parseGroupExpression = parseGroupExpression;
|
|
extra.parseLeftHandSideExpression = parseLeftHandSideExpression;
|
|
extra.parseLeftHandSideExpressionAllowCall = parseLeftHandSideExpressionAllowCall;
|
|
parseGroupExpression = trackGroupExpression;
|
|
parseLeftHandSideExpression = trackLeftHandSideExpression;
|
|
parseLeftHandSideExpressionAllowCall = trackLeftHandSideExpressionAllowCall;
|
|
|
|
wrapTracking = wrapTrackingFunction(extra.range, extra.loc);
|
|
|
|
extra.parseAdditiveExpression = parseAdditiveExpression;
|
|
extra.parseAssignmentExpression = parseAssignmentExpression;
|
|
extra.parseBitwiseANDExpression = parseBitwiseANDExpression;
|
|
extra.parseBitwiseORExpression = parseBitwiseORExpression;
|
|
extra.parseBitwiseXORExpression = parseBitwiseXORExpression;
|
|
extra.parseBlock = parseBlock;
|
|
extra.parseFunctionSourceElements = parseFunctionSourceElements;
|
|
extra.parseCatchClause = parseCatchClause;
|
|
extra.parseComputedMember = parseComputedMember;
|
|
extra.parseConditionalExpression = parseConditionalExpression;
|
|
extra.parseConstLetDeclaration = parseConstLetDeclaration;
|
|
extra.parseEqualityExpression = parseEqualityExpression;
|
|
extra.parseExpression = parseExpression;
|
|
extra.parseForVariableDeclaration = parseForVariableDeclaration;
|
|
extra.parseFunctionDeclaration = parseFunctionDeclaration;
|
|
extra.parseFunctionExpression = parseFunctionExpression;
|
|
extra.parseLogicalANDExpression = parseLogicalANDExpression;
|
|
extra.parseLogicalORExpression = parseLogicalORExpression;
|
|
extra.parseMultiplicativeExpression = parseMultiplicativeExpression;
|
|
extra.parseNewExpression = parseNewExpression;
|
|
extra.parseNonComputedProperty = parseNonComputedProperty;
|
|
extra.parseObjectProperty = parseObjectProperty;
|
|
extra.parseObjectPropertyKey = parseObjectPropertyKey;
|
|
extra.parsePostfixExpression = parsePostfixExpression;
|
|
extra.parsePrimaryExpression = parsePrimaryExpression;
|
|
extra.parseProgram = parseProgram;
|
|
extra.parsePropertyFunction = parsePropertyFunction;
|
|
extra.parseRelationalExpression = parseRelationalExpression;
|
|
extra.parseStatement = parseStatement;
|
|
extra.parseShiftExpression = parseShiftExpression;
|
|
extra.parseSwitchCase = parseSwitchCase;
|
|
extra.parseUnaryExpression = parseUnaryExpression;
|
|
extra.parseVariableDeclaration = parseVariableDeclaration;
|
|
extra.parseVariableIdentifier = parseVariableIdentifier;
|
|
|
|
parseAdditiveExpression = wrapTracking(extra.parseAdditiveExpression);
|
|
parseAssignmentExpression = wrapTracking(extra.parseAssignmentExpression);
|
|
parseBitwiseANDExpression = wrapTracking(extra.parseBitwiseANDExpression);
|
|
parseBitwiseORExpression = wrapTracking(extra.parseBitwiseORExpression);
|
|
parseBitwiseXORExpression = wrapTracking(extra.parseBitwiseXORExpression);
|
|
parseBlock = wrapTracking(extra.parseBlock);
|
|
parseFunctionSourceElements = wrapTracking(extra.parseFunctionSourceElements);
|
|
parseCatchClause = wrapTracking(extra.parseCatchClause);
|
|
parseComputedMember = wrapTracking(extra.parseComputedMember);
|
|
parseConditionalExpression = wrapTracking(extra.parseConditionalExpression);
|
|
parseConstLetDeclaration = wrapTracking(extra.parseConstLetDeclaration);
|
|
parseEqualityExpression = wrapTracking(extra.parseEqualityExpression);
|
|
parseExpression = wrapTracking(extra.parseExpression);
|
|
parseForVariableDeclaration = wrapTracking(extra.parseForVariableDeclaration);
|
|
parseFunctionDeclaration = wrapTracking(extra.parseFunctionDeclaration);
|
|
parseFunctionExpression = wrapTracking(extra.parseFunctionExpression);
|
|
parseLeftHandSideExpression = wrapTracking(parseLeftHandSideExpression);
|
|
parseLogicalANDExpression = wrapTracking(extra.parseLogicalANDExpression);
|
|
parseLogicalORExpression = wrapTracking(extra.parseLogicalORExpression);
|
|
parseMultiplicativeExpression = wrapTracking(extra.parseMultiplicativeExpression);
|
|
parseNewExpression = wrapTracking(extra.parseNewExpression);
|
|
parseNonComputedProperty = wrapTracking(extra.parseNonComputedProperty);
|
|
parseObjectProperty = wrapTracking(extra.parseObjectProperty);
|
|
parseObjectPropertyKey = wrapTracking(extra.parseObjectPropertyKey);
|
|
parsePostfixExpression = wrapTracking(extra.parsePostfixExpression);
|
|
parsePrimaryExpression = wrapTracking(extra.parsePrimaryExpression);
|
|
parseProgram = wrapTracking(extra.parseProgram);
|
|
parsePropertyFunction = wrapTracking(extra.parsePropertyFunction);
|
|
parseRelationalExpression = wrapTracking(extra.parseRelationalExpression);
|
|
parseStatement = wrapTracking(extra.parseStatement);
|
|
parseShiftExpression = wrapTracking(extra.parseShiftExpression);
|
|
parseSwitchCase = wrapTracking(extra.parseSwitchCase);
|
|
parseUnaryExpression = wrapTracking(extra.parseUnaryExpression);
|
|
parseVariableDeclaration = wrapTracking(extra.parseVariableDeclaration);
|
|
parseVariableIdentifier = wrapTracking(extra.parseVariableIdentifier);
|
|
}
|
|
|
|
if (typeof extra.tokens !== 'undefined') {
|
|
extra.advance = advance;
|
|
extra.scanRegExp = scanRegExp;
|
|
|
|
advance = collectToken;
|
|
scanRegExp = collectRegex;
|
|
}
|
|
}
|
|
|
|
function unpatch() {
|
|
if (typeof extra.skipComment === 'function') {
|
|
skipComment = extra.skipComment;
|
|
}
|
|
|
|
if (extra.raw) {
|
|
createLiteral = extra.createLiteral;
|
|
}
|
|
|
|
if (extra.range || extra.loc) {
|
|
parseAdditiveExpression = extra.parseAdditiveExpression;
|
|
parseAssignmentExpression = extra.parseAssignmentExpression;
|
|
parseBitwiseANDExpression = extra.parseBitwiseANDExpression;
|
|
parseBitwiseORExpression = extra.parseBitwiseORExpression;
|
|
parseBitwiseXORExpression = extra.parseBitwiseXORExpression;
|
|
parseBlock = extra.parseBlock;
|
|
parseFunctionSourceElements = extra.parseFunctionSourceElements;
|
|
parseCatchClause = extra.parseCatchClause;
|
|
parseComputedMember = extra.parseComputedMember;
|
|
parseConditionalExpression = extra.parseConditionalExpression;
|
|
parseConstLetDeclaration = extra.parseConstLetDeclaration;
|
|
parseEqualityExpression = extra.parseEqualityExpression;
|
|
parseExpression = extra.parseExpression;
|
|
parseForVariableDeclaration = extra.parseForVariableDeclaration;
|
|
parseFunctionDeclaration = extra.parseFunctionDeclaration;
|
|
parseFunctionExpression = extra.parseFunctionExpression;
|
|
parseGroupExpression = extra.parseGroupExpression;
|
|
parseLeftHandSideExpression = extra.parseLeftHandSideExpression;
|
|
parseLeftHandSideExpressionAllowCall = extra.parseLeftHandSideExpressionAllowCall;
|
|
parseLogicalANDExpression = extra.parseLogicalANDExpression;
|
|
parseLogicalORExpression = extra.parseLogicalORExpression;
|
|
parseMultiplicativeExpression = extra.parseMultiplicativeExpression;
|
|
parseNewExpression = extra.parseNewExpression;
|
|
parseNonComputedProperty = extra.parseNonComputedProperty;
|
|
parseObjectProperty = extra.parseObjectProperty;
|
|
parseObjectPropertyKey = extra.parseObjectPropertyKey;
|
|
parsePrimaryExpression = extra.parsePrimaryExpression;
|
|
parsePostfixExpression = extra.parsePostfixExpression;
|
|
parseProgram = extra.parseProgram;
|
|
parsePropertyFunction = extra.parsePropertyFunction;
|
|
parseRelationalExpression = extra.parseRelationalExpression;
|
|
parseStatement = extra.parseStatement;
|
|
parseShiftExpression = extra.parseShiftExpression;
|
|
parseSwitchCase = extra.parseSwitchCase;
|
|
parseUnaryExpression = extra.parseUnaryExpression;
|
|
parseVariableDeclaration = extra.parseVariableDeclaration;
|
|
parseVariableIdentifier = extra.parseVariableIdentifier;
|
|
}
|
|
|
|
if (typeof extra.scanRegExp === 'function') {
|
|
advance = extra.advance;
|
|
scanRegExp = extra.scanRegExp;
|
|
}
|
|
}
|
|
|
|
function stringToArray(str) {
|
|
var length = str.length,
|
|
result = [],
|
|
i;
|
|
for (i = 0; i < length; ++i) {
|
|
result[i] = str.charAt(i);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
function parse(code, options) {
|
|
var program, toString;
|
|
|
|
toString = String;
|
|
if (typeof code !== 'string' && !(code instanceof String)) {
|
|
code = toString(code);
|
|
}
|
|
|
|
source = code;
|
|
index = 0;
|
|
lineNumber = (source.length > 0) ? 1 : 0;
|
|
lineStart = 0;
|
|
length = source.length;
|
|
buffer = null;
|
|
state = {
|
|
allowIn: true,
|
|
labelSet: {},
|
|
inFunctionBody: false,
|
|
inIteration: false,
|
|
inSwitch: false
|
|
};
|
|
|
|
extra = {};
|
|
if (typeof options !== 'undefined') {
|
|
extra.range = (typeof options.range === 'boolean') && options.range;
|
|
extra.loc = (typeof options.loc === 'boolean') && options.loc;
|
|
extra.raw = (typeof options.raw === 'boolean') && options.raw;
|
|
if (typeof options.tokens === 'boolean' && options.tokens) {
|
|
extra.tokens = [];
|
|
}
|
|
if (typeof options.comment === 'boolean' && options.comment) {
|
|
extra.comments = [];
|
|
}
|
|
if (typeof options.tolerant === 'boolean' && options.tolerant) {
|
|
extra.errors = [];
|
|
}
|
|
}
|
|
|
|
if (length > 0) {
|
|
if (typeof source[0] === 'undefined') {
|
|
// Try first to convert to a string. This is good as fast path
|
|
// for old IE which understands string indexing for string
|
|
// literals only and not for string object.
|
|
if (code instanceof String) {
|
|
source = code.valueOf();
|
|
}
|
|
|
|
// Force accessing the characters via an array.
|
|
if (typeof source[0] === 'undefined') {
|
|
source = stringToArray(code);
|
|
}
|
|
}
|
|
}
|
|
|
|
patch();
|
|
try {
|
|
program = parseProgram();
|
|
if (typeof extra.comments !== 'undefined') {
|
|
filterCommentLocation();
|
|
program.comments = extra.comments;
|
|
}
|
|
if (typeof extra.tokens !== 'undefined') {
|
|
filterTokenLocation();
|
|
program.tokens = extra.tokens;
|
|
}
|
|
if (typeof extra.errors !== 'undefined') {
|
|
program.errors = extra.errors;
|
|
}
|
|
if (extra.range || extra.loc) {
|
|
program.body = filterGroup(program.body);
|
|
}
|
|
} catch (e) {
|
|
throw e;
|
|
} finally {
|
|
unpatch();
|
|
extra = {};
|
|
}
|
|
|
|
return program;
|
|
}
|
|
|
|
// Sync with package.json.
|
|
exports.version = '1.0.4';
|
|
|
|
exports.parse = parse;
|
|
|
|
// Deep copy.
|
|
exports.Syntax = (function () {
|
|
var name, types = {};
|
|
|
|
if (typeof Object.create === 'function') {
|
|
types = Object.create(null);
|
|
}
|
|
|
|
for (name in Syntax) {
|
|
if (Syntax.hasOwnProperty(name)) {
|
|
types[name] = Syntax[name];
|
|
}
|
|
}
|
|
|
|
if (typeof Object.freeze === 'function') {
|
|
Object.freeze(types);
|
|
}
|
|
|
|
return types;
|
|
}());
|
|
|
|
}));
|
|
/* vim: set sw=4 ts=4 et tw=80 : */
|
|
|
|
},{}],50:[function(require,module,exports){
|
|
var global=typeof self !== "undefined" ? self : typeof window !== "undefined" ? window : {};/**
|
|
* @license
|
|
* Lo-Dash 2.3.0 (Custom Build) <http://lodash.com/>
|
|
* Build: `lodash modern -o ./dist/lodash.js`
|
|
* Copyright 2012-2013 The Dojo Foundation <http://dojofoundation.org/>
|
|
* Based on Underscore.js 1.5.2 <http://underscorejs.org/LICENSE>
|
|
* Copyright 2009-2013 Jeremy Ashkenas, DocumentCloud and Investigative Reporters & Editors
|
|
* Available under MIT license <http://lodash.com/license>
|
|
*/
|
|
;(function() {
|
|
|
|
/** Used as a safe reference for `undefined` in pre ES5 environments */
|
|
var undefined;
|
|
|
|
/** Used to pool arrays and objects used internally */
|
|
var arrayPool = [],
|
|
objectPool = [];
|
|
|
|
/** Used to generate unique IDs */
|
|
var idCounter = 0;
|
|
|
|
/** Used to prefix keys to avoid issues with `__proto__` and properties on `Object.prototype` */
|
|
var keyPrefix = +new Date + '';
|
|
|
|
/** Used as the size when optimizations are enabled for large arrays */
|
|
var largeArraySize = 75;
|
|
|
|
/** Used as the max size of the `arrayPool` and `objectPool` */
|
|
var maxPoolSize = 40;
|
|
|
|
/** Used to detect and test whitespace */
|
|
var whitespace = (
|
|
// whitespace
|
|
' \t\x0B\f\xA0\ufeff' +
|
|
|
|
// line terminators
|
|
'\n\r\u2028\u2029' +
|
|
|
|
// unicode category "Zs" space separators
|
|
'\u1680\u180e\u2000\u2001\u2002\u2003\u2004\u2005\u2006\u2007\u2008\u2009\u200a\u202f\u205f\u3000'
|
|
);
|
|
|
|
/** Used to match empty string literals in compiled template source */
|
|
var reEmptyStringLeading = /\b__p \+= '';/g,
|
|
reEmptyStringMiddle = /\b(__p \+=) '' \+/g,
|
|
reEmptyStringTrailing = /(__e\(.*?\)|\b__t\)) \+\n'';/g;
|
|
|
|
/**
|
|
* Used to match ES6 template delimiters
|
|
* http://people.mozilla.org/~jorendorff/es6-draft.html#sec-7.8.6
|
|
*/
|
|
var reEsTemplate = /\$\{([^\\}]*(?:\\.[^\\}]*)*)\}/g;
|
|
|
|
/** Used to match regexp flags from their coerced string values */
|
|
var reFlags = /\w*$/;
|
|
|
|
/** Used to detected named functions */
|
|
var reFuncName = /^\s*function[ \n\r\t]+\w/;
|
|
|
|
/** Used to match "interpolate" template delimiters */
|
|
var reInterpolate = /<%=([\s\S]+?)%>/g;
|
|
|
|
/** Used to match leading whitespace and zeros to be removed */
|
|
var reLeadingSpacesAndZeros = RegExp('^[' + whitespace + ']*0+(?=.$)');
|
|
|
|
/** Used to ensure capturing order of template delimiters */
|
|
var reNoMatch = /($^)/;
|
|
|
|
/** Used to detect functions containing a `this` reference */
|
|
var reThis = /\bthis\b/;
|
|
|
|
/** Used to match unescaped characters in compiled string literals */
|
|
var reUnescapedString = /['\n\r\t\u2028\u2029\\]/g;
|
|
|
|
/** Used to assign default `context` object properties */
|
|
var contextProps = [
|
|
'Array', 'Boolean', 'Date', 'Function', 'Math', 'Number', 'Object',
|
|
'RegExp', 'String', '_', 'attachEvent', 'clearTimeout', 'isFinite', 'isNaN',
|
|
'parseInt', 'setImmediate', 'setTimeout'
|
|
];
|
|
|
|
/** Used to make template sourceURLs easier to identify */
|
|
var templateCounter = 0;
|
|
|
|
/** `Object#toString` result shortcuts */
|
|
var argsClass = '[object Arguments]',
|
|
arrayClass = '[object Array]',
|
|
boolClass = '[object Boolean]',
|
|
dateClass = '[object Date]',
|
|
funcClass = '[object Function]',
|
|
numberClass = '[object Number]',
|
|
objectClass = '[object Object]',
|
|
regexpClass = '[object RegExp]',
|
|
stringClass = '[object String]';
|
|
|
|
/** Used to identify object classifications that `_.clone` supports */
|
|
var cloneableClasses = {};
|
|
cloneableClasses[funcClass] = false;
|
|
cloneableClasses[argsClass] = cloneableClasses[arrayClass] =
|
|
cloneableClasses[boolClass] = cloneableClasses[dateClass] =
|
|
cloneableClasses[numberClass] = cloneableClasses[objectClass] =
|
|
cloneableClasses[regexpClass] = cloneableClasses[stringClass] = true;
|
|
|
|
/** Used as an internal `_.debounce` options object */
|
|
var debounceOptions = {
|
|
'leading': false,
|
|
'maxWait': 0,
|
|
'trailing': false
|
|
};
|
|
|
|
/** Used as the property descriptor for `__bindData__` */
|
|
var descriptor = {
|
|
'configurable': false,
|
|
'enumerable': false,
|
|
'value': null,
|
|
'writable': false
|
|
};
|
|
|
|
/** Used to determine if values are of the language type Object */
|
|
var objectTypes = {
|
|
'boolean': false,
|
|
'function': true,
|
|
'object': true,
|
|
'number': false,
|
|
'string': false,
|
|
'undefined': false
|
|
};
|
|
|
|
/** Used to escape characters for inclusion in compiled string literals */
|
|
var stringEscapes = {
|
|
'\\': '\\',
|
|
"'": "'",
|
|
'\n': 'n',
|
|
'\r': 'r',
|
|
'\t': 't',
|
|
'\u2028': 'u2028',
|
|
'\u2029': 'u2029'
|
|
};
|
|
|
|
/** Used as a reference to the global object */
|
|
var root = (objectTypes[typeof window] && window) || this;
|
|
|
|
/** Detect free variable `exports` */
|
|
var freeExports = objectTypes[typeof exports] && exports && !exports.nodeType && exports;
|
|
|
|
/** Detect free variable `module` */
|
|
var freeModule = objectTypes[typeof module] && module && !module.nodeType && module;
|
|
|
|
/** Detect the popular CommonJS extension `module.exports` */
|
|
var moduleExports = freeModule && freeModule.exports === freeExports && freeExports;
|
|
|
|
/** Detect free variable `global` from Node.js or Browserified code and use it as `root` */
|
|
var freeGlobal = objectTypes[typeof global] && global;
|
|
if (freeGlobal && (freeGlobal.global === freeGlobal || freeGlobal.window === freeGlobal)) {
|
|
root = freeGlobal;
|
|
}
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
|
|
/**
|
|
* The base implementation of `_.indexOf` without support for binary searches
|
|
* or `fromIndex` constraints.
|
|
*
|
|
* @private
|
|
* @param {Array} array The array to search.
|
|
* @param {*} value The value to search for.
|
|
* @param {number} [fromIndex=0] The index to search from.
|
|
* @returns {number} Returns the index of the matched value or `-1`.
|
|
*/
|
|
function baseIndexOf(array, value, fromIndex) {
|
|
var index = (fromIndex || 0) - 1,
|
|
length = array ? array.length : 0;
|
|
|
|
while (++index < length) {
|
|
if (array[index] === value) {
|
|
return index;
|
|
}
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
/**
|
|
* An implementation of `_.contains` for cache objects that mimics the return
|
|
* signature of `_.indexOf` by returning `0` if the value is found, else `-1`.
|
|
*
|
|
* @private
|
|
* @param {Object} cache The cache object to inspect.
|
|
* @param {*} value The value to search for.
|
|
* @returns {number} Returns `0` if `value` is found, else `-1`.
|
|
*/
|
|
function cacheIndexOf(cache, value) {
|
|
var type = typeof value;
|
|
cache = cache.cache;
|
|
|
|
if (type == 'boolean' || value == null) {
|
|
return cache[value] ? 0 : -1;
|
|
}
|
|
if (type != 'number' && type != 'string') {
|
|
type = 'object';
|
|
}
|
|
var key = type == 'number' ? value : keyPrefix + value;
|
|
cache = (cache = cache[type]) && cache[key];
|
|
|
|
return type == 'object'
|
|
? (cache && baseIndexOf(cache, value) > -1 ? 0 : -1)
|
|
: (cache ? 0 : -1);
|
|
}
|
|
|
|
/**
|
|
* Adds a given value to the corresponding cache object.
|
|
*
|
|
* @private
|
|
* @param {*} value The value to add to the cache.
|
|
*/
|
|
function cachePush(value) {
|
|
var cache = this.cache,
|
|
type = typeof value;
|
|
|
|
if (type == 'boolean' || value == null) {
|
|
cache[value] = true;
|
|
} else {
|
|
if (type != 'number' && type != 'string') {
|
|
type = 'object';
|
|
}
|
|
var key = type == 'number' ? value : keyPrefix + value,
|
|
typeCache = cache[type] || (cache[type] = {});
|
|
|
|
if (type == 'object') {
|
|
(typeCache[key] || (typeCache[key] = [])).push(value);
|
|
} else {
|
|
typeCache[key] = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Used by `_.max` and `_.min` as the default callback when a given
|
|
* collection is a string value.
|
|
*
|
|
* @private
|
|
* @param {string} value The character to inspect.
|
|
* @returns {number} Returns the code unit of given character.
|
|
*/
|
|
function charAtCallback(value) {
|
|
return value.charCodeAt(0);
|
|
}
|
|
|
|
/**
|
|
* Used by `sortBy` to compare transformed `collection` elements, stable sorting
|
|
* them in ascending order.
|
|
*
|
|
* @private
|
|
* @param {Object} a The object to compare to `b`.
|
|
* @param {Object} b The object to compare to `a`.
|
|
* @returns {number} Returns the sort order indicator of `1` or `-1`.
|
|
*/
|
|
function compareAscending(a, b) {
|
|
var ac = a.criteria,
|
|
bc = b.criteria;
|
|
|
|
// ensure a stable sort in V8 and other engines
|
|
// http://code.google.com/p/v8/issues/detail?id=90
|
|
if (ac !== bc) {
|
|
if (ac > bc || typeof ac == 'undefined') {
|
|
return 1;
|
|
}
|
|
if (ac < bc || typeof bc == 'undefined') {
|
|
return -1;
|
|
}
|
|
}
|
|
// The JS engine embedded in Adobe applications like InDesign has a buggy
|
|
// `Array#sort` implementation that causes it, under certain circumstances,
|
|
// to return the same value for `a` and `b`.
|
|
// See https://github.com/jashkenas/underscore/pull/1247
|
|
return a.index - b.index;
|
|
}
|
|
|
|
/**
|
|
* Creates a cache object to optimize linear searches of large arrays.
|
|
*
|
|
* @private
|
|
* @param {Array} [array=[]] The array to search.
|
|
* @returns {null|Object} Returns the cache object or `null` if caching should not be used.
|
|
*/
|
|
function createCache(array) {
|
|
var index = -1,
|
|
length = array.length,
|
|
first = array[0],
|
|
mid = array[(length / 2) | 0],
|
|
last = array[length - 1];
|
|
|
|
if (first && typeof first == 'object' &&
|
|
mid && typeof mid == 'object' && last && typeof last == 'object') {
|
|
return false;
|
|
}
|
|
var cache = getObject();
|
|
cache['false'] = cache['null'] = cache['true'] = cache['undefined'] = false;
|
|
|
|
var result = getObject();
|
|
result.array = array;
|
|
result.cache = cache;
|
|
result.push = cachePush;
|
|
|
|
while (++index < length) {
|
|
result.push(array[index]);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Used by `template` to escape characters for inclusion in compiled
|
|
* string literals.
|
|
*
|
|
* @private
|
|
* @param {string} match The matched character to escape.
|
|
* @returns {string} Returns the escaped character.
|
|
*/
|
|
function escapeStringChar(match) {
|
|
return '\\' + stringEscapes[match];
|
|
}
|
|
|
|
/**
|
|
* Gets an array from the array pool or creates a new one if the pool is empty.
|
|
*
|
|
* @private
|
|
* @returns {Array} The array from the pool.
|
|
*/
|
|
function getArray() {
|
|
return arrayPool.pop() || [];
|
|
}
|
|
|
|
/**
|
|
* Gets an object from the object pool or creates a new one if the pool is empty.
|
|
*
|
|
* @private
|
|
* @returns {Object} The object from the pool.
|
|
*/
|
|
function getObject() {
|
|
return objectPool.pop() || {
|
|
'array': null,
|
|
'cache': null,
|
|
'criteria': null,
|
|
'false': false,
|
|
'index': 0,
|
|
'null': false,
|
|
'number': null,
|
|
'object': null,
|
|
'push': null,
|
|
'string': null,
|
|
'true': false,
|
|
'undefined': false,
|
|
'value': null
|
|
};
|
|
}
|
|
|
|
/**
|
|
* Releases the given array back to the array pool.
|
|
*
|
|
* @private
|
|
* @param {Array} [array] The array to release.
|
|
*/
|
|
function releaseArray(array) {
|
|
array.length = 0;
|
|
if (arrayPool.length < maxPoolSize) {
|
|
arrayPool.push(array);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Releases the given object back to the object pool.
|
|
*
|
|
* @private
|
|
* @param {Object} [object] The object to release.
|
|
*/
|
|
function releaseObject(object) {
|
|
var cache = object.cache;
|
|
if (cache) {
|
|
releaseObject(cache);
|
|
}
|
|
object.array = object.cache = object.criteria = object.object = object.number = object.string = object.value = null;
|
|
if (objectPool.length < maxPoolSize) {
|
|
objectPool.push(object);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Slices the `collection` from the `start` index up to, but not including,
|
|
* the `end` index.
|
|
*
|
|
* Note: This function is used instead of `Array#slice` to support node lists
|
|
* in IE < 9 and to ensure dense arrays are returned.
|
|
*
|
|
* @private
|
|
* @param {Array|Object|string} collection The collection to slice.
|
|
* @param {number} start The start index.
|
|
* @param {number} end The end index.
|
|
* @returns {Array} Returns the new array.
|
|
*/
|
|
function slice(array, start, end) {
|
|
start || (start = 0);
|
|
if (typeof end == 'undefined') {
|
|
end = array ? array.length : 0;
|
|
}
|
|
var index = -1,
|
|
length = end - start || 0,
|
|
result = Array(length < 0 ? 0 : length);
|
|
|
|
while (++index < length) {
|
|
result[index] = array[start + index];
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
|
|
/**
|
|
* Create a new `lodash` function using the given context object.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Utilities
|
|
* @param {Object} [context=root] The context object.
|
|
* @returns {Function} Returns the `lodash` function.
|
|
*/
|
|
function runInContext(context) {
|
|
// Avoid issues with some ES3 environments that attempt to use values, named
|
|
// after built-in constructors like `Object`, for the creation of literals.
|
|
// ES5 clears this up by stating that literals must use built-in constructors.
|
|
// See http://es5.github.io/#x11.1.5.
|
|
context = context ? _.defaults(root.Object(), context, _.pick(root, contextProps)) : root;
|
|
|
|
/** Native constructor references */
|
|
var Array = context.Array,
|
|
Boolean = context.Boolean,
|
|
Date = context.Date,
|
|
Function = context.Function,
|
|
Math = context.Math,
|
|
Number = context.Number,
|
|
Object = context.Object,
|
|
RegExp = context.RegExp,
|
|
String = context.String,
|
|
TypeError = context.TypeError;
|
|
|
|
/**
|
|
* Used for `Array` method references.
|
|
*
|
|
* Normally `Array.prototype` would suffice, however, using an array literal
|
|
* avoids issues in Narwhal.
|
|
*/
|
|
var arrayRef = [];
|
|
|
|
/** Used for native method references */
|
|
var objectProto = Object.prototype;
|
|
|
|
/** Used to restore the original `_` reference in `noConflict` */
|
|
var oldDash = context._;
|
|
|
|
/** Used to resolve the internal [[Class]] of values */
|
|
var toString = objectProto.toString;
|
|
|
|
/** Used to detect if a method is native */
|
|
var reNative = RegExp('^' +
|
|
String(toString)
|
|
.replace(/[.*+?^${}()|[\]\\]/g, '\\$&')
|
|
.replace(/toString| for [^\]]+/g, '.*?') + '$'
|
|
);
|
|
|
|
/** Native method shortcuts */
|
|
var ceil = Math.ceil,
|
|
clearTimeout = context.clearTimeout,
|
|
floor = Math.floor,
|
|
fnToString = Function.prototype.toString,
|
|
getPrototypeOf = reNative.test(getPrototypeOf = Object.getPrototypeOf) && getPrototypeOf,
|
|
hasOwnProperty = objectProto.hasOwnProperty,
|
|
now = reNative.test(now = Date.now) && now || function() { return +new Date; },
|
|
push = arrayRef.push,
|
|
setTimeout = context.setTimeout,
|
|
splice = arrayRef.splice;
|
|
|
|
/** Used to detect `setImmediate` in Node.js */
|
|
var setImmediate = typeof (setImmediate = freeGlobal && moduleExports && freeGlobal.setImmediate) == 'function' &&
|
|
!reNative.test(setImmediate) && setImmediate;
|
|
|
|
/** Used to set meta data on functions */
|
|
var defineProperty = (function() {
|
|
// IE 8 only accepts DOM elements
|
|
try {
|
|
var o = {},
|
|
func = reNative.test(func = Object.defineProperty) && func,
|
|
result = func(o, o, o) && func;
|
|
} catch(e) { }
|
|
return result;
|
|
}());
|
|
|
|
/* Native method shortcuts for methods with the same name as other `lodash` methods */
|
|
var nativeCreate = reNative.test(nativeCreate = Object.create) && nativeCreate,
|
|
nativeIsArray = reNative.test(nativeIsArray = Array.isArray) && nativeIsArray,
|
|
nativeIsFinite = context.isFinite,
|
|
nativeIsNaN = context.isNaN,
|
|
nativeKeys = reNative.test(nativeKeys = Object.keys) && nativeKeys,
|
|
nativeMax = Math.max,
|
|
nativeMin = Math.min,
|
|
nativeParseInt = context.parseInt,
|
|
nativeRandom = Math.random;
|
|
|
|
/** Used to lookup a built-in constructor by [[Class]] */
|
|
var ctorByClass = {};
|
|
ctorByClass[arrayClass] = Array;
|
|
ctorByClass[boolClass] = Boolean;
|
|
ctorByClass[dateClass] = Date;
|
|
ctorByClass[funcClass] = Function;
|
|
ctorByClass[objectClass] = Object;
|
|
ctorByClass[numberClass] = Number;
|
|
ctorByClass[regexpClass] = RegExp;
|
|
ctorByClass[stringClass] = String;
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
|
|
/**
|
|
* Creates a `lodash` object which wraps the given value to enable intuitive
|
|
* method chaining.
|
|
*
|
|
* In addition to Lo-Dash methods, wrappers also have the following `Array` methods:
|
|
* `concat`, `join`, `pop`, `push`, `reverse`, `shift`, `slice`, `sort`, `splice`,
|
|
* and `unshift`
|
|
*
|
|
* Chaining is supported in custom builds as long as the `value` method is
|
|
* implicitly or explicitly included in the build.
|
|
*
|
|
* The chainable wrapper functions are:
|
|
* `after`, `assign`, `bind`, `bindAll`, `bindKey`, `chain`, `compact`,
|
|
* `compose`, `concat`, `countBy`, `create`, `createCallback`, `curry`,
|
|
* `debounce`, `defaults`, `defer`, `delay`, `difference`, `filter`, `flatten`,
|
|
* `forEach`, `forEachRight`, `forIn`, `forInRight`, `forOwn`, `forOwnRight`,
|
|
* `functions`, `groupBy`, `indexBy`, `initial`, `intersection`, `invert`,
|
|
* `invoke`, `keys`, `map`, `max`, `memoize`, `merge`, `min`, `object`, `omit`,
|
|
* `once`, `pairs`, `partial`, `partialRight`, `pick`, `pluck`, `pull`, `push`,
|
|
* `range`, `reject`, `remove`, `rest`, `reverse`, `shuffle`, `slice`, `sort`,
|
|
* `sortBy`, `splice`, `tap`, `throttle`, `times`, `toArray`, `transform`,
|
|
* `union`, `uniq`, `unshift`, `unzip`, `values`, `where`, `without`, `wrap`,
|
|
* and `zip`
|
|
*
|
|
* The non-chainable wrapper functions are:
|
|
* `clone`, `cloneDeep`, `contains`, `escape`, `every`, `find`, `findIndex`,
|
|
* `findKey`, `findLast`, `findLastIndex`, `findLastKey`, `has`, `identity`,
|
|
* `indexOf`, `isArguments`, `isArray`, `isBoolean`, `isDate`, `isElement`,
|
|
* `isEmpty`, `isEqual`, `isFinite`, `isFunction`, `isNaN`, `isNull`, `isNumber`,
|
|
* `isObject`, `isPlainObject`, `isRegExp`, `isString`, `isUndefined`, `join`,
|
|
* `lastIndexOf`, `mixin`, `noConflict`, `parseInt`, `pop`, `random`, `reduce`,
|
|
* `reduceRight`, `result`, `shift`, `size`, `some`, `sortedIndex`, `runInContext`,
|
|
* `template`, `unescape`, `uniqueId`, and `value`
|
|
*
|
|
* The wrapper functions `first` and `last` return wrapped values when `n` is
|
|
* provided, otherwise they return unwrapped values.
|
|
*
|
|
* Explicit chaining can be enabled by using the `_.chain` method.
|
|
*
|
|
* @name _
|
|
* @constructor
|
|
* @category Chaining
|
|
* @param {*} value The value to wrap in a `lodash` instance.
|
|
* @returns {Object} Returns a `lodash` instance.
|
|
* @example
|
|
*
|
|
* var wrapped = _([1, 2, 3]);
|
|
*
|
|
* // returns an unwrapped value
|
|
* wrapped.reduce(function(sum, num) {
|
|
* return sum + num;
|
|
* });
|
|
* // => 6
|
|
*
|
|
* // returns a wrapped value
|
|
* var squares = wrapped.map(function(num) {
|
|
* return num * num;
|
|
* });
|
|
*
|
|
* _.isArray(squares);
|
|
* // => false
|
|
*
|
|
* _.isArray(squares.value());
|
|
* // => true
|
|
*/
|
|
function lodash(value) {
|
|
// don't wrap if already wrapped, even if wrapped by a different `lodash` constructor
|
|
return (value && typeof value == 'object' && !isArray(value) && hasOwnProperty.call(value, '__wrapped__'))
|
|
? value
|
|
: new lodashWrapper(value);
|
|
}
|
|
|
|
/**
|
|
* A fast path for creating `lodash` wrapper objects.
|
|
*
|
|
* @private
|
|
* @param {*} value The value to wrap in a `lodash` instance.
|
|
* @param {boolean} chainAll A flag to enable chaining for all methods
|
|
* @returns {Object} Returns a `lodash` instance.
|
|
*/
|
|
function lodashWrapper(value, chainAll) {
|
|
this.__chain__ = !!chainAll;
|
|
this.__wrapped__ = value;
|
|
}
|
|
// ensure `new lodashWrapper` is an instance of `lodash`
|
|
lodashWrapper.prototype = lodash.prototype;
|
|
|
|
/**
|
|
* An object used to flag environments features.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @type Object
|
|
*/
|
|
var support = lodash.support = {};
|
|
|
|
/**
|
|
* Detect if functions can be decompiled by `Function#toString`
|
|
* (all but PS3 and older Opera mobile browsers & avoided in Windows 8 apps).
|
|
*
|
|
* @memberOf _.support
|
|
* @type boolean
|
|
*/
|
|
support.funcDecomp = !reNative.test(context.WinRTError) && reThis.test(runInContext);
|
|
|
|
/**
|
|
* Detect if `Function#name` is supported (all but IE).
|
|
*
|
|
* @memberOf _.support
|
|
* @type boolean
|
|
*/
|
|
support.funcNames = typeof Function.name == 'string';
|
|
|
|
/**
|
|
* By default, the template delimiters used by Lo-Dash are similar to those in
|
|
* embedded Ruby (ERB). Change the following template settings to use alternative
|
|
* delimiters.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @type Object
|
|
*/
|
|
lodash.templateSettings = {
|
|
|
|
/**
|
|
* Used to detect `data` property values to be HTML-escaped.
|
|
*
|
|
* @memberOf _.templateSettings
|
|
* @type RegExp
|
|
*/
|
|
'escape': /<%-([\s\S]+?)%>/g,
|
|
|
|
/**
|
|
* Used to detect code to be evaluated.
|
|
*
|
|
* @memberOf _.templateSettings
|
|
* @type RegExp
|
|
*/
|
|
'evaluate': /<%([\s\S]+?)%>/g,
|
|
|
|
/**
|
|
* Used to detect `data` property values to inject.
|
|
*
|
|
* @memberOf _.templateSettings
|
|
* @type RegExp
|
|
*/
|
|
'interpolate': reInterpolate,
|
|
|
|
/**
|
|
* Used to reference the data object in the template text.
|
|
*
|
|
* @memberOf _.templateSettings
|
|
* @type string
|
|
*/
|
|
'variable': '',
|
|
|
|
/**
|
|
* Used to import variables into the compiled template.
|
|
*
|
|
* @memberOf _.templateSettings
|
|
* @type Object
|
|
*/
|
|
'imports': {
|
|
|
|
/**
|
|
* A reference to the `lodash` function.
|
|
*
|
|
* @memberOf _.templateSettings.imports
|
|
* @type Function
|
|
*/
|
|
'_': lodash
|
|
}
|
|
};
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
|
|
/**
|
|
* The base implementation of `_.bind` that creates the bound function and
|
|
* sets its meta data.
|
|
*
|
|
* @private
|
|
* @param {Array} bindData The bind data array.
|
|
* @returns {Function} Returns the new bound function.
|
|
*/
|
|
function baseBind(bindData) {
|
|
var func = bindData[0],
|
|
partialArgs = bindData[2],
|
|
thisArg = bindData[4];
|
|
|
|
function bound() {
|
|
// `Function#bind` spec
|
|
// http://es5.github.io/#x15.3.4.5
|
|
if (partialArgs) {
|
|
var args = partialArgs.slice();
|
|
push.apply(args, arguments);
|
|
}
|
|
// mimic the constructor's `return` behavior
|
|
// http://es5.github.io/#x13.2.2
|
|
if (this instanceof bound) {
|
|
// ensure `new bound` is an instance of `func`
|
|
var thisBinding = baseCreate(func.prototype),
|
|
result = func.apply(thisBinding, args || arguments);
|
|
return isObject(result) ? result : thisBinding;
|
|
}
|
|
return func.apply(thisArg, args || arguments);
|
|
}
|
|
setBindData(bound, bindData);
|
|
return bound;
|
|
}
|
|
|
|
/**
|
|
* The base implementation of `_.clone` without argument juggling or support
|
|
* for `thisArg` binding.
|
|
*
|
|
* @private
|
|
* @param {*} value The value to clone.
|
|
* @param {boolean} [isDeep=false] Specify a deep clone.
|
|
* @param {Function} [callback] The function to customize cloning values.
|
|
* @param {Array} [stackA=[]] Tracks traversed source objects.
|
|
* @param {Array} [stackB=[]] Associates clones with source counterparts.
|
|
* @returns {*} Returns the cloned value.
|
|
*/
|
|
function baseClone(value, isDeep, callback, stackA, stackB) {
|
|
if (callback) {
|
|
var result = callback(value);
|
|
if (typeof result != 'undefined') {
|
|
return result;
|
|
}
|
|
}
|
|
// inspect [[Class]]
|
|
var isObj = isObject(value);
|
|
if (isObj) {
|
|
var className = toString.call(value);
|
|
if (!cloneableClasses[className]) {
|
|
return value;
|
|
}
|
|
var ctor = ctorByClass[className];
|
|
switch (className) {
|
|
case boolClass:
|
|
case dateClass:
|
|
return new ctor(+value);
|
|
|
|
case numberClass:
|
|
case stringClass:
|
|
return new ctor(value);
|
|
|
|
case regexpClass:
|
|
result = ctor(value.source, reFlags.exec(value));
|
|
result.lastIndex = value.lastIndex;
|
|
return result;
|
|
}
|
|
} else {
|
|
return value;
|
|
}
|
|
var isArr = isArray(value);
|
|
if (isDeep) {
|
|
// check for circular references and return corresponding clone
|
|
var initedStack = !stackA;
|
|
stackA || (stackA = getArray());
|
|
stackB || (stackB = getArray());
|
|
|
|
var length = stackA.length;
|
|
while (length--) {
|
|
if (stackA[length] == value) {
|
|
return stackB[length];
|
|
}
|
|
}
|
|
result = isArr ? ctor(value.length) : {};
|
|
}
|
|
else {
|
|
result = isArr ? slice(value) : assign({}, value);
|
|
}
|
|
// add array properties assigned by `RegExp#exec`
|
|
if (isArr) {
|
|
if (hasOwnProperty.call(value, 'index')) {
|
|
result.index = value.index;
|
|
}
|
|
if (hasOwnProperty.call(value, 'input')) {
|
|
result.input = value.input;
|
|
}
|
|
}
|
|
// exit for shallow clone
|
|
if (!isDeep) {
|
|
return result;
|
|
}
|
|
// add the source value to the stack of traversed objects
|
|
// and associate it with its clone
|
|
stackA.push(value);
|
|
stackB.push(result);
|
|
|
|
// recursively populate clone (susceptible to call stack limits)
|
|
(isArr ? forEach : forOwn)(value, function(objValue, key) {
|
|
result[key] = baseClone(objValue, isDeep, callback, stackA, stackB);
|
|
});
|
|
|
|
if (initedStack) {
|
|
releaseArray(stackA);
|
|
releaseArray(stackB);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* The base implementation of `_.create` without support for assigning
|
|
* properties to the created object.
|
|
*
|
|
* @private
|
|
* @param {Object} prototype The object to inherit from.
|
|
* @returns {Object} Returns the new object.
|
|
*/
|
|
function baseCreate(prototype, properties) {
|
|
return isObject(prototype) ? nativeCreate(prototype) : {};
|
|
}
|
|
// fallback for browsers without `Object.create`
|
|
if (!nativeCreate) {
|
|
baseCreate = (function() {
|
|
function Object() {}
|
|
return function(prototype) {
|
|
if (isObject(prototype)) {
|
|
Object.prototype = prototype;
|
|
var result = new Object;
|
|
Object.prototype = null;
|
|
}
|
|
return result || context.Object();
|
|
};
|
|
}());
|
|
}
|
|
|
|
/**
|
|
* The base implementation of `_.createCallback` without support for creating
|
|
* "_.pluck" or "_.where" style callbacks.
|
|
*
|
|
* @private
|
|
* @param {*} [func=identity] The value to convert to a callback.
|
|
* @param {*} [thisArg] The `this` binding of the created callback.
|
|
* @param {number} [argCount] The number of arguments the callback accepts.
|
|
* @returns {Function} Returns a callback function.
|
|
*/
|
|
function baseCreateCallback(func, thisArg, argCount) {
|
|
if (typeof func != 'function') {
|
|
return identity;
|
|
}
|
|
// exit early for no `thisArg` or already bound by `Function#bind`
|
|
if (typeof thisArg == 'undefined' || !('prototype' in func)) {
|
|
return func;
|
|
}
|
|
var bindData = func.__bindData__;
|
|
if (typeof bindData == 'undefined') {
|
|
if (support.funcNames) {
|
|
bindData = !func.name;
|
|
}
|
|
bindData = bindData || !support.funcDecomp;
|
|
if (!bindData) {
|
|
var source = fnToString.call(func);
|
|
if (!support.funcNames) {
|
|
bindData = !reFuncName.test(source);
|
|
}
|
|
if (!bindData) {
|
|
// checks if `func` references the `this` keyword and stores the result
|
|
bindData = reThis.test(source);
|
|
setBindData(func, bindData);
|
|
}
|
|
}
|
|
}
|
|
// exit early if there are no `this` references or `func` is bound
|
|
if (bindData === false || (bindData !== true && bindData[1] & 1)) {
|
|
return func;
|
|
}
|
|
switch (argCount) {
|
|
case 1: return function(value) {
|
|
return func.call(thisArg, value);
|
|
};
|
|
case 2: return function(a, b) {
|
|
return func.call(thisArg, a, b);
|
|
};
|
|
case 3: return function(value, index, collection) {
|
|
return func.call(thisArg, value, index, collection);
|
|
};
|
|
case 4: return function(accumulator, value, index, collection) {
|
|
return func.call(thisArg, accumulator, value, index, collection);
|
|
};
|
|
}
|
|
return bind(func, thisArg);
|
|
}
|
|
|
|
/**
|
|
* The base implementation of `createWrapper` that creates the wrapper and
|
|
* sets its meta data.
|
|
*
|
|
* @private
|
|
* @param {Array} bindData The bind data array.
|
|
* @returns {Function} Returns the new function.
|
|
*/
|
|
function baseCreateWrapper(bindData) {
|
|
var func = bindData[0],
|
|
bitmask = bindData[1],
|
|
partialArgs = bindData[2],
|
|
partialRightArgs = bindData[3],
|
|
thisArg = bindData[4],
|
|
arity = bindData[5];
|
|
|
|
var isBind = bitmask & 1,
|
|
isBindKey = bitmask & 2,
|
|
isCurry = bitmask & 4,
|
|
isCurryBound = bitmask & 8,
|
|
key = func;
|
|
|
|
function bound() {
|
|
var thisBinding = isBind ? thisArg : this;
|
|
if (partialArgs) {
|
|
var args = partialArgs.slice();
|
|
push.apply(args, arguments);
|
|
}
|
|
if (partialRightArgs || isCurry) {
|
|
args || (args = slice(arguments));
|
|
if (partialRightArgs) {
|
|
push.apply(args, partialRightArgs);
|
|
}
|
|
if (isCurry && args.length < arity) {
|
|
bitmask |= 16 & ~32;
|
|
return baseCreateWrapper([func, (isCurryBound ? bitmask : bitmask & ~3), args, null, thisArg, arity]);
|
|
}
|
|
}
|
|
args || (args = arguments);
|
|
if (isBindKey) {
|
|
func = thisBinding[key];
|
|
}
|
|
if (this instanceof bound) {
|
|
thisBinding = baseCreate(func.prototype);
|
|
var result = func.apply(thisBinding, args);
|
|
return isObject(result) ? result : thisBinding;
|
|
}
|
|
return func.apply(thisBinding, args);
|
|
}
|
|
setBindData(bound, bindData);
|
|
return bound;
|
|
}
|
|
|
|
/**
|
|
* The base implementation of `_.difference` that accepts a single array
|
|
* of values to exclude.
|
|
*
|
|
* @private
|
|
* @param {Array} array The array to process.
|
|
* @param {Array} [values] The array of values to exclude.
|
|
* @returns {Array} Returns a new array of filtered values.
|
|
*/
|
|
function baseDifference(array, values) {
|
|
var index = -1,
|
|
indexOf = getIndexOf(),
|
|
length = array ? array.length : 0,
|
|
isLarge = length >= largeArraySize && indexOf === baseIndexOf,
|
|
result = [];
|
|
|
|
if (isLarge) {
|
|
var cache = createCache(values);
|
|
if (cache) {
|
|
indexOf = cacheIndexOf;
|
|
values = cache;
|
|
} else {
|
|
isLarge = false;
|
|
}
|
|
}
|
|
while (++index < length) {
|
|
var value = array[index];
|
|
if (indexOf(values, value) < 0) {
|
|
result.push(value);
|
|
}
|
|
}
|
|
if (isLarge) {
|
|
releaseObject(values);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* The base implementation of `_.flatten` without support for callback
|
|
* shorthands or `thisArg` binding.
|
|
*
|
|
* @private
|
|
* @param {Array} array The array to flatten.
|
|
* @param {boolean} [isShallow=false] A flag to restrict flattening to a single level.
|
|
* @param {boolean} [isStrict=false] A flag to restrict flattening to arrays and `arguments` objects.
|
|
* @param {number} [fromIndex=0] The index to start from.
|
|
* @returns {Array} Returns a new flattened array.
|
|
*/
|
|
function baseFlatten(array, isShallow, isStrict, fromIndex) {
|
|
var index = (fromIndex || 0) - 1,
|
|
length = array ? array.length : 0,
|
|
result = [];
|
|
|
|
while (++index < length) {
|
|
var value = array[index];
|
|
|
|
if (value && typeof value == 'object' && typeof value.length == 'number'
|
|
&& (isArray(value) || isArguments(value))) {
|
|
// recursively flatten arrays (susceptible to call stack limits)
|
|
if (!isShallow) {
|
|
value = baseFlatten(value, isShallow, isStrict);
|
|
}
|
|
var valIndex = -1,
|
|
valLength = value.length,
|
|
resIndex = result.length;
|
|
|
|
result.length += valLength;
|
|
while (++valIndex < valLength) {
|
|
result[resIndex++] = value[valIndex];
|
|
}
|
|
} else if (!isStrict) {
|
|
result.push(value);
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* The base implementation of `_.isEqual`, without support for `thisArg` binding,
|
|
* that allows partial "_.where" style comparisons.
|
|
*
|
|
* @private
|
|
* @param {*} a The value to compare.
|
|
* @param {*} b The other value to compare.
|
|
* @param {Function} [callback] The function to customize comparing values.
|
|
* @param {Function} [isWhere=false] A flag to indicate performing partial comparisons.
|
|
* @param {Array} [stackA=[]] Tracks traversed `a` objects.
|
|
* @param {Array} [stackB=[]] Tracks traversed `b` objects.
|
|
* @returns {boolean} Returns `true` if the values are equivalent, else `false`.
|
|
*/
|
|
function baseIsEqual(a, b, callback, isWhere, stackA, stackB) {
|
|
// used to indicate that when comparing objects, `a` has at least the properties of `b`
|
|
if (callback) {
|
|
var result = callback(a, b);
|
|
if (typeof result != 'undefined') {
|
|
return !!result;
|
|
}
|
|
}
|
|
// exit early for identical values
|
|
if (a === b) {
|
|
// treat `+0` vs. `-0` as not equal
|
|
return a !== 0 || (1 / a == 1 / b);
|
|
}
|
|
var type = typeof a,
|
|
otherType = typeof b;
|
|
|
|
// exit early for unlike primitive values
|
|
if (a === a &&
|
|
!(a && objectTypes[type]) &&
|
|
!(b && objectTypes[otherType])) {
|
|
return false;
|
|
}
|
|
// exit early for `null` and `undefined` avoiding ES3's Function#call behavior
|
|
// http://es5.github.io/#x15.3.4.4
|
|
if (a == null || b == null) {
|
|
return a === b;
|
|
}
|
|
// compare [[Class]] names
|
|
var className = toString.call(a),
|
|
otherClass = toString.call(b);
|
|
|
|
if (className == argsClass) {
|
|
className = objectClass;
|
|
}
|
|
if (otherClass == argsClass) {
|
|
otherClass = objectClass;
|
|
}
|
|
if (className != otherClass) {
|
|
return false;
|
|
}
|
|
switch (className) {
|
|
case boolClass:
|
|
case dateClass:
|
|
// coerce dates and booleans to numbers, dates to milliseconds and booleans
|
|
// to `1` or `0` treating invalid dates coerced to `NaN` as not equal
|
|
return +a == +b;
|
|
|
|
case numberClass:
|
|
// treat `NaN` vs. `NaN` as equal
|
|
return (a != +a)
|
|
? b != +b
|
|
// but treat `+0` vs. `-0` as not equal
|
|
: (a == 0 ? (1 / a == 1 / b) : a == +b);
|
|
|
|
case regexpClass:
|
|
case stringClass:
|
|
// coerce regexes to strings (http://es5.github.io/#x15.10.6.4)
|
|
// treat string primitives and their corresponding object instances as equal
|
|
return a == String(b);
|
|
}
|
|
var isArr = className == arrayClass;
|
|
if (!isArr) {
|
|
// unwrap any `lodash` wrapped values
|
|
var aWrapped = hasOwnProperty.call(a, '__wrapped__'),
|
|
bWrapped = hasOwnProperty.call(b, '__wrapped__');
|
|
|
|
if (aWrapped || bWrapped) {
|
|
return baseIsEqual(aWrapped ? a.__wrapped__ : a, bWrapped ? b.__wrapped__ : b, callback, isWhere, stackA, stackB);
|
|
}
|
|
// exit for functions and DOM nodes
|
|
if (className != objectClass) {
|
|
return false;
|
|
}
|
|
// in older versions of Opera, `arguments` objects have `Array` constructors
|
|
var ctorA = a.constructor,
|
|
ctorB = b.constructor;
|
|
|
|
// non `Object` object instances with different constructors are not equal
|
|
if (ctorA != ctorB &&
|
|
!(isFunction(ctorA) && ctorA instanceof ctorA && isFunction(ctorB) && ctorB instanceof ctorB) &&
|
|
('constructor' in a && 'constructor' in b)
|
|
) {
|
|
return false;
|
|
}
|
|
}
|
|
// assume cyclic structures are equal
|
|
// the algorithm for detecting cyclic structures is adapted from ES 5.1
|
|
// section 15.12.3, abstract operation `JO` (http://es5.github.io/#x15.12.3)
|
|
var initedStack = !stackA;
|
|
stackA || (stackA = getArray());
|
|
stackB || (stackB = getArray());
|
|
|
|
var length = stackA.length;
|
|
while (length--) {
|
|
if (stackA[length] == a) {
|
|
return stackB[length] == b;
|
|
}
|
|
}
|
|
var size = 0;
|
|
result = true;
|
|
|
|
// add `a` and `b` to the stack of traversed objects
|
|
stackA.push(a);
|
|
stackB.push(b);
|
|
|
|
// recursively compare objects and arrays (susceptible to call stack limits)
|
|
if (isArr) {
|
|
length = a.length;
|
|
size = b.length;
|
|
|
|
// compare lengths to determine if a deep comparison is necessary
|
|
result = size == a.length;
|
|
if (!result && !isWhere) {
|
|
return result;
|
|
}
|
|
// deep compare the contents, ignoring non-numeric properties
|
|
while (size--) {
|
|
var index = length,
|
|
value = b[size];
|
|
|
|
if (isWhere) {
|
|
while (index--) {
|
|
if ((result = baseIsEqual(a[index], value, callback, isWhere, stackA, stackB))) {
|
|
break;
|
|
}
|
|
}
|
|
} else if (!(result = baseIsEqual(a[size], value, callback, isWhere, stackA, stackB))) {
|
|
break;
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
// deep compare objects using `forIn`, instead of `forOwn`, to avoid `Object.keys`
|
|
// which, in this case, is more costly
|
|
forIn(b, function(value, key, b) {
|
|
if (hasOwnProperty.call(b, key)) {
|
|
// count the number of properties.
|
|
size++;
|
|
// deep compare each property value.
|
|
return (result = hasOwnProperty.call(a, key) && baseIsEqual(a[key], value, callback, isWhere, stackA, stackB));
|
|
}
|
|
});
|
|
|
|
if (result && !isWhere) {
|
|
// ensure both objects have the same number of properties
|
|
forIn(a, function(value, key, a) {
|
|
if (hasOwnProperty.call(a, key)) {
|
|
// `size` will be `-1` if `a` has more properties than `b`
|
|
return (result = --size > -1);
|
|
}
|
|
});
|
|
}
|
|
if (initedStack) {
|
|
releaseArray(stackA);
|
|
releaseArray(stackB);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* The base implementation of `_.merge` without argument juggling or support
|
|
* for `thisArg` binding.
|
|
*
|
|
* @private
|
|
* @param {Object} object The destination object.
|
|
* @param {Object} source The source object.
|
|
* @param {Function} [callback] The function to customize merging properties.
|
|
* @param {Array} [stackA=[]] Tracks traversed source objects.
|
|
* @param {Array} [stackB=[]] Associates values with source counterparts.
|
|
*/
|
|
function baseMerge(object, source, callback, stackA, stackB) {
|
|
(isArray(source) ? forEach : forOwn)(source, function(source, key) {
|
|
var found,
|
|
isArr,
|
|
result = source,
|
|
value = object[key];
|
|
|
|
if (source && ((isArr = isArray(source)) || isPlainObject(source))) {
|
|
// avoid merging previously merged cyclic sources
|
|
var stackLength = stackA.length;
|
|
while (stackLength--) {
|
|
if ((found = stackA[stackLength] == source)) {
|
|
value = stackB[stackLength];
|
|
break;
|
|
}
|
|
}
|
|
if (!found) {
|
|
var isShallow;
|
|
if (callback) {
|
|
result = callback(value, source);
|
|
if ((isShallow = typeof result != 'undefined')) {
|
|
value = result;
|
|
}
|
|
}
|
|
if (!isShallow) {
|
|
value = isArr
|
|
? (isArray(value) ? value : [])
|
|
: (isPlainObject(value) ? value : {});
|
|
}
|
|
// add `source` and associated `value` to the stack of traversed objects
|
|
stackA.push(source);
|
|
stackB.push(value);
|
|
|
|
// recursively merge objects and arrays (susceptible to call stack limits)
|
|
if (!isShallow) {
|
|
baseMerge(value, source, callback, stackA, stackB);
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
if (callback) {
|
|
result = callback(value, source);
|
|
if (typeof result == 'undefined') {
|
|
result = source;
|
|
}
|
|
}
|
|
if (typeof result != 'undefined') {
|
|
value = result;
|
|
}
|
|
}
|
|
object[key] = value;
|
|
});
|
|
}
|
|
|
|
/**
|
|
* The base implementation of `_.random` without argument juggling or support
|
|
* for returning floating-point numbers.
|
|
*
|
|
* @private
|
|
* @param {number} min The minimum possible value.
|
|
* @param {number} max The maximum possible value.
|
|
* @returns {number} Returns a random number.
|
|
*/
|
|
function baseRandom(min, max) {
|
|
return min + floor(nativeRandom() * (max - min + 1));
|
|
}
|
|
|
|
/**
|
|
* The base implementation of `_.uniq` without support for callback shorthands
|
|
* or `thisArg` binding.
|
|
*
|
|
* @private
|
|
* @param {Array} array The array to process.
|
|
* @param {boolean} [isSorted=false] A flag to indicate that `array` is sorted.
|
|
* @param {Function} [callback] The function called per iteration.
|
|
* @returns {Array} Returns a duplicate-value-free array.
|
|
*/
|
|
function baseUniq(array, isSorted, callback) {
|
|
var index = -1,
|
|
indexOf = getIndexOf(),
|
|
length = array ? array.length : 0,
|
|
result = [];
|
|
|
|
var isLarge = !isSorted && length >= largeArraySize && indexOf === baseIndexOf,
|
|
seen = (callback || isLarge) ? getArray() : result;
|
|
|
|
if (isLarge) {
|
|
var cache = createCache(seen);
|
|
if (cache) {
|
|
indexOf = cacheIndexOf;
|
|
seen = cache;
|
|
} else {
|
|
isLarge = false;
|
|
seen = callback ? seen : (releaseArray(seen), result);
|
|
}
|
|
}
|
|
while (++index < length) {
|
|
var value = array[index],
|
|
computed = callback ? callback(value, index, array) : value;
|
|
|
|
if (isSorted
|
|
? !index || seen[seen.length - 1] !== computed
|
|
: indexOf(seen, computed) < 0
|
|
) {
|
|
if (callback || isLarge) {
|
|
seen.push(computed);
|
|
}
|
|
result.push(value);
|
|
}
|
|
}
|
|
if (isLarge) {
|
|
releaseArray(seen.array);
|
|
releaseObject(seen);
|
|
} else if (callback) {
|
|
releaseArray(seen);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Creates a function that aggregates a collection, creating an object composed
|
|
* of keys generated from the results of running each element of the collection
|
|
* through a callback. The given `setter` function sets the keys and values
|
|
* of the composed object.
|
|
*
|
|
* @private
|
|
* @param {Function} setter The setter function.
|
|
* @returns {Function} Returns the new aggregator function.
|
|
*/
|
|
function createAggregator(setter) {
|
|
return function(collection, callback, thisArg) {
|
|
var result = {};
|
|
callback = lodash.createCallback(callback, thisArg, 3);
|
|
|
|
var index = -1,
|
|
length = collection ? collection.length : 0;
|
|
|
|
if (typeof length == 'number') {
|
|
while (++index < length) {
|
|
var value = collection[index];
|
|
setter(result, value, callback(value, index, collection), collection);
|
|
}
|
|
} else {
|
|
forOwn(collection, function(value, key, collection) {
|
|
setter(result, value, callback(value, key, collection), collection);
|
|
});
|
|
}
|
|
return result;
|
|
};
|
|
}
|
|
|
|
/**
|
|
* Creates a function that, when called, either curries or invokes `func`
|
|
* with an optional `this` binding and partially applied arguments.
|
|
*
|
|
* @private
|
|
* @param {Function|string} func The function or method name to reference.
|
|
* @param {number} bitmask The bitmask of method flags to compose.
|
|
* The bitmask may be composed of the following flags:
|
|
* 1 - `_.bind`
|
|
* 2 - `_.bindKey`
|
|
* 4 - `_.curry`
|
|
* 8 - `_.curry` (bound)
|
|
* 16 - `_.partial`
|
|
* 32 - `_.partialRight`
|
|
* @param {Array} [partialArgs] An array of arguments to prepend to those
|
|
* provided to the new function.
|
|
* @param {Array} [partialRightArgs] An array of arguments to append to those
|
|
* provided to the new function.
|
|
* @param {*} [thisArg] The `this` binding of `func`.
|
|
* @param {number} [arity] The arity of `func`.
|
|
* @returns {Function} Returns the new function.
|
|
*/
|
|
function createWrapper(func, bitmask, partialArgs, partialRightArgs, thisArg, arity) {
|
|
var isBind = bitmask & 1,
|
|
isBindKey = bitmask & 2,
|
|
isCurry = bitmask & 4,
|
|
isCurryBound = bitmask & 8,
|
|
isPartial = bitmask & 16,
|
|
isPartialRight = bitmask & 32;
|
|
|
|
if (!isBindKey && !isFunction(func)) {
|
|
throw new TypeError;
|
|
}
|
|
if (isPartial && !partialArgs.length) {
|
|
bitmask &= ~16;
|
|
isPartial = partialArgs = false;
|
|
}
|
|
if (isPartialRight && !partialRightArgs.length) {
|
|
bitmask &= ~32;
|
|
isPartialRight = partialRightArgs = false;
|
|
}
|
|
var bindData = func && func.__bindData__;
|
|
if (bindData && bindData !== true) {
|
|
bindData = bindData.slice();
|
|
|
|
// set `thisBinding` is not previously bound
|
|
if (isBind && !(bindData[1] & 1)) {
|
|
bindData[4] = thisArg;
|
|
}
|
|
// set if previously bound but not currently (subsequent curried functions)
|
|
if (!isBind && bindData[1] & 1) {
|
|
bitmask |= 8;
|
|
}
|
|
// set curried arity if not yet set
|
|
if (isCurry && !(bindData[1] & 4)) {
|
|
bindData[5] = arity;
|
|
}
|
|
// append partial left arguments
|
|
if (isPartial) {
|
|
push.apply(bindData[2] || (bindData[2] = []), partialArgs);
|
|
}
|
|
// append partial right arguments
|
|
if (isPartialRight) {
|
|
push.apply(bindData[3] || (bindData[3] = []), partialRightArgs);
|
|
}
|
|
// merge flags
|
|
bindData[1] |= bitmask;
|
|
return createWrapper.apply(null, bindData);
|
|
}
|
|
// fast path for `_.bind`
|
|
var creater = (bitmask == 1 || bitmask === 17) ? baseBind : baseCreateWrapper;
|
|
return creater([func, bitmask, partialArgs, partialRightArgs, thisArg, arity]);
|
|
}
|
|
|
|
/**
|
|
* Used by `escape` to convert characters to HTML entities.
|
|
*
|
|
* @private
|
|
* @param {string} match The matched character to escape.
|
|
* @returns {string} Returns the escaped character.
|
|
*/
|
|
function escapeHtmlChar(match) {
|
|
return htmlEscapes[match];
|
|
}
|
|
|
|
/**
|
|
* Gets the appropriate "indexOf" function. If the `_.indexOf` method is
|
|
* customized, this method returns the custom method, otherwise it returns
|
|
* the `baseIndexOf` function.
|
|
*
|
|
* @private
|
|
* @returns {Function} Returns the "indexOf" function.
|
|
*/
|
|
function getIndexOf() {
|
|
var result = (result = lodash.indexOf) === indexOf ? baseIndexOf : result;
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Sets `this` binding data on a given function.
|
|
*
|
|
* @private
|
|
* @param {Function} func The function to set data on.
|
|
* @param {Array} value The data array to set.
|
|
*/
|
|
var setBindData = !defineProperty ? noop : function(func, value) {
|
|
descriptor.value = value;
|
|
defineProperty(func, '__bindData__', descriptor);
|
|
};
|
|
|
|
/**
|
|
* A fallback implementation of `isPlainObject` which checks if a given value
|
|
* is an object created by the `Object` constructor, assuming objects created
|
|
* by the `Object` constructor have no inherited enumerable properties and that
|
|
* there are no `Object.prototype` extensions.
|
|
*
|
|
* @private
|
|
* @param {*} value The value to check.
|
|
* @returns {boolean} Returns `true` if `value` is a plain object, else `false`.
|
|
*/
|
|
function shimIsPlainObject(value) {
|
|
var ctor,
|
|
result;
|
|
|
|
// avoid non Object objects, `arguments` objects, and DOM elements
|
|
if (!(value && toString.call(value) == objectClass) ||
|
|
(ctor = value.constructor, isFunction(ctor) && !(ctor instanceof ctor))) {
|
|
return false;
|
|
}
|
|
// In most environments an object's own properties are iterated before
|
|
// its inherited properties. If the last iterated property is an object's
|
|
// own property then there are no inherited enumerable properties.
|
|
forIn(value, function(value, key) {
|
|
result = key;
|
|
});
|
|
return typeof result == 'undefined' || hasOwnProperty.call(value, result);
|
|
}
|
|
|
|
/**
|
|
* Used by `unescape` to convert HTML entities to characters.
|
|
*
|
|
* @private
|
|
* @param {string} match The matched character to unescape.
|
|
* @returns {string} Returns the unescaped character.
|
|
*/
|
|
function unescapeHtmlChar(match) {
|
|
return htmlUnescapes[match];
|
|
}
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
|
|
/**
|
|
* Checks if `value` is an `arguments` object.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Objects
|
|
* @param {*} value The value to check.
|
|
* @returns {boolean} Returns `true` if the `value` is an `arguments` object, else `false`.
|
|
* @example
|
|
*
|
|
* (function() { return _.isArguments(arguments); })(1, 2, 3);
|
|
* // => true
|
|
*
|
|
* _.isArguments([1, 2, 3]);
|
|
* // => false
|
|
*/
|
|
function isArguments(value) {
|
|
return value && typeof value == 'object' && typeof value.length == 'number' &&
|
|
toString.call(value) == argsClass || false;
|
|
}
|
|
|
|
/**
|
|
* Checks if `value` is an array.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @type Function
|
|
* @category Objects
|
|
* @param {*} value The value to check.
|
|
* @returns {boolean} Returns `true` if the `value` is an array, else `false`.
|
|
* @example
|
|
*
|
|
* (function() { return _.isArray(arguments); })();
|
|
* // => false
|
|
*
|
|
* _.isArray([1, 2, 3]);
|
|
* // => true
|
|
*/
|
|
var isArray = nativeIsArray || function(value) {
|
|
return value && typeof value == 'object' && typeof value.length == 'number' &&
|
|
toString.call(value) == arrayClass || false;
|
|
};
|
|
|
|
/**
|
|
* A fallback implementation of `Object.keys` which produces an array of the
|
|
* given object's own enumerable property names.
|
|
*
|
|
* @private
|
|
* @type Function
|
|
* @param {Object} object The object to inspect.
|
|
* @returns {Array} Returns an array of property names.
|
|
*/
|
|
var shimKeys = function(object) {
|
|
var index, iterable = object, result = [];
|
|
if (!iterable) return result;
|
|
if (!(objectTypes[typeof object])) return result;
|
|
for (index in iterable) {
|
|
if (hasOwnProperty.call(iterable, index)) {
|
|
result.push(index);
|
|
}
|
|
}
|
|
return result
|
|
};
|
|
|
|
/**
|
|
* Creates an array composed of the own enumerable property names of an object.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Objects
|
|
* @param {Object} object The object to inspect.
|
|
* @returns {Array} Returns an array of property names.
|
|
* @example
|
|
*
|
|
* _.keys({ 'one': 1, 'two': 2, 'three': 3 });
|
|
* // => ['one', 'two', 'three'] (property order is not guaranteed across environments)
|
|
*/
|
|
var keys = !nativeKeys ? shimKeys : function(object) {
|
|
if (!isObject(object)) {
|
|
return [];
|
|
}
|
|
return nativeKeys(object);
|
|
};
|
|
|
|
/**
|
|
* Used to convert characters to HTML entities:
|
|
*
|
|
* Though the `>` character is escaped for symmetry, characters like `>` and `/`
|
|
* don't require escaping in HTML and have no special meaning unless they're part
|
|
* of a tag or an unquoted attribute value.
|
|
* http://mathiasbynens.be/notes/ambiguous-ampersands (under "semi-related fun fact")
|
|
*/
|
|
var htmlEscapes = {
|
|
'&': '&',
|
|
'<': '<',
|
|
'>': '>',
|
|
'"': '"',
|
|
"'": '''
|
|
};
|
|
|
|
/** Used to convert HTML entities to characters */
|
|
var htmlUnescapes = invert(htmlEscapes);
|
|
|
|
/** Used to match HTML entities and HTML characters */
|
|
var reEscapedHtml = RegExp('(' + keys(htmlUnescapes).join('|') + ')', 'g'),
|
|
reUnescapedHtml = RegExp('[' + keys(htmlEscapes).join('') + ']', 'g');
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
|
|
/**
|
|
* Assigns own enumerable properties of source object(s) to the destination
|
|
* object. Subsequent sources will overwrite property assignments of previous
|
|
* sources. If a callback is provided it will be executed to produce the
|
|
* assigned values. The callback is bound to `thisArg` and invoked with two
|
|
* arguments; (objectValue, sourceValue).
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @type Function
|
|
* @alias extend
|
|
* @category Objects
|
|
* @param {Object} object The destination object.
|
|
* @param {...Object} [source] The source objects.
|
|
* @param {Function} [callback] The function to customize assigning values.
|
|
* @param {*} [thisArg] The `this` binding of `callback`.
|
|
* @returns {Object} Returns the destination object.
|
|
* @example
|
|
*
|
|
* _.assign({ 'name': 'fred' }, { 'employer': 'slate' });
|
|
* // => { 'name': 'fred', 'employer': 'slate' }
|
|
*
|
|
* var defaults = _.partialRight(_.assign, function(a, b) {
|
|
* return typeof a == 'undefined' ? b : a;
|
|
* });
|
|
*
|
|
* var object = { 'name': 'barney' };
|
|
* defaults(object, { 'name': 'fred', 'employer': 'slate' });
|
|
* // => { 'name': 'barney', 'employer': 'slate' }
|
|
*/
|
|
var assign = function(object, source, guard) {
|
|
var index, iterable = object, result = iterable;
|
|
if (!iterable) return result;
|
|
var args = arguments,
|
|
argsIndex = 0,
|
|
argsLength = typeof guard == 'number' ? 2 : args.length;
|
|
if (argsLength > 3 && typeof args[argsLength - 2] == 'function') {
|
|
var callback = baseCreateCallback(args[--argsLength - 1], args[argsLength--], 2);
|
|
} else if (argsLength > 2 && typeof args[argsLength - 1] == 'function') {
|
|
callback = args[--argsLength];
|
|
}
|
|
while (++argsIndex < argsLength) {
|
|
iterable = args[argsIndex];
|
|
if (iterable && objectTypes[typeof iterable]) {
|
|
var ownIndex = -1,
|
|
ownProps = objectTypes[typeof iterable] && keys(iterable),
|
|
length = ownProps ? ownProps.length : 0;
|
|
|
|
while (++ownIndex < length) {
|
|
index = ownProps[ownIndex];
|
|
result[index] = callback ? callback(result[index], iterable[index]) : iterable[index];
|
|
}
|
|
}
|
|
}
|
|
return result
|
|
};
|
|
|
|
/**
|
|
* Creates a clone of `value`. If `isDeep` is `true` nested objects will also
|
|
* be cloned, otherwise they will be assigned by reference. If a callback
|
|
* is provided it will be executed to produce the cloned values. If the
|
|
* callback returns `undefined` cloning will be handled by the method instead.
|
|
* The callback is bound to `thisArg` and invoked with one argument; (value).
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Objects
|
|
* @param {*} value The value to clone.
|
|
* @param {boolean} [isDeep=false] Specify a deep clone.
|
|
* @param {Function} [callback] The function to customize cloning values.
|
|
* @param {*} [thisArg] The `this` binding of `callback`.
|
|
* @returns {*} Returns the cloned value.
|
|
* @example
|
|
*
|
|
* var characters = [
|
|
* { 'name': 'barney', 'age': 36 },
|
|
* { 'name': 'fred', 'age': 40 }
|
|
* ];
|
|
*
|
|
* var shallow = _.clone(characters);
|
|
* shallow[0] === characters[0];
|
|
* // => true
|
|
*
|
|
* var deep = _.clone(characters, true);
|
|
* deep[0] === characters[0];
|
|
* // => false
|
|
*
|
|
* _.mixin({
|
|
* 'clone': _.partialRight(_.clone, function(value) {
|
|
* return _.isElement(value) ? value.cloneNode(false) : undefined;
|
|
* })
|
|
* });
|
|
*
|
|
* var clone = _.clone(document.body);
|
|
* clone.childNodes.length;
|
|
* // => 0
|
|
*/
|
|
function clone(value, isDeep, callback, thisArg) {
|
|
// allows working with "Collections" methods without using their `index`
|
|
// and `collection` arguments for `isDeep` and `callback`
|
|
if (typeof isDeep != 'boolean' && isDeep != null) {
|
|
thisArg = callback;
|
|
callback = isDeep;
|
|
isDeep = false;
|
|
}
|
|
return baseClone(value, isDeep, typeof callback == 'function' && baseCreateCallback(callback, thisArg, 1));
|
|
}
|
|
|
|
/**
|
|
* Creates a deep clone of `value`. If a callback is provided it will be
|
|
* executed to produce the cloned values. If the callback returns `undefined`
|
|
* cloning will be handled by the method instead. The callback is bound to
|
|
* `thisArg` and invoked with one argument; (value).
|
|
*
|
|
* Note: This method is loosely based on the structured clone algorithm. Functions
|
|
* and DOM nodes are **not** cloned. The enumerable properties of `arguments` objects and
|
|
* objects created by constructors other than `Object` are cloned to plain `Object` objects.
|
|
* See http://www.w3.org/TR/html5/infrastructure.html#internal-structured-cloning-algorithm.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Objects
|
|
* @param {*} value The value to deep clone.
|
|
* @param {Function} [callback] The function to customize cloning values.
|
|
* @param {*} [thisArg] The `this` binding of `callback`.
|
|
* @returns {*} Returns the deep cloned value.
|
|
* @example
|
|
*
|
|
* var characters = [
|
|
* { 'name': 'barney', 'age': 36 },
|
|
* { 'name': 'fred', 'age': 40 }
|
|
* ];
|
|
*
|
|
* var deep = _.cloneDeep(characters);
|
|
* deep[0] === characters[0];
|
|
* // => false
|
|
*
|
|
* var view = {
|
|
* 'label': 'docs',
|
|
* 'node': element
|
|
* };
|
|
*
|
|
* var clone = _.cloneDeep(view, function(value) {
|
|
* return _.isElement(value) ? value.cloneNode(true) : undefined;
|
|
* });
|
|
*
|
|
* clone.node == view.node;
|
|
* // => false
|
|
*/
|
|
function cloneDeep(value, callback, thisArg) {
|
|
return baseClone(value, true, typeof callback == 'function' && baseCreateCallback(callback, thisArg, 1));
|
|
}
|
|
|
|
/**
|
|
* Creates an object that inherits from the given `prototype` object. If a
|
|
* `properties` object is provided its own enumerable properties are assigned
|
|
* to the created object.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Objects
|
|
* @param {Object} prototype The object to inherit from.
|
|
* @param {Object} [properties] The properties to assign to the object.
|
|
* @returns {Object} Returns the new object.
|
|
* @example
|
|
*
|
|
* function Shape() {
|
|
* this.x = 0;
|
|
* this.y = 0;
|
|
* }
|
|
*
|
|
* function Circle() {
|
|
* Shape.call(this);
|
|
* }
|
|
*
|
|
* Circle.prototype = _.create(Shape.prototype, { 'constructor': Circle });
|
|
*
|
|
* var circle = new Circle;
|
|
* circle instanceof Circle;
|
|
* // => true
|
|
*
|
|
* circle instanceof Shape;
|
|
* // => true
|
|
*/
|
|
function create(prototype, properties) {
|
|
var result = baseCreate(prototype);
|
|
return properties ? assign(result, properties) : result;
|
|
}
|
|
|
|
/**
|
|
* Assigns own enumerable properties of source object(s) to the destination
|
|
* object for all destination properties that resolve to `undefined`. Once a
|
|
* property is set, additional defaults of the same property will be ignored.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @type Function
|
|
* @category Objects
|
|
* @param {Object} object The destination object.
|
|
* @param {...Object} [source] The source objects.
|
|
* @param- {Object} [guard] Allows working with `_.reduce` without using its
|
|
* `key` and `object` arguments as sources.
|
|
* @returns {Object} Returns the destination object.
|
|
* @example
|
|
*
|
|
* var object = { 'name': 'barney' };
|
|
* _.defaults(object, { 'name': 'fred', 'employer': 'slate' });
|
|
* // => { 'name': 'barney', 'employer': 'slate' }
|
|
*/
|
|
var defaults = function(object, source, guard) {
|
|
var index, iterable = object, result = iterable;
|
|
if (!iterable) return result;
|
|
var args = arguments,
|
|
argsIndex = 0,
|
|
argsLength = typeof guard == 'number' ? 2 : args.length;
|
|
while (++argsIndex < argsLength) {
|
|
iterable = args[argsIndex];
|
|
if (iterable && objectTypes[typeof iterable]) {
|
|
var ownIndex = -1,
|
|
ownProps = objectTypes[typeof iterable] && keys(iterable),
|
|
length = ownProps ? ownProps.length : 0;
|
|
|
|
while (++ownIndex < length) {
|
|
index = ownProps[ownIndex];
|
|
if (typeof result[index] == 'undefined') result[index] = iterable[index];
|
|
}
|
|
}
|
|
}
|
|
return result
|
|
};
|
|
|
|
/**
|
|
* This method is like `_.findIndex` except that it returns the key of the
|
|
* first element that passes the callback check, instead of the element itself.
|
|
*
|
|
* If a property name is provided for `callback` the created "_.pluck" style
|
|
* callback will return the property value of the given element.
|
|
*
|
|
* If an object is provided for `callback` the created "_.where" style callback
|
|
* will return `true` for elements that have the properties of the given object,
|
|
* else `false`.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Objects
|
|
* @param {Object} object The object to search.
|
|
* @param {Function|Object|string} [callback=identity] The function called per
|
|
* iteration. If a property name or object is provided it will be used to
|
|
* create a "_.pluck" or "_.where" style callback, respectively.
|
|
* @param {*} [thisArg] The `this` binding of `callback`.
|
|
* @returns {string|undefined} Returns the key of the found element, else `undefined`.
|
|
* @example
|
|
*
|
|
* var characters = {
|
|
* 'barney': { 'age': 36, 'blocked': false },
|
|
* 'fred': { 'age': 40, 'blocked': true },
|
|
* 'pebbles': { 'age': 1, 'blocked': false }
|
|
* };
|
|
*
|
|
* _.findKey(characters, function(chr) {
|
|
* return chr.age < 40;
|
|
* });
|
|
* // => 'barney' (property order is not guaranteed across environments)
|
|
*
|
|
* // using "_.where" callback shorthand
|
|
* _.findKey(characters, { 'age': 1 });
|
|
* // => 'pebbles'
|
|
*
|
|
* // using "_.pluck" callback shorthand
|
|
* _.findKey(characters, 'blocked');
|
|
* // => 'fred'
|
|
*/
|
|
function findKey(object, callback, thisArg) {
|
|
var result;
|
|
callback = lodash.createCallback(callback, thisArg, 3);
|
|
forOwn(object, function(value, key, object) {
|
|
if (callback(value, key, object)) {
|
|
result = key;
|
|
return false;
|
|
}
|
|
});
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* This method is like `_.findKey` except that it iterates over elements
|
|
* of a `collection` in the opposite order.
|
|
*
|
|
* If a property name is provided for `callback` the created "_.pluck" style
|
|
* callback will return the property value of the given element.
|
|
*
|
|
* If an object is provided for `callback` the created "_.where" style callback
|
|
* will return `true` for elements that have the properties of the given object,
|
|
* else `false`.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Objects
|
|
* @param {Object} object The object to search.
|
|
* @param {Function|Object|string} [callback=identity] The function called per
|
|
* iteration. If a property name or object is provided it will be used to
|
|
* create a "_.pluck" or "_.where" style callback, respectively.
|
|
* @param {*} [thisArg] The `this` binding of `callback`.
|
|
* @returns {string|undefined} Returns the key of the found element, else `undefined`.
|
|
* @example
|
|
*
|
|
* var characters = {
|
|
* 'barney': { 'age': 36, 'blocked': true },
|
|
* 'fred': { 'age': 40, 'blocked': false },
|
|
* 'pebbles': { 'age': 1, 'blocked': true }
|
|
* };
|
|
*
|
|
* _.findLastKey(characters, function(chr) {
|
|
* return chr.age < 40;
|
|
* });
|
|
* // => returns `pebbles`, assuming `_.findKey` returns `barney`
|
|
*
|
|
* // using "_.where" callback shorthand
|
|
* _.findLastKey(characters, { 'age': 40 });
|
|
* // => 'fred'
|
|
*
|
|
* // using "_.pluck" callback shorthand
|
|
* _.findLastKey(characters, 'blocked');
|
|
* // => 'pebbles'
|
|
*/
|
|
function findLastKey(object, callback, thisArg) {
|
|
var result;
|
|
callback = lodash.createCallback(callback, thisArg, 3);
|
|
forOwnRight(object, function(value, key, object) {
|
|
if (callback(value, key, object)) {
|
|
result = key;
|
|
return false;
|
|
}
|
|
});
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Iterates over own and inherited enumerable properties of an object,
|
|
* executing the callback for each property. The callback is bound to `thisArg`
|
|
* and invoked with three arguments; (value, key, object). Callbacks may exit
|
|
* iteration early by explicitly returning `false`.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @type Function
|
|
* @category Objects
|
|
* @param {Object} object The object to iterate over.
|
|
* @param {Function} [callback=identity] The function called per iteration.
|
|
* @param {*} [thisArg] The `this` binding of `callback`.
|
|
* @returns {Object} Returns `object`.
|
|
* @example
|
|
*
|
|
* function Shape() {
|
|
* this.x = 0;
|
|
* this.y = 0;
|
|
* }
|
|
*
|
|
* Shape.prototype.move = function(x, y) {
|
|
* this.x += x;
|
|
* this.y += y;
|
|
* };
|
|
*
|
|
* _.forIn(new Shape, function(value, key) {
|
|
* console.log(key);
|
|
* });
|
|
* // => logs 'x', 'y', and 'move' (property order is not guaranteed across environments)
|
|
*/
|
|
var forIn = function(collection, callback, thisArg) {
|
|
var index, iterable = collection, result = iterable;
|
|
if (!iterable) return result;
|
|
if (!objectTypes[typeof iterable]) return result;
|
|
callback = callback && typeof thisArg == 'undefined' ? callback : baseCreateCallback(callback, thisArg, 3);
|
|
for (index in iterable) {
|
|
if (callback(iterable[index], index, collection) === false) return result;
|
|
}
|
|
return result
|
|
};
|
|
|
|
/**
|
|
* This method is like `_.forIn` except that it iterates over elements
|
|
* of a `collection` in the opposite order.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Objects
|
|
* @param {Object} object The object to iterate over.
|
|
* @param {Function} [callback=identity] The function called per iteration.
|
|
* @param {*} [thisArg] The `this` binding of `callback`.
|
|
* @returns {Object} Returns `object`.
|
|
* @example
|
|
*
|
|
* function Shape() {
|
|
* this.x = 0;
|
|
* this.y = 0;
|
|
* }
|
|
*
|
|
* Shape.prototype.move = function(x, y) {
|
|
* this.x += x;
|
|
* this.y += y;
|
|
* };
|
|
*
|
|
* _.forInRight(new Shape, function(value, key) {
|
|
* console.log(key);
|
|
* });
|
|
* // => logs 'move', 'y', and 'x' assuming `_.forIn ` logs 'x', 'y', and 'move'
|
|
*/
|
|
function forInRight(object, callback, thisArg) {
|
|
var pairs = [];
|
|
|
|
forIn(object, function(value, key) {
|
|
pairs.push(key, value);
|
|
});
|
|
|
|
var length = pairs.length;
|
|
callback = baseCreateCallback(callback, thisArg, 3);
|
|
while (length--) {
|
|
if (callback(pairs[length--], pairs[length], object) === false) {
|
|
break;
|
|
}
|
|
}
|
|
return object;
|
|
}
|
|
|
|
/**
|
|
* Iterates over own enumerable properties of an object, executing the callback
|
|
* for each property. The callback is bound to `thisArg` and invoked with three
|
|
* arguments; (value, key, object). Callbacks may exit iteration early by
|
|
* explicitly returning `false`.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @type Function
|
|
* @category Objects
|
|
* @param {Object} object The object to iterate over.
|
|
* @param {Function} [callback=identity] The function called per iteration.
|
|
* @param {*} [thisArg] The `this` binding of `callback`.
|
|
* @returns {Object} Returns `object`.
|
|
* @example
|
|
*
|
|
* _.forOwn({ '0': 'zero', '1': 'one', 'length': 2 }, function(num, key) {
|
|
* console.log(key);
|
|
* });
|
|
* // => logs '0', '1', and 'length' (property order is not guaranteed across environments)
|
|
*/
|
|
var forOwn = function(collection, callback, thisArg) {
|
|
var index, iterable = collection, result = iterable;
|
|
if (!iterable) return result;
|
|
if (!objectTypes[typeof iterable]) return result;
|
|
callback = callback && typeof thisArg == 'undefined' ? callback : baseCreateCallback(callback, thisArg, 3);
|
|
var ownIndex = -1,
|
|
ownProps = objectTypes[typeof iterable] && keys(iterable),
|
|
length = ownProps ? ownProps.length : 0;
|
|
|
|
while (++ownIndex < length) {
|
|
index = ownProps[ownIndex];
|
|
if (callback(iterable[index], index, collection) === false) return result;
|
|
}
|
|
return result
|
|
};
|
|
|
|
/**
|
|
* This method is like `_.forOwn` except that it iterates over elements
|
|
* of a `collection` in the opposite order.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Objects
|
|
* @param {Object} object The object to iterate over.
|
|
* @param {Function} [callback=identity] The function called per iteration.
|
|
* @param {*} [thisArg] The `this` binding of `callback`.
|
|
* @returns {Object} Returns `object`.
|
|
* @example
|
|
*
|
|
* _.forOwnRight({ '0': 'zero', '1': 'one', 'length': 2 }, function(num, key) {
|
|
* console.log(key);
|
|
* });
|
|
* // => logs 'length', '1', and '0' assuming `_.forOwn` logs '0', '1', and 'length'
|
|
*/
|
|
function forOwnRight(object, callback, thisArg) {
|
|
var props = keys(object),
|
|
length = props.length;
|
|
|
|
callback = baseCreateCallback(callback, thisArg, 3);
|
|
while (length--) {
|
|
var key = props[length];
|
|
if (callback(object[key], key, object) === false) {
|
|
break;
|
|
}
|
|
}
|
|
return object;
|
|
}
|
|
|
|
/**
|
|
* Creates a sorted array of property names of all enumerable properties,
|
|
* own and inherited, of `object` that have function values.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @alias methods
|
|
* @category Objects
|
|
* @param {Object} object The object to inspect.
|
|
* @returns {Array} Returns an array of property names that have function values.
|
|
* @example
|
|
*
|
|
* _.functions(_);
|
|
* // => ['all', 'any', 'bind', 'bindAll', 'clone', 'compact', 'compose', ...]
|
|
*/
|
|
function functions(object) {
|
|
var result = [];
|
|
forIn(object, function(value, key) {
|
|
if (isFunction(value)) {
|
|
result.push(key);
|
|
}
|
|
});
|
|
return result.sort();
|
|
}
|
|
|
|
/**
|
|
* Checks if the specified object `property` exists and is a direct property,
|
|
* instead of an inherited property.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Objects
|
|
* @param {Object} object The object to check.
|
|
* @param {string} property The property to check for.
|
|
* @returns {boolean} Returns `true` if key is a direct property, else `false`.
|
|
* @example
|
|
*
|
|
* _.has({ 'a': 1, 'b': 2, 'c': 3 }, 'b');
|
|
* // => true
|
|
*/
|
|
function has(object, property) {
|
|
return object ? hasOwnProperty.call(object, property) : false;
|
|
}
|
|
|
|
/**
|
|
* Creates an object composed of the inverted keys and values of the given object.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Objects
|
|
* @param {Object} object The object to invert.
|
|
* @returns {Object} Returns the created inverted object.
|
|
* @example
|
|
*
|
|
* _.invert({ 'first': 'fred', 'second': 'barney' });
|
|
* // => { 'fred': 'first', 'barney': 'second' }
|
|
*/
|
|
function invert(object) {
|
|
var index = -1,
|
|
props = keys(object),
|
|
length = props.length,
|
|
result = {};
|
|
|
|
while (++index < length) {
|
|
var key = props[index];
|
|
result[object[key]] = key;
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Checks if `value` is a boolean value.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Objects
|
|
* @param {*} value The value to check.
|
|
* @returns {boolean} Returns `true` if the `value` is a boolean value, else `false`.
|
|
* @example
|
|
*
|
|
* _.isBoolean(null);
|
|
* // => false
|
|
*/
|
|
function isBoolean(value) {
|
|
return value === true || value === false ||
|
|
value && typeof value == 'object' && toString.call(value) == boolClass || false;
|
|
}
|
|
|
|
/**
|
|
* Checks if `value` is a date.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Objects
|
|
* @param {*} value The value to check.
|
|
* @returns {boolean} Returns `true` if the `value` is a date, else `false`.
|
|
* @example
|
|
*
|
|
* _.isDate(new Date);
|
|
* // => true
|
|
*/
|
|
function isDate(value) {
|
|
return value && typeof value == 'object' && toString.call(value) == dateClass || false;
|
|
}
|
|
|
|
/**
|
|
* Checks if `value` is a DOM element.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Objects
|
|
* @param {*} value The value to check.
|
|
* @returns {boolean} Returns `true` if the `value` is a DOM element, else `false`.
|
|
* @example
|
|
*
|
|
* _.isElement(document.body);
|
|
* // => true
|
|
*/
|
|
function isElement(value) {
|
|
return value && value.nodeType === 1 || false;
|
|
}
|
|
|
|
/**
|
|
* Checks if `value` is empty. Arrays, strings, or `arguments` objects with a
|
|
* length of `0` and objects with no own enumerable properties are considered
|
|
* "empty".
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Objects
|
|
* @param {Array|Object|string} value The value to inspect.
|
|
* @returns {boolean} Returns `true` if the `value` is empty, else `false`.
|
|
* @example
|
|
*
|
|
* _.isEmpty([1, 2, 3]);
|
|
* // => false
|
|
*
|
|
* _.isEmpty({});
|
|
* // => true
|
|
*
|
|
* _.isEmpty('');
|
|
* // => true
|
|
*/
|
|
function isEmpty(value) {
|
|
var result = true;
|
|
if (!value) {
|
|
return result;
|
|
}
|
|
var className = toString.call(value),
|
|
length = value.length;
|
|
|
|
if ((className == arrayClass || className == stringClass || className == argsClass ) ||
|
|
(className == objectClass && typeof length == 'number' && isFunction(value.splice))) {
|
|
return !length;
|
|
}
|
|
forOwn(value, function() {
|
|
return (result = false);
|
|
});
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Performs a deep comparison between two values to determine if they are
|
|
* equivalent to each other. If a callback is provided it will be executed
|
|
* to compare values. If the callback returns `undefined` comparisons will
|
|
* be handled by the method instead. The callback is bound to `thisArg` and
|
|
* invoked with two arguments; (a, b).
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Objects
|
|
* @param {*} a The value to compare.
|
|
* @param {*} b The other value to compare.
|
|
* @param {Function} [callback] The function to customize comparing values.
|
|
* @param {*} [thisArg] The `this` binding of `callback`.
|
|
* @returns {boolean} Returns `true` if the values are equivalent, else `false`.
|
|
* @example
|
|
*
|
|
* var object = { 'name': 'fred' };
|
|
* var copy = { 'name': 'fred' };
|
|
*
|
|
* object == copy;
|
|
* // => false
|
|
*
|
|
* _.isEqual(object, copy);
|
|
* // => true
|
|
*
|
|
* var words = ['hello', 'goodbye'];
|
|
* var otherWords = ['hi', 'goodbye'];
|
|
*
|
|
* _.isEqual(words, otherWords, function(a, b) {
|
|
* var reGreet = /^(?:hello|hi)$/i,
|
|
* aGreet = _.isString(a) && reGreet.test(a),
|
|
* bGreet = _.isString(b) && reGreet.test(b);
|
|
*
|
|
* return (aGreet || bGreet) ? (aGreet == bGreet) : undefined;
|
|
* });
|
|
* // => true
|
|
*/
|
|
function isEqual(a, b, callback, thisArg) {
|
|
return baseIsEqual(a, b, typeof callback == 'function' && baseCreateCallback(callback, thisArg, 2));
|
|
}
|
|
|
|
/**
|
|
* Checks if `value` is, or can be coerced to, a finite number.
|
|
*
|
|
* Note: This is not the same as native `isFinite` which will return true for
|
|
* booleans and empty strings. See http://es5.github.io/#x15.1.2.5.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Objects
|
|
* @param {*} value The value to check.
|
|
* @returns {boolean} Returns `true` if the `value` is finite, else `false`.
|
|
* @example
|
|
*
|
|
* _.isFinite(-101);
|
|
* // => true
|
|
*
|
|
* _.isFinite('10');
|
|
* // => true
|
|
*
|
|
* _.isFinite(true);
|
|
* // => false
|
|
*
|
|
* _.isFinite('');
|
|
* // => false
|
|
*
|
|
* _.isFinite(Infinity);
|
|
* // => false
|
|
*/
|
|
function isFinite(value) {
|
|
return nativeIsFinite(value) && !nativeIsNaN(parseFloat(value));
|
|
}
|
|
|
|
/**
|
|
* Checks if `value` is a function.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Objects
|
|
* @param {*} value The value to check.
|
|
* @returns {boolean} Returns `true` if the `value` is a function, else `false`.
|
|
* @example
|
|
*
|
|
* _.isFunction(_);
|
|
* // => true
|
|
*/
|
|
function isFunction(value) {
|
|
return typeof value == 'function';
|
|
}
|
|
|
|
/**
|
|
* Checks if `value` is the language type of Object.
|
|
* (e.g. arrays, functions, objects, regexes, `new Number(0)`, and `new String('')`)
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Objects
|
|
* @param {*} value The value to check.
|
|
* @returns {boolean} Returns `true` if the `value` is an object, else `false`.
|
|
* @example
|
|
*
|
|
* _.isObject({});
|
|
* // => true
|
|
*
|
|
* _.isObject([1, 2, 3]);
|
|
* // => true
|
|
*
|
|
* _.isObject(1);
|
|
* // => false
|
|
*/
|
|
function isObject(value) {
|
|
// check if the value is the ECMAScript language type of Object
|
|
// http://es5.github.io/#x8
|
|
// and avoid a V8 bug
|
|
// http://code.google.com/p/v8/issues/detail?id=2291
|
|
return !!(value && objectTypes[typeof value]);
|
|
}
|
|
|
|
/**
|
|
* Checks if `value` is `NaN`.
|
|
*
|
|
* Note: This is not the same as native `isNaN` which will return `true` for
|
|
* `undefined` and other non-numeric values. See http://es5.github.io/#x15.1.2.4.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Objects
|
|
* @param {*} value The value to check.
|
|
* @returns {boolean} Returns `true` if the `value` is `NaN`, else `false`.
|
|
* @example
|
|
*
|
|
* _.isNaN(NaN);
|
|
* // => true
|
|
*
|
|
* _.isNaN(new Number(NaN));
|
|
* // => true
|
|
*
|
|
* isNaN(undefined);
|
|
* // => true
|
|
*
|
|
* _.isNaN(undefined);
|
|
* // => false
|
|
*/
|
|
function isNaN(value) {
|
|
// `NaN` as a primitive is the only value that is not equal to itself
|
|
// (perform the [[Class]] check first to avoid errors with some host objects in IE)
|
|
return isNumber(value) && value != +value;
|
|
}
|
|
|
|
/**
|
|
* Checks if `value` is `null`.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Objects
|
|
* @param {*} value The value to check.
|
|
* @returns {boolean} Returns `true` if the `value` is `null`, else `false`.
|
|
* @example
|
|
*
|
|
* _.isNull(null);
|
|
* // => true
|
|
*
|
|
* _.isNull(undefined);
|
|
* // => false
|
|
*/
|
|
function isNull(value) {
|
|
return value === null;
|
|
}
|
|
|
|
/**
|
|
* Checks if `value` is a number.
|
|
*
|
|
* Note: `NaN` is considered a number. See http://es5.github.io/#x8.5.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Objects
|
|
* @param {*} value The value to check.
|
|
* @returns {boolean} Returns `true` if the `value` is a number, else `false`.
|
|
* @example
|
|
*
|
|
* _.isNumber(8.4 * 5);
|
|
* // => true
|
|
*/
|
|
function isNumber(value) {
|
|
return typeof value == 'number' ||
|
|
value && typeof value == 'object' && toString.call(value) == numberClass || false;
|
|
}
|
|
|
|
/**
|
|
* Checks if `value` is an object created by the `Object` constructor.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Objects
|
|
* @param {*} value The value to check.
|
|
* @returns {boolean} Returns `true` if `value` is a plain object, else `false`.
|
|
* @example
|
|
*
|
|
* function Shape() {
|
|
* this.x = 0;
|
|
* this.y = 0;
|
|
* }
|
|
*
|
|
* _.isPlainObject(new Shape);
|
|
* // => false
|
|
*
|
|
* _.isPlainObject([1, 2, 3]);
|
|
* // => false
|
|
*
|
|
* _.isPlainObject({ 'x': 0, 'y': 0 });
|
|
* // => true
|
|
*/
|
|
var isPlainObject = function(value) {
|
|
if (!(value && toString.call(value) == objectClass)) {
|
|
return false;
|
|
}
|
|
var valueOf = value.valueOf,
|
|
objProto = typeof valueOf == 'function' && (objProto = getPrototypeOf(valueOf)) && getPrototypeOf(objProto);
|
|
|
|
return objProto
|
|
? (value == objProto || getPrototypeOf(value) == objProto)
|
|
: shimIsPlainObject(value);
|
|
};
|
|
|
|
/**
|
|
* Checks if `value` is a regular expression.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Objects
|
|
* @param {*} value The value to check.
|
|
* @returns {boolean} Returns `true` if the `value` is a regular expression, else `false`.
|
|
* @example
|
|
*
|
|
* _.isRegExp(/fred/);
|
|
* // => true
|
|
*/
|
|
function isRegExp(value) {
|
|
return value && typeof value == 'object' && toString.call(value) == regexpClass || false;
|
|
}
|
|
|
|
/**
|
|
* Checks if `value` is a string.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Objects
|
|
* @param {*} value The value to check.
|
|
* @returns {boolean} Returns `true` if the `value` is a string, else `false`.
|
|
* @example
|
|
*
|
|
* _.isString('fred');
|
|
* // => true
|
|
*/
|
|
function isString(value) {
|
|
return typeof value == 'string' ||
|
|
value && typeof value == 'object' && toString.call(value) == stringClass || false;
|
|
}
|
|
|
|
/**
|
|
* Checks if `value` is `undefined`.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Objects
|
|
* @param {*} value The value to check.
|
|
* @returns {boolean} Returns `true` if the `value` is `undefined`, else `false`.
|
|
* @example
|
|
*
|
|
* _.isUndefined(void 0);
|
|
* // => true
|
|
*/
|
|
function isUndefined(value) {
|
|
return typeof value == 'undefined';
|
|
}
|
|
|
|
/**
|
|
* Recursively merges own enumerable properties of the source object(s), that
|
|
* don't resolve to `undefined` into the destination object. Subsequent sources
|
|
* will overwrite property assignments of previous sources. If a callback is
|
|
* provided it will be executed to produce the merged values of the destination
|
|
* and source properties. If the callback returns `undefined` merging will
|
|
* be handled by the method instead. The callback is bound to `thisArg` and
|
|
* invoked with two arguments; (objectValue, sourceValue).
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Objects
|
|
* @param {Object} object The destination object.
|
|
* @param {...Object} [source] The source objects.
|
|
* @param {Function} [callback] The function to customize merging properties.
|
|
* @param {*} [thisArg] The `this` binding of `callback`.
|
|
* @returns {Object} Returns the destination object.
|
|
* @example
|
|
*
|
|
* var names = {
|
|
* 'characters': [
|
|
* { 'name': 'barney' },
|
|
* { 'name': 'fred' }
|
|
* ]
|
|
* };
|
|
*
|
|
* var ages = {
|
|
* 'characters': [
|
|
* { 'age': 36 },
|
|
* { 'age': 40 }
|
|
* ]
|
|
* };
|
|
*
|
|
* _.merge(names, ages);
|
|
* // => { 'characters': [{ 'name': 'barney', 'age': 36 }, { 'name': 'fred', 'age': 40 }] }
|
|
*
|
|
* var food = {
|
|
* 'fruits': ['apple'],
|
|
* 'vegetables': ['beet']
|
|
* };
|
|
*
|
|
* var otherFood = {
|
|
* 'fruits': ['banana'],
|
|
* 'vegetables': ['carrot']
|
|
* };
|
|
*
|
|
* _.merge(food, otherFood, function(a, b) {
|
|
* return _.isArray(a) ? a.concat(b) : undefined;
|
|
* });
|
|
* // => { 'fruits': ['apple', 'banana'], 'vegetables': ['beet', 'carrot] }
|
|
*/
|
|
function merge(object) {
|
|
var args = arguments,
|
|
length = 2;
|
|
|
|
if (!isObject(object)) {
|
|
return object;
|
|
}
|
|
|
|
// allows working with `_.reduce` and `_.reduceRight` without using
|
|
// their `index` and `collection` arguments
|
|
if (typeof args[2] != 'number') {
|
|
length = args.length;
|
|
}
|
|
if (length > 3 && typeof args[length - 2] == 'function') {
|
|
var callback = baseCreateCallback(args[--length - 1], args[length--], 2);
|
|
} else if (length > 2 && typeof args[length - 1] == 'function') {
|
|
callback = args[--length];
|
|
}
|
|
var sources = slice(arguments, 1, length),
|
|
index = -1,
|
|
stackA = getArray(),
|
|
stackB = getArray();
|
|
|
|
while (++index < length) {
|
|
baseMerge(object, sources[index], callback, stackA, stackB);
|
|
}
|
|
releaseArray(stackA);
|
|
releaseArray(stackB);
|
|
return object;
|
|
}
|
|
|
|
/**
|
|
* Creates a shallow clone of `object` excluding the specified properties.
|
|
* Property names may be specified as individual arguments or as arrays of
|
|
* property names. If a callback is provided it will be executed for each
|
|
* property of `object` omitting the properties the callback returns truey
|
|
* for. The callback is bound to `thisArg` and invoked with three arguments;
|
|
* (value, key, object).
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Objects
|
|
* @param {Object} object The source object.
|
|
* @param {Function|...string|string[]} [callback] The properties to omit or the
|
|
* function called per iteration.
|
|
* @param {*} [thisArg] The `this` binding of `callback`.
|
|
* @returns {Object} Returns an object without the omitted properties.
|
|
* @example
|
|
*
|
|
* _.omit({ 'name': 'fred', 'age': 40 }, 'age');
|
|
* // => { 'name': 'fred' }
|
|
*
|
|
* _.omit({ 'name': 'fred', 'age': 40 }, function(value) {
|
|
* return typeof value == 'number';
|
|
* });
|
|
* // => { 'name': 'fred' }
|
|
*/
|
|
function omit(object, callback, thisArg) {
|
|
var result = {};
|
|
if (typeof callback != 'function') {
|
|
var props = [];
|
|
forIn(object, function(value, key) {
|
|
props.push(key);
|
|
});
|
|
props = baseDifference(props, baseFlatten(arguments, true, false, 1));
|
|
|
|
var index = -1,
|
|
length = props.length;
|
|
|
|
while (++index < length) {
|
|
var key = props[index];
|
|
result[key] = object[key];
|
|
}
|
|
} else {
|
|
callback = lodash.createCallback(callback, thisArg, 3);
|
|
forIn(object, function(value, key, object) {
|
|
if (!callback(value, key, object)) {
|
|
result[key] = value;
|
|
}
|
|
});
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Creates a two dimensional array of an object's key-value pairs,
|
|
* i.e. `[[key1, value1], [key2, value2]]`.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Objects
|
|
* @param {Object} object The object to inspect.
|
|
* @returns {Array} Returns new array of key-value pairs.
|
|
* @example
|
|
*
|
|
* _.pairs({ 'barney': 36, 'fred': 40 });
|
|
* // => [['barney', 36], ['fred', 40]] (property order is not guaranteed across environments)
|
|
*/
|
|
function pairs(object) {
|
|
var index = -1,
|
|
props = keys(object),
|
|
length = props.length,
|
|
result = Array(length);
|
|
|
|
while (++index < length) {
|
|
var key = props[index];
|
|
result[index] = [key, object[key]];
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Creates a shallow clone of `object` composed of the specified properties.
|
|
* Property names may be specified as individual arguments or as arrays of
|
|
* property names. If a callback is provided it will be executed for each
|
|
* property of `object` picking the properties the callback returns truey
|
|
* for. The callback is bound to `thisArg` and invoked with three arguments;
|
|
* (value, key, object).
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Objects
|
|
* @param {Object} object The source object.
|
|
* @param {Function|...string|string[]} [callback] The function called per
|
|
* iteration or property names to pick, specified as individual property
|
|
* names or arrays of property names.
|
|
* @param {*} [thisArg] The `this` binding of `callback`.
|
|
* @returns {Object} Returns an object composed of the picked properties.
|
|
* @example
|
|
*
|
|
* _.pick({ 'name': 'fred', '_userid': 'fred1' }, 'name');
|
|
* // => { 'name': 'fred' }
|
|
*
|
|
* _.pick({ 'name': 'fred', '_userid': 'fred1' }, function(value, key) {
|
|
* return key.charAt(0) != '_';
|
|
* });
|
|
* // => { 'name': 'fred' }
|
|
*/
|
|
function pick(object, callback, thisArg) {
|
|
var result = {};
|
|
if (typeof callback != 'function') {
|
|
var index = -1,
|
|
props = baseFlatten(arguments, true, false, 1),
|
|
length = isObject(object) ? props.length : 0;
|
|
|
|
while (++index < length) {
|
|
var key = props[index];
|
|
if (key in object) {
|
|
result[key] = object[key];
|
|
}
|
|
}
|
|
} else {
|
|
callback = lodash.createCallback(callback, thisArg, 3);
|
|
forIn(object, function(value, key, object) {
|
|
if (callback(value, key, object)) {
|
|
result[key] = value;
|
|
}
|
|
});
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* An alternative to `_.reduce` this method transforms `object` to a new
|
|
* `accumulator` object which is the result of running each of its elements
|
|
* through a callback, with each callback execution potentially mutating
|
|
* the `accumulator` object. The callback is bound to `thisArg` and invoked
|
|
* with four arguments; (accumulator, value, key, object). Callbacks may exit
|
|
* iteration early by explicitly returning `false`.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Objects
|
|
* @param {Array|Object} object The object to iterate over.
|
|
* @param {Function} [callback=identity] The function called per iteration.
|
|
* @param {*} [accumulator] The custom accumulator value.
|
|
* @param {*} [thisArg] The `this` binding of `callback`.
|
|
* @returns {*} Returns the accumulated value.
|
|
* @example
|
|
*
|
|
* var squares = _.transform([1, 2, 3, 4, 5, 6, 7, 8, 9, 10], function(result, num) {
|
|
* num *= num;
|
|
* if (num % 2) {
|
|
* return result.push(num) < 3;
|
|
* }
|
|
* });
|
|
* // => [1, 9, 25]
|
|
*
|
|
* var mapped = _.transform({ 'a': 1, 'b': 2, 'c': 3 }, function(result, num, key) {
|
|
* result[key] = num * 3;
|
|
* });
|
|
* // => { 'a': 3, 'b': 6, 'c': 9 }
|
|
*/
|
|
function transform(object, callback, accumulator, thisArg) {
|
|
var isArr = isArray(object);
|
|
if (accumulator == null) {
|
|
if (isArr) {
|
|
accumulator = [];
|
|
} else {
|
|
var ctor = object && object.constructor,
|
|
proto = ctor && ctor.prototype;
|
|
|
|
accumulator = baseCreate(proto);
|
|
}
|
|
}
|
|
if (callback) {
|
|
callback = lodash.createCallback(callback, thisArg, 4);
|
|
(isArr ? forEach : forOwn)(object, function(value, index, object) {
|
|
return callback(accumulator, value, index, object);
|
|
});
|
|
}
|
|
return accumulator;
|
|
}
|
|
|
|
/**
|
|
* Creates an array composed of the own enumerable property values of `object`.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Objects
|
|
* @param {Object} object The object to inspect.
|
|
* @returns {Array} Returns an array of property values.
|
|
* @example
|
|
*
|
|
* _.values({ 'one': 1, 'two': 2, 'three': 3 });
|
|
* // => [1, 2, 3] (property order is not guaranteed across environments)
|
|
*/
|
|
function values(object) {
|
|
var index = -1,
|
|
props = keys(object),
|
|
length = props.length,
|
|
result = Array(length);
|
|
|
|
while (++index < length) {
|
|
result[index] = object[props[index]];
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
|
|
/**
|
|
* Creates an array of elements from the specified indexes, or keys, of the
|
|
* `collection`. Indexes may be specified as individual arguments or as arrays
|
|
* of indexes.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Collections
|
|
* @param {Array|Object|string} collection The collection to iterate over.
|
|
* @param {...(number|number[]|string|string[])} [index] The indexes of `collection`
|
|
* to retrieve, specified as individual indexes or arrays of indexes.
|
|
* @returns {Array} Returns a new array of elements corresponding to the
|
|
* provided indexes.
|
|
* @example
|
|
*
|
|
* _.at(['a', 'b', 'c', 'd', 'e'], [0, 2, 4]);
|
|
* // => ['a', 'c', 'e']
|
|
*
|
|
* _.at(['fred', 'barney', 'pebbles'], 0, 2);
|
|
* // => ['fred', 'pebbles']
|
|
*/
|
|
function at(collection) {
|
|
var args = arguments,
|
|
index = -1,
|
|
props = baseFlatten(args, true, false, 1),
|
|
length = (args[2] && args[2][args[1]] === collection) ? 1 : props.length,
|
|
result = Array(length);
|
|
|
|
while(++index < length) {
|
|
result[index] = collection[props[index]];
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Checks if a given value is present in a collection using strict equality
|
|
* for comparisons, i.e. `===`. If `fromIndex` is negative, it is used as the
|
|
* offset from the end of the collection.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @alias include
|
|
* @category Collections
|
|
* @param {Array|Object|string} collection The collection to iterate over.
|
|
* @param {*} target The value to check for.
|
|
* @param {number} [fromIndex=0] The index to search from.
|
|
* @returns {boolean} Returns `true` if the `target` element is found, else `false`.
|
|
* @example
|
|
*
|
|
* _.contains([1, 2, 3], 1);
|
|
* // => true
|
|
*
|
|
* _.contains([1, 2, 3], 1, 2);
|
|
* // => false
|
|
*
|
|
* _.contains({ 'name': 'fred', 'age': 40 }, 'fred');
|
|
* // => true
|
|
*
|
|
* _.contains('pebbles', 'eb');
|
|
* // => true
|
|
*/
|
|
function contains(collection, target, fromIndex) {
|
|
var index = -1,
|
|
indexOf = getIndexOf(),
|
|
length = collection ? collection.length : 0,
|
|
result = false;
|
|
|
|
fromIndex = (fromIndex < 0 ? nativeMax(0, length + fromIndex) : fromIndex) || 0;
|
|
if (isArray(collection)) {
|
|
result = indexOf(collection, target, fromIndex) > -1;
|
|
} else if (typeof length == 'number') {
|
|
result = (isString(collection) ? collection.indexOf(target, fromIndex) : indexOf(collection, target, fromIndex)) > -1;
|
|
} else {
|
|
forOwn(collection, function(value) {
|
|
if (++index >= fromIndex) {
|
|
return !(result = value === target);
|
|
}
|
|
});
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Creates an object composed of keys generated from the results of running
|
|
* each element of `collection` through the callback. The corresponding value
|
|
* of each key is the number of times the key was returned by the callback.
|
|
* The callback is bound to `thisArg` and invoked with three arguments;
|
|
* (value, index|key, collection).
|
|
*
|
|
* If a property name is provided for `callback` the created "_.pluck" style
|
|
* callback will return the property value of the given element.
|
|
*
|
|
* If an object is provided for `callback` the created "_.where" style callback
|
|
* will return `true` for elements that have the properties of the given object,
|
|
* else `false`.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Collections
|
|
* @param {Array|Object|string} collection The collection to iterate over.
|
|
* @param {Function|Object|string} [callback=identity] The function called
|
|
* per iteration. If a property name or object is provided it will be used
|
|
* to create a "_.pluck" or "_.where" style callback, respectively.
|
|
* @param {*} [thisArg] The `this` binding of `callback`.
|
|
* @returns {Object} Returns the composed aggregate object.
|
|
* @example
|
|
*
|
|
* _.countBy([4.3, 6.1, 6.4], function(num) { return Math.floor(num); });
|
|
* // => { '4': 1, '6': 2 }
|
|
*
|
|
* _.countBy([4.3, 6.1, 6.4], function(num) { return this.floor(num); }, Math);
|
|
* // => { '4': 1, '6': 2 }
|
|
*
|
|
* _.countBy(['one', 'two', 'three'], 'length');
|
|
* // => { '3': 2, '5': 1 }
|
|
*/
|
|
var countBy = createAggregator(function(result, value, key) {
|
|
(hasOwnProperty.call(result, key) ? result[key]++ : result[key] = 1);
|
|
});
|
|
|
|
/**
|
|
* Checks if the given callback returns truey value for **all** elements of
|
|
* a collection. The callback is bound to `thisArg` and invoked with three
|
|
* arguments; (value, index|key, collection).
|
|
*
|
|
* If a property name is provided for `callback` the created "_.pluck" style
|
|
* callback will return the property value of the given element.
|
|
*
|
|
* If an object is provided for `callback` the created "_.where" style callback
|
|
* will return `true` for elements that have the properties of the given object,
|
|
* else `false`.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @alias all
|
|
* @category Collections
|
|
* @param {Array|Object|string} collection The collection to iterate over.
|
|
* @param {Function|Object|string} [callback=identity] The function called
|
|
* per iteration. If a property name or object is provided it will be used
|
|
* to create a "_.pluck" or "_.where" style callback, respectively.
|
|
* @param {*} [thisArg] The `this` binding of `callback`.
|
|
* @returns {boolean} Returns `true` if all elements passed the callback check,
|
|
* else `false`.
|
|
* @example
|
|
*
|
|
* _.every([true, 1, null, 'yes']);
|
|
* // => false
|
|
*
|
|
* var characters = [
|
|
* { 'name': 'barney', 'age': 36 },
|
|
* { 'name': 'fred', 'age': 40 }
|
|
* ];
|
|
*
|
|
* // using "_.pluck" callback shorthand
|
|
* _.every(characters, 'age');
|
|
* // => true
|
|
*
|
|
* // using "_.where" callback shorthand
|
|
* _.every(characters, { 'age': 36 });
|
|
* // => false
|
|
*/
|
|
function every(collection, callback, thisArg) {
|
|
var result = true;
|
|
callback = lodash.createCallback(callback, thisArg, 3);
|
|
|
|
var index = -1,
|
|
length = collection ? collection.length : 0;
|
|
|
|
if (typeof length == 'number') {
|
|
while (++index < length) {
|
|
if (!(result = !!callback(collection[index], index, collection))) {
|
|
break;
|
|
}
|
|
}
|
|
} else {
|
|
forOwn(collection, function(value, index, collection) {
|
|
return (result = !!callback(value, index, collection));
|
|
});
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Iterates over elements of a collection, returning an array of all elements
|
|
* the callback returns truey for. The callback is bound to `thisArg` and
|
|
* invoked with three arguments; (value, index|key, collection).
|
|
*
|
|
* If a property name is provided for `callback` the created "_.pluck" style
|
|
* callback will return the property value of the given element.
|
|
*
|
|
* If an object is provided for `callback` the created "_.where" style callback
|
|
* will return `true` for elements that have the properties of the given object,
|
|
* else `false`.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @alias select
|
|
* @category Collections
|
|
* @param {Array|Object|string} collection The collection to iterate over.
|
|
* @param {Function|Object|string} [callback=identity] The function called
|
|
* per iteration. If a property name or object is provided it will be used
|
|
* to create a "_.pluck" or "_.where" style callback, respectively.
|
|
* @param {*} [thisArg] The `this` binding of `callback`.
|
|
* @returns {Array} Returns a new array of elements that passed the callback check.
|
|
* @example
|
|
*
|
|
* var evens = _.filter([1, 2, 3, 4, 5, 6], function(num) { return num % 2 == 0; });
|
|
* // => [2, 4, 6]
|
|
*
|
|
* var characters = [
|
|
* { 'name': 'barney', 'age': 36, 'blocked': false },
|
|
* { 'name': 'fred', 'age': 40, 'blocked': true }
|
|
* ];
|
|
*
|
|
* // using "_.pluck" callback shorthand
|
|
* _.filter(characters, 'blocked');
|
|
* // => [{ 'name': 'fred', 'age': 40, 'blocked': true }]
|
|
*
|
|
* // using "_.where" callback shorthand
|
|
* _.filter(characters, { 'age': 36 });
|
|
* // => [{ 'name': 'barney', 'age': 36, 'blocked': false }]
|
|
*/
|
|
function filter(collection, callback, thisArg) {
|
|
var result = [];
|
|
callback = lodash.createCallback(callback, thisArg, 3);
|
|
|
|
var index = -1,
|
|
length = collection ? collection.length : 0;
|
|
|
|
if (typeof length == 'number') {
|
|
while (++index < length) {
|
|
var value = collection[index];
|
|
if (callback(value, index, collection)) {
|
|
result.push(value);
|
|
}
|
|
}
|
|
} else {
|
|
forOwn(collection, function(value, index, collection) {
|
|
if (callback(value, index, collection)) {
|
|
result.push(value);
|
|
}
|
|
});
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Iterates over elements of a collection, returning the first element that
|
|
* the callback returns truey for. The callback is bound to `thisArg` and
|
|
* invoked with three arguments; (value, index|key, collection).
|
|
*
|
|
* If a property name is provided for `callback` the created "_.pluck" style
|
|
* callback will return the property value of the given element.
|
|
*
|
|
* If an object is provided for `callback` the created "_.where" style callback
|
|
* will return `true` for elements that have the properties of the given object,
|
|
* else `false`.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @alias detect, findWhere
|
|
* @category Collections
|
|
* @param {Array|Object|string} collection The collection to iterate over.
|
|
* @param {Function|Object|string} [callback=identity] The function called
|
|
* per iteration. If a property name or object is provided it will be used
|
|
* to create a "_.pluck" or "_.where" style callback, respectively.
|
|
* @param {*} [thisArg] The `this` binding of `callback`.
|
|
* @returns {*} Returns the found element, else `undefined`.
|
|
* @example
|
|
*
|
|
* var characters = [
|
|
* { 'name': 'barney', 'age': 36, 'blocked': false },
|
|
* { 'name': 'fred', 'age': 40, 'blocked': true },
|
|
* { 'name': 'pebbles', 'age': 1, 'blocked': false }
|
|
* ];
|
|
*
|
|
* _.find(characters, function(chr) {
|
|
* return chr.age < 40;
|
|
* });
|
|
* // => { 'name': 'barney', 'age': 36, 'blocked': false }
|
|
*
|
|
* // using "_.where" callback shorthand
|
|
* _.find(characters, { 'age': 1 });
|
|
* // => { 'name': 'pebbles', 'age': 1, 'blocked': false }
|
|
*
|
|
* // using "_.pluck" callback shorthand
|
|
* _.find(characters, 'blocked');
|
|
* // => { 'name': 'fred', 'age': 40, 'blocked': true }
|
|
*/
|
|
function find(collection, callback, thisArg) {
|
|
callback = lodash.createCallback(callback, thisArg, 3);
|
|
|
|
var index = -1,
|
|
length = collection ? collection.length : 0;
|
|
|
|
if (typeof length == 'number') {
|
|
while (++index < length) {
|
|
var value = collection[index];
|
|
if (callback(value, index, collection)) {
|
|
return value;
|
|
}
|
|
}
|
|
} else {
|
|
var result;
|
|
forOwn(collection, function(value, index, collection) {
|
|
if (callback(value, index, collection)) {
|
|
result = value;
|
|
return false;
|
|
}
|
|
});
|
|
return result;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* This method is like `_.find` except that it iterates over elements
|
|
* of a `collection` from right to left.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Collections
|
|
* @param {Array|Object|string} collection The collection to iterate over.
|
|
* @param {Function|Object|string} [callback=identity] The function called
|
|
* per iteration. If a property name or object is provided it will be used
|
|
* to create a "_.pluck" or "_.where" style callback, respectively.
|
|
* @param {*} [thisArg] The `this` binding of `callback`.
|
|
* @returns {*} Returns the found element, else `undefined`.
|
|
* @example
|
|
*
|
|
* _.findLast([1, 2, 3, 4], function(num) {
|
|
* return num % 2 == 1;
|
|
* });
|
|
* // => 3
|
|
*/
|
|
function findLast(collection, callback, thisArg) {
|
|
var result;
|
|
callback = lodash.createCallback(callback, thisArg, 3);
|
|
forEachRight(collection, function(value, index, collection) {
|
|
if (callback(value, index, collection)) {
|
|
result = value;
|
|
return false;
|
|
}
|
|
});
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Iterates over elements of a collection, executing the callback for each
|
|
* element. The callback is bound to `thisArg` and invoked with three arguments;
|
|
* (value, index|key, collection). Callbacks may exit iteration early by
|
|
* explicitly returning `false`.
|
|
*
|
|
* Note: As with other "Collections" methods, objects with a `length` property
|
|
* are iterated like arrays. To avoid this behavior `_.forIn` or `_.forOwn`
|
|
* may be used for object iteration.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @alias each
|
|
* @category Collections
|
|
* @param {Array|Object|string} collection The collection to iterate over.
|
|
* @param {Function} [callback=identity] The function called per iteration.
|
|
* @param {*} [thisArg] The `this` binding of `callback`.
|
|
* @returns {Array|Object|string} Returns `collection`.
|
|
* @example
|
|
*
|
|
* _([1, 2, 3]).forEach(function(num) { console.log(num); }).join(',');
|
|
* // => logs each number and returns '1,2,3'
|
|
*
|
|
* _.forEach({ 'one': 1, 'two': 2, 'three': 3 }, function(num) { console.log(num); });
|
|
* // => logs each number and returns the object (property order is not guaranteed across environments)
|
|
*/
|
|
function forEach(collection, callback, thisArg) {
|
|
var index = -1,
|
|
length = collection ? collection.length : 0;
|
|
|
|
callback = callback && typeof thisArg == 'undefined' ? callback : baseCreateCallback(callback, thisArg, 3);
|
|
if (typeof length == 'number') {
|
|
while (++index < length) {
|
|
if (callback(collection[index], index, collection) === false) {
|
|
break;
|
|
}
|
|
}
|
|
} else {
|
|
forOwn(collection, callback);
|
|
}
|
|
return collection;
|
|
}
|
|
|
|
/**
|
|
* This method is like `_.forEach` except that it iterates over elements
|
|
* of a `collection` from right to left.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @alias eachRight
|
|
* @category Collections
|
|
* @param {Array|Object|string} collection The collection to iterate over.
|
|
* @param {Function} [callback=identity] The function called per iteration.
|
|
* @param {*} [thisArg] The `this` binding of `callback`.
|
|
* @returns {Array|Object|string} Returns `collection`.
|
|
* @example
|
|
*
|
|
* _([1, 2, 3]).forEachRight(function(num) { console.log(num); }).join(',');
|
|
* // => logs each number from right to left and returns '3,2,1'
|
|
*/
|
|
function forEachRight(collection, callback, thisArg) {
|
|
var length = collection ? collection.length : 0;
|
|
callback = callback && typeof thisArg == 'undefined' ? callback : baseCreateCallback(callback, thisArg, 3);
|
|
if (typeof length == 'number') {
|
|
while (length--) {
|
|
if (callback(collection[length], length, collection) === false) {
|
|
break;
|
|
}
|
|
}
|
|
} else {
|
|
var props = keys(collection);
|
|
length = props.length;
|
|
forOwn(collection, function(value, key, collection) {
|
|
key = props ? props[--length] : --length;
|
|
return callback(collection[key], key, collection);
|
|
});
|
|
}
|
|
return collection;
|
|
}
|
|
|
|
/**
|
|
* Creates an object composed of keys generated from the results of running
|
|
* each element of a collection through the callback. The corresponding value
|
|
* of each key is an array of the elements responsible for generating the key.
|
|
* The callback is bound to `thisArg` and invoked with three arguments;
|
|
* (value, index|key, collection).
|
|
*
|
|
* If a property name is provided for `callback` the created "_.pluck" style
|
|
* callback will return the property value of the given element.
|
|
*
|
|
* If an object is provided for `callback` the created "_.where" style callback
|
|
* will return `true` for elements that have the properties of the given object,
|
|
* else `false`
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Collections
|
|
* @param {Array|Object|string} collection The collection to iterate over.
|
|
* @param {Function|Object|string} [callback=identity] The function called
|
|
* per iteration. If a property name or object is provided it will be used
|
|
* to create a "_.pluck" or "_.where" style callback, respectively.
|
|
* @param {*} [thisArg] The `this` binding of `callback`.
|
|
* @returns {Object} Returns the composed aggregate object.
|
|
* @example
|
|
*
|
|
* _.groupBy([4.2, 6.1, 6.4], function(num) { return Math.floor(num); });
|
|
* // => { '4': [4.2], '6': [6.1, 6.4] }
|
|
*
|
|
* _.groupBy([4.2, 6.1, 6.4], function(num) { return this.floor(num); }, Math);
|
|
* // => { '4': [4.2], '6': [6.1, 6.4] }
|
|
*
|
|
* // using "_.pluck" callback shorthand
|
|
* _.groupBy(['one', 'two', 'three'], 'length');
|
|
* // => { '3': ['one', 'two'], '5': ['three'] }
|
|
*/
|
|
var groupBy = createAggregator(function(result, value, key) {
|
|
(hasOwnProperty.call(result, key) ? result[key] : result[key] = []).push(value);
|
|
});
|
|
|
|
/**
|
|
* Creates an object composed of keys generated from the results of running
|
|
* each element of the collection through the given callback. The corresponding
|
|
* value of each key is the last element responsible for generating the key.
|
|
* The callback is bound to `thisArg` and invoked with three arguments;
|
|
* (value, index|key, collection).
|
|
*
|
|
* If a property name is provided for `callback` the created "_.pluck" style
|
|
* callback will return the property value of the given element.
|
|
*
|
|
* If an object is provided for `callback` the created "_.where" style callback
|
|
* will return `true` for elements that have the properties of the given object,
|
|
* else `false`.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Collections
|
|
* @param {Array|Object|string} collection The collection to iterate over.
|
|
* @param {Function|Object|string} [callback=identity] The function called
|
|
* per iteration. If a property name or object is provided it will be used
|
|
* to create a "_.pluck" or "_.where" style callback, respectively.
|
|
* @param {*} [thisArg] The `this` binding of `callback`.
|
|
* @returns {Object} Returns the composed aggregate object.
|
|
* @example
|
|
*
|
|
* var keys = [
|
|
* { 'dir': 'left', 'code': 97 },
|
|
* { 'dir': 'right', 'code': 100 }
|
|
* ];
|
|
*
|
|
* _.indexBy(keys, 'dir');
|
|
* // => { 'left': { 'dir': 'left', 'code': 97 }, 'right': { 'dir': 'right', 'code': 100 } }
|
|
*
|
|
* _.indexBy(keys, function(key) { return String.fromCharCode(key.code); });
|
|
* // => { 'a': { 'dir': 'left', 'code': 97 }, 'd': { 'dir': 'right', 'code': 100 } }
|
|
*
|
|
* _.indexBy(characters, function(key) { this.fromCharCode(key.code); }, String);
|
|
* // => { 'a': { 'dir': 'left', 'code': 97 }, 'd': { 'dir': 'right', 'code': 100 } }
|
|
*/
|
|
var indexBy = createAggregator(function(result, value, key) {
|
|
result[key] = value;
|
|
});
|
|
|
|
/**
|
|
* Invokes the method named by `methodName` on each element in the `collection`
|
|
* returning an array of the results of each invoked method. Additional arguments
|
|
* will be provided to each invoked method. If `methodName` is a function it
|
|
* will be invoked for, and `this` bound to, each element in the `collection`.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Collections
|
|
* @param {Array|Object|string} collection The collection to iterate over.
|
|
* @param {Function|string} methodName The name of the method to invoke or
|
|
* the function invoked per iteration.
|
|
* @param {...*} [arg] Arguments to invoke the method with.
|
|
* @returns {Array} Returns a new array of the results of each invoked method.
|
|
* @example
|
|
*
|
|
* _.invoke([[5, 1, 7], [3, 2, 1]], 'sort');
|
|
* // => [[1, 5, 7], [1, 2, 3]]
|
|
*
|
|
* _.invoke([123, 456], String.prototype.split, '');
|
|
* // => [['1', '2', '3'], ['4', '5', '6']]
|
|
*/
|
|
function invoke(collection, methodName) {
|
|
var args = slice(arguments, 2),
|
|
index = -1,
|
|
isFunc = typeof methodName == 'function',
|
|
length = collection ? collection.length : 0,
|
|
result = Array(typeof length == 'number' ? length : 0);
|
|
|
|
forEach(collection, function(value) {
|
|
result[++index] = (isFunc ? methodName : value[methodName]).apply(value, args);
|
|
});
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Creates an array of values by running each element in the collection
|
|
* through the callback. The callback is bound to `thisArg` and invoked with
|
|
* three arguments; (value, index|key, collection).
|
|
*
|
|
* If a property name is provided for `callback` the created "_.pluck" style
|
|
* callback will return the property value of the given element.
|
|
*
|
|
* If an object is provided for `callback` the created "_.where" style callback
|
|
* will return `true` for elements that have the properties of the given object,
|
|
* else `false`.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @alias collect
|
|
* @category Collections
|
|
* @param {Array|Object|string} collection The collection to iterate over.
|
|
* @param {Function|Object|string} [callback=identity] The function called
|
|
* per iteration. If a property name or object is provided it will be used
|
|
* to create a "_.pluck" or "_.where" style callback, respectively.
|
|
* @param {*} [thisArg] The `this` binding of `callback`.
|
|
* @returns {Array} Returns a new array of the results of each `callback` execution.
|
|
* @example
|
|
*
|
|
* _.map([1, 2, 3], function(num) { return num * 3; });
|
|
* // => [3, 6, 9]
|
|
*
|
|
* _.map({ 'one': 1, 'two': 2, 'three': 3 }, function(num) { return num * 3; });
|
|
* // => [3, 6, 9] (property order is not guaranteed across environments)
|
|
*
|
|
* var characters = [
|
|
* { 'name': 'barney', 'age': 36 },
|
|
* { 'name': 'fred', 'age': 40 }
|
|
* ];
|
|
*
|
|
* // using "_.pluck" callback shorthand
|
|
* _.map(characters, 'name');
|
|
* // => ['barney', 'fred']
|
|
*/
|
|
function map(collection, callback, thisArg) {
|
|
var index = -1,
|
|
length = collection ? collection.length : 0;
|
|
|
|
callback = lodash.createCallback(callback, thisArg, 3);
|
|
if (typeof length == 'number') {
|
|
var result = Array(length);
|
|
while (++index < length) {
|
|
result[index] = callback(collection[index], index, collection);
|
|
}
|
|
} else {
|
|
result = [];
|
|
forOwn(collection, function(value, key, collection) {
|
|
result[++index] = callback(value, key, collection);
|
|
});
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Retrieves the maximum value of a collection. If the collection is empty or
|
|
* falsey `-Infinity` is returned. If a callback is provided it will be executed
|
|
* for each value in the collection to generate the criterion by which the value
|
|
* is ranked. The callback is bound to `thisArg` and invoked with three
|
|
* arguments; (value, index, collection).
|
|
*
|
|
* If a property name is provided for `callback` the created "_.pluck" style
|
|
* callback will return the property value of the given element.
|
|
*
|
|
* If an object is provided for `callback` the created "_.where" style callback
|
|
* will return `true` for elements that have the properties of the given object,
|
|
* else `false`.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Collections
|
|
* @param {Array|Object|string} collection The collection to iterate over.
|
|
* @param {Function|Object|string} [callback=identity] The function called
|
|
* per iteration. If a property name or object is provided it will be used
|
|
* to create a "_.pluck" or "_.where" style callback, respectively.
|
|
* @param {*} [thisArg] The `this` binding of `callback`.
|
|
* @returns {*} Returns the maximum value.
|
|
* @example
|
|
*
|
|
* _.max([4, 2, 8, 6]);
|
|
* // => 8
|
|
*
|
|
* var characters = [
|
|
* { 'name': 'barney', 'age': 36 },
|
|
* { 'name': 'fred', 'age': 40 }
|
|
* ];
|
|
*
|
|
* _.max(characters, function(chr) { return chr.age; });
|
|
* // => { 'name': 'fred', 'age': 40 };
|
|
*
|
|
* // using "_.pluck" callback shorthand
|
|
* _.max(characters, 'age');
|
|
* // => { 'name': 'fred', 'age': 40 };
|
|
*/
|
|
function max(collection, callback, thisArg) {
|
|
var computed = -Infinity,
|
|
result = computed;
|
|
|
|
// allows working with functions like `_.map` without using
|
|
// their `index` argument as a callback
|
|
if (typeof callback != 'function' && thisArg && thisArg[callback] === collection) {
|
|
callback = null;
|
|
}
|
|
if (callback == null && isArray(collection)) {
|
|
var index = -1,
|
|
length = collection.length;
|
|
|
|
while (++index < length) {
|
|
var value = collection[index];
|
|
if (value > result) {
|
|
result = value;
|
|
}
|
|
}
|
|
} else {
|
|
callback = (callback == null && isString(collection))
|
|
? charAtCallback
|
|
: lodash.createCallback(callback, thisArg, 3);
|
|
|
|
forEach(collection, function(value, index, collection) {
|
|
var current = callback(value, index, collection);
|
|
if (current > computed) {
|
|
computed = current;
|
|
result = value;
|
|
}
|
|
});
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Retrieves the minimum value of a collection. If the collection is empty or
|
|
* falsey `Infinity` is returned. If a callback is provided it will be executed
|
|
* for each value in the collection to generate the criterion by which the value
|
|
* is ranked. The callback is bound to `thisArg` and invoked with three
|
|
* arguments; (value, index, collection).
|
|
*
|
|
* If a property name is provided for `callback` the created "_.pluck" style
|
|
* callback will return the property value of the given element.
|
|
*
|
|
* If an object is provided for `callback` the created "_.where" style callback
|
|
* will return `true` for elements that have the properties of the given object,
|
|
* else `false`.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Collections
|
|
* @param {Array|Object|string} collection The collection to iterate over.
|
|
* @param {Function|Object|string} [callback=identity] The function called
|
|
* per iteration. If a property name or object is provided it will be used
|
|
* to create a "_.pluck" or "_.where" style callback, respectively.
|
|
* @param {*} [thisArg] The `this` binding of `callback`.
|
|
* @returns {*} Returns the minimum value.
|
|
* @example
|
|
*
|
|
* _.min([4, 2, 8, 6]);
|
|
* // => 2
|
|
*
|
|
* var characters = [
|
|
* { 'name': 'barney', 'age': 36 },
|
|
* { 'name': 'fred', 'age': 40 }
|
|
* ];
|
|
*
|
|
* _.min(characters, function(chr) { return chr.age; });
|
|
* // => { 'name': 'barney', 'age': 36 };
|
|
*
|
|
* // using "_.pluck" callback shorthand
|
|
* _.min(characters, 'age');
|
|
* // => { 'name': 'barney', 'age': 36 };
|
|
*/
|
|
function min(collection, callback, thisArg) {
|
|
var computed = Infinity,
|
|
result = computed;
|
|
|
|
// allows working with functions like `_.map` without using
|
|
// their `index` argument as a callback
|
|
if (typeof callback != 'function' && thisArg && thisArg[callback] === collection) {
|
|
callback = null;
|
|
}
|
|
if (callback == null && isArray(collection)) {
|
|
var index = -1,
|
|
length = collection.length;
|
|
|
|
while (++index < length) {
|
|
var value = collection[index];
|
|
if (value < result) {
|
|
result = value;
|
|
}
|
|
}
|
|
} else {
|
|
callback = (callback == null && isString(collection))
|
|
? charAtCallback
|
|
: lodash.createCallback(callback, thisArg, 3);
|
|
|
|
forEach(collection, function(value, index, collection) {
|
|
var current = callback(value, index, collection);
|
|
if (current < computed) {
|
|
computed = current;
|
|
result = value;
|
|
}
|
|
});
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Retrieves the value of a specified property from all elements in the collection.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @type Function
|
|
* @category Collections
|
|
* @param {Array|Object|string} collection The collection to iterate over.
|
|
* @param {string} property The property to pluck.
|
|
* @returns {Array} Returns a new array of property values.
|
|
* @example
|
|
*
|
|
* var characters = [
|
|
* { 'name': 'barney', 'age': 36 },
|
|
* { 'name': 'fred', 'age': 40 }
|
|
* ];
|
|
*
|
|
* _.pluck(characters, 'name');
|
|
* // => ['barney', 'fred']
|
|
*/
|
|
function pluck(collection, property) {
|
|
var index = -1,
|
|
length = collection ? collection.length : 0;
|
|
|
|
if (typeof length == 'number') {
|
|
var result = Array(length);
|
|
while (++index < length) {
|
|
result[index] = collection[index][property];
|
|
}
|
|
}
|
|
return result || map(collection, property);
|
|
}
|
|
|
|
/**
|
|
* Reduces a collection to a value which is the accumulated result of running
|
|
* each element in the collection through the callback, where each successive
|
|
* callback execution consumes the return value of the previous execution. If
|
|
* `accumulator` is not provided the first element of the collection will be
|
|
* used as the initial `accumulator` value. The callback is bound to `thisArg`
|
|
* and invoked with four arguments; (accumulator, value, index|key, collection).
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @alias foldl, inject
|
|
* @category Collections
|
|
* @param {Array|Object|string} collection The collection to iterate over.
|
|
* @param {Function} [callback=identity] The function called per iteration.
|
|
* @param {*} [accumulator] Initial value of the accumulator.
|
|
* @param {*} [thisArg] The `this` binding of `callback`.
|
|
* @returns {*} Returns the accumulated value.
|
|
* @example
|
|
*
|
|
* var sum = _.reduce([1, 2, 3], function(sum, num) {
|
|
* return sum + num;
|
|
* });
|
|
* // => 6
|
|
*
|
|
* var mapped = _.reduce({ 'a': 1, 'b': 2, 'c': 3 }, function(result, num, key) {
|
|
* result[key] = num * 3;
|
|
* return result;
|
|
* }, {});
|
|
* // => { 'a': 3, 'b': 6, 'c': 9 }
|
|
*/
|
|
function reduce(collection, callback, accumulator, thisArg) {
|
|
if (!collection) return accumulator;
|
|
var noaccum = arguments.length < 3;
|
|
callback = lodash.createCallback(callback, thisArg, 4);
|
|
|
|
var index = -1,
|
|
length = collection.length;
|
|
|
|
if (typeof length == 'number') {
|
|
if (noaccum) {
|
|
accumulator = collection[++index];
|
|
}
|
|
while (++index < length) {
|
|
accumulator = callback(accumulator, collection[index], index, collection);
|
|
}
|
|
} else {
|
|
forOwn(collection, function(value, index, collection) {
|
|
accumulator = noaccum
|
|
? (noaccum = false, value)
|
|
: callback(accumulator, value, index, collection)
|
|
});
|
|
}
|
|
return accumulator;
|
|
}
|
|
|
|
/**
|
|
* This method is like `_.reduce` except that it iterates over elements
|
|
* of a `collection` from right to left.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @alias foldr
|
|
* @category Collections
|
|
* @param {Array|Object|string} collection The collection to iterate over.
|
|
* @param {Function} [callback=identity] The function called per iteration.
|
|
* @param {*} [accumulator] Initial value of the accumulator.
|
|
* @param {*} [thisArg] The `this` binding of `callback`.
|
|
* @returns {*} Returns the accumulated value.
|
|
* @example
|
|
*
|
|
* var list = [[0, 1], [2, 3], [4, 5]];
|
|
* var flat = _.reduceRight(list, function(a, b) { return a.concat(b); }, []);
|
|
* // => [4, 5, 2, 3, 0, 1]
|
|
*/
|
|
function reduceRight(collection, callback, accumulator, thisArg) {
|
|
var noaccum = arguments.length < 3;
|
|
callback = lodash.createCallback(callback, thisArg, 4);
|
|
forEachRight(collection, function(value, index, collection) {
|
|
accumulator = noaccum
|
|
? (noaccum = false, value)
|
|
: callback(accumulator, value, index, collection);
|
|
});
|
|
return accumulator;
|
|
}
|
|
|
|
/**
|
|
* The opposite of `_.filter` this method returns the elements of a
|
|
* collection that the callback does **not** return truey for.
|
|
*
|
|
* If a property name is provided for `callback` the created "_.pluck" style
|
|
* callback will return the property value of the given element.
|
|
*
|
|
* If an object is provided for `callback` the created "_.where" style callback
|
|
* will return `true` for elements that have the properties of the given object,
|
|
* else `false`.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Collections
|
|
* @param {Array|Object|string} collection The collection to iterate over.
|
|
* @param {Function|Object|string} [callback=identity] The function called
|
|
* per iteration. If a property name or object is provided it will be used
|
|
* to create a "_.pluck" or "_.where" style callback, respectively.
|
|
* @param {*} [thisArg] The `this` binding of `callback`.
|
|
* @returns {Array} Returns a new array of elements that failed the callback check.
|
|
* @example
|
|
*
|
|
* var odds = _.reject([1, 2, 3, 4, 5, 6], function(num) { return num % 2 == 0; });
|
|
* // => [1, 3, 5]
|
|
*
|
|
* var characters = [
|
|
* { 'name': 'barney', 'age': 36, 'blocked': false },
|
|
* { 'name': 'fred', 'age': 40, 'blocked': true }
|
|
* ];
|
|
*
|
|
* // using "_.pluck" callback shorthand
|
|
* _.reject(characters, 'blocked');
|
|
* // => [{ 'name': 'barney', 'age': 36, 'blocked': false }]
|
|
*
|
|
* // using "_.where" callback shorthand
|
|
* _.reject(characters, { 'age': 36 });
|
|
* // => [{ 'name': 'fred', 'age': 40, 'blocked': true }]
|
|
*/
|
|
function reject(collection, callback, thisArg) {
|
|
callback = lodash.createCallback(callback, thisArg, 3);
|
|
return filter(collection, function(value, index, collection) {
|
|
return !callback(value, index, collection);
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Retrieves a random element or `n` random elements from a collection.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Collections
|
|
* @param {Array|Object|string} collection The collection to sample.
|
|
* @param {number} [n] The number of elements to sample.
|
|
* @param- {Object} [guard] Allows working with functions like `_.map`
|
|
* without using their `index` arguments as `n`.
|
|
* @returns {Array} Returns the random sample(s) of `collection`.
|
|
* @example
|
|
*
|
|
* _.sample([1, 2, 3, 4]);
|
|
* // => 2
|
|
*
|
|
* _.sample([1, 2, 3, 4], 2);
|
|
* // => [3, 1]
|
|
*/
|
|
function sample(collection, n, guard) {
|
|
if (collection && typeof collection.length != 'number') {
|
|
collection = values(collection);
|
|
}
|
|
if (n == null || guard) {
|
|
return collection ? collection[baseRandom(0, collection.length - 1)] : undefined;
|
|
}
|
|
var result = shuffle(collection);
|
|
result.length = nativeMin(nativeMax(0, n), result.length);
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Creates an array of shuffled values, using a version of the Fisher-Yates
|
|
* shuffle. See http://en.wikipedia.org/wiki/Fisher-Yates_shuffle.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Collections
|
|
* @param {Array|Object|string} collection The collection to shuffle.
|
|
* @returns {Array} Returns a new shuffled collection.
|
|
* @example
|
|
*
|
|
* _.shuffle([1, 2, 3, 4, 5, 6]);
|
|
* // => [4, 1, 6, 3, 5, 2]
|
|
*/
|
|
function shuffle(collection) {
|
|
var index = -1,
|
|
length = collection ? collection.length : 0,
|
|
result = Array(typeof length == 'number' ? length : 0);
|
|
|
|
forEach(collection, function(value) {
|
|
var rand = baseRandom(0, ++index);
|
|
result[index] = result[rand];
|
|
result[rand] = value;
|
|
});
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Gets the size of the `collection` by returning `collection.length` for arrays
|
|
* and array-like objects or the number of own enumerable properties for objects.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Collections
|
|
* @param {Array|Object|string} collection The collection to inspect.
|
|
* @returns {number} Returns `collection.length` or number of own enumerable properties.
|
|
* @example
|
|
*
|
|
* _.size([1, 2]);
|
|
* // => 2
|
|
*
|
|
* _.size({ 'one': 1, 'two': 2, 'three': 3 });
|
|
* // => 3
|
|
*
|
|
* _.size('pebbles');
|
|
* // => 5
|
|
*/
|
|
function size(collection) {
|
|
var length = collection ? collection.length : 0;
|
|
return typeof length == 'number' ? length : keys(collection).length;
|
|
}
|
|
|
|
/**
|
|
* Checks if the callback returns a truey value for **any** element of a
|
|
* collection. The function returns as soon as it finds a passing value and
|
|
* does not iterate over the entire collection. The callback is bound to
|
|
* `thisArg` and invoked with three arguments; (value, index|key, collection).
|
|
*
|
|
* If a property name is provided for `callback` the created "_.pluck" style
|
|
* callback will return the property value of the given element.
|
|
*
|
|
* If an object is provided for `callback` the created "_.where" style callback
|
|
* will return `true` for elements that have the properties of the given object,
|
|
* else `false`.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @alias any
|
|
* @category Collections
|
|
* @param {Array|Object|string} collection The collection to iterate over.
|
|
* @param {Function|Object|string} [callback=identity] The function called
|
|
* per iteration. If a property name or object is provided it will be used
|
|
* to create a "_.pluck" or "_.where" style callback, respectively.
|
|
* @param {*} [thisArg] The `this` binding of `callback`.
|
|
* @returns {boolean} Returns `true` if any element passed the callback check,
|
|
* else `false`.
|
|
* @example
|
|
*
|
|
* _.some([null, 0, 'yes', false], Boolean);
|
|
* // => true
|
|
*
|
|
* var characters = [
|
|
* { 'name': 'barney', 'age': 36, 'blocked': false },
|
|
* { 'name': 'fred', 'age': 40, 'blocked': true }
|
|
* ];
|
|
*
|
|
* // using "_.pluck" callback shorthand
|
|
* _.some(characters, 'blocked');
|
|
* // => true
|
|
*
|
|
* // using "_.where" callback shorthand
|
|
* _.some(characters, { 'age': 1 });
|
|
* // => false
|
|
*/
|
|
function some(collection, callback, thisArg) {
|
|
var result;
|
|
callback = lodash.createCallback(callback, thisArg, 3);
|
|
|
|
var index = -1,
|
|
length = collection ? collection.length : 0;
|
|
|
|
if (typeof length == 'number') {
|
|
while (++index < length) {
|
|
if ((result = callback(collection[index], index, collection))) {
|
|
break;
|
|
}
|
|
}
|
|
} else {
|
|
forOwn(collection, function(value, index, collection) {
|
|
return !(result = callback(value, index, collection));
|
|
});
|
|
}
|
|
return !!result;
|
|
}
|
|
|
|
/**
|
|
* Creates an array of elements, sorted in ascending order by the results of
|
|
* running each element in a collection through the callback. This method
|
|
* performs a stable sort, that is, it will preserve the original sort order
|
|
* of equal elements. The callback is bound to `thisArg` and invoked with
|
|
* three arguments; (value, index|key, collection).
|
|
*
|
|
* If a property name is provided for `callback` the created "_.pluck" style
|
|
* callback will return the property value of the given element.
|
|
*
|
|
* If an object is provided for `callback` the created "_.where" style callback
|
|
* will return `true` for elements that have the properties of the given object,
|
|
* else `false`.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Collections
|
|
* @param {Array|Object|string} collection The collection to iterate over.
|
|
* @param {Function|Object|string} [callback=identity] The function called
|
|
* per iteration. If a property name or object is provided it will be used
|
|
* to create a "_.pluck" or "_.where" style callback, respectively.
|
|
* @param {*} [thisArg] The `this` binding of `callback`.
|
|
* @returns {Array} Returns a new array of sorted elements.
|
|
* @example
|
|
*
|
|
* _.sortBy([1, 2, 3], function(num) { return Math.sin(num); });
|
|
* // => [3, 1, 2]
|
|
*
|
|
* _.sortBy([1, 2, 3], function(num) { return this.sin(num); }, Math);
|
|
* // => [3, 1, 2]
|
|
*
|
|
* // using "_.pluck" callback shorthand
|
|
* _.sortBy(['banana', 'strawberry', 'apple'], 'length');
|
|
* // => ['apple', 'banana', 'strawberry']
|
|
*/
|
|
function sortBy(collection, callback, thisArg) {
|
|
var index = -1,
|
|
length = collection ? collection.length : 0,
|
|
result = Array(typeof length == 'number' ? length : 0);
|
|
|
|
callback = lodash.createCallback(callback, thisArg, 3);
|
|
forEach(collection, function(value, key, collection) {
|
|
var object = result[++index] = getObject();
|
|
object.criteria = callback(value, key, collection);
|
|
object.index = index;
|
|
object.value = value;
|
|
});
|
|
|
|
length = result.length;
|
|
result.sort(compareAscending);
|
|
while (length--) {
|
|
var object = result[length];
|
|
result[length] = object.value;
|
|
releaseObject(object);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Converts the `collection` to an array.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Collections
|
|
* @param {Array|Object|string} collection The collection to convert.
|
|
* @returns {Array} Returns the new converted array.
|
|
* @example
|
|
*
|
|
* (function() { return _.toArray(arguments).slice(1); })(1, 2, 3, 4);
|
|
* // => [2, 3, 4]
|
|
*/
|
|
function toArray(collection) {
|
|
if (collection && typeof collection.length == 'number') {
|
|
return slice(collection);
|
|
}
|
|
return values(collection);
|
|
}
|
|
|
|
/**
|
|
* Performs a deep comparison of each element in a `collection` to the given
|
|
* `properties` object, returning an array of all elements that have equivalent
|
|
* property values.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @type Function
|
|
* @category Collections
|
|
* @param {Array|Object|string} collection The collection to iterate over.
|
|
* @param {Object} properties The object of property values to filter by.
|
|
* @returns {Array} Returns a new array of elements that have the given properties.
|
|
* @example
|
|
*
|
|
* var characters = [
|
|
* { 'name': 'barney', 'age': 36, 'pets': ['hoppy'] },
|
|
* { 'name': 'fred', 'age': 40, 'pets': ['baby puss', 'dino'] }
|
|
* ];
|
|
*
|
|
* _.where(characters, { 'age': 36 });
|
|
* // => [{ 'name': 'barney', 'age': 36, 'pets': ['hoppy'] }]
|
|
*
|
|
* _.where(characters, { 'pets': ['dino'] });
|
|
* // => [{ 'name': 'fred', 'age': 40, 'pets': ['baby puss', 'dino'] }]
|
|
*/
|
|
var where = filter;
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
|
|
/**
|
|
* Creates an array with all falsey values removed. The values `false`, `null`,
|
|
* `0`, `""`, `undefined`, and `NaN` are all falsey.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Arrays
|
|
* @param {Array} array The array to compact.
|
|
* @returns {Array} Returns a new array of filtered values.
|
|
* @example
|
|
*
|
|
* _.compact([0, 1, false, 2, '', 3]);
|
|
* // => [1, 2, 3]
|
|
*/
|
|
function compact(array) {
|
|
var index = -1,
|
|
length = array ? array.length : 0,
|
|
result = [];
|
|
|
|
while (++index < length) {
|
|
var value = array[index];
|
|
if (value) {
|
|
result.push(value);
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Creates an array excluding all values of the provided arrays using strict
|
|
* equality for comparisons, i.e. `===`.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Arrays
|
|
* @param {Array} array The array to process.
|
|
* @param {...Array} [values] The arrays of values to exclude.
|
|
* @returns {Array} Returns a new array of filtered values.
|
|
* @example
|
|
*
|
|
* _.difference([1, 2, 3, 4, 5], [5, 2, 10]);
|
|
* // => [1, 3, 4]
|
|
*/
|
|
function difference(array) {
|
|
return baseDifference(array, baseFlatten(arguments, true, true, 1));
|
|
}
|
|
|
|
/**
|
|
* This method is like `_.find` except that it returns the index of the first
|
|
* element that passes the callback check, instead of the element itself.
|
|
*
|
|
* If a property name is provided for `callback` the created "_.pluck" style
|
|
* callback will return the property value of the given element.
|
|
*
|
|
* If an object is provided for `callback` the created "_.where" style callback
|
|
* will return `true` for elements that have the properties of the given object,
|
|
* else `false`.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Arrays
|
|
* @param {Array} array The array to search.
|
|
* @param {Function|Object|string} [callback=identity] The function called
|
|
* per iteration. If a property name or object is provided it will be used
|
|
* to create a "_.pluck" or "_.where" style callback, respectively.
|
|
* @param {*} [thisArg] The `this` binding of `callback`.
|
|
* @returns {number} Returns the index of the found element, else `-1`.
|
|
* @example
|
|
*
|
|
* var characters = [
|
|
* { 'name': 'barney', 'age': 36, 'blocked': false },
|
|
* { 'name': 'fred', 'age': 40, 'blocked': true },
|
|
* { 'name': 'pebbles', 'age': 1, 'blocked': false }
|
|
* ];
|
|
*
|
|
* _.findIndex(characters, function(chr) {
|
|
* return chr.age < 20;
|
|
* });
|
|
* // => 2
|
|
*
|
|
* // using "_.where" callback shorthand
|
|
* _.findIndex(characters, { 'age': 36 });
|
|
* // => 0
|
|
*
|
|
* // using "_.pluck" callback shorthand
|
|
* _.findIndex(characters, 'blocked');
|
|
* // => 1
|
|
*/
|
|
function findIndex(array, callback, thisArg) {
|
|
var index = -1,
|
|
length = array ? array.length : 0;
|
|
|
|
callback = lodash.createCallback(callback, thisArg, 3);
|
|
while (++index < length) {
|
|
if (callback(array[index], index, array)) {
|
|
return index;
|
|
}
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
/**
|
|
* This method is like `_.findIndex` except that it iterates over elements
|
|
* of a `collection` from right to left.
|
|
*
|
|
* If a property name is provided for `callback` the created "_.pluck" style
|
|
* callback will return the property value of the given element.
|
|
*
|
|
* If an object is provided for `callback` the created "_.where" style callback
|
|
* will return `true` for elements that have the properties of the given object,
|
|
* else `false`.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Arrays
|
|
* @param {Array} array The array to search.
|
|
* @param {Function|Object|string} [callback=identity] The function called
|
|
* per iteration. If a property name or object is provided it will be used
|
|
* to create a "_.pluck" or "_.where" style callback, respectively.
|
|
* @param {*} [thisArg] The `this` binding of `callback`.
|
|
* @returns {number} Returns the index of the found element, else `-1`.
|
|
* @example
|
|
*
|
|
* var characters = [
|
|
* { 'name': 'barney', 'age': 36, 'blocked': true },
|
|
* { 'name': 'fred', 'age': 40, 'blocked': false },
|
|
* { 'name': 'pebbles', 'age': 1, 'blocked': true }
|
|
* ];
|
|
*
|
|
* _.findLastIndex(characters, function(chr) {
|
|
* return chr.age > 30;
|
|
* });
|
|
* // => 1
|
|
*
|
|
* // using "_.where" callback shorthand
|
|
* _.findLastIndex(characters, { 'age': 36 });
|
|
* // => 0
|
|
*
|
|
* // using "_.pluck" callback shorthand
|
|
* _.findLastIndex(characters, 'blocked');
|
|
* // => 2
|
|
*/
|
|
function findLastIndex(array, callback, thisArg) {
|
|
var length = array ? array.length : 0;
|
|
callback = lodash.createCallback(callback, thisArg, 3);
|
|
while (length--) {
|
|
if (callback(array[length], length, array)) {
|
|
return length;
|
|
}
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
/**
|
|
* Gets the first element or first `n` elements of an array. If a callback
|
|
* is provided elements at the beginning of the array are returned as long
|
|
* as the callback returns truey. The callback is bound to `thisArg` and
|
|
* invoked with three arguments; (value, index, array).
|
|
*
|
|
* If a property name is provided for `callback` the created "_.pluck" style
|
|
* callback will return the property value of the given element.
|
|
*
|
|
* If an object is provided for `callback` the created "_.where" style callback
|
|
* will return `true` for elements that have the properties of the given object,
|
|
* else `false`.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @alias head, take
|
|
* @category Arrays
|
|
* @param {Array} array The array to query.
|
|
* @param {Function|Object|number|string} [callback] The function called
|
|
* per element or the number of elements to return. If a property name or
|
|
* object is provided it will be used to create a "_.pluck" or "_.where"
|
|
* style callback, respectively.
|
|
* @param {*} [thisArg] The `this` binding of `callback`.
|
|
* @returns {*} Returns the first element(s) of `array`.
|
|
* @example
|
|
*
|
|
* _.first([1, 2, 3]);
|
|
* // => 1
|
|
*
|
|
* _.first([1, 2, 3], 2);
|
|
* // => [1, 2]
|
|
*
|
|
* _.first([1, 2, 3], function(num) {
|
|
* return num < 3;
|
|
* });
|
|
* // => [1, 2]
|
|
*
|
|
* var characters = [
|
|
* { 'name': 'barney', 'blocked': true, 'employer': 'slate' },
|
|
* { 'name': 'fred', 'blocked': false, 'employer': 'slate' },
|
|
* { 'name': 'pebbles', 'blocked': true, 'employer': 'na' }
|
|
* ];
|
|
*
|
|
* // using "_.pluck" callback shorthand
|
|
* _.first(characters, 'blocked');
|
|
* // => [{ 'name': 'barney', 'blocked': true, 'employer': 'slate' }]
|
|
*
|
|
* // using "_.where" callback shorthand
|
|
* _.pluck(_.first(characters, { 'employer': 'slate' }), 'name');
|
|
* // => ['barney', 'fred']
|
|
*/
|
|
function first(array, callback, thisArg) {
|
|
var n = 0,
|
|
length = array ? array.length : 0;
|
|
|
|
if (typeof callback != 'number' && callback != null) {
|
|
var index = -1;
|
|
callback = lodash.createCallback(callback, thisArg, 3);
|
|
while (++index < length && callback(array[index], index, array)) {
|
|
n++;
|
|
}
|
|
} else {
|
|
n = callback;
|
|
if (n == null || thisArg) {
|
|
return array ? array[0] : undefined;
|
|
}
|
|
}
|
|
return slice(array, 0, nativeMin(nativeMax(0, n), length));
|
|
}
|
|
|
|
/**
|
|
* Flattens a nested array (the nesting can be to any depth). If `isShallow`
|
|
* is truey, the array will only be flattened a single level. If a callback
|
|
* is provided each element of the array is passed through the callback before
|
|
* flattening. The callback is bound to `thisArg` and invoked with three
|
|
* arguments; (value, index, array).
|
|
*
|
|
* If a property name is provided for `callback` the created "_.pluck" style
|
|
* callback will return the property value of the given element.
|
|
*
|
|
* If an object is provided for `callback` the created "_.where" style callback
|
|
* will return `true` for elements that have the properties of the given object,
|
|
* else `false`.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Arrays
|
|
* @param {Array} array The array to flatten.
|
|
* @param {boolean} [isShallow=false] A flag to restrict flattening to a single level.
|
|
* @param {Function|Object|string} [callback=identity] The function called
|
|
* per iteration. If a property name or object is provided it will be used
|
|
* to create a "_.pluck" or "_.where" style callback, respectively.
|
|
* @param {*} [thisArg] The `this` binding of `callback`.
|
|
* @returns {Array} Returns a new flattened array.
|
|
* @example
|
|
*
|
|
* _.flatten([1, [2], [3, [[4]]]]);
|
|
* // => [1, 2, 3, 4];
|
|
*
|
|
* _.flatten([1, [2], [3, [[4]]]], true);
|
|
* // => [1, 2, 3, [[4]]];
|
|
*
|
|
* var characters = [
|
|
* { 'name': 'barney', 'age': 30, 'pets': ['hoppy'] },
|
|
* { 'name': 'fred', 'age': 40, 'pets': ['baby puss', 'dino'] }
|
|
* ];
|
|
*
|
|
* // using "_.pluck" callback shorthand
|
|
* _.flatten(characters, 'pets');
|
|
* // => ['hoppy', 'baby puss', 'dino']
|
|
*/
|
|
function flatten(array, isShallow, callback, thisArg) {
|
|
// juggle arguments
|
|
if (typeof isShallow != 'boolean' && isShallow != null) {
|
|
thisArg = callback;
|
|
callback = (typeof isShallow != 'function' && thisArg && thisArg[isShallow] === array) ? null : isShallow;
|
|
isShallow = false;
|
|
}
|
|
if (callback != null) {
|
|
array = map(array, callback, thisArg);
|
|
}
|
|
return baseFlatten(array, isShallow);
|
|
}
|
|
|
|
/**
|
|
* Gets the index at which the first occurrence of `value` is found using
|
|
* strict equality for comparisons, i.e. `===`. If the array is already sorted
|
|
* providing `true` for `fromIndex` will run a faster binary search.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Arrays
|
|
* @param {Array} array The array to search.
|
|
* @param {*} value The value to search for.
|
|
* @param {boolean|number} [fromIndex=0] The index to search from or `true`
|
|
* to perform a binary search on a sorted array.
|
|
* @returns {number} Returns the index of the matched value or `-1`.
|
|
* @example
|
|
*
|
|
* _.indexOf([1, 2, 3, 1, 2, 3], 2);
|
|
* // => 1
|
|
*
|
|
* _.indexOf([1, 2, 3, 1, 2, 3], 2, 3);
|
|
* // => 4
|
|
*
|
|
* _.indexOf([1, 1, 2, 2, 3, 3], 2, true);
|
|
* // => 2
|
|
*/
|
|
function indexOf(array, value, fromIndex) {
|
|
if (typeof fromIndex == 'number') {
|
|
var length = array ? array.length : 0;
|
|
fromIndex = (fromIndex < 0 ? nativeMax(0, length + fromIndex) : fromIndex || 0);
|
|
} else if (fromIndex) {
|
|
var index = sortedIndex(array, value);
|
|
return array[index] === value ? index : -1;
|
|
}
|
|
return baseIndexOf(array, value, fromIndex);
|
|
}
|
|
|
|
/**
|
|
* Gets all but the last element or last `n` elements of an array. If a
|
|
* callback is provided elements at the end of the array are excluded from
|
|
* the result as long as the callback returns truey. The callback is bound
|
|
* to `thisArg` and invoked with three arguments; (value, index, array).
|
|
*
|
|
* If a property name is provided for `callback` the created "_.pluck" style
|
|
* callback will return the property value of the given element.
|
|
*
|
|
* If an object is provided for `callback` the created "_.where" style callback
|
|
* will return `true` for elements that have the properties of the given object,
|
|
* else `false`.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Arrays
|
|
* @param {Array} array The array to query.
|
|
* @param {Function|Object|number|string} [callback=1] The function called
|
|
* per element or the number of elements to exclude. If a property name or
|
|
* object is provided it will be used to create a "_.pluck" or "_.where"
|
|
* style callback, respectively.
|
|
* @param {*} [thisArg] The `this` binding of `callback`.
|
|
* @returns {Array} Returns a slice of `array`.
|
|
* @example
|
|
*
|
|
* _.initial([1, 2, 3]);
|
|
* // => [1, 2]
|
|
*
|
|
* _.initial([1, 2, 3], 2);
|
|
* // => [1]
|
|
*
|
|
* _.initial([1, 2, 3], function(num) {
|
|
* return num > 1;
|
|
* });
|
|
* // => [1]
|
|
*
|
|
* var characters = [
|
|
* { 'name': 'barney', 'blocked': false, 'employer': 'slate' },
|
|
* { 'name': 'fred', 'blocked': true, 'employer': 'slate' },
|
|
* { 'name': 'pebbles', 'blocked': true, 'employer': 'na' }
|
|
* ];
|
|
*
|
|
* // using "_.pluck" callback shorthand
|
|
* _.initial(characters, 'blocked');
|
|
* // => [{ 'name': 'barney', 'blocked': false, 'employer': 'slate' }]
|
|
*
|
|
* // using "_.where" callback shorthand
|
|
* _.pluck(_.initial(characters, { 'employer': 'na' }), 'name');
|
|
* // => ['barney', 'fred']
|
|
*/
|
|
function initial(array, callback, thisArg) {
|
|
var n = 0,
|
|
length = array ? array.length : 0;
|
|
|
|
if (typeof callback != 'number' && callback != null) {
|
|
var index = length;
|
|
callback = lodash.createCallback(callback, thisArg, 3);
|
|
while (index-- && callback(array[index], index, array)) {
|
|
n++;
|
|
}
|
|
} else {
|
|
n = (callback == null || thisArg) ? 1 : callback || n;
|
|
}
|
|
return slice(array, 0, nativeMin(nativeMax(0, length - n), length));
|
|
}
|
|
|
|
/**
|
|
* Creates an array of unique values present in all provided arrays using
|
|
* strict equality for comparisons, i.e. `===`.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Arrays
|
|
* @param {...Array} [array] The arrays to inspect.
|
|
* @returns {Array} Returns an array of composite values.
|
|
* @example
|
|
*
|
|
* _.intersection([1, 2, 3], [101, 2, 1, 10], [2, 1]);
|
|
* // => [1, 2]
|
|
*/
|
|
function intersection(array) {
|
|
var args = arguments,
|
|
argsLength = args.length,
|
|
argsIndex = -1,
|
|
caches = getArray(),
|
|
index = -1,
|
|
indexOf = getIndexOf(),
|
|
length = array ? array.length : 0,
|
|
result = [],
|
|
seen = getArray();
|
|
|
|
while (++argsIndex < argsLength) {
|
|
var value = args[argsIndex];
|
|
caches[argsIndex] = indexOf === baseIndexOf &&
|
|
(value ? value.length : 0) >= largeArraySize &&
|
|
createCache(argsIndex ? args[argsIndex] : seen);
|
|
}
|
|
outer:
|
|
while (++index < length) {
|
|
var cache = caches[0];
|
|
value = array[index];
|
|
|
|
if ((cache ? cacheIndexOf(cache, value) : indexOf(seen, value)) < 0) {
|
|
argsIndex = argsLength;
|
|
(cache || seen).push(value);
|
|
while (--argsIndex) {
|
|
cache = caches[argsIndex];
|
|
if ((cache ? cacheIndexOf(cache, value) : indexOf(args[argsIndex], value)) < 0) {
|
|
continue outer;
|
|
}
|
|
}
|
|
result.push(value);
|
|
}
|
|
}
|
|
while (argsLength--) {
|
|
cache = caches[argsLength];
|
|
if (cache) {
|
|
releaseObject(cache);
|
|
}
|
|
}
|
|
releaseArray(caches);
|
|
releaseArray(seen);
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Gets the last element or last `n` elements of an array. If a callback is
|
|
* provided elements at the end of the array are returned as long as the
|
|
* callback returns truey. The callback is bound to `thisArg` and invoked
|
|
* with three arguments; (value, index, array).
|
|
*
|
|
* If a property name is provided for `callback` the created "_.pluck" style
|
|
* callback will return the property value of the given element.
|
|
*
|
|
* If an object is provided for `callback` the created "_.where" style callback
|
|
* will return `true` for elements that have the properties of the given object,
|
|
* else `false`.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Arrays
|
|
* @param {Array} array The array to query.
|
|
* @param {Function|Object|number|string} [callback] The function called
|
|
* per element or the number of elements to return. If a property name or
|
|
* object is provided it will be used to create a "_.pluck" or "_.where"
|
|
* style callback, respectively.
|
|
* @param {*} [thisArg] The `this` binding of `callback`.
|
|
* @returns {*} Returns the last element(s) of `array`.
|
|
* @example
|
|
*
|
|
* _.last([1, 2, 3]);
|
|
* // => 3
|
|
*
|
|
* _.last([1, 2, 3], 2);
|
|
* // => [2, 3]
|
|
*
|
|
* _.last([1, 2, 3], function(num) {
|
|
* return num > 1;
|
|
* });
|
|
* // => [2, 3]
|
|
*
|
|
* var characters = [
|
|
* { 'name': 'barney', 'blocked': false, 'employer': 'slate' },
|
|
* { 'name': 'fred', 'blocked': true, 'employer': 'slate' },
|
|
* { 'name': 'pebbles', 'blocked': true, 'employer': 'na' }
|
|
* ];
|
|
*
|
|
* // using "_.pluck" callback shorthand
|
|
* _.pluck(_.last(characters, 'blocked'), 'name');
|
|
* // => ['fred', 'pebbles']
|
|
*
|
|
* // using "_.where" callback shorthand
|
|
* _.last(characters, { 'employer': 'na' });
|
|
* // => [{ 'name': 'pebbles', 'blocked': true, 'employer': 'na' }]
|
|
*/
|
|
function last(array, callback, thisArg) {
|
|
var n = 0,
|
|
length = array ? array.length : 0;
|
|
|
|
if (typeof callback != 'number' && callback != null) {
|
|
var index = length;
|
|
callback = lodash.createCallback(callback, thisArg, 3);
|
|
while (index-- && callback(array[index], index, array)) {
|
|
n++;
|
|
}
|
|
} else {
|
|
n = callback;
|
|
if (n == null || thisArg) {
|
|
return array ? array[length - 1] : undefined;
|
|
}
|
|
}
|
|
return slice(array, nativeMax(0, length - n));
|
|
}
|
|
|
|
/**
|
|
* Gets the index at which the last occurrence of `value` is found using strict
|
|
* equality for comparisons, i.e. `===`. If `fromIndex` is negative, it is used
|
|
* as the offset from the end of the collection.
|
|
*
|
|
* If a property name is provided for `callback` the created "_.pluck" style
|
|
* callback will return the property value of the given element.
|
|
*
|
|
* If an object is provided for `callback` the created "_.where" style callback
|
|
* will return `true` for elements that have the properties of the given object,
|
|
* else `false`.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Arrays
|
|
* @param {Array} array The array to search.
|
|
* @param {*} value The value to search for.
|
|
* @param {number} [fromIndex=array.length-1] The index to search from.
|
|
* @returns {number} Returns the index of the matched value or `-1`.
|
|
* @example
|
|
*
|
|
* _.lastIndexOf([1, 2, 3, 1, 2, 3], 2);
|
|
* // => 4
|
|
*
|
|
* _.lastIndexOf([1, 2, 3, 1, 2, 3], 2, 3);
|
|
* // => 1
|
|
*/
|
|
function lastIndexOf(array, value, fromIndex) {
|
|
var index = array ? array.length : 0;
|
|
if (typeof fromIndex == 'number') {
|
|
index = (fromIndex < 0 ? nativeMax(0, index + fromIndex) : nativeMin(fromIndex, index - 1)) + 1;
|
|
}
|
|
while (index--) {
|
|
if (array[index] === value) {
|
|
return index;
|
|
}
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
/**
|
|
* Removes all provided values from the given array using strict equality for
|
|
* comparisons, i.e. `===`.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Arrays
|
|
* @param {Array} array The array to modify.
|
|
* @param {...*} [value] The values to remove.
|
|
* @returns {Array} Returns `array`.
|
|
* @example
|
|
*
|
|
* var array = [1, 2, 3, 1, 2, 3];
|
|
* _.pull(array, 2, 3);
|
|
* console.log(array);
|
|
* // => [1, 1]
|
|
*/
|
|
function pull(array) {
|
|
var args = arguments,
|
|
argsIndex = 0,
|
|
argsLength = args.length,
|
|
length = array ? array.length : 0;
|
|
|
|
while (++argsIndex < argsLength) {
|
|
var index = -1,
|
|
value = args[argsIndex];
|
|
while (++index < length) {
|
|
if (array[index] === value) {
|
|
splice.call(array, index--, 1);
|
|
length--;
|
|
}
|
|
}
|
|
}
|
|
return array;
|
|
}
|
|
|
|
/**
|
|
* Creates an array of numbers (positive and/or negative) progressing from
|
|
* `start` up to but not including `end`. If `start` is less than `stop` a
|
|
* zero-length range is created unless a negative `step` is specified.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Arrays
|
|
* @param {number} [start=0] The start of the range.
|
|
* @param {number} end The end of the range.
|
|
* @param {number} [step=1] The value to increment or decrement by.
|
|
* @returns {Array} Returns a new range array.
|
|
* @example
|
|
*
|
|
* _.range(4);
|
|
* // => [0, 1, 2, 3]
|
|
*
|
|
* _.range(1, 5);
|
|
* // => [1, 2, 3, 4]
|
|
*
|
|
* _.range(0, 20, 5);
|
|
* // => [0, 5, 10, 15]
|
|
*
|
|
* _.range(0, -4, -1);
|
|
* // => [0, -1, -2, -3]
|
|
*
|
|
* _.range(1, 4, 0);
|
|
* // => [1, 1, 1]
|
|
*
|
|
* _.range(0);
|
|
* // => []
|
|
*/
|
|
function range(start, end, step) {
|
|
start = +start || 0;
|
|
step = typeof step == 'number' ? step : (+step || 1);
|
|
|
|
if (end == null) {
|
|
end = start;
|
|
start = 0;
|
|
}
|
|
// use `Array(length)` so engines like Chakra and V8 avoid slower modes
|
|
// http://youtu.be/XAqIpGU8ZZk#t=17m25s
|
|
var index = -1,
|
|
length = nativeMax(0, ceil((end - start) / (step || 1))),
|
|
result = Array(length);
|
|
|
|
while (++index < length) {
|
|
result[index] = start;
|
|
start += step;
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Removes all elements from an array that the callback returns truey for
|
|
* and returns an array of removed elements. The callback is bound to `thisArg`
|
|
* and invoked with three arguments; (value, index, array).
|
|
*
|
|
* If a property name is provided for `callback` the created "_.pluck" style
|
|
* callback will return the property value of the given element.
|
|
*
|
|
* If an object is provided for `callback` the created "_.where" style callback
|
|
* will return `true` for elements that have the properties of the given object,
|
|
* else `false`.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Arrays
|
|
* @param {Array} array The array to modify.
|
|
* @param {Function|Object|string} [callback=identity] The function called
|
|
* per iteration. If a property name or object is provided it will be used
|
|
* to create a "_.pluck" or "_.where" style callback, respectively.
|
|
* @param {*} [thisArg] The `this` binding of `callback`.
|
|
* @returns {Array} Returns a new array of removed elements.
|
|
* @example
|
|
*
|
|
* var array = [1, 2, 3, 4, 5, 6];
|
|
* var evens = _.remove(array, function(num) { return num % 2 == 0; });
|
|
*
|
|
* console.log(array);
|
|
* // => [1, 3, 5]
|
|
*
|
|
* console.log(evens);
|
|
* // => [2, 4, 6]
|
|
*/
|
|
function remove(array, callback, thisArg) {
|
|
var index = -1,
|
|
length = array ? array.length : 0,
|
|
result = [];
|
|
|
|
callback = lodash.createCallback(callback, thisArg, 3);
|
|
while (++index < length) {
|
|
var value = array[index];
|
|
if (callback(value, index, array)) {
|
|
result.push(value);
|
|
splice.call(array, index--, 1);
|
|
length--;
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* The opposite of `_.initial` this method gets all but the first element or
|
|
* first `n` elements of an array. If a callback function is provided elements
|
|
* at the beginning of the array are excluded from the result as long as the
|
|
* callback returns truey. The callback is bound to `thisArg` and invoked
|
|
* with three arguments; (value, index, array).
|
|
*
|
|
* If a property name is provided for `callback` the created "_.pluck" style
|
|
* callback will return the property value of the given element.
|
|
*
|
|
* If an object is provided for `callback` the created "_.where" style callback
|
|
* will return `true` for elements that have the properties of the given object,
|
|
* else `false`.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @alias drop, tail
|
|
* @category Arrays
|
|
* @param {Array} array The array to query.
|
|
* @param {Function|Object|number|string} [callback=1] The function called
|
|
* per element or the number of elements to exclude. If a property name or
|
|
* object is provided it will be used to create a "_.pluck" or "_.where"
|
|
* style callback, respectively.
|
|
* @param {*} [thisArg] The `this` binding of `callback`.
|
|
* @returns {Array} Returns a slice of `array`.
|
|
* @example
|
|
*
|
|
* _.rest([1, 2, 3]);
|
|
* // => [2, 3]
|
|
*
|
|
* _.rest([1, 2, 3], 2);
|
|
* // => [3]
|
|
*
|
|
* _.rest([1, 2, 3], function(num) {
|
|
* return num < 3;
|
|
* });
|
|
* // => [3]
|
|
*
|
|
* var characters = [
|
|
* { 'name': 'barney', 'blocked': true, 'employer': 'slate' },
|
|
* { 'name': 'fred', 'blocked': false, 'employer': 'slate' },
|
|
* { 'name': 'pebbles', 'blocked': true, 'employer': 'na' }
|
|
* ];
|
|
*
|
|
* // using "_.pluck" callback shorthand
|
|
* _.pluck(_.rest(characters, 'blocked'), 'name');
|
|
* // => ['fred', 'pebbles']
|
|
*
|
|
* // using "_.where" callback shorthand
|
|
* _.rest(characters, { 'employer': 'slate' });
|
|
* // => [{ 'name': 'pebbles', 'blocked': true, 'employer': 'na' }]
|
|
*/
|
|
function rest(array, callback, thisArg) {
|
|
if (typeof callback != 'number' && callback != null) {
|
|
var n = 0,
|
|
index = -1,
|
|
length = array ? array.length : 0;
|
|
|
|
callback = lodash.createCallback(callback, thisArg, 3);
|
|
while (++index < length && callback(array[index], index, array)) {
|
|
n++;
|
|
}
|
|
} else {
|
|
n = (callback == null || thisArg) ? 1 : nativeMax(0, callback);
|
|
}
|
|
return slice(array, n);
|
|
}
|
|
|
|
/**
|
|
* Uses a binary search to determine the smallest index at which a value
|
|
* should be inserted into a given sorted array in order to maintain the sort
|
|
* order of the array. If a callback is provided it will be executed for
|
|
* `value` and each element of `array` to compute their sort ranking. The
|
|
* callback is bound to `thisArg` and invoked with one argument; (value).
|
|
*
|
|
* If a property name is provided for `callback` the created "_.pluck" style
|
|
* callback will return the property value of the given element.
|
|
*
|
|
* If an object is provided for `callback` the created "_.where" style callback
|
|
* will return `true` for elements that have the properties of the given object,
|
|
* else `false`.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Arrays
|
|
* @param {Array} array The array to inspect.
|
|
* @param {*} value The value to evaluate.
|
|
* @param {Function|Object|string} [callback=identity] The function called
|
|
* per iteration. If a property name or object is provided it will be used
|
|
* to create a "_.pluck" or "_.where" style callback, respectively.
|
|
* @param {*} [thisArg] The `this` binding of `callback`.
|
|
* @returns {number} Returns the index at which `value` should be inserted
|
|
* into `array`.
|
|
* @example
|
|
*
|
|
* _.sortedIndex([20, 30, 50], 40);
|
|
* // => 2
|
|
*
|
|
* // using "_.pluck" callback shorthand
|
|
* _.sortedIndex([{ 'x': 20 }, { 'x': 30 }, { 'x': 50 }], { 'x': 40 }, 'x');
|
|
* // => 2
|
|
*
|
|
* var dict = {
|
|
* 'wordToNumber': { 'twenty': 20, 'thirty': 30, 'fourty': 40, 'fifty': 50 }
|
|
* };
|
|
*
|
|
* _.sortedIndex(['twenty', 'thirty', 'fifty'], 'fourty', function(word) {
|
|
* return dict.wordToNumber[word];
|
|
* });
|
|
* // => 2
|
|
*
|
|
* _.sortedIndex(['twenty', 'thirty', 'fifty'], 'fourty', function(word) {
|
|
* return this.wordToNumber[word];
|
|
* }, dict);
|
|
* // => 2
|
|
*/
|
|
function sortedIndex(array, value, callback, thisArg) {
|
|
var low = 0,
|
|
high = array ? array.length : low;
|
|
|
|
// explicitly reference `identity` for better inlining in Firefox
|
|
callback = callback ? lodash.createCallback(callback, thisArg, 1) : identity;
|
|
value = callback(value);
|
|
|
|
while (low < high) {
|
|
var mid = (low + high) >>> 1;
|
|
(callback(array[mid]) < value)
|
|
? low = mid + 1
|
|
: high = mid;
|
|
}
|
|
return low;
|
|
}
|
|
|
|
/**
|
|
* Creates an array of unique values, in order, of the provided arrays using
|
|
* strict equality for comparisons, i.e. `===`.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Arrays
|
|
* @param {...Array} [array] The arrays to inspect.
|
|
* @returns {Array} Returns an array of composite values.
|
|
* @example
|
|
*
|
|
* _.union([1, 2, 3], [101, 2, 1, 10], [2, 1]);
|
|
* // => [1, 2, 3, 101, 10]
|
|
*/
|
|
function union(array) {
|
|
return baseUniq(baseFlatten(arguments, true, true));
|
|
}
|
|
|
|
/**
|
|
* Creates a duplicate-value-free version of an array using strict equality
|
|
* for comparisons, i.e. `===`. If the array is sorted, providing
|
|
* `true` for `isSorted` will use a faster algorithm. If a callback is provided
|
|
* each element of `array` is passed through the callback before uniqueness
|
|
* is computed. The callback is bound to `thisArg` and invoked with three
|
|
* arguments; (value, index, array).
|
|
*
|
|
* If a property name is provided for `callback` the created "_.pluck" style
|
|
* callback will return the property value of the given element.
|
|
*
|
|
* If an object is provided for `callback` the created "_.where" style callback
|
|
* will return `true` for elements that have the properties of the given object,
|
|
* else `false`.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @alias unique
|
|
* @category Arrays
|
|
* @param {Array} array The array to process.
|
|
* @param {boolean} [isSorted=false] A flag to indicate that `array` is sorted.
|
|
* @param {Function|Object|string} [callback=identity] The function called
|
|
* per iteration. If a property name or object is provided it will be used
|
|
* to create a "_.pluck" or "_.where" style callback, respectively.
|
|
* @param {*} [thisArg] The `this` binding of `callback`.
|
|
* @returns {Array} Returns a duplicate-value-free array.
|
|
* @example
|
|
*
|
|
* _.uniq([1, 2, 1, 3, 1]);
|
|
* // => [1, 2, 3]
|
|
*
|
|
* _.uniq([1, 1, 2, 2, 3], true);
|
|
* // => [1, 2, 3]
|
|
*
|
|
* _.uniq(['A', 'b', 'C', 'a', 'B', 'c'], function(letter) { return letter.toLowerCase(); });
|
|
* // => ['A', 'b', 'C']
|
|
*
|
|
* _.uniq([1, 2.5, 3, 1.5, 2, 3.5], function(num) { return this.floor(num); }, Math);
|
|
* // => [1, 2.5, 3]
|
|
*
|
|
* // using "_.pluck" callback shorthand
|
|
* _.uniq([{ 'x': 1 }, { 'x': 2 }, { 'x': 1 }], 'x');
|
|
* // => [{ 'x': 1 }, { 'x': 2 }]
|
|
*/
|
|
function uniq(array, isSorted, callback, thisArg) {
|
|
// juggle arguments
|
|
if (typeof isSorted != 'boolean' && isSorted != null) {
|
|
thisArg = callback;
|
|
callback = (typeof isSorted != 'function' && thisArg && thisArg[isSorted] === array) ? null : isSorted;
|
|
isSorted = false;
|
|
}
|
|
if (callback != null) {
|
|
callback = lodash.createCallback(callback, thisArg, 3);
|
|
}
|
|
return baseUniq(array, isSorted, callback);
|
|
}
|
|
|
|
/**
|
|
* Creates an array excluding all provided values using strict equality for
|
|
* comparisons, i.e. `===`.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Arrays
|
|
* @param {Array} array The array to filter.
|
|
* @param {...*} [value] The values to exclude.
|
|
* @returns {Array} Returns a new array of filtered values.
|
|
* @example
|
|
*
|
|
* _.without([1, 2, 1, 0, 3, 1, 4], 0, 1);
|
|
* // => [2, 3, 4]
|
|
*/
|
|
function without(array) {
|
|
return baseDifference(array, slice(arguments, 1));
|
|
}
|
|
|
|
/**
|
|
* Creates an array of grouped elements, the first of which contains the first
|
|
* elements of the given arrays, the second of which contains the second
|
|
* elements of the given arrays, and so on.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @alias unzip
|
|
* @category Arrays
|
|
* @param {...Array} [array] Arrays to process.
|
|
* @returns {Array} Returns a new array of grouped elements.
|
|
* @example
|
|
*
|
|
* _.zip(['fred', 'barney'], [30, 40], [true, false]);
|
|
* // => [['fred', 30, true], ['barney', 40, false]]
|
|
*/
|
|
function zip() {
|
|
var array = arguments.length > 1 ? arguments : arguments[0],
|
|
index = -1,
|
|
length = array ? max(pluck(array, 'length')) : 0,
|
|
result = Array(length < 0 ? 0 : length);
|
|
|
|
while (++index < length) {
|
|
result[index] = pluck(array, index);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Creates an object composed from arrays of `keys` and `values`. Provide
|
|
* either a single two dimensional array, i.e. `[[key1, value1], [key2, value2]]`
|
|
* or two arrays, one of `keys` and one of corresponding `values`.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @alias object
|
|
* @category Arrays
|
|
* @param {Array} keys The array of keys.
|
|
* @param {Array} [values=[]] The array of values.
|
|
* @returns {Object} Returns an object composed of the given keys and
|
|
* corresponding values.
|
|
* @example
|
|
*
|
|
* _.zipObject(['fred', 'barney'], [30, 40]);
|
|
* // => { 'fred': 30, 'barney': 40 }
|
|
*/
|
|
function zipObject(keys, values) {
|
|
var index = -1,
|
|
length = keys ? keys.length : 0,
|
|
result = {};
|
|
|
|
while (++index < length) {
|
|
var key = keys[index];
|
|
if (values) {
|
|
result[key] = values[index];
|
|
} else if (key) {
|
|
result[key[0]] = key[1];
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
|
|
/**
|
|
* Creates a function that executes `func`, with the `this` binding and
|
|
* arguments of the created function, only after being called `n` times.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Functions
|
|
* @param {number} n The number of times the function must be called before
|
|
* `func` is executed.
|
|
* @param {Function} func The function to restrict.
|
|
* @returns {Function} Returns the new restricted function.
|
|
* @example
|
|
*
|
|
* var saves = ['profile', 'settings'];
|
|
*
|
|
* var done = _.after(saves.length, function() {
|
|
* console.log('Done saving!');
|
|
* });
|
|
*
|
|
* _.forEach(saves, function(type) {
|
|
* asyncSave({ 'type': type, 'complete': done });
|
|
* });
|
|
* // => logs 'Done saving!', after all saves have completed
|
|
*/
|
|
function after(n, func) {
|
|
if (!isFunction(func)) {
|
|
throw new TypeError;
|
|
}
|
|
return function() {
|
|
if (--n < 1) {
|
|
return func.apply(this, arguments);
|
|
}
|
|
};
|
|
}
|
|
|
|
/**
|
|
* Creates a function that, when called, invokes `func` with the `this`
|
|
* binding of `thisArg` and prepends any additional `bind` arguments to those
|
|
* provided to the bound function.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Functions
|
|
* @param {Function} func The function to bind.
|
|
* @param {*} [thisArg] The `this` binding of `func`.
|
|
* @param {...*} [arg] Arguments to be partially applied.
|
|
* @returns {Function} Returns the new bound function.
|
|
* @example
|
|
*
|
|
* var func = function(greeting) {
|
|
* return greeting + ' ' + this.name;
|
|
* };
|
|
*
|
|
* func = _.bind(func, { 'name': 'fred' }, 'hi');
|
|
* func();
|
|
* // => 'hi fred'
|
|
*/
|
|
function bind(func, thisArg) {
|
|
return arguments.length > 2
|
|
? createWrapper(func, 17, slice(arguments, 2), null, thisArg)
|
|
: createWrapper(func, 1, null, null, thisArg);
|
|
}
|
|
|
|
/**
|
|
* Binds methods of an object to the object itself, overwriting the existing
|
|
* method. Method names may be specified as individual arguments or as arrays
|
|
* of method names. If no method names are provided all the function properties
|
|
* of `object` will be bound.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Functions
|
|
* @param {Object} object The object to bind and assign the bound methods to.
|
|
* @param {...string} [methodName] The object method names to
|
|
* bind, specified as individual method names or arrays of method names.
|
|
* @returns {Object} Returns `object`.
|
|
* @example
|
|
*
|
|
* var view = {
|
|
* 'label': 'docs',
|
|
* 'onClick': function() { console.log('clicked ' + this.label); }
|
|
* };
|
|
*
|
|
* _.bindAll(view);
|
|
* jQuery('#docs').on('click', view.onClick);
|
|
* // => logs 'clicked docs', when the button is clicked
|
|
*/
|
|
function bindAll(object) {
|
|
var funcs = arguments.length > 1 ? baseFlatten(arguments, true, false, 1) : functions(object),
|
|
index = -1,
|
|
length = funcs.length;
|
|
|
|
while (++index < length) {
|
|
var key = funcs[index];
|
|
object[key] = createWrapper(object[key], 1, null, null, object);
|
|
}
|
|
return object;
|
|
}
|
|
|
|
/**
|
|
* Creates a function that, when called, invokes the method at `object[key]`
|
|
* and prepends any additional `bindKey` arguments to those provided to the bound
|
|
* function. This method differs from `_.bind` by allowing bound functions to
|
|
* reference methods that will be redefined or don't yet exist.
|
|
* See http://michaux.ca/articles/lazy-function-definition-pattern.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Functions
|
|
* @param {Object} object The object the method belongs to.
|
|
* @param {string} key The key of the method.
|
|
* @param {...*} [arg] Arguments to be partially applied.
|
|
* @returns {Function} Returns the new bound function.
|
|
* @example
|
|
*
|
|
* var object = {
|
|
* 'name': 'fred',
|
|
* 'greet': function(greeting) {
|
|
* return greeting + ' ' + this.name;
|
|
* }
|
|
* };
|
|
*
|
|
* var func = _.bindKey(object, 'greet', 'hi');
|
|
* func();
|
|
* // => 'hi fred'
|
|
*
|
|
* object.greet = function(greeting) {
|
|
* return greeting + 'ya ' + this.name + '!';
|
|
* };
|
|
*
|
|
* func();
|
|
* // => 'hiya fred!'
|
|
*/
|
|
function bindKey(object, key) {
|
|
return arguments.length > 2
|
|
? createWrapper(key, 19, slice(arguments, 2), null, object)
|
|
: createWrapper(key, 3, null, null, object);
|
|
}
|
|
|
|
/**
|
|
* Creates a function that is the composition of the provided functions,
|
|
* where each function consumes the return value of the function that follows.
|
|
* For example, composing the functions `f()`, `g()`, and `h()` produces `f(g(h()))`.
|
|
* Each function is executed with the `this` binding of the composed function.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Functions
|
|
* @param {...Function} [func] Functions to compose.
|
|
* @returns {Function} Returns the new composed function.
|
|
* @example
|
|
*
|
|
* var realNameMap = {
|
|
* 'pebbles': 'penelope'
|
|
* };
|
|
*
|
|
* var format = function(name) {
|
|
* name = realNameMap[name.toLowerCase()] || name;
|
|
* return name.charAt(0).toUpperCase() + name.slice(1).toLowerCase();
|
|
* };
|
|
*
|
|
* var greet = function(formatted) {
|
|
* return 'Hiya ' + formatted + '!';
|
|
* };
|
|
*
|
|
* var welcome = _.compose(greet, format);
|
|
* welcome('pebbles');
|
|
* // => 'Hiya Penelope!'
|
|
*/
|
|
function compose() {
|
|
var funcs = arguments,
|
|
length = funcs.length;
|
|
|
|
while (length--) {
|
|
if (!isFunction(funcs[length])) {
|
|
throw new TypeError;
|
|
}
|
|
}
|
|
return function() {
|
|
var args = arguments,
|
|
length = funcs.length;
|
|
|
|
while (length--) {
|
|
args = [funcs[length].apply(this, args)];
|
|
}
|
|
return args[0];
|
|
};
|
|
}
|
|
|
|
/**
|
|
* Produces a callback bound to an optional `thisArg`. If `func` is a property
|
|
* name the created callback will return the property value for a given element.
|
|
* If `func` is an object the created callback will return `true` for elements
|
|
* that contain the equivalent object properties, otherwise it will return `false`.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Functions
|
|
* @param {*} [func=identity] The value to convert to a callback.
|
|
* @param {*} [thisArg] The `this` binding of the created callback.
|
|
* @param {number} [argCount] The number of arguments the callback accepts.
|
|
* @returns {Function} Returns a callback function.
|
|
* @example
|
|
*
|
|
* var characters = [
|
|
* { 'name': 'barney', 'age': 36 },
|
|
* { 'name': 'fred', 'age': 40 }
|
|
* ];
|
|
*
|
|
* // wrap to create custom callback shorthands
|
|
* _.createCallback = _.wrap(_.createCallback, function(func, callback, thisArg) {
|
|
* var match = /^(.+?)__([gl]t)(.+)$/.exec(callback);
|
|
* return !match ? func(callback, thisArg) : function(object) {
|
|
* return match[2] == 'gt' ? object[match[1]] > match[3] : object[match[1]] < match[3];
|
|
* };
|
|
* });
|
|
*
|
|
* _.filter(characters, 'age__gt38');
|
|
* // => [{ 'name': 'fred', 'age': 40 }]
|
|
*/
|
|
function createCallback(func, thisArg, argCount) {
|
|
var type = typeof func;
|
|
if (func == null || type == 'function') {
|
|
return baseCreateCallback(func, thisArg, argCount);
|
|
}
|
|
// handle "_.pluck" style callback shorthands
|
|
if (type != 'object') {
|
|
return function(object) {
|
|
return object[func];
|
|
};
|
|
}
|
|
var props = keys(func),
|
|
key = props[0],
|
|
a = func[key];
|
|
|
|
// handle "_.where" style callback shorthands
|
|
if (props.length == 1 && a === a && !isObject(a)) {
|
|
// fast path the common case of providing an object with a single
|
|
// property containing a primitive value
|
|
return function(object) {
|
|
var b = object[key];
|
|
return a === b && (a !== 0 || (1 / a == 1 / b));
|
|
};
|
|
}
|
|
return function(object) {
|
|
var length = props.length,
|
|
result = false;
|
|
|
|
while (length--) {
|
|
if (!(result = baseIsEqual(object[props[length]], func[props[length]], null, true))) {
|
|
break;
|
|
}
|
|
}
|
|
return result;
|
|
};
|
|
}
|
|
|
|
/**
|
|
* Creates a function which accepts one or more arguments of `func` that when
|
|
* invoked either executes `func` returning its result, if all `func` arguments
|
|
* have been provided, or returns a function that accepts one or more of the
|
|
* remaining `func` arguments, and so on. The arity of `func` can be specified
|
|
* if `func.length` is not sufficient.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Functions
|
|
* @param {Function} func The function to curry.
|
|
* @param {number} [arity=func.length] The arity of `func`.
|
|
* @returns {Function} Returns the new curried function.
|
|
* @example
|
|
*
|
|
* var curried = _.curry(function(a, b, c) {
|
|
* console.log(a + b + c);
|
|
* });
|
|
*
|
|
* curried(1)(2)(3);
|
|
* // => 6
|
|
*
|
|
* curried(1, 2)(3);
|
|
* // => 6
|
|
*
|
|
* curried(1, 2, 3);
|
|
* // => 6
|
|
*/
|
|
function curry(func, arity) {
|
|
arity = typeof arity == 'number' ? arity : (+arity || func.length);
|
|
return createWrapper(func, 4, null, null, null, arity);
|
|
}
|
|
|
|
/**
|
|
* Creates a function that will delay the execution of `func` until after
|
|
* `wait` milliseconds have elapsed since the last time it was invoked.
|
|
* Provide an options object to indicate that `func` should be invoked on
|
|
* the leading and/or trailing edge of the `wait` timeout. Subsequent calls
|
|
* to the debounced function will return the result of the last `func` call.
|
|
*
|
|
* Note: If `leading` and `trailing` options are `true` `func` will be called
|
|
* on the trailing edge of the timeout only if the the debounced function is
|
|
* invoked more than once during the `wait` timeout.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Functions
|
|
* @param {Function} func The function to debounce.
|
|
* @param {number} wait The number of milliseconds to delay.
|
|
* @param {Object} [options] The options object.
|
|
* @param {boolean} [options.leading=false] Specify execution on the leading edge of the timeout.
|
|
* @param {number} [options.maxWait] The maximum time `func` is allowed to be delayed before it's called.
|
|
* @param {boolean} [options.trailing=true] Specify execution on the trailing edge of the timeout.
|
|
* @returns {Function} Returns the new debounced function.
|
|
* @example
|
|
*
|
|
* // avoid costly calculations while the window size is in flux
|
|
* var lazyLayout = _.debounce(calculateLayout, 150);
|
|
* jQuery(window).on('resize', lazyLayout);
|
|
*
|
|
* // execute `sendMail` when the click event is fired, debouncing subsequent calls
|
|
* jQuery('#postbox').on('click', _.debounce(sendMail, 300, {
|
|
* 'leading': true,
|
|
* 'trailing': false
|
|
* });
|
|
*
|
|
* // ensure `batchLog` is executed once after 1 second of debounced calls
|
|
* var source = new EventSource('/stream');
|
|
* source.addEventListener('message', _.debounce(batchLog, 250, {
|
|
* 'maxWait': 1000
|
|
* }, false);
|
|
*/
|
|
function debounce(func, wait, options) {
|
|
var args,
|
|
maxTimeoutId,
|
|
result,
|
|
stamp,
|
|
thisArg,
|
|
timeoutId,
|
|
trailingCall,
|
|
lastCalled = 0,
|
|
maxWait = false,
|
|
trailing = true;
|
|
|
|
if (!isFunction(func)) {
|
|
throw new TypeError;
|
|
}
|
|
wait = nativeMax(0, wait) || 0;
|
|
if (options === true) {
|
|
var leading = true;
|
|
trailing = false;
|
|
} else if (isObject(options)) {
|
|
leading = options.leading;
|
|
maxWait = 'maxWait' in options && (nativeMax(wait, options.maxWait) || 0);
|
|
trailing = 'trailing' in options ? options.trailing : trailing;
|
|
}
|
|
var delayed = function() {
|
|
var remaining = wait - (now() - stamp);
|
|
if (remaining <= 0) {
|
|
if (maxTimeoutId) {
|
|
clearTimeout(maxTimeoutId);
|
|
}
|
|
var isCalled = trailingCall;
|
|
maxTimeoutId = timeoutId = trailingCall = undefined;
|
|
if (isCalled) {
|
|
lastCalled = now();
|
|
result = func.apply(thisArg, args);
|
|
if (!timeoutId && !maxTimeoutId) {
|
|
args = thisArg = null;
|
|
}
|
|
}
|
|
} else {
|
|
timeoutId = setTimeout(delayed, remaining);
|
|
}
|
|
};
|
|
|
|
var maxDelayed = function() {
|
|
if (timeoutId) {
|
|
clearTimeout(timeoutId);
|
|
}
|
|
maxTimeoutId = timeoutId = trailingCall = undefined;
|
|
if (trailing || (maxWait !== wait)) {
|
|
lastCalled = now();
|
|
result = func.apply(thisArg, args);
|
|
if (!timeoutId && !maxTimeoutId) {
|
|
args = thisArg = null;
|
|
}
|
|
}
|
|
};
|
|
|
|
return function() {
|
|
args = arguments;
|
|
stamp = now();
|
|
thisArg = this;
|
|
trailingCall = trailing && (timeoutId || !leading);
|
|
|
|
if (maxWait === false) {
|
|
var leadingCall = leading && !timeoutId;
|
|
} else {
|
|
if (!maxTimeoutId && !leading) {
|
|
lastCalled = stamp;
|
|
}
|
|
var remaining = maxWait - (stamp - lastCalled),
|
|
isCalled = remaining <= 0;
|
|
|
|
if (isCalled) {
|
|
if (maxTimeoutId) {
|
|
maxTimeoutId = clearTimeout(maxTimeoutId);
|
|
}
|
|
lastCalled = stamp;
|
|
result = func.apply(thisArg, args);
|
|
}
|
|
else if (!maxTimeoutId) {
|
|
maxTimeoutId = setTimeout(maxDelayed, remaining);
|
|
}
|
|
}
|
|
if (isCalled && timeoutId) {
|
|
timeoutId = clearTimeout(timeoutId);
|
|
}
|
|
else if (!timeoutId && wait !== maxWait) {
|
|
timeoutId = setTimeout(delayed, wait);
|
|
}
|
|
if (leadingCall) {
|
|
isCalled = true;
|
|
result = func.apply(thisArg, args);
|
|
}
|
|
if (isCalled && !timeoutId && !maxTimeoutId) {
|
|
args = thisArg = null;
|
|
}
|
|
return result;
|
|
};
|
|
}
|
|
|
|
/**
|
|
* Defers executing the `func` function until the current call stack has cleared.
|
|
* Additional arguments will be provided to `func` when it is invoked.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Functions
|
|
* @param {Function} func The function to defer.
|
|
* @param {...*} [arg] Arguments to invoke the function with.
|
|
* @returns {number} Returns the timer id.
|
|
* @example
|
|
*
|
|
* _.defer(function() { console.log('deferred'); });
|
|
* // returns from the function before 'deferred' is logged
|
|
*/
|
|
function defer(func) {
|
|
if (!isFunction(func)) {
|
|
throw new TypeError;
|
|
}
|
|
var args = slice(arguments, 1);
|
|
return setTimeout(function() { func.apply(undefined, args); }, 1);
|
|
}
|
|
// use `setImmediate` if available in Node.js
|
|
if (setImmediate) {
|
|
defer = function(func) {
|
|
if (!isFunction(func)) {
|
|
throw new TypeError;
|
|
}
|
|
return setImmediate.apply(context, arguments);
|
|
};
|
|
}
|
|
|
|
/**
|
|
* Executes the `func` function after `wait` milliseconds. Additional arguments
|
|
* will be provided to `func` when it is invoked.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Functions
|
|
* @param {Function} func The function to delay.
|
|
* @param {number} wait The number of milliseconds to delay execution.
|
|
* @param {...*} [arg] Arguments to invoke the function with.
|
|
* @returns {number} Returns the timer id.
|
|
* @example
|
|
*
|
|
* var log = _.bind(console.log, console);
|
|
* _.delay(log, 1000, 'logged later');
|
|
* // => 'logged later' (Appears after one second.)
|
|
*/
|
|
function delay(func, wait) {
|
|
if (!isFunction(func)) {
|
|
throw new TypeError;
|
|
}
|
|
var args = slice(arguments, 2);
|
|
return setTimeout(function() { func.apply(undefined, args); }, wait);
|
|
}
|
|
|
|
/**
|
|
* Creates a function that memoizes the result of `func`. If `resolver` is
|
|
* provided it will be used to determine the cache key for storing the result
|
|
* based on the arguments provided to the memoized function. By default, the
|
|
* first argument provided to the memoized function is used as the cache key.
|
|
* The `func` is executed with the `this` binding of the memoized function.
|
|
* The result cache is exposed as the `cache` property on the memoized function.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Functions
|
|
* @param {Function} func The function to have its output memoized.
|
|
* @param {Function} [resolver] A function used to resolve the cache key.
|
|
* @returns {Function} Returns the new memoizing function.
|
|
* @example
|
|
*
|
|
* var fibonacci = _.memoize(function(n) {
|
|
* return n < 2 ? n : fibonacci(n - 1) + fibonacci(n - 2);
|
|
* });
|
|
*
|
|
* fibonacci(9)
|
|
* // => 34
|
|
*
|
|
* var data = {
|
|
* 'fred': { 'name': 'fred', 'age': 40 },
|
|
* 'pebbles': { 'name': 'pebbles', 'age': 1 }
|
|
* };
|
|
*
|
|
* // modifying the result cache
|
|
* var get = _.memoize(function(name) { return data[name]; }, _.identity);
|
|
* get('pebbles');
|
|
* // => { 'name': 'pebbles', 'age': 1 }
|
|
*
|
|
* get.cache.pebbles.name = 'penelope';
|
|
* get('pebbles');
|
|
* // => { 'name': 'penelope', 'age': 1 }
|
|
*/
|
|
function memoize(func, resolver) {
|
|
if (!isFunction(func)) {
|
|
throw new TypeError;
|
|
}
|
|
var memoized = function() {
|
|
var cache = memoized.cache,
|
|
key = resolver ? resolver.apply(this, arguments) : keyPrefix + arguments[0];
|
|
|
|
return hasOwnProperty.call(cache, key)
|
|
? cache[key]
|
|
: (cache[key] = func.apply(this, arguments));
|
|
}
|
|
memoized.cache = {};
|
|
return memoized;
|
|
}
|
|
|
|
/**
|
|
* Creates a function that is restricted to execute `func` once. Repeat calls to
|
|
* the function will return the value of the first call. The `func` is executed
|
|
* with the `this` binding of the created function.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Functions
|
|
* @param {Function} func The function to restrict.
|
|
* @returns {Function} Returns the new restricted function.
|
|
* @example
|
|
*
|
|
* var initialize = _.once(createApplication);
|
|
* initialize();
|
|
* initialize();
|
|
* // `initialize` executes `createApplication` once
|
|
*/
|
|
function once(func) {
|
|
var ran,
|
|
result;
|
|
|
|
if (!isFunction(func)) {
|
|
throw new TypeError;
|
|
}
|
|
return function() {
|
|
if (ran) {
|
|
return result;
|
|
}
|
|
ran = true;
|
|
result = func.apply(this, arguments);
|
|
|
|
// clear the `func` variable so the function may be garbage collected
|
|
func = null;
|
|
return result;
|
|
};
|
|
}
|
|
|
|
/**
|
|
* Creates a function that, when called, invokes `func` with any additional
|
|
* `partial` arguments prepended to those provided to the new function. This
|
|
* method is similar to `_.bind` except it does **not** alter the `this` binding.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Functions
|
|
* @param {Function} func The function to partially apply arguments to.
|
|
* @param {...*} [arg] Arguments to be partially applied.
|
|
* @returns {Function} Returns the new partially applied function.
|
|
* @example
|
|
*
|
|
* var greet = function(greeting, name) { return greeting + ' ' + name; };
|
|
* var hi = _.partial(greet, 'hi');
|
|
* hi('fred');
|
|
* // => 'hi fred'
|
|
*/
|
|
function partial(func) {
|
|
return createWrapper(func, 16, slice(arguments, 1));
|
|
}
|
|
|
|
/**
|
|
* This method is like `_.partial` except that `partial` arguments are
|
|
* appended to those provided to the new function.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Functions
|
|
* @param {Function} func The function to partially apply arguments to.
|
|
* @param {...*} [arg] Arguments to be partially applied.
|
|
* @returns {Function} Returns the new partially applied function.
|
|
* @example
|
|
*
|
|
* var defaultsDeep = _.partialRight(_.merge, _.defaults);
|
|
*
|
|
* var options = {
|
|
* 'variable': 'data',
|
|
* 'imports': { 'jq': $ }
|
|
* };
|
|
*
|
|
* defaultsDeep(options, _.templateSettings);
|
|
*
|
|
* options.variable
|
|
* // => 'data'
|
|
*
|
|
* options.imports
|
|
* // => { '_': _, 'jq': $ }
|
|
*/
|
|
function partialRight(func) {
|
|
return createWrapper(func, 32, null, slice(arguments, 1));
|
|
}
|
|
|
|
/**
|
|
* Creates a function that, when executed, will only call the `func` function
|
|
* at most once per every `wait` milliseconds. Provide an options object to
|
|
* indicate that `func` should be invoked on the leading and/or trailing edge
|
|
* of the `wait` timeout. Subsequent calls to the throttled function will
|
|
* return the result of the last `func` call.
|
|
*
|
|
* Note: If `leading` and `trailing` options are `true` `func` will be called
|
|
* on the trailing edge of the timeout only if the the throttled function is
|
|
* invoked more than once during the `wait` timeout.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Functions
|
|
* @param {Function} func The function to throttle.
|
|
* @param {number} wait The number of milliseconds to throttle executions to.
|
|
* @param {Object} [options] The options object.
|
|
* @param {boolean} [options.leading=true] Specify execution on the leading edge of the timeout.
|
|
* @param {boolean} [options.trailing=true] Specify execution on the trailing edge of the timeout.
|
|
* @returns {Function} Returns the new throttled function.
|
|
* @example
|
|
*
|
|
* // avoid excessively updating the position while scrolling
|
|
* var throttled = _.throttle(updatePosition, 100);
|
|
* jQuery(window).on('scroll', throttled);
|
|
*
|
|
* // execute `renewToken` when the click event is fired, but not more than once every 5 minutes
|
|
* jQuery('.interactive').on('click', _.throttle(renewToken, 300000, {
|
|
* 'trailing': false
|
|
* }));
|
|
*/
|
|
function throttle(func, wait, options) {
|
|
var leading = true,
|
|
trailing = true;
|
|
|
|
if (!isFunction(func)) {
|
|
throw new TypeError;
|
|
}
|
|
if (options === false) {
|
|
leading = false;
|
|
} else if (isObject(options)) {
|
|
leading = 'leading' in options ? options.leading : leading;
|
|
trailing = 'trailing' in options ? options.trailing : trailing;
|
|
}
|
|
debounceOptions.leading = leading;
|
|
debounceOptions.maxWait = wait;
|
|
debounceOptions.trailing = trailing;
|
|
|
|
return debounce(func, wait, debounceOptions);
|
|
}
|
|
|
|
/**
|
|
* Creates a function that provides `value` to the wrapper function as its
|
|
* first argument. Additional arguments provided to the function are appended
|
|
* to those provided to the wrapper function. The wrapper is executed with
|
|
* the `this` binding of the created function.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Functions
|
|
* @param {*} value The value to wrap.
|
|
* @param {Function} wrapper The wrapper function.
|
|
* @returns {Function} Returns the new function.
|
|
* @example
|
|
*
|
|
* var p = _.wrap(_.escape, function(func, text) {
|
|
* return '<p>' + func(text) + '</p>';
|
|
* });
|
|
*
|
|
* p('Fred, Wilma, & Pebbles');
|
|
* // => '<p>Fred, Wilma, & Pebbles</p>'
|
|
*/
|
|
function wrap(value, wrapper) {
|
|
return createWrapper(wrapper, 16, [value]);
|
|
}
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
|
|
/**
|
|
* Converts the characters `&`, `<`, `>`, `"`, and `'` in `string` to their
|
|
* corresponding HTML entities.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Utilities
|
|
* @param {string} string The string to escape.
|
|
* @returns {string} Returns the escaped string.
|
|
* @example
|
|
*
|
|
* _.escape('Fred, Wilma, & Pebbles');
|
|
* // => 'Fred, Wilma, & Pebbles'
|
|
*/
|
|
function escape(string) {
|
|
return string == null ? '' : String(string).replace(reUnescapedHtml, escapeHtmlChar);
|
|
}
|
|
|
|
/**
|
|
* This method returns the first argument provided to it.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Utilities
|
|
* @param {*} value Any value.
|
|
* @returns {*} Returns `value`.
|
|
* @example
|
|
*
|
|
* var object = { 'name': 'fred' };
|
|
* _.identity(object) === object;
|
|
* // => true
|
|
*/
|
|
function identity(value) {
|
|
return value;
|
|
}
|
|
|
|
/**
|
|
* Adds function properties of a source object to the `lodash` function and
|
|
* chainable wrapper.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Utilities
|
|
* @param {Object} object The object of function properties to add to `lodash`.
|
|
* @param {Object} object The object of function properties to add to `lodash`.
|
|
* @example
|
|
*
|
|
* _.mixin({
|
|
* 'capitalize': function(string) {
|
|
* return string.charAt(0).toUpperCase() + string.slice(1).toLowerCase();
|
|
* }
|
|
* });
|
|
*
|
|
* _.capitalize('fred');
|
|
* // => 'Fred'
|
|
*
|
|
* _('fred').capitalize();
|
|
* // => 'Fred'
|
|
*/
|
|
function mixin(object, source) {
|
|
var ctor = object,
|
|
isFunc = !source || isFunction(ctor);
|
|
|
|
if (!source) {
|
|
ctor = lodashWrapper;
|
|
source = object;
|
|
object = lodash;
|
|
}
|
|
forEach(functions(source), function(methodName) {
|
|
var func = object[methodName] = source[methodName];
|
|
if (isFunc) {
|
|
ctor.prototype[methodName] = function() {
|
|
var value = this.__wrapped__,
|
|
args = [value];
|
|
|
|
push.apply(args, arguments);
|
|
var result = func.apply(object, args);
|
|
if (value && typeof value == 'object' && value === result) {
|
|
return this;
|
|
}
|
|
result = new ctor(result);
|
|
result.__chain__ = this.__chain__;
|
|
return result;
|
|
};
|
|
}
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Reverts the '_' variable to its previous value and returns a reference to
|
|
* the `lodash` function.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Utilities
|
|
* @returns {Function} Returns the `lodash` function.
|
|
* @example
|
|
*
|
|
* var lodash = _.noConflict();
|
|
*/
|
|
function noConflict() {
|
|
context._ = oldDash;
|
|
return this;
|
|
}
|
|
|
|
/**
|
|
* A no-operation function.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Utilities
|
|
* @example
|
|
*
|
|
* var object = { 'name': 'fred' };
|
|
* _.noop(object) === undefined;
|
|
* // => true
|
|
*/
|
|
function noop() {
|
|
// no operation performed
|
|
}
|
|
|
|
/**
|
|
* Converts the given value into an integer of the specified radix.
|
|
* If `radix` is `undefined` or `0` a `radix` of `10` is used unless the
|
|
* `value` is a hexadecimal, in which case a `radix` of `16` is used.
|
|
*
|
|
* Note: This method avoids differences in native ES3 and ES5 `parseInt`
|
|
* implementations. See http://es5.github.io/#E.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Utilities
|
|
* @param {string} value The value to parse.
|
|
* @param {number} [radix] The radix used to interpret the value to parse.
|
|
* @returns {number} Returns the new integer value.
|
|
* @example
|
|
*
|
|
* _.parseInt('08');
|
|
* // => 8
|
|
*/
|
|
var parseInt = nativeParseInt(whitespace + '08') == 8 ? nativeParseInt : function(value, radix) {
|
|
// Firefox < 21 and Opera < 15 follow the ES3 specified implementation of `parseInt`
|
|
return nativeParseInt(isString(value) ? value.replace(reLeadingSpacesAndZeros, '') : value, radix || 0);
|
|
};
|
|
|
|
/**
|
|
* Produces a random number between `min` and `max` (inclusive). If only one
|
|
* argument is provided a number between `0` and the given number will be
|
|
* returned. If `floating` is truey or either `min` or `max` are floats a
|
|
* floating-point number will be returned instead of an integer.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Utilities
|
|
* @param {number} [min=0] The minimum possible value.
|
|
* @param {number} [max=1] The maximum possible value.
|
|
* @param {boolean} [floating=false] Specify returning a floating-point number.
|
|
* @returns {number} Returns a random number.
|
|
* @example
|
|
*
|
|
* _.random(0, 5);
|
|
* // => an integer between 0 and 5
|
|
*
|
|
* _.random(5);
|
|
* // => also an integer between 0 and 5
|
|
*
|
|
* _.random(5, true);
|
|
* // => a floating-point number between 0 and 5
|
|
*
|
|
* _.random(1.2, 5.2);
|
|
* // => a floating-point number between 1.2 and 5.2
|
|
*/
|
|
function random(min, max, floating) {
|
|
var noMin = min == null,
|
|
noMax = max == null;
|
|
|
|
if (floating == null) {
|
|
if (typeof min == 'boolean' && noMax) {
|
|
floating = min;
|
|
min = 1;
|
|
}
|
|
else if (!noMax && typeof max == 'boolean') {
|
|
floating = max;
|
|
noMax = true;
|
|
}
|
|
}
|
|
if (noMin && noMax) {
|
|
max = 1;
|
|
}
|
|
min = +min || 0;
|
|
if (noMax) {
|
|
max = min;
|
|
min = 0;
|
|
} else {
|
|
max = +max || 0;
|
|
}
|
|
if (floating || min % 1 || max % 1) {
|
|
var rand = nativeRandom();
|
|
return nativeMin(min + (rand * (max - min + parseFloat('1e-' + ((rand +'').length - 1)))), max);
|
|
}
|
|
return baseRandom(min, max);
|
|
}
|
|
|
|
/**
|
|
* Resolves the value of `property` on `object`. If `property` is a function
|
|
* it will be invoked with the `this` binding of `object` and its result returned,
|
|
* else the property value is returned. If `object` is falsey then `undefined`
|
|
* is returned.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Utilities
|
|
* @param {Object} object The object to inspect.
|
|
* @param {string} property The property to get the value of.
|
|
* @returns {*} Returns the resolved value.
|
|
* @example
|
|
*
|
|
* var object = {
|
|
* 'cheese': 'crumpets',
|
|
* 'stuff': function() {
|
|
* return 'nonsense';
|
|
* }
|
|
* };
|
|
*
|
|
* _.result(object, 'cheese');
|
|
* // => 'crumpets'
|
|
*
|
|
* _.result(object, 'stuff');
|
|
* // => 'nonsense'
|
|
*/
|
|
function result(object, property) {
|
|
if (object) {
|
|
var value = object[property];
|
|
return isFunction(value) ? object[property]() : value;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* A micro-templating method that handles arbitrary delimiters, preserves
|
|
* whitespace, and correctly escapes quotes within interpolated code.
|
|
*
|
|
* Note: In the development build, `_.template` utilizes sourceURLs for easier
|
|
* debugging. See http://www.html5rocks.com/en/tutorials/developertools/sourcemaps/#toc-sourceurl
|
|
*
|
|
* For more information on precompiling templates see:
|
|
* http://lodash.com/custom-builds
|
|
*
|
|
* For more information on Chrome extension sandboxes see:
|
|
* http://developer.chrome.com/stable/extensions/sandboxingEval.html
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Utilities
|
|
* @param {string} text The template text.
|
|
* @param {Object} data The data object used to populate the text.
|
|
* @param {Object} [options] The options object.
|
|
* @param {RegExp} [options.escape] The "escape" delimiter.
|
|
* @param {RegExp} [options.evaluate] The "evaluate" delimiter.
|
|
* @param {Object} [options.imports] An object to import into the template as local variables.
|
|
* @param {RegExp} [options.interpolate] The "interpolate" delimiter.
|
|
* @param {string} [sourceURL] The sourceURL of the template's compiled source.
|
|
* @param {string} [variable] The data object variable name.
|
|
* @returns {Function|string} Returns a compiled function when no `data` object
|
|
* is given, else it returns the interpolated text.
|
|
* @example
|
|
*
|
|
* // using the "interpolate" delimiter to create a compiled template
|
|
* var compiled = _.template('hello <%= name %>');
|
|
* compiled({ 'name': 'fred' });
|
|
* // => 'hello fred'
|
|
*
|
|
* // using the "escape" delimiter to escape HTML in data property values
|
|
* _.template('<b><%- value %></b>', { 'value': '<script>' });
|
|
* // => '<b><script></b>'
|
|
*
|
|
* // using the "evaluate" delimiter to generate HTML
|
|
* var list = '<% _.forEach(people, function(name) { %><li><%- name %></li><% }); %>';
|
|
* _.template(list, { 'people': ['fred', 'barney'] });
|
|
* // => '<li>fred</li><li>barney</li>'
|
|
*
|
|
* // using the ES6 delimiter as an alternative to the default "interpolate" delimiter
|
|
* _.template('hello ${ name }', { 'name': 'pebbles' });
|
|
* // => 'hello pebbles'
|
|
*
|
|
* // using the internal `print` function in "evaluate" delimiters
|
|
* _.template('<% print("hello " + name); %>!', { 'name': 'barney' });
|
|
* // => 'hello barney!'
|
|
*
|
|
* // using a custom template delimiters
|
|
* _.templateSettings = {
|
|
* 'interpolate': /{{([\s\S]+?)}}/g
|
|
* };
|
|
*
|
|
* _.template('hello {{ name }}!', { 'name': 'mustache' });
|
|
* // => 'hello mustache!'
|
|
*
|
|
* // using the `imports` option to import jQuery
|
|
* var list = '<% $.each(people, function(name) { %><li><%- name %></li><% }); %>';
|
|
* _.template(list, { 'people': ['fred', 'barney'] }, { 'imports': { '$': jQuery } });
|
|
* // => '<li>fred</li><li>barney</li>'
|
|
*
|
|
* // using the `sourceURL` option to specify a custom sourceURL for the template
|
|
* var compiled = _.template('hello <%= name %>', null, { 'sourceURL': '/basic/greeting.jst' });
|
|
* compiled(data);
|
|
* // => find the source of "greeting.jst" under the Sources tab or Resources panel of the web inspector
|
|
*
|
|
* // using the `variable` option to ensure a with-statement isn't used in the compiled template
|
|
* var compiled = _.template('hi <%= data.name %>!', null, { 'variable': 'data' });
|
|
* compiled.source;
|
|
* // => function(data) {
|
|
* var __t, __p = '', __e = _.escape;
|
|
* __p += 'hi ' + ((__t = ( data.name )) == null ? '' : __t) + '!';
|
|
* return __p;
|
|
* }
|
|
*
|
|
* // using the `source` property to inline compiled templates for meaningful
|
|
* // line numbers in error messages and a stack trace
|
|
* fs.writeFileSync(path.join(cwd, 'jst.js'), '\
|
|
* var JST = {\
|
|
* "main": ' + _.template(mainText).source + '\
|
|
* };\
|
|
* ');
|
|
*/
|
|
function template(text, data, options) {
|
|
// based on John Resig's `tmpl` implementation
|
|
// http://ejohn.org/blog/javascript-micro-templating/
|
|
// and Laura Doktorova's doT.js
|
|
// https://github.com/olado/doT
|
|
var settings = lodash.templateSettings;
|
|
text = String(text || '');
|
|
|
|
// avoid missing dependencies when `iteratorTemplate` is not defined
|
|
options = defaults({}, options, settings);
|
|
|
|
var imports = defaults({}, options.imports, settings.imports),
|
|
importsKeys = keys(imports),
|
|
importsValues = values(imports);
|
|
|
|
var isEvaluating,
|
|
index = 0,
|
|
interpolate = options.interpolate || reNoMatch,
|
|
source = "__p += '";
|
|
|
|
// compile the regexp to match each delimiter
|
|
var reDelimiters = RegExp(
|
|
(options.escape || reNoMatch).source + '|' +
|
|
interpolate.source + '|' +
|
|
(interpolate === reInterpolate ? reEsTemplate : reNoMatch).source + '|' +
|
|
(options.evaluate || reNoMatch).source + '|$'
|
|
, 'g');
|
|
|
|
text.replace(reDelimiters, function(match, escapeValue, interpolateValue, esTemplateValue, evaluateValue, offset) {
|
|
interpolateValue || (interpolateValue = esTemplateValue);
|
|
|
|
// escape characters that cannot be included in string literals
|
|
source += text.slice(index, offset).replace(reUnescapedString, escapeStringChar);
|
|
|
|
// replace delimiters with snippets
|
|
if (escapeValue) {
|
|
source += "' +\n__e(" + escapeValue + ") +\n'";
|
|
}
|
|
if (evaluateValue) {
|
|
isEvaluating = true;
|
|
source += "';\n" + evaluateValue + ";\n__p += '";
|
|
}
|
|
if (interpolateValue) {
|
|
source += "' +\n((__t = (" + interpolateValue + ")) == null ? '' : __t) +\n'";
|
|
}
|
|
index = offset + match.length;
|
|
|
|
// the JS engine embedded in Adobe products requires returning the `match`
|
|
// string in order to produce the correct `offset` value
|
|
return match;
|
|
});
|
|
|
|
source += "';\n";
|
|
|
|
// if `variable` is not specified, wrap a with-statement around the generated
|
|
// code to add the data object to the top of the scope chain
|
|
var variable = options.variable,
|
|
hasVariable = variable;
|
|
|
|
if (!hasVariable) {
|
|
variable = 'obj';
|
|
source = 'with (' + variable + ') {\n' + source + '\n}\n';
|
|
}
|
|
// cleanup code by stripping empty strings
|
|
source = (isEvaluating ? source.replace(reEmptyStringLeading, '') : source)
|
|
.replace(reEmptyStringMiddle, '$1')
|
|
.replace(reEmptyStringTrailing, '$1;');
|
|
|
|
// frame code as the function body
|
|
source = 'function(' + variable + ') {\n' +
|
|
(hasVariable ? '' : variable + ' || (' + variable + ' = {});\n') +
|
|
"var __t, __p = '', __e = _.escape" +
|
|
(isEvaluating
|
|
? ', __j = Array.prototype.join;\n' +
|
|
"function print() { __p += __j.call(arguments, '') }\n"
|
|
: ';\n'
|
|
) +
|
|
source +
|
|
'return __p\n}';
|
|
|
|
// Use a sourceURL for easier debugging.
|
|
// http://www.html5rocks.com/en/tutorials/developertools/sourcemaps/#toc-sourceurl
|
|
var sourceURL = '\n/*\n//# sourceURL=' + (options.sourceURL || '/lodash/template/source[' + (templateCounter++) + ']') + '\n*/';
|
|
|
|
try {
|
|
var result = Function(importsKeys, 'return ' + source + sourceURL).apply(undefined, importsValues);
|
|
} catch(e) {
|
|
e.source = source;
|
|
throw e;
|
|
}
|
|
if (data) {
|
|
return result(data);
|
|
}
|
|
// provide the compiled function's source by its `toString` method, in
|
|
// supported environments, or the `source` property as a convenience for
|
|
// inlining compiled templates during the build process
|
|
result.source = source;
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* Executes the callback `n` times, returning an array of the results
|
|
* of each callback execution. The callback is bound to `thisArg` and invoked
|
|
* with one argument; (index).
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Utilities
|
|
* @param {number} n The number of times to execute the callback.
|
|
* @param {Function} callback The function called per iteration.
|
|
* @param {*} [thisArg] The `this` binding of `callback`.
|
|
* @returns {Array} Returns an array of the results of each `callback` execution.
|
|
* @example
|
|
*
|
|
* var diceRolls = _.times(3, _.partial(_.random, 1, 6));
|
|
* // => [3, 6, 4]
|
|
*
|
|
* _.times(3, function(n) { mage.castSpell(n); });
|
|
* // => calls `mage.castSpell(n)` three times, passing `n` of `0`, `1`, and `2` respectively
|
|
*
|
|
* _.times(3, function(n) { this.cast(n); }, mage);
|
|
* // => also calls `mage.castSpell(n)` three times
|
|
*/
|
|
function times(n, callback, thisArg) {
|
|
n = (n = +n) > -1 ? n : 0;
|
|
var index = -1,
|
|
result = Array(n);
|
|
|
|
callback = baseCreateCallback(callback, thisArg, 1);
|
|
while (++index < n) {
|
|
result[index] = callback(index);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* The inverse of `_.escape` this method converts the HTML entities
|
|
* `&`, `<`, `>`, `"`, and `'` in `string` to their
|
|
* corresponding characters.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Utilities
|
|
* @param {string} string The string to unescape.
|
|
* @returns {string} Returns the unescaped string.
|
|
* @example
|
|
*
|
|
* _.unescape('Fred, Barney & Pebbles');
|
|
* // => 'Fred, Barney & Pebbles'
|
|
*/
|
|
function unescape(string) {
|
|
return string == null ? '' : String(string).replace(reEscapedHtml, unescapeHtmlChar);
|
|
}
|
|
|
|
/**
|
|
* Generates a unique ID. If `prefix` is provided the ID will be appended to it.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Utilities
|
|
* @param {string} [prefix] The value to prefix the ID with.
|
|
* @returns {string} Returns the unique ID.
|
|
* @example
|
|
*
|
|
* _.uniqueId('contact_');
|
|
* // => 'contact_104'
|
|
*
|
|
* _.uniqueId();
|
|
* // => '105'
|
|
*/
|
|
function uniqueId(prefix) {
|
|
var id = ++idCounter;
|
|
return String(prefix == null ? '' : prefix) + id;
|
|
}
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
|
|
/**
|
|
* Creates a `lodash` object that wraps the given value with explicit
|
|
* method chaining enabled.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Chaining
|
|
* @param {*} value The value to wrap.
|
|
* @returns {Object} Returns the wrapper object.
|
|
* @example
|
|
*
|
|
* var characters = [
|
|
* { 'name': 'barney', 'age': 36 },
|
|
* { 'name': 'fred', 'age': 40 },
|
|
* { 'name': 'pebbles', 'age': 1 }
|
|
* ];
|
|
*
|
|
* var youngest = _.chain(characters)
|
|
* .sortBy('age')
|
|
* .map(function(chr) { return chr.name + ' is ' + chr.age; })
|
|
* .first()
|
|
* .value();
|
|
* // => 'pebbles is 1'
|
|
*/
|
|
function chain(value) {
|
|
value = new lodashWrapper(value);
|
|
value.__chain__ = true;
|
|
return value;
|
|
}
|
|
|
|
/**
|
|
* Invokes `interceptor` with the `value` as the first argument and then
|
|
* returns `value`. The purpose of this method is to "tap into" a method
|
|
* chain in order to perform operations on intermediate results within
|
|
* the chain.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @category Chaining
|
|
* @param {*} value The value to provide to `interceptor`.
|
|
* @param {Function} interceptor The function to invoke.
|
|
* @returns {*} Returns `value`.
|
|
* @example
|
|
*
|
|
* _([1, 2, 3, 4])
|
|
* .tap(function(array) { array.pop(); })
|
|
* .reverse()
|
|
* .value();
|
|
* // => [3, 2, 1]
|
|
*/
|
|
function tap(value, interceptor) {
|
|
interceptor(value);
|
|
return value;
|
|
}
|
|
|
|
/**
|
|
* Enables explicit method chaining on the wrapper object.
|
|
*
|
|
* @name chain
|
|
* @memberOf _
|
|
* @category Chaining
|
|
* @returns {*} Returns the wrapper object.
|
|
* @example
|
|
*
|
|
* var characters = [
|
|
* { 'name': 'barney', 'age': 36 },
|
|
* { 'name': 'fred', 'age': 40 }
|
|
* ];
|
|
*
|
|
* // without explicit chaining
|
|
* _(characters).first();
|
|
* // => { 'name': 'barney', 'age': 36 }
|
|
*
|
|
* // with explicit chaining
|
|
* _(characters).chain()
|
|
* .first()
|
|
* .pick('age')
|
|
* .value()
|
|
* // => { 'age': 36 }
|
|
*/
|
|
function wrapperChain() {
|
|
this.__chain__ = true;
|
|
return this;
|
|
}
|
|
|
|
/**
|
|
* Produces the `toString` result of the wrapped value.
|
|
*
|
|
* @name toString
|
|
* @memberOf _
|
|
* @category Chaining
|
|
* @returns {string} Returns the string result.
|
|
* @example
|
|
*
|
|
* _([1, 2, 3]).toString();
|
|
* // => '1,2,3'
|
|
*/
|
|
function wrapperToString() {
|
|
return String(this.__wrapped__);
|
|
}
|
|
|
|
/**
|
|
* Extracts the wrapped value.
|
|
*
|
|
* @name valueOf
|
|
* @memberOf _
|
|
* @alias value
|
|
* @category Chaining
|
|
* @returns {*} Returns the wrapped value.
|
|
* @example
|
|
*
|
|
* _([1, 2, 3]).valueOf();
|
|
* // => [1, 2, 3]
|
|
*/
|
|
function wrapperValueOf() {
|
|
return this.__wrapped__;
|
|
}
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
|
|
// add functions that return wrapped values when chaining
|
|
lodash.after = after;
|
|
lodash.assign = assign;
|
|
lodash.at = at;
|
|
lodash.bind = bind;
|
|
lodash.bindAll = bindAll;
|
|
lodash.bindKey = bindKey;
|
|
lodash.chain = chain;
|
|
lodash.compact = compact;
|
|
lodash.compose = compose;
|
|
lodash.countBy = countBy;
|
|
lodash.create = create;
|
|
lodash.createCallback = createCallback;
|
|
lodash.curry = curry;
|
|
lodash.debounce = debounce;
|
|
lodash.defaults = defaults;
|
|
lodash.defer = defer;
|
|
lodash.delay = delay;
|
|
lodash.difference = difference;
|
|
lodash.filter = filter;
|
|
lodash.flatten = flatten;
|
|
lodash.forEach = forEach;
|
|
lodash.forEachRight = forEachRight;
|
|
lodash.forIn = forIn;
|
|
lodash.forInRight = forInRight;
|
|
lodash.forOwn = forOwn;
|
|
lodash.forOwnRight = forOwnRight;
|
|
lodash.functions = functions;
|
|
lodash.groupBy = groupBy;
|
|
lodash.indexBy = indexBy;
|
|
lodash.initial = initial;
|
|
lodash.intersection = intersection;
|
|
lodash.invert = invert;
|
|
lodash.invoke = invoke;
|
|
lodash.keys = keys;
|
|
lodash.map = map;
|
|
lodash.max = max;
|
|
lodash.memoize = memoize;
|
|
lodash.merge = merge;
|
|
lodash.min = min;
|
|
lodash.omit = omit;
|
|
lodash.once = once;
|
|
lodash.pairs = pairs;
|
|
lodash.partial = partial;
|
|
lodash.partialRight = partialRight;
|
|
lodash.pick = pick;
|
|
lodash.pluck = pluck;
|
|
lodash.pull = pull;
|
|
lodash.range = range;
|
|
lodash.reject = reject;
|
|
lodash.remove = remove;
|
|
lodash.rest = rest;
|
|
lodash.shuffle = shuffle;
|
|
lodash.sortBy = sortBy;
|
|
lodash.tap = tap;
|
|
lodash.throttle = throttle;
|
|
lodash.times = times;
|
|
lodash.toArray = toArray;
|
|
lodash.transform = transform;
|
|
lodash.union = union;
|
|
lodash.uniq = uniq;
|
|
lodash.values = values;
|
|
lodash.where = where;
|
|
lodash.without = without;
|
|
lodash.wrap = wrap;
|
|
lodash.zip = zip;
|
|
lodash.zipObject = zipObject;
|
|
|
|
// add aliases
|
|
lodash.collect = map;
|
|
lodash.drop = rest;
|
|
lodash.each = forEach;
|
|
lodash.eachRight = forEachRight;
|
|
lodash.extend = assign;
|
|
lodash.methods = functions;
|
|
lodash.object = zipObject;
|
|
lodash.select = filter;
|
|
lodash.tail = rest;
|
|
lodash.unique = uniq;
|
|
lodash.unzip = zip;
|
|
|
|
// add functions to `lodash.prototype`
|
|
mixin(lodash);
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
|
|
// add functions that return unwrapped values when chaining
|
|
lodash.clone = clone;
|
|
lodash.cloneDeep = cloneDeep;
|
|
lodash.contains = contains;
|
|
lodash.escape = escape;
|
|
lodash.every = every;
|
|
lodash.find = find;
|
|
lodash.findIndex = findIndex;
|
|
lodash.findKey = findKey;
|
|
lodash.findLast = findLast;
|
|
lodash.findLastIndex = findLastIndex;
|
|
lodash.findLastKey = findLastKey;
|
|
lodash.has = has;
|
|
lodash.identity = identity;
|
|
lodash.indexOf = indexOf;
|
|
lodash.isArguments = isArguments;
|
|
lodash.isArray = isArray;
|
|
lodash.isBoolean = isBoolean;
|
|
lodash.isDate = isDate;
|
|
lodash.isElement = isElement;
|
|
lodash.isEmpty = isEmpty;
|
|
lodash.isEqual = isEqual;
|
|
lodash.isFinite = isFinite;
|
|
lodash.isFunction = isFunction;
|
|
lodash.isNaN = isNaN;
|
|
lodash.isNull = isNull;
|
|
lodash.isNumber = isNumber;
|
|
lodash.isObject = isObject;
|
|
lodash.isPlainObject = isPlainObject;
|
|
lodash.isRegExp = isRegExp;
|
|
lodash.isString = isString;
|
|
lodash.isUndefined = isUndefined;
|
|
lodash.lastIndexOf = lastIndexOf;
|
|
lodash.mixin = mixin;
|
|
lodash.noConflict = noConflict;
|
|
lodash.noop = noop;
|
|
lodash.parseInt = parseInt;
|
|
lodash.random = random;
|
|
lodash.reduce = reduce;
|
|
lodash.reduceRight = reduceRight;
|
|
lodash.result = result;
|
|
lodash.runInContext = runInContext;
|
|
lodash.size = size;
|
|
lodash.some = some;
|
|
lodash.sortedIndex = sortedIndex;
|
|
lodash.template = template;
|
|
lodash.unescape = unescape;
|
|
lodash.uniqueId = uniqueId;
|
|
|
|
// add aliases
|
|
lodash.all = every;
|
|
lodash.any = some;
|
|
lodash.detect = find;
|
|
lodash.findWhere = find;
|
|
lodash.foldl = reduce;
|
|
lodash.foldr = reduceRight;
|
|
lodash.include = contains;
|
|
lodash.inject = reduce;
|
|
|
|
forOwn(lodash, function(func, methodName) {
|
|
if (!lodash.prototype[methodName]) {
|
|
lodash.prototype[methodName] = function() {
|
|
var args = [this.__wrapped__],
|
|
chainAll = this.__chain__;
|
|
|
|
push.apply(args, arguments);
|
|
var result = func.apply(lodash, args);
|
|
return chainAll
|
|
? new lodashWrapper(result, chainAll)
|
|
: result;
|
|
};
|
|
}
|
|
});
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
|
|
// add functions capable of returning wrapped and unwrapped values when chaining
|
|
lodash.first = first;
|
|
lodash.last = last;
|
|
lodash.sample = sample;
|
|
|
|
// add aliases
|
|
lodash.take = first;
|
|
lodash.head = first;
|
|
|
|
forOwn(lodash, function(func, methodName) {
|
|
var callbackable = methodName !== 'sample';
|
|
if (!lodash.prototype[methodName]) {
|
|
lodash.prototype[methodName]= function(n, guard) {
|
|
var chainAll = this.__chain__,
|
|
result = func(this.__wrapped__, n, guard);
|
|
|
|
return !chainAll && (n == null || (guard && !(callbackable && typeof n == 'function')))
|
|
? result
|
|
: new lodashWrapper(result, chainAll);
|
|
};
|
|
}
|
|
});
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
|
|
/**
|
|
* The semantic version number.
|
|
*
|
|
* @static
|
|
* @memberOf _
|
|
* @type string
|
|
*/
|
|
lodash.VERSION = '2.3.0';
|
|
|
|
// add "Chaining" functions to the wrapper
|
|
lodash.prototype.chain = wrapperChain;
|
|
lodash.prototype.toString = wrapperToString;
|
|
lodash.prototype.value = wrapperValueOf;
|
|
lodash.prototype.valueOf = wrapperValueOf;
|
|
|
|
// add `Array` functions that return unwrapped values
|
|
forEach(['join', 'pop', 'shift'], function(methodName) {
|
|
var func = arrayRef[methodName];
|
|
lodash.prototype[methodName] = function() {
|
|
var chainAll = this.__chain__,
|
|
result = func.apply(this.__wrapped__, arguments);
|
|
|
|
return chainAll
|
|
? new lodashWrapper(result, chainAll)
|
|
: result;
|
|
};
|
|
});
|
|
|
|
// add `Array` functions that return the wrapped value
|
|
forEach(['push', 'reverse', 'sort', 'unshift'], function(methodName) {
|
|
var func = arrayRef[methodName];
|
|
lodash.prototype[methodName] = function() {
|
|
func.apply(this.__wrapped__, arguments);
|
|
return this;
|
|
};
|
|
});
|
|
|
|
// add `Array` functions that return new wrapped values
|
|
forEach(['concat', 'slice', 'splice'], function(methodName) {
|
|
var func = arrayRef[methodName];
|
|
lodash.prototype[methodName] = function() {
|
|
return new lodashWrapper(func.apply(this.__wrapped__, arguments), this.__chain__);
|
|
};
|
|
});
|
|
|
|
return lodash;
|
|
}
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
|
|
// expose Lo-Dash
|
|
var _ = runInContext();
|
|
|
|
// some AMD build optimizers like r.js check for condition patterns like the following:
|
|
if (typeof define == 'function' && typeof define.amd == 'object' && define.amd) {
|
|
// Expose Lo-Dash to the global object even when an AMD loader is present in
|
|
// case Lo-Dash was injected by a third-party script and not intended to be
|
|
// loaded as a module. The global assignment can be reverted in the Lo-Dash
|
|
// module by its `noConflict()` method.
|
|
root._ = _;
|
|
|
|
// define as an anonymous module so, through path mapping, it can be
|
|
// referenced as the "underscore" module
|
|
define(function() {
|
|
return _;
|
|
});
|
|
}
|
|
// check for `exports` after `define` in case a build optimizer adds an `exports` object
|
|
else if (freeExports && freeModule) {
|
|
// in Node.js or RingoJS
|
|
if (moduleExports) {
|
|
(freeModule.exports = _)._ = _;
|
|
}
|
|
// in Narwhal or Rhino -require
|
|
else {
|
|
freeExports._ = _;
|
|
}
|
|
}
|
|
else {
|
|
// in a browser or Rhino
|
|
root._ = _;
|
|
}
|
|
}.call(this));
|
|
|
|
},{}],51:[function(require,module,exports){
|
|
var process=require("__browserify_process");;(function (require, exports, module, platform) {
|
|
|
|
if (module) module.exports = minimatch
|
|
else exports.minimatch = minimatch
|
|
|
|
if (!require) {
|
|
require = function (id) {
|
|
switch (id) {
|
|
case "sigmund": return function sigmund (obj) {
|
|
return JSON.stringify(obj)
|
|
}
|
|
case "path": return { basename: function (f) {
|
|
f = f.split(/[\/\\]/)
|
|
var e = f.pop()
|
|
if (!e) e = f.pop()
|
|
return e
|
|
}}
|
|
case "lru-cache": return function LRUCache () {
|
|
// not quite an LRU, but still space-limited.
|
|
var cache = {}
|
|
var cnt = 0
|
|
this.set = function (k, v) {
|
|
cnt ++
|
|
if (cnt >= 100) cache = {}
|
|
cache[k] = v
|
|
}
|
|
this.get = function (k) { return cache[k] }
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
minimatch.Minimatch = Minimatch
|
|
|
|
var LRU = require("lru-cache")
|
|
, cache = minimatch.cache = new LRU({max: 100})
|
|
, GLOBSTAR = minimatch.GLOBSTAR = Minimatch.GLOBSTAR = {}
|
|
, sigmund = require("sigmund")
|
|
|
|
var path = require("path")
|
|
// any single thing other than /
|
|
// don't need to escape / when using new RegExp()
|
|
, qmark = "[^/]"
|
|
|
|
// * => any number of characters
|
|
, star = qmark + "*?"
|
|
|
|
// ** when dots are allowed. Anything goes, except .. and .
|
|
// not (^ or / followed by one or two dots followed by $ or /),
|
|
// followed by anything, any number of times.
|
|
, twoStarDot = "(?:(?!(?:\\\/|^)(?:\\.{1,2})($|\\\/)).)*?"
|
|
|
|
// not a ^ or / followed by a dot,
|
|
// followed by anything, any number of times.
|
|
, twoStarNoDot = "(?:(?!(?:\\\/|^)\\.).)*?"
|
|
|
|
// characters that need to be escaped in RegExp.
|
|
, reSpecials = charSet("().*{}+?[]^$\\!")
|
|
|
|
// "abc" -> { a:true, b:true, c:true }
|
|
function charSet (s) {
|
|
return s.split("").reduce(function (set, c) {
|
|
set[c] = true
|
|
return set
|
|
}, {})
|
|
}
|
|
|
|
// normalizes slashes.
|
|
var slashSplit = /\/+/
|
|
|
|
minimatch.monkeyPatch = monkeyPatch
|
|
function monkeyPatch () {
|
|
var desc = Object.getOwnPropertyDescriptor(String.prototype, "match")
|
|
var orig = desc.value
|
|
desc.value = function (p) {
|
|
if (p instanceof Minimatch) return p.match(this)
|
|
return orig.call(this, p)
|
|
}
|
|
Object.defineProperty(String.prototype, desc)
|
|
}
|
|
|
|
minimatch.filter = filter
|
|
function filter (pattern, options) {
|
|
options = options || {}
|
|
return function (p, i, list) {
|
|
return minimatch(p, pattern, options)
|
|
}
|
|
}
|
|
|
|
function ext (a, b) {
|
|
a = a || {}
|
|
b = b || {}
|
|
var t = {}
|
|
Object.keys(b).forEach(function (k) {
|
|
t[k] = b[k]
|
|
})
|
|
Object.keys(a).forEach(function (k) {
|
|
t[k] = a[k]
|
|
})
|
|
return t
|
|
}
|
|
|
|
minimatch.defaults = function (def) {
|
|
if (!def || !Object.keys(def).length) return minimatch
|
|
|
|
var orig = minimatch
|
|
|
|
var m = function minimatch (p, pattern, options) {
|
|
return orig.minimatch(p, pattern, ext(def, options))
|
|
}
|
|
|
|
m.Minimatch = function Minimatch (pattern, options) {
|
|
return new orig.Minimatch(pattern, ext(def, options))
|
|
}
|
|
|
|
return m
|
|
}
|
|
|
|
Minimatch.defaults = function (def) {
|
|
if (!def || !Object.keys(def).length) return Minimatch
|
|
return minimatch.defaults(def).Minimatch
|
|
}
|
|
|
|
|
|
function minimatch (p, pattern, options) {
|
|
if (typeof pattern !== "string") {
|
|
throw new TypeError("glob pattern string required")
|
|
}
|
|
|
|
if (!options) options = {}
|
|
|
|
// shortcut: comments match nothing.
|
|
if (!options.nocomment && pattern.charAt(0) === "#") {
|
|
return false
|
|
}
|
|
|
|
// "" only matches ""
|
|
if (pattern.trim() === "") return p === ""
|
|
|
|
return new Minimatch(pattern, options).match(p)
|
|
}
|
|
|
|
function Minimatch (pattern, options) {
|
|
if (!(this instanceof Minimatch)) {
|
|
return new Minimatch(pattern, options, cache)
|
|
}
|
|
|
|
if (typeof pattern !== "string") {
|
|
throw new TypeError("glob pattern string required")
|
|
}
|
|
|
|
if (!options) options = {}
|
|
pattern = pattern.trim()
|
|
|
|
// windows: need to use /, not \
|
|
// On other platforms, \ is a valid (albeit bad) filename char.
|
|
if (platform === "win32") {
|
|
pattern = pattern.split("\\").join("/")
|
|
}
|
|
|
|
// lru storage.
|
|
// these things aren't particularly big, but walking down the string
|
|
// and turning it into a regexp can get pretty costly.
|
|
var cacheKey = pattern + "\n" + sigmund(options)
|
|
var cached = minimatch.cache.get(cacheKey)
|
|
if (cached) return cached
|
|
minimatch.cache.set(cacheKey, this)
|
|
|
|
this.options = options
|
|
this.set = []
|
|
this.pattern = pattern
|
|
this.regexp = null
|
|
this.negate = false
|
|
this.comment = false
|
|
this.empty = false
|
|
|
|
// make the set of regexps etc.
|
|
this.make()
|
|
}
|
|
|
|
Minimatch.prototype.make = make
|
|
function make () {
|
|
// don't do it more than once.
|
|
if (this._made) return
|
|
|
|
var pattern = this.pattern
|
|
var options = this.options
|
|
|
|
// empty patterns and comments match nothing.
|
|
if (!options.nocomment && pattern.charAt(0) === "#") {
|
|
this.comment = true
|
|
return
|
|
}
|
|
if (!pattern) {
|
|
this.empty = true
|
|
return
|
|
}
|
|
|
|
// step 1: figure out negation, etc.
|
|
this.parseNegate()
|
|
|
|
// step 2: expand braces
|
|
var set = this.globSet = this.braceExpand()
|
|
|
|
if (options.debug) console.error(this.pattern, set)
|
|
|
|
// step 3: now we have a set, so turn each one into a series of path-portion
|
|
// matching patterns.
|
|
// These will be regexps, except in the case of "**", which is
|
|
// set to the GLOBSTAR object for globstar behavior,
|
|
// and will not contain any / characters
|
|
set = this.globParts = set.map(function (s) {
|
|
return s.split(slashSplit)
|
|
})
|
|
|
|
if (options.debug) console.error(this.pattern, set)
|
|
|
|
// glob --> regexps
|
|
set = set.map(function (s, si, set) {
|
|
return s.map(this.parse, this)
|
|
}, this)
|
|
|
|
if (options.debug) console.error(this.pattern, set)
|
|
|
|
// filter out everything that didn't compile properly.
|
|
set = set.filter(function (s) {
|
|
return -1 === s.indexOf(false)
|
|
})
|
|
|
|
if (options.debug) console.error(this.pattern, set)
|
|
|
|
this.set = set
|
|
}
|
|
|
|
Minimatch.prototype.parseNegate = parseNegate
|
|
function parseNegate () {
|
|
var pattern = this.pattern
|
|
, negate = false
|
|
, options = this.options
|
|
, negateOffset = 0
|
|
|
|
if (options.nonegate) return
|
|
|
|
for ( var i = 0, l = pattern.length
|
|
; i < l && pattern.charAt(i) === "!"
|
|
; i ++) {
|
|
negate = !negate
|
|
negateOffset ++
|
|
}
|
|
|
|
if (negateOffset) this.pattern = pattern.substr(negateOffset)
|
|
this.negate = negate
|
|
}
|
|
|
|
// Brace expansion:
|
|
// a{b,c}d -> abd acd
|
|
// a{b,}c -> abc ac
|
|
// a{0..3}d -> a0d a1d a2d a3d
|
|
// a{b,c{d,e}f}g -> abg acdfg acefg
|
|
// a{b,c}d{e,f}g -> abdeg acdeg abdeg abdfg
|
|
//
|
|
// Invalid sets are not expanded.
|
|
// a{2..}b -> a{2..}b
|
|
// a{b}c -> a{b}c
|
|
minimatch.braceExpand = function (pattern, options) {
|
|
return new Minimatch(pattern, options).braceExpand()
|
|
}
|
|
|
|
Minimatch.prototype.braceExpand = braceExpand
|
|
function braceExpand (pattern, options) {
|
|
options = options || this.options
|
|
pattern = typeof pattern === "undefined"
|
|
? this.pattern : pattern
|
|
|
|
if (typeof pattern === "undefined") {
|
|
throw new Error("undefined pattern")
|
|
}
|
|
|
|
if (options.nobrace ||
|
|
!pattern.match(/\{.*\}/)) {
|
|
// shortcut. no need to expand.
|
|
return [pattern]
|
|
}
|
|
|
|
var escaping = false
|
|
|
|
// examples and comments refer to this crazy pattern:
|
|
// a{b,c{d,e},{f,g}h}x{y,z}
|
|
// expected:
|
|
// abxy
|
|
// abxz
|
|
// acdxy
|
|
// acdxz
|
|
// acexy
|
|
// acexz
|
|
// afhxy
|
|
// afhxz
|
|
// aghxy
|
|
// aghxz
|
|
|
|
// everything before the first \{ is just a prefix.
|
|
// So, we pluck that off, and work with the rest,
|
|
// and then prepend it to everything we find.
|
|
if (pattern.charAt(0) !== "{") {
|
|
// console.error(pattern)
|
|
var prefix = null
|
|
for (var i = 0, l = pattern.length; i < l; i ++) {
|
|
var c = pattern.charAt(i)
|
|
// console.error(i, c)
|
|
if (c === "\\") {
|
|
escaping = !escaping
|
|
} else if (c === "{" && !escaping) {
|
|
prefix = pattern.substr(0, i)
|
|
break
|
|
}
|
|
}
|
|
|
|
// actually no sets, all { were escaped.
|
|
if (prefix === null) {
|
|
// console.error("no sets")
|
|
return [pattern]
|
|
}
|
|
|
|
var tail = braceExpand(pattern.substr(i), options)
|
|
return tail.map(function (t) {
|
|
return prefix + t
|
|
})
|
|
}
|
|
|
|
// now we have something like:
|
|
// {b,c{d,e},{f,g}h}x{y,z}
|
|
// walk through the set, expanding each part, until
|
|
// the set ends. then, we'll expand the suffix.
|
|
// If the set only has a single member, then'll put the {} back
|
|
|
|
// first, handle numeric sets, since they're easier
|
|
var numset = pattern.match(/^\{(-?[0-9]+)\.\.(-?[0-9]+)\}/)
|
|
if (numset) {
|
|
// console.error("numset", numset[1], numset[2])
|
|
var suf = braceExpand(pattern.substr(numset[0].length), options)
|
|
, start = +numset[1]
|
|
, end = +numset[2]
|
|
, inc = start > end ? -1 : 1
|
|
, set = []
|
|
for (var i = start; i != (end + inc); i += inc) {
|
|
// append all the suffixes
|
|
for (var ii = 0, ll = suf.length; ii < ll; ii ++) {
|
|
set.push(i + suf[ii])
|
|
}
|
|
}
|
|
return set
|
|
}
|
|
|
|
// ok, walk through the set
|
|
// We hope, somewhat optimistically, that there
|
|
// will be a } at the end.
|
|
// If the closing brace isn't found, then the pattern is
|
|
// interpreted as braceExpand("\\" + pattern) so that
|
|
// the leading \{ will be interpreted literally.
|
|
var i = 1 // skip the \{
|
|
, depth = 1
|
|
, set = []
|
|
, member = ""
|
|
, sawEnd = false
|
|
, escaping = false
|
|
|
|
function addMember () {
|
|
set.push(member)
|
|
member = ""
|
|
}
|
|
|
|
// console.error("Entering for")
|
|
FOR: for (i = 1, l = pattern.length; i < l; i ++) {
|
|
var c = pattern.charAt(i)
|
|
// console.error("", i, c)
|
|
|
|
if (escaping) {
|
|
escaping = false
|
|
member += "\\" + c
|
|
} else {
|
|
switch (c) {
|
|
case "\\":
|
|
escaping = true
|
|
continue
|
|
|
|
case "{":
|
|
depth ++
|
|
member += "{"
|
|
continue
|
|
|
|
case "}":
|
|
depth --
|
|
// if this closes the actual set, then we're done
|
|
if (depth === 0) {
|
|
addMember()
|
|
// pluck off the close-brace
|
|
i ++
|
|
break FOR
|
|
} else {
|
|
member += c
|
|
continue
|
|
}
|
|
|
|
case ",":
|
|
if (depth === 1) {
|
|
addMember()
|
|
} else {
|
|
member += c
|
|
}
|
|
continue
|
|
|
|
default:
|
|
member += c
|
|
continue
|
|
} // switch
|
|
} // else
|
|
} // for
|
|
|
|
// now we've either finished the set, and the suffix is
|
|
// pattern.substr(i), or we have *not* closed the set,
|
|
// and need to escape the leading brace
|
|
if (depth !== 0) {
|
|
// console.error("didn't close", pattern)
|
|
return braceExpand("\\" + pattern, options)
|
|
}
|
|
|
|
// x{y,z} -> ["xy", "xz"]
|
|
// console.error("set", set)
|
|
// console.error("suffix", pattern.substr(i))
|
|
var suf = braceExpand(pattern.substr(i), options)
|
|
// ["b", "c{d,e}","{f,g}h"] ->
|
|
// [["b"], ["cd", "ce"], ["fh", "gh"]]
|
|
var addBraces = set.length === 1
|
|
// console.error("set pre-expanded", set)
|
|
set = set.map(function (p) {
|
|
return braceExpand(p, options)
|
|
})
|
|
// console.error("set expanded", set)
|
|
|
|
|
|
// [["b"], ["cd", "ce"], ["fh", "gh"]] ->
|
|
// ["b", "cd", "ce", "fh", "gh"]
|
|
set = set.reduce(function (l, r) {
|
|
return l.concat(r)
|
|
})
|
|
|
|
if (addBraces) {
|
|
set = set.map(function (s) {
|
|
return "{" + s + "}"
|
|
})
|
|
}
|
|
|
|
// now attach the suffixes.
|
|
var ret = []
|
|
for (var i = 0, l = set.length; i < l; i ++) {
|
|
for (var ii = 0, ll = suf.length; ii < ll; ii ++) {
|
|
ret.push(set[i] + suf[ii])
|
|
}
|
|
}
|
|
return ret
|
|
}
|
|
|
|
// parse a component of the expanded set.
|
|
// At this point, no pattern may contain "/" in it
|
|
// so we're going to return a 2d array, where each entry is the full
|
|
// pattern, split on '/', and then turned into a regular expression.
|
|
// A regexp is made at the end which joins each array with an
|
|
// escaped /, and another full one which joins each regexp with |.
|
|
//
|
|
// Following the lead of Bash 4.1, note that "**" only has special meaning
|
|
// when it is the *only* thing in a path portion. Otherwise, any series
|
|
// of * is equivalent to a single *. Globstar behavior is enabled by
|
|
// default, and can be disabled by setting options.noglobstar.
|
|
Minimatch.prototype.parse = parse
|
|
var SUBPARSE = {}
|
|
function parse (pattern, isSub) {
|
|
var options = this.options
|
|
|
|
// shortcuts
|
|
if (!options.noglobstar && pattern === "**") return GLOBSTAR
|
|
if (pattern === "") return ""
|
|
|
|
var re = ""
|
|
, hasMagic = !!options.nocase
|
|
, escaping = false
|
|
// ? => one single character
|
|
, patternListStack = []
|
|
, plType
|
|
, stateChar
|
|
, inClass = false
|
|
, reClassStart = -1
|
|
, classStart = -1
|
|
// . and .. never match anything that doesn't start with .,
|
|
// even when options.dot is set.
|
|
, patternStart = pattern.charAt(0) === "." ? "" // anything
|
|
// not (start or / followed by . or .. followed by / or end)
|
|
: options.dot ? "(?!(?:^|\\\/)\\.{1,2}(?:$|\\\/))"
|
|
: "(?!\\.)"
|
|
|
|
function clearStateChar () {
|
|
if (stateChar) {
|
|
// we had some state-tracking character
|
|
// that wasn't consumed by this pass.
|
|
switch (stateChar) {
|
|
case "*":
|
|
re += star
|
|
hasMagic = true
|
|
break
|
|
case "?":
|
|
re += qmark
|
|
hasMagic = true
|
|
break
|
|
default:
|
|
re += "\\"+stateChar
|
|
break
|
|
}
|
|
stateChar = false
|
|
}
|
|
}
|
|
|
|
for ( var i = 0, len = pattern.length, c
|
|
; (i < len) && (c = pattern.charAt(i))
|
|
; i ++ ) {
|
|
|
|
if (options.debug) {
|
|
console.error("%s\t%s %s %j", pattern, i, re, c)
|
|
}
|
|
|
|
// skip over any that are escaped.
|
|
if (escaping && reSpecials[c]) {
|
|
re += "\\" + c
|
|
escaping = false
|
|
continue
|
|
}
|
|
|
|
SWITCH: switch (c) {
|
|
case "/":
|
|
// completely not allowed, even escaped.
|
|
// Should already be path-split by now.
|
|
return false
|
|
|
|
case "\\":
|
|
clearStateChar()
|
|
escaping = true
|
|
continue
|
|
|
|
// the various stateChar values
|
|
// for the "extglob" stuff.
|
|
case "?":
|
|
case "*":
|
|
case "+":
|
|
case "@":
|
|
case "!":
|
|
if (options.debug) {
|
|
console.error("%s\t%s %s %j <-- stateChar", pattern, i, re, c)
|
|
}
|
|
|
|
// all of those are literals inside a class, except that
|
|
// the glob [!a] means [^a] in regexp
|
|
if (inClass) {
|
|
if (c === "!" && i === classStart + 1) c = "^"
|
|
re += c
|
|
continue
|
|
}
|
|
|
|
// if we already have a stateChar, then it means
|
|
// that there was something like ** or +? in there.
|
|
// Handle the stateChar, then proceed with this one.
|
|
clearStateChar()
|
|
stateChar = c
|
|
// if extglob is disabled, then +(asdf|foo) isn't a thing.
|
|
// just clear the statechar *now*, rather than even diving into
|
|
// the patternList stuff.
|
|
if (options.noext) clearStateChar()
|
|
continue
|
|
|
|
case "(":
|
|
if (inClass) {
|
|
re += "("
|
|
continue
|
|
}
|
|
|
|
if (!stateChar) {
|
|
re += "\\("
|
|
continue
|
|
}
|
|
|
|
plType = stateChar
|
|
patternListStack.push({ type: plType
|
|
, start: i - 1
|
|
, reStart: re.length })
|
|
// negation is (?:(?!js)[^/]*)
|
|
re += stateChar === "!" ? "(?:(?!" : "(?:"
|
|
stateChar = false
|
|
continue
|
|
|
|
case ")":
|
|
if (inClass || !patternListStack.length) {
|
|
re += "\\)"
|
|
continue
|
|
}
|
|
|
|
hasMagic = true
|
|
re += ")"
|
|
plType = patternListStack.pop().type
|
|
// negation is (?:(?!js)[^/]*)
|
|
// The others are (?:<pattern>)<type>
|
|
switch (plType) {
|
|
case "!":
|
|
re += "[^/]*?)"
|
|
break
|
|
case "?":
|
|
case "+":
|
|
case "*": re += plType
|
|
case "@": break // the default anyway
|
|
}
|
|
continue
|
|
|
|
case "|":
|
|
if (inClass || !patternListStack.length || escaping) {
|
|
re += "\\|"
|
|
escaping = false
|
|
continue
|
|
}
|
|
|
|
re += "|"
|
|
continue
|
|
|
|
// these are mostly the same in regexp and glob
|
|
case "[":
|
|
// swallow any state-tracking char before the [
|
|
clearStateChar()
|
|
|
|
if (inClass) {
|
|
re += "\\" + c
|
|
continue
|
|
}
|
|
|
|
inClass = true
|
|
classStart = i
|
|
reClassStart = re.length
|
|
re += c
|
|
continue
|
|
|
|
case "]":
|
|
// a right bracket shall lose its special
|
|
// meaning and represent itself in
|
|
// a bracket expression if it occurs
|
|
// first in the list. -- POSIX.2 2.8.3.2
|
|
if (i === classStart + 1 || !inClass) {
|
|
re += "\\" + c
|
|
escaping = false
|
|
continue
|
|
}
|
|
|
|
// finish up the class.
|
|
hasMagic = true
|
|
inClass = false
|
|
re += c
|
|
continue
|
|
|
|
default:
|
|
// swallow any state char that wasn't consumed
|
|
clearStateChar()
|
|
|
|
if (escaping) {
|
|
// no need
|
|
escaping = false
|
|
} else if (reSpecials[c]
|
|
&& !(c === "^" && inClass)) {
|
|
re += "\\"
|
|
}
|
|
|
|
re += c
|
|
|
|
} // switch
|
|
} // for
|
|
|
|
|
|
// handle the case where we left a class open.
|
|
// "[abc" is valid, equivalent to "\[abc"
|
|
if (inClass) {
|
|
// split where the last [ was, and escape it
|
|
// this is a huge pita. We now have to re-walk
|
|
// the contents of the would-be class to re-translate
|
|
// any characters that were passed through as-is
|
|
var cs = pattern.substr(classStart + 1)
|
|
, sp = this.parse(cs, SUBPARSE)
|
|
re = re.substr(0, reClassStart) + "\\[" + sp[0]
|
|
hasMagic = hasMagic || sp[1]
|
|
}
|
|
|
|
// handle the case where we had a +( thing at the *end*
|
|
// of the pattern.
|
|
// each pattern list stack adds 3 chars, and we need to go through
|
|
// and escape any | chars that were passed through as-is for the regexp.
|
|
// Go through and escape them, taking care not to double-escape any
|
|
// | chars that were already escaped.
|
|
var pl
|
|
while (pl = patternListStack.pop()) {
|
|
var tail = re.slice(pl.reStart + 3)
|
|
// maybe some even number of \, then maybe 1 \, followed by a |
|
|
tail = tail.replace(/((?:\\{2})*)(\\?)\|/g, function (_, $1, $2) {
|
|
if (!$2) {
|
|
// the | isn't already escaped, so escape it.
|
|
$2 = "\\"
|
|
}
|
|
|
|
// need to escape all those slashes *again*, without escaping the
|
|
// one that we need for escaping the | character. As it works out,
|
|
// escaping an even number of slashes can be done by simply repeating
|
|
// it exactly after itself. That's why this trick works.
|
|
//
|
|
// I am sorry that you have to see this.
|
|
return $1 + $1 + $2 + "|"
|
|
})
|
|
|
|
// console.error("tail=%j\n %s", tail, tail)
|
|
var t = pl.type === "*" ? star
|
|
: pl.type === "?" ? qmark
|
|
: "\\" + pl.type
|
|
|
|
hasMagic = true
|
|
re = re.slice(0, pl.reStart)
|
|
+ t + "\\("
|
|
+ tail
|
|
}
|
|
|
|
// handle trailing things that only matter at the very end.
|
|
clearStateChar()
|
|
if (escaping) {
|
|
// trailing \\
|
|
re += "\\\\"
|
|
}
|
|
|
|
// only need to apply the nodot start if the re starts with
|
|
// something that could conceivably capture a dot
|
|
var addPatternStart = false
|
|
switch (re.charAt(0)) {
|
|
case ".":
|
|
case "[":
|
|
case "(": addPatternStart = true
|
|
}
|
|
|
|
// if the re is not "" at this point, then we need to make sure
|
|
// it doesn't match against an empty path part.
|
|
// Otherwise a/* will match a/, which it should not.
|
|
if (re !== "" && hasMagic) re = "(?=.)" + re
|
|
|
|
if (addPatternStart) re = patternStart + re
|
|
|
|
// parsing just a piece of a larger pattern.
|
|
if (isSub === SUBPARSE) {
|
|
return [ re, hasMagic ]
|
|
}
|
|
|
|
// skip the regexp for non-magical patterns
|
|
// unescape anything in it, though, so that it'll be
|
|
// an exact match against a file etc.
|
|
if (!hasMagic) {
|
|
return globUnescape(pattern)
|
|
}
|
|
|
|
var flags = options.nocase ? "i" : ""
|
|
, regExp = new RegExp("^" + re + "$", flags)
|
|
|
|
regExp._glob = pattern
|
|
regExp._src = re
|
|
|
|
return regExp
|
|
}
|
|
|
|
minimatch.makeRe = function (pattern, options) {
|
|
return new Minimatch(pattern, options || {}).makeRe()
|
|
}
|
|
|
|
Minimatch.prototype.makeRe = makeRe
|
|
function makeRe () {
|
|
if (this.regexp || this.regexp === false) return this.regexp
|
|
|
|
// at this point, this.set is a 2d array of partial
|
|
// pattern strings, or "**".
|
|
//
|
|
// It's better to use .match(). This function shouldn't
|
|
// be used, really, but it's pretty convenient sometimes,
|
|
// when you just want to work with a regex.
|
|
var set = this.set
|
|
|
|
if (!set.length) return this.regexp = false
|
|
var options = this.options
|
|
|
|
var twoStar = options.noglobstar ? star
|
|
: options.dot ? twoStarDot
|
|
: twoStarNoDot
|
|
, flags = options.nocase ? "i" : ""
|
|
|
|
var re = set.map(function (pattern) {
|
|
return pattern.map(function (p) {
|
|
return (p === GLOBSTAR) ? twoStar
|
|
: (typeof p === "string") ? regExpEscape(p)
|
|
: p._src
|
|
}).join("\\\/")
|
|
}).join("|")
|
|
|
|
// must match entire pattern
|
|
// ending in a * or ** will make it less strict.
|
|
re = "^(?:" + re + ")$"
|
|
|
|
// can match anything, as long as it's not this.
|
|
if (this.negate) re = "^(?!" + re + ").*$"
|
|
|
|
try {
|
|
return this.regexp = new RegExp(re, flags)
|
|
} catch (ex) {
|
|
return this.regexp = false
|
|
}
|
|
}
|
|
|
|
minimatch.match = function (list, pattern, options) {
|
|
var mm = new Minimatch(pattern, options)
|
|
list = list.filter(function (f) {
|
|
return mm.match(f)
|
|
})
|
|
if (options.nonull && !list.length) {
|
|
list.push(pattern)
|
|
}
|
|
return list
|
|
}
|
|
|
|
Minimatch.prototype.match = match
|
|
function match (f, partial) {
|
|
// console.error("match", f, this.pattern)
|
|
// short-circuit in the case of busted things.
|
|
// comments, etc.
|
|
if (this.comment) return false
|
|
if (this.empty) return f === ""
|
|
|
|
if (f === "/" && partial) return true
|
|
|
|
var options = this.options
|
|
|
|
// windows: need to use /, not \
|
|
// On other platforms, \ is a valid (albeit bad) filename char.
|
|
if (platform === "win32") {
|
|
f = f.split("\\").join("/")
|
|
}
|
|
|
|
// treat the test path as a set of pathparts.
|
|
f = f.split(slashSplit)
|
|
if (options.debug) {
|
|
console.error(this.pattern, "split", f)
|
|
}
|
|
|
|
// just ONE of the pattern sets in this.set needs to match
|
|
// in order for it to be valid. If negating, then just one
|
|
// match means that we have failed.
|
|
// Either way, return on the first hit.
|
|
|
|
var set = this.set
|
|
// console.error(this.pattern, "set", set)
|
|
|
|
for (var i = 0, l = set.length; i < l; i ++) {
|
|
var pattern = set[i]
|
|
var hit = this.matchOne(f, pattern, partial)
|
|
if (hit) {
|
|
if (options.flipNegate) return true
|
|
return !this.negate
|
|
}
|
|
}
|
|
|
|
// didn't get any hits. this is success if it's a negative
|
|
// pattern, failure otherwise.
|
|
if (options.flipNegate) return false
|
|
return this.negate
|
|
}
|
|
|
|
// set partial to true to test if, for example,
|
|
// "/a/b" matches the start of "/*/b/*/d"
|
|
// Partial means, if you run out of file before you run
|
|
// out of pattern, then that's fine, as long as all
|
|
// the parts match.
|
|
Minimatch.prototype.matchOne = function (file, pattern, partial) {
|
|
var options = this.options
|
|
|
|
if (options.debug) {
|
|
console.error("matchOne",
|
|
{ "this": this
|
|
, file: file
|
|
, pattern: pattern })
|
|
}
|
|
|
|
if (options.matchBase && pattern.length === 1) {
|
|
file = path.basename(file.join("/")).split("/")
|
|
}
|
|
|
|
if (options.debug) {
|
|
console.error("matchOne", file.length, pattern.length)
|
|
}
|
|
|
|
for ( var fi = 0
|
|
, pi = 0
|
|
, fl = file.length
|
|
, pl = pattern.length
|
|
; (fi < fl) && (pi < pl)
|
|
; fi ++, pi ++ ) {
|
|
|
|
if (options.debug) {
|
|
console.error("matchOne loop")
|
|
}
|
|
var p = pattern[pi]
|
|
, f = file[fi]
|
|
|
|
if (options.debug) {
|
|
console.error(pattern, p, f)
|
|
}
|
|
|
|
// should be impossible.
|
|
// some invalid regexp stuff in the set.
|
|
if (p === false) return false
|
|
|
|
if (p === GLOBSTAR) {
|
|
if (options.debug)
|
|
console.error('GLOBSTAR', [pattern, p, f])
|
|
|
|
// "**"
|
|
// a/**/b/**/c would match the following:
|
|
// a/b/x/y/z/c
|
|
// a/x/y/z/b/c
|
|
// a/b/x/b/x/c
|
|
// a/b/c
|
|
// To do this, take the rest of the pattern after
|
|
// the **, and see if it would match the file remainder.
|
|
// If so, return success.
|
|
// If not, the ** "swallows" a segment, and try again.
|
|
// This is recursively awful.
|
|
//
|
|
// a/**/b/**/c matching a/b/x/y/z/c
|
|
// - a matches a
|
|
// - doublestar
|
|
// - matchOne(b/x/y/z/c, b/**/c)
|
|
// - b matches b
|
|
// - doublestar
|
|
// - matchOne(x/y/z/c, c) -> no
|
|
// - matchOne(y/z/c, c) -> no
|
|
// - matchOne(z/c, c) -> no
|
|
// - matchOne(c, c) yes, hit
|
|
var fr = fi
|
|
, pr = pi + 1
|
|
if (pr === pl) {
|
|
if (options.debug)
|
|
console.error('** at the end')
|
|
// a ** at the end will just swallow the rest.
|
|
// We have found a match.
|
|
// however, it will not swallow /.x, unless
|
|
// options.dot is set.
|
|
// . and .. are *never* matched by **, for explosively
|
|
// exponential reasons.
|
|
for ( ; fi < fl; fi ++) {
|
|
if (file[fi] === "." || file[fi] === ".." ||
|
|
(!options.dot && file[fi].charAt(0) === ".")) return false
|
|
}
|
|
return true
|
|
}
|
|
|
|
// ok, let's see if we can swallow whatever we can.
|
|
WHILE: while (fr < fl) {
|
|
var swallowee = file[fr]
|
|
|
|
if (options.debug) {
|
|
console.error('\nglobstar while',
|
|
file, fr, pattern, pr, swallowee)
|
|
}
|
|
|
|
// XXX remove this slice. Just pass the start index.
|
|
if (this.matchOne(file.slice(fr), pattern.slice(pr), partial)) {
|
|
if (options.debug)
|
|
console.error('globstar found match!', fr, fl, swallowee)
|
|
// found a match.
|
|
return true
|
|
} else {
|
|
// can't swallow "." or ".." ever.
|
|
// can only swallow ".foo" when explicitly asked.
|
|
if (swallowee === "." || swallowee === ".." ||
|
|
(!options.dot && swallowee.charAt(0) === ".")) {
|
|
if (options.debug)
|
|
console.error("dot detected!", file, fr, pattern, pr)
|
|
break WHILE
|
|
}
|
|
|
|
// ** swallows a segment, and continue.
|
|
if (options.debug)
|
|
console.error('globstar swallow a segment, and continue')
|
|
fr ++
|
|
}
|
|
}
|
|
// no match was found.
|
|
// However, in partial mode, we can't say this is necessarily over.
|
|
// If there's more *pattern* left, then
|
|
if (partial) {
|
|
// ran out of file
|
|
// console.error("\n>>> no match, partial?", file, fr, pattern, pr)
|
|
if (fr === fl) return true
|
|
}
|
|
return false
|
|
}
|
|
|
|
// something other than **
|
|
// non-magic patterns just have to match exactly
|
|
// patterns with magic have been turned into regexps.
|
|
var hit
|
|
if (typeof p === "string") {
|
|
if (options.nocase) {
|
|
hit = f.toLowerCase() === p.toLowerCase()
|
|
} else {
|
|
hit = f === p
|
|
}
|
|
if (options.debug) {
|
|
console.error("string match", p, f, hit)
|
|
}
|
|
} else {
|
|
hit = f.match(p)
|
|
if (options.debug) {
|
|
console.error("pattern match", p, f, hit)
|
|
}
|
|
}
|
|
|
|
if (!hit) return false
|
|
}
|
|
|
|
// Note: ending in / means that we'll get a final ""
|
|
// at the end of the pattern. This can only match a
|
|
// corresponding "" at the end of the file.
|
|
// If the file ends in /, then it can only match a
|
|
// a pattern that ends in /, unless the pattern just
|
|
// doesn't have any more for it. But, a/b/ should *not*
|
|
// match "a/b/*", even though "" matches against the
|
|
// [^/]*? pattern, except in partial mode, where it might
|
|
// simply not be reached yet.
|
|
// However, a/b/ should still satisfy a/*
|
|
|
|
// now either we fell off the end of the pattern, or we're done.
|
|
if (fi === fl && pi === pl) {
|
|
// ran out of pattern and filename at the same time.
|
|
// an exact hit!
|
|
return true
|
|
} else if (fi === fl) {
|
|
// ran out of file, but still had pattern left.
|
|
// this is ok if we're doing the match as part of
|
|
// a glob fs traversal.
|
|
return partial
|
|
} else if (pi === pl) {
|
|
// ran out of pattern, still have file left.
|
|
// this is only acceptable if we're on the very last
|
|
// empty segment of a file with a trailing slash.
|
|
// a/* should match a/b/
|
|
var emptyFileEnd = (fi === fl - 1) && (file[fi] === "")
|
|
return emptyFileEnd
|
|
}
|
|
|
|
// should be unreachable.
|
|
throw new Error("wtf?")
|
|
}
|
|
|
|
|
|
// replace stuff like \* with *
|
|
function globUnescape (s) {
|
|
return s.replace(/\\(.)/g, "$1")
|
|
}
|
|
|
|
|
|
function regExpEscape (s) {
|
|
return s.replace(/[-[\]{}()*+?.,\\^$|#\s]/g, "\\$&")
|
|
}
|
|
|
|
})( typeof require === "function" ? require : null,
|
|
this,
|
|
typeof module === "object" ? module : null,
|
|
typeof process === "object" ? process.platform : "win32"
|
|
)
|
|
|
|
},{"__browserify_process":15,"lru-cache":52,"path":7,"sigmund":53}],52:[function(require,module,exports){
|
|
;(function () { // closure for web browsers
|
|
|
|
if (typeof module === 'object' && module.exports) {
|
|
module.exports = LRUCache
|
|
} else {
|
|
// just set the global for non-node platforms.
|
|
this.LRUCache = LRUCache
|
|
}
|
|
|
|
function hOP (obj, key) {
|
|
return Object.prototype.hasOwnProperty.call(obj, key)
|
|
}
|
|
|
|
function naiveLength () { return 1 }
|
|
|
|
function LRUCache (options) {
|
|
if (!(this instanceof LRUCache))
|
|
return new LRUCache(options)
|
|
|
|
if (typeof options === 'number')
|
|
options = { max: options }
|
|
|
|
if (!options)
|
|
options = {}
|
|
|
|
this._max = options.max
|
|
// Kind of weird to have a default max of Infinity, but oh well.
|
|
if (!this._max || !(typeof this._max === "number") || this._max <= 0 )
|
|
this._max = Infinity
|
|
|
|
this._lengthCalculator = options.length || naiveLength
|
|
if (typeof this._lengthCalculator !== "function")
|
|
this._lengthCalculator = naiveLength
|
|
|
|
this._allowStale = options.stale || false
|
|
this._maxAge = options.maxAge || null
|
|
this._dispose = options.dispose
|
|
this.reset()
|
|
}
|
|
|
|
// resize the cache when the max changes.
|
|
Object.defineProperty(LRUCache.prototype, "max",
|
|
{ set : function (mL) {
|
|
if (!mL || !(typeof mL === "number") || mL <= 0 ) mL = Infinity
|
|
this._max = mL
|
|
if (this._length > this._max) trim(this)
|
|
}
|
|
, get : function () { return this._max }
|
|
, enumerable : true
|
|
})
|
|
|
|
// resize the cache when the lengthCalculator changes.
|
|
Object.defineProperty(LRUCache.prototype, "lengthCalculator",
|
|
{ set : function (lC) {
|
|
if (typeof lC !== "function") {
|
|
this._lengthCalculator = naiveLength
|
|
this._length = this._itemCount
|
|
for (var key in this._cache) {
|
|
this._cache[key].length = 1
|
|
}
|
|
} else {
|
|
this._lengthCalculator = lC
|
|
this._length = 0
|
|
for (var key in this._cache) {
|
|
this._cache[key].length = this._lengthCalculator(this._cache[key].value)
|
|
this._length += this._cache[key].length
|
|
}
|
|
}
|
|
|
|
if (this._length > this._max) trim(this)
|
|
}
|
|
, get : function () { return this._lengthCalculator }
|
|
, enumerable : true
|
|
})
|
|
|
|
Object.defineProperty(LRUCache.prototype, "length",
|
|
{ get : function () { return this._length }
|
|
, enumerable : true
|
|
})
|
|
|
|
|
|
Object.defineProperty(LRUCache.prototype, "itemCount",
|
|
{ get : function () { return this._itemCount }
|
|
, enumerable : true
|
|
})
|
|
|
|
LRUCache.prototype.forEach = function (fn, thisp) {
|
|
thisp = thisp || this
|
|
var i = 0;
|
|
for (var k = this._mru - 1; k >= 0 && i < this._itemCount; k--) if (this._lruList[k]) {
|
|
i++
|
|
var hit = this._lruList[k]
|
|
if (this._maxAge && (Date.now() - hit.now > this._maxAge)) {
|
|
del(this, hit)
|
|
if (!this._allowStale) hit = undefined
|
|
}
|
|
if (hit) {
|
|
fn.call(thisp, hit.value, hit.key, this)
|
|
}
|
|
}
|
|
}
|
|
|
|
LRUCache.prototype.keys = function () {
|
|
var keys = new Array(this._itemCount)
|
|
var i = 0
|
|
for (var k = this._mru - 1; k >= 0 && i < this._itemCount; k--) if (this._lruList[k]) {
|
|
var hit = this._lruList[k]
|
|
keys[i++] = hit.key
|
|
}
|
|
return keys
|
|
}
|
|
|
|
LRUCache.prototype.values = function () {
|
|
var values = new Array(this._itemCount)
|
|
var i = 0
|
|
for (var k = this._mru - 1; k >= 0 && i < this._itemCount; k--) if (this._lruList[k]) {
|
|
var hit = this._lruList[k]
|
|
values[i++] = hit.value
|
|
}
|
|
return values
|
|
}
|
|
|
|
LRUCache.prototype.reset = function () {
|
|
if (this._dispose && this._cache) {
|
|
for (var k in this._cache) {
|
|
this._dispose(k, this._cache[k].value)
|
|
}
|
|
}
|
|
|
|
this._cache = Object.create(null) // hash of items by key
|
|
this._lruList = Object.create(null) // list of items in order of use recency
|
|
this._mru = 0 // most recently used
|
|
this._lru = 0 // least recently used
|
|
this._length = 0 // number of items in the list
|
|
this._itemCount = 0
|
|
}
|
|
|
|
// Provided for debugging/dev purposes only. No promises whatsoever that
|
|
// this API stays stable.
|
|
LRUCache.prototype.dump = function () {
|
|
return this._cache
|
|
}
|
|
|
|
LRUCache.prototype.dumpLru = function () {
|
|
return this._lruList
|
|
}
|
|
|
|
LRUCache.prototype.set = function (key, value) {
|
|
if (hOP(this._cache, key)) {
|
|
// dispose of the old one before overwriting
|
|
if (this._dispose) this._dispose(key, this._cache[key].value)
|
|
if (this._maxAge) this._cache[key].now = Date.now()
|
|
this._cache[key].value = value
|
|
this.get(key)
|
|
return true
|
|
}
|
|
|
|
var len = this._lengthCalculator(value)
|
|
var age = this._maxAge ? Date.now() : 0
|
|
var hit = new Entry(key, value, this._mru++, len, age)
|
|
|
|
// oversized objects fall out of cache automatically.
|
|
if (hit.length > this._max) {
|
|
if (this._dispose) this._dispose(key, value)
|
|
return false
|
|
}
|
|
|
|
this._length += hit.length
|
|
this._lruList[hit.lu] = this._cache[key] = hit
|
|
this._itemCount ++
|
|
|
|
if (this._length > this._max) trim(this)
|
|
return true
|
|
}
|
|
|
|
LRUCache.prototype.has = function (key) {
|
|
if (!hOP(this._cache, key)) return false
|
|
var hit = this._cache[key]
|
|
if (this._maxAge && (Date.now() - hit.now > this._maxAge)) {
|
|
return false
|
|
}
|
|
return true
|
|
}
|
|
|
|
LRUCache.prototype.get = function (key) {
|
|
return get(this, key, true)
|
|
}
|
|
|
|
LRUCache.prototype.peek = function (key) {
|
|
return get(this, key, false)
|
|
}
|
|
|
|
LRUCache.prototype.pop = function () {
|
|
var hit = this._lruList[this._lru]
|
|
del(this, hit)
|
|
return hit || null
|
|
}
|
|
|
|
LRUCache.prototype.del = function (key) {
|
|
del(this, this._cache[key])
|
|
}
|
|
|
|
function get (self, key, doUse) {
|
|
var hit = self._cache[key]
|
|
if (hit) {
|
|
if (self._maxAge && (Date.now() - hit.now > self._maxAge)) {
|
|
del(self, hit)
|
|
if (!self._allowStale) hit = undefined
|
|
} else {
|
|
if (doUse) use(self, hit)
|
|
}
|
|
if (hit) hit = hit.value
|
|
}
|
|
return hit
|
|
}
|
|
|
|
function use (self, hit) {
|
|
shiftLU(self, hit)
|
|
hit.lu = self._mru ++
|
|
self._lruList[hit.lu] = hit
|
|
}
|
|
|
|
function trim (self) {
|
|
while (self._lru < self._mru && self._length > self._max)
|
|
del(self, self._lruList[self._lru])
|
|
}
|
|
|
|
function shiftLU (self, hit) {
|
|
delete self._lruList[ hit.lu ]
|
|
while (self._lru < self._mru && !self._lruList[self._lru]) self._lru ++
|
|
}
|
|
|
|
function del (self, hit) {
|
|
if (hit) {
|
|
if (self._dispose) self._dispose(hit.key, hit.value)
|
|
self._length -= hit.length
|
|
self._itemCount --
|
|
delete self._cache[ hit.key ]
|
|
shiftLU(self, hit)
|
|
}
|
|
}
|
|
|
|
// classy, since V8 prefers predictable objects.
|
|
function Entry (key, value, lu, length, now) {
|
|
this.key = key
|
|
this.value = value
|
|
this.lu = lu
|
|
this.length = length
|
|
this.now = now
|
|
}
|
|
|
|
})()
|
|
|
|
},{}],53:[function(require,module,exports){
|
|
module.exports = sigmund
|
|
function sigmund (subject, maxSessions) {
|
|
maxSessions = maxSessions || 10;
|
|
var notes = [];
|
|
var analysis = '';
|
|
var RE = RegExp;
|
|
|
|
function psychoAnalyze (subject, session) {
|
|
if (session > maxSessions) return;
|
|
|
|
if (typeof subject === 'function' ||
|
|
typeof subject === 'undefined') {
|
|
return;
|
|
}
|
|
|
|
if (typeof subject !== 'object' || !subject ||
|
|
(subject instanceof RE)) {
|
|
analysis += subject;
|
|
return;
|
|
}
|
|
|
|
if (notes.indexOf(subject) !== -1 || session === maxSessions) return;
|
|
|
|
notes.push(subject);
|
|
analysis += '{';
|
|
Object.keys(subject).forEach(function (issue, _, __) {
|
|
// pseudo-private values. skip those.
|
|
if (issue.charAt(0) === '_') return;
|
|
var to = typeof subject[issue];
|
|
if (to === 'function' || to === 'undefined') return;
|
|
analysis += issue;
|
|
psychoAnalyze(subject[issue], session + 1);
|
|
});
|
|
}
|
|
psychoAnalyze(subject, 0);
|
|
return analysis;
|
|
}
|
|
|
|
// vim: set softtabstop=4 shiftwidth=4:
|
|
|
|
},{}],54:[function(require,module,exports){
|
|
var process=require("__browserify_process");var path = require('path');
|
|
var minimist = require('minimist');
|
|
var wordwrap = require('wordwrap');
|
|
|
|
/* Hack an instance of Argv with process.argv into Argv
|
|
so people can do
|
|
require('optimist')(['--beeble=1','-z','zizzle']).argv
|
|
to parse a list of args and
|
|
require('optimist').argv
|
|
to get a parsed version of process.argv.
|
|
*/
|
|
|
|
var inst = Argv(process.argv.slice(2));
|
|
Object.keys(inst).forEach(function (key) {
|
|
Argv[key] = typeof inst[key] == 'function'
|
|
? inst[key].bind(inst)
|
|
: inst[key];
|
|
});
|
|
|
|
var exports = module.exports = Argv;
|
|
function Argv (processArgs, cwd) {
|
|
var self = {};
|
|
if (!cwd) cwd = process.cwd();
|
|
|
|
self.$0 = process.argv
|
|
.slice(0,2)
|
|
.map(function (x) {
|
|
var b = rebase(cwd, x);
|
|
return x.match(/^\//) && b.length < x.length
|
|
? b : x
|
|
})
|
|
.join(' ')
|
|
;
|
|
|
|
if (process.env._ != undefined && process.argv[1] == process.env._) {
|
|
self.$0 = process.env._.replace(
|
|
path.dirname(process.execPath) + '/', ''
|
|
);
|
|
}
|
|
|
|
var options = {
|
|
boolean: [],
|
|
string: [],
|
|
alias: {},
|
|
default: []
|
|
};
|
|
|
|
self.boolean = function (bools) {
|
|
options.boolean.push.apply(options.boolean, [].concat(bools));
|
|
return self;
|
|
};
|
|
|
|
self.string = function (strings) {
|
|
options.string.push.apply(options.string, [].concat(strings));
|
|
return self;
|
|
};
|
|
|
|
self.default = function (key, value) {
|
|
if (typeof key === 'object') {
|
|
Object.keys(key).forEach(function (k) {
|
|
self.default(k, key[k]);
|
|
});
|
|
}
|
|
else {
|
|
options.default[key] = value;
|
|
}
|
|
return self;
|
|
};
|
|
|
|
self.alias = function (x, y) {
|
|
if (typeof x === 'object') {
|
|
Object.keys(x).forEach(function (key) {
|
|
self.alias(key, x[key]);
|
|
});
|
|
}
|
|
else {
|
|
options.alias[x] = (options.alias[x] || []).concat(y);
|
|
}
|
|
return self;
|
|
};
|
|
|
|
var demanded = {};
|
|
self.demand = function (keys) {
|
|
if (typeof keys == 'number') {
|
|
if (!demanded._) demanded._ = 0;
|
|
demanded._ += keys;
|
|
}
|
|
else if (Array.isArray(keys)) {
|
|
keys.forEach(function (key) {
|
|
self.demand(key);
|
|
});
|
|
}
|
|
else {
|
|
demanded[keys] = true;
|
|
}
|
|
|
|
return self;
|
|
};
|
|
|
|
var usage;
|
|
self.usage = function (msg, opts) {
|
|
if (!opts && typeof msg === 'object') {
|
|
opts = msg;
|
|
msg = null;
|
|
}
|
|
|
|
usage = msg;
|
|
|
|
if (opts) self.options(opts);
|
|
|
|
return self;
|
|
};
|
|
|
|
function fail (msg) {
|
|
self.showHelp();
|
|
if (msg) console.error(msg);
|
|
process.exit(1);
|
|
}
|
|
|
|
var checks = [];
|
|
self.check = function (f) {
|
|
checks.push(f);
|
|
return self;
|
|
};
|
|
|
|
var descriptions = {};
|
|
self.describe = function (key, desc) {
|
|
if (typeof key === 'object') {
|
|
Object.keys(key).forEach(function (k) {
|
|
self.describe(k, key[k]);
|
|
});
|
|
}
|
|
else {
|
|
descriptions[key] = desc;
|
|
}
|
|
return self;
|
|
};
|
|
|
|
self.parse = function (args) {
|
|
return parseArgs(args);
|
|
};
|
|
|
|
self.option = self.options = function (key, opt) {
|
|
if (typeof key === 'object') {
|
|
Object.keys(key).forEach(function (k) {
|
|
self.options(k, key[k]);
|
|
});
|
|
}
|
|
else {
|
|
if (opt.alias) self.alias(key, opt.alias);
|
|
if (opt.demand) self.demand(key);
|
|
if (typeof opt.default !== 'undefined') {
|
|
self.default(key, opt.default);
|
|
}
|
|
|
|
if (opt.boolean || opt.type === 'boolean') {
|
|
self.boolean(key);
|
|
}
|
|
if (opt.string || opt.type === 'string') {
|
|
self.string(key);
|
|
}
|
|
|
|
var desc = opt.describe || opt.description || opt.desc;
|
|
if (desc) {
|
|
self.describe(key, desc);
|
|
}
|
|
}
|
|
|
|
return self;
|
|
};
|
|
|
|
var wrap = null;
|
|
self.wrap = function (cols) {
|
|
wrap = cols;
|
|
return self;
|
|
};
|
|
|
|
self.showHelp = function (fn) {
|
|
if (!fn) fn = console.error;
|
|
fn(self.help());
|
|
};
|
|
|
|
self.help = function () {
|
|
var keys = Object.keys(
|
|
Object.keys(descriptions)
|
|
.concat(Object.keys(demanded))
|
|
.concat(Object.keys(options.default))
|
|
.reduce(function (acc, key) {
|
|
if (key !== '_') acc[key] = true;
|
|
return acc;
|
|
}, {})
|
|
);
|
|
|
|
var help = keys.length ? [ 'Options:' ] : [];
|
|
|
|
if (usage) {
|
|
help.unshift(usage.replace(/\$0/g, self.$0), '');
|
|
}
|
|
|
|
var switches = keys.reduce(function (acc, key) {
|
|
acc[key] = [ key ].concat(options.alias[key] || [])
|
|
.map(function (sw) {
|
|
return (sw.length > 1 ? '--' : '-') + sw
|
|
})
|
|
.join(', ')
|
|
;
|
|
return acc;
|
|
}, {});
|
|
|
|
var switchlen = longest(Object.keys(switches).map(function (s) {
|
|
return switches[s] || '';
|
|
}));
|
|
|
|
var desclen = longest(Object.keys(descriptions).map(function (d) {
|
|
return descriptions[d] || '';
|
|
}));
|
|
|
|
keys.forEach(function (key) {
|
|
var kswitch = switches[key];
|
|
var desc = descriptions[key] || '';
|
|
|
|
if (wrap) {
|
|
desc = wordwrap(switchlen + 4, wrap)(desc)
|
|
.slice(switchlen + 4)
|
|
;
|
|
}
|
|
|
|
var spadding = new Array(
|
|
Math.max(switchlen - kswitch.length + 3, 0)
|
|
).join(' ');
|
|
|
|
var dpadding = new Array(
|
|
Math.max(desclen - desc.length + 1, 0)
|
|
).join(' ');
|
|
|
|
var type = null;
|
|
|
|
if (options.boolean[key]) type = '[boolean]';
|
|
if (options.string[key]) type = '[string]';
|
|
|
|
if (!wrap && dpadding.length > 0) {
|
|
desc += dpadding;
|
|
}
|
|
|
|
var prelude = ' ' + kswitch + spadding;
|
|
var extra = [
|
|
type,
|
|
demanded[key]
|
|
? '[required]'
|
|
: null
|
|
,
|
|
options.default[key] !== undefined
|
|
? '[default: ' + JSON.stringify(options.default[key]) + ']'
|
|
: null
|
|
,
|
|
].filter(Boolean).join(' ');
|
|
|
|
var body = [ desc, extra ].filter(Boolean).join(' ');
|
|
|
|
if (wrap) {
|
|
var dlines = desc.split('\n');
|
|
var dlen = dlines.slice(-1)[0].length
|
|
+ (dlines.length === 1 ? prelude.length : 0)
|
|
|
|
body = desc + (dlen + extra.length > wrap - 2
|
|
? '\n'
|
|
+ new Array(wrap - extra.length + 1).join(' ')
|
|
+ extra
|
|
: new Array(wrap - extra.length - dlen + 1).join(' ')
|
|
+ extra
|
|
);
|
|
}
|
|
|
|
help.push(prelude + body);
|
|
});
|
|
|
|
help.push('');
|
|
return help.join('\n');
|
|
};
|
|
|
|
Object.defineProperty(self, 'argv', {
|
|
get : function () { return parseArgs(processArgs) },
|
|
enumerable : true,
|
|
});
|
|
|
|
function parseArgs (args) {
|
|
var argv = minimist(args, options);
|
|
argv.$0 = self.$0;
|
|
|
|
if (demanded._ && argv._.length < demanded._) {
|
|
fail('Not enough non-option arguments: got '
|
|
+ argv._.length + ', need at least ' + demanded._
|
|
);
|
|
}
|
|
|
|
var missing = [];
|
|
Object.keys(demanded).forEach(function (key) {
|
|
if (!argv[key]) missing.push(key);
|
|
});
|
|
|
|
if (missing.length) {
|
|
fail('Missing required arguments: ' + missing.join(', '));
|
|
}
|
|
|
|
checks.forEach(function (f) {
|
|
try {
|
|
if (f(argv) === false) {
|
|
fail('Argument check failed: ' + f.toString());
|
|
}
|
|
}
|
|
catch (err) {
|
|
fail(err)
|
|
}
|
|
});
|
|
|
|
return argv;
|
|
}
|
|
|
|
function longest (xs) {
|
|
return Math.max.apply(
|
|
null,
|
|
xs.map(function (x) { return x.length })
|
|
);
|
|
}
|
|
|
|
return self;
|
|
};
|
|
|
|
// rebase an absolute path to a relative one with respect to a base directory
|
|
// exported for tests
|
|
exports.rebase = rebase;
|
|
function rebase (base, dir) {
|
|
var ds = path.normalize(dir).split('/').slice(1);
|
|
var bs = path.normalize(base).split('/').slice(1);
|
|
|
|
for (var i = 0; ds[i] && ds[i] == bs[i]; i++);
|
|
ds.splice(0, i); bs.splice(0, i);
|
|
|
|
var p = path.normalize(
|
|
bs.map(function () { return '..' }).concat(ds).join('/')
|
|
).replace(/\/$/,'').replace(/^$/, '.');
|
|
return p.match(/^[.\/]/) ? p : './' + p;
|
|
};
|
|
|
|
},{"__browserify_process":15,"minimist":55,"path":7,"wordwrap":56}],55:[function(require,module,exports){
|
|
module.exports = function (args, opts) {
|
|
if (!opts) opts = {};
|
|
|
|
var flags = { bools : {}, strings : {} };
|
|
|
|
[].concat(opts['boolean']).filter(Boolean).forEach(function (key) {
|
|
flags.bools[key] = true;
|
|
});
|
|
|
|
[].concat(opts.string).filter(Boolean).forEach(function (key) {
|
|
flags.strings[key] = true;
|
|
});
|
|
|
|
var aliases = {};
|
|
Object.keys(opts.alias || {}).forEach(function (key) {
|
|
aliases[key] = [].concat(opts.alias[key]);
|
|
aliases[key].forEach(function (x) {
|
|
aliases[x] = [key].concat(aliases[key].filter(function (y) {
|
|
return x !== y;
|
|
}));
|
|
});
|
|
});
|
|
|
|
var defaults = opts['default'] || {};
|
|
|
|
var argv = { _ : [] };
|
|
Object.keys(flags.bools).forEach(function (key) {
|
|
setArg(key, defaults[key] === undefined ? false : defaults[key]);
|
|
});
|
|
|
|
var notFlags = [];
|
|
|
|
if (args.indexOf('--') !== -1) {
|
|
notFlags = args.slice(args.indexOf('--')+1);
|
|
args = args.slice(0, args.indexOf('--'));
|
|
}
|
|
|
|
function setArg (key, val) {
|
|
var value = !flags.strings[key] && isNumber(val)
|
|
? Number(val) : val
|
|
;
|
|
setKey(argv, key.split('.'), value);
|
|
|
|
(aliases[key] || []).forEach(function (x) {
|
|
setKey(argv, x.split('.'), value);
|
|
});
|
|
}
|
|
|
|
for (var i = 0; i < args.length; i++) {
|
|
var arg = args[i];
|
|
|
|
if (arg.match(/^--.+=/)) {
|
|
// Using [\s\S] instead of . because js doesn't support the
|
|
// 'dotall' regex modifier. See:
|
|
// http://stackoverflow.com/a/1068308/13216
|
|
var m = arg.match(/^--([^=]+)=([\s\S]*)$/);
|
|
setArg(m[1], m[2]);
|
|
}
|
|
else if (arg.match(/^--no-.+/)) {
|
|
var key = arg.match(/^--no-(.+)/)[1];
|
|
setArg(key, false);
|
|
}
|
|
else if (arg.match(/^--.+/)) {
|
|
var key = arg.match(/^--(.+)/)[1];
|
|
var next = args[i + 1];
|
|
if (next !== undefined && !next.match(/^-/)
|
|
&& !flags.bools[key]
|
|
&& (aliases[key] ? !flags.bools[aliases[key]] : true)) {
|
|
setArg(key, next);
|
|
i++;
|
|
}
|
|
else if (/^(true|false)$/.test(next)) {
|
|
setArg(key, next === 'true');
|
|
i++;
|
|
}
|
|
else {
|
|
setArg(key, true);
|
|
}
|
|
}
|
|
else if (arg.match(/^-[^-]+/)) {
|
|
var letters = arg.slice(1,-1).split('');
|
|
|
|
var broken = false;
|
|
for (var j = 0; j < letters.length; j++) {
|
|
var next = arg.slice(j+2);
|
|
|
|
if (letters[j+1] && letters[j+1] === '=') {
|
|
setArg(letters[j], arg.slice(j+3));
|
|
broken = true;
|
|
break;
|
|
}
|
|
|
|
if (next === '-') {
|
|
setArg(letters[j], next)
|
|
continue;
|
|
}
|
|
|
|
if (/[A-Za-z]/.test(letters[j])
|
|
&& /-?\d+(\.\d*)?(e-?\d+)?$/.test(next)) {
|
|
setArg(letters[j], next);
|
|
broken = true;
|
|
break;
|
|
}
|
|
|
|
if (letters[j+1] && letters[j+1].match(/\W/)) {
|
|
setArg(letters[j], arg.slice(j+2));
|
|
broken = true;
|
|
break;
|
|
}
|
|
else {
|
|
setArg(letters[j], true);
|
|
}
|
|
}
|
|
|
|
var key = arg.slice(-1)[0];
|
|
if (!broken && key !== '-') {
|
|
if (args[i+1] && !/^(-|--)[^-]/.test(args[i+1])
|
|
&& !flags.bools[key]
|
|
&& (aliases[key] ? !flags.bools[aliases[key]] : true)) {
|
|
setArg(key, args[i+1]);
|
|
i++;
|
|
}
|
|
else if (args[i+1] && /true|false/.test(args[i+1])) {
|
|
setArg(key, args[i+1] === 'true');
|
|
i++;
|
|
}
|
|
else {
|
|
setArg(key, true);
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
argv._.push(
|
|
flags.strings['_'] || !isNumber(arg) ? arg : Number(arg)
|
|
);
|
|
}
|
|
}
|
|
|
|
Object.keys(defaults).forEach(function (key) {
|
|
if (!hasKey(argv, key.split('.'))) {
|
|
setKey(argv, key.split('.'), defaults[key]);
|
|
|
|
(aliases[key] || []).forEach(function (x) {
|
|
setKey(argv, x.split('.'), defaults[key]);
|
|
});
|
|
}
|
|
});
|
|
|
|
notFlags.forEach(function(key) {
|
|
argv._.push(key);
|
|
});
|
|
|
|
return argv;
|
|
};
|
|
|
|
function hasKey (obj, keys) {
|
|
var o = obj;
|
|
keys.slice(0,-1).forEach(function (key) {
|
|
o = (o[key] || {});
|
|
});
|
|
|
|
var key = keys[keys.length - 1];
|
|
return key in o;
|
|
}
|
|
|
|
function setKey (obj, keys, value) {
|
|
var o = obj;
|
|
keys.slice(0,-1).forEach(function (key) {
|
|
if (o[key] === undefined) o[key] = {};
|
|
o = o[key];
|
|
});
|
|
|
|
var key = keys[keys.length - 1];
|
|
if (o[key] === undefined || typeof o[key] === 'boolean') {
|
|
o[key] = value;
|
|
}
|
|
else if (Array.isArray(o[key])) {
|
|
o[key].push(value);
|
|
}
|
|
else {
|
|
o[key] = [ o[key], value ];
|
|
}
|
|
}
|
|
|
|
function isNumber (x) {
|
|
if (typeof x === 'number') return true;
|
|
if (/^0x[0-9a-f]+$/i.test(x)) return true;
|
|
return /^[-+]?(?:\d+(?:\.\d*)?|\.\d+)(e[-+]?\d+)?$/.test(x);
|
|
}
|
|
|
|
function longest (xs) {
|
|
return Math.max.apply(null, xs.map(function (x) { return x.length }));
|
|
}
|
|
|
|
},{}],56:[function(require,module,exports){
|
|
var wordwrap = module.exports = function (start, stop, params) {
|
|
if (typeof start === 'object') {
|
|
params = start;
|
|
start = params.start;
|
|
stop = params.stop;
|
|
}
|
|
|
|
if (typeof stop === 'object') {
|
|
params = stop;
|
|
start = start || params.start;
|
|
stop = undefined;
|
|
}
|
|
|
|
if (!stop) {
|
|
stop = start;
|
|
start = 0;
|
|
}
|
|
|
|
if (!params) params = {};
|
|
var mode = params.mode || 'soft';
|
|
var re = mode === 'hard' ? /\b/ : /(\S+\s+)/;
|
|
|
|
return function (text) {
|
|
var chunks = text.toString()
|
|
.split(re)
|
|
.reduce(function (acc, x) {
|
|
if (mode === 'hard') {
|
|
for (var i = 0; i < x.length; i += stop - start) {
|
|
acc.push(x.slice(i, i + stop - start));
|
|
}
|
|
}
|
|
else acc.push(x)
|
|
return acc;
|
|
}, [])
|
|
;
|
|
|
|
return chunks.reduce(function (lines, rawChunk) {
|
|
if (rawChunk === '') return lines;
|
|
|
|
var chunk = rawChunk.replace(/\t/g, ' ');
|
|
|
|
var i = lines.length - 1;
|
|
if (lines[i].length + chunk.length > stop) {
|
|
lines[i] = lines[i].replace(/\s+$/, '');
|
|
|
|
chunk.split(/\n/).forEach(function (c) {
|
|
lines.push(
|
|
new Array(start + 1).join(' ')
|
|
+ c.replace(/^\s+/, '')
|
|
);
|
|
});
|
|
}
|
|
else if (chunk.match(/\n/)) {
|
|
var xs = chunk.split(/\n/);
|
|
lines[i] += xs.shift();
|
|
xs.forEach(function (c) {
|
|
lines.push(
|
|
new Array(start + 1).join(' ')
|
|
+ c.replace(/^\s+/, '')
|
|
);
|
|
});
|
|
}
|
|
else {
|
|
lines[i] += chunk;
|
|
}
|
|
|
|
return lines;
|
|
}, [ new Array(start + 1).join(' ') ]).join('\n');
|
|
};
|
|
};
|
|
|
|
wordwrap.soft = wordwrap;
|
|
|
|
wordwrap.hard = function (start, stop) {
|
|
return wordwrap(start, stop, { mode : 'hard' });
|
|
};
|
|
|
|
},{}],57:[function(require,module,exports){
|
|
var process=require("__browserify_process");/** @license MIT License (c) copyright 2011-2013 original author or authors */
|
|
|
|
/**
|
|
* A lightweight CommonJS Promises/A and when() implementation
|
|
* when is part of the cujo.js family of libraries (http://cujojs.com/)
|
|
*
|
|
* Licensed under the MIT License at:
|
|
* http://www.opensource.org/licenses/mit-license.php
|
|
*
|
|
* @author Brian Cavalier
|
|
* @author John Hann
|
|
* @version 2.6.0
|
|
*/
|
|
(function(define, global) { 'use strict';
|
|
define(function (require) {
|
|
|
|
// Public API
|
|
|
|
when.promise = promise; // Create a pending promise
|
|
when.resolve = resolve; // Create a resolved promise
|
|
when.reject = reject; // Create a rejected promise
|
|
when.defer = defer; // Create a {promise, resolver} pair
|
|
|
|
when.join = join; // Join 2 or more promises
|
|
|
|
when.all = all; // Resolve a list of promises
|
|
when.map = map; // Array.map() for promises
|
|
when.reduce = reduce; // Array.reduce() for promises
|
|
when.settle = settle; // Settle a list of promises
|
|
|
|
when.any = any; // One-winner race
|
|
when.some = some; // Multi-winner race
|
|
|
|
when.isPromise = isPromiseLike; // DEPRECATED: use isPromiseLike
|
|
when.isPromiseLike = isPromiseLike; // Is something promise-like, aka thenable
|
|
|
|
/**
|
|
* Register an observer for a promise or immediate value.
|
|
*
|
|
* @param {*} promiseOrValue
|
|
* @param {function?} [onFulfilled] callback to be called when promiseOrValue is
|
|
* successfully fulfilled. If promiseOrValue is an immediate value, callback
|
|
* will be invoked immediately.
|
|
* @param {function?} [onRejected] callback to be called when promiseOrValue is
|
|
* rejected.
|
|
* @param {function?} [onProgress] callback to be called when progress updates
|
|
* are issued for promiseOrValue.
|
|
* @returns {Promise} a new {@link Promise} that will complete with the return
|
|
* value of callback or errback or the completion value of promiseOrValue if
|
|
* callback and/or errback is not supplied.
|
|
*/
|
|
function when(promiseOrValue, onFulfilled, onRejected, onProgress) {
|
|
// Get a trusted promise for the input promiseOrValue, and then
|
|
// register promise handlers
|
|
return cast(promiseOrValue).then(onFulfilled, onRejected, onProgress);
|
|
}
|
|
|
|
function cast(x) {
|
|
return x instanceof Promise ? x : resolve(x);
|
|
}
|
|
|
|
/**
|
|
* Trusted Promise constructor. A Promise created from this constructor is
|
|
* a trusted when.js promise. Any other duck-typed promise is considered
|
|
* untrusted.
|
|
* @constructor
|
|
* @param {function} sendMessage function to deliver messages to the promise's handler
|
|
* @param {function?} inspect function that reports the promise's state
|
|
* @name Promise
|
|
*/
|
|
function Promise(sendMessage, inspect) {
|
|
this._message = sendMessage;
|
|
this.inspect = inspect;
|
|
}
|
|
|
|
Promise.prototype = {
|
|
/**
|
|
* Register handlers for this promise.
|
|
* @param [onFulfilled] {Function} fulfillment handler
|
|
* @param [onRejected] {Function} rejection handler
|
|
* @param [onProgress] {Function} progress handler
|
|
* @return {Promise} new Promise
|
|
*/
|
|
then: function(onFulfilled, onRejected, onProgress) {
|
|
/*jshint unused:false*/
|
|
var args, sendMessage;
|
|
|
|
args = arguments;
|
|
sendMessage = this._message;
|
|
|
|
return _promise(function(resolve, reject, notify) {
|
|
sendMessage('when', args, resolve, notify);
|
|
}, this._status && this._status.observed());
|
|
},
|
|
|
|
/**
|
|
* Register a rejection handler. Shortcut for .then(undefined, onRejected)
|
|
* @param {function?} onRejected
|
|
* @return {Promise}
|
|
*/
|
|
otherwise: function(onRejected) {
|
|
return this.then(undef, onRejected);
|
|
},
|
|
|
|
/**
|
|
* Ensures that onFulfilledOrRejected will be called regardless of whether
|
|
* this promise is fulfilled or rejected. onFulfilledOrRejected WILL NOT
|
|
* receive the promises' value or reason. Any returned value will be disregarded.
|
|
* onFulfilledOrRejected may throw or return a rejected promise to signal
|
|
* an additional error.
|
|
* @param {function} onFulfilledOrRejected handler to be called regardless of
|
|
* fulfillment or rejection
|
|
* @returns {Promise}
|
|
*/
|
|
ensure: function(onFulfilledOrRejected) {
|
|
return typeof onFulfilledOrRejected === 'function'
|
|
? this.then(injectHandler, injectHandler)['yield'](this)
|
|
: this;
|
|
|
|
function injectHandler() {
|
|
return resolve(onFulfilledOrRejected());
|
|
}
|
|
},
|
|
|
|
/**
|
|
* Terminate a promise chain by handling the ultimate fulfillment value or
|
|
* rejection reason, and assuming responsibility for all errors. if an
|
|
* error propagates out of handleResult or handleFatalError, it will be
|
|
* rethrown to the host, resulting in a loud stack track on most platforms
|
|
* and a crash on some.
|
|
* @param {function?} handleResult
|
|
* @param {function?} handleError
|
|
* @returns {undefined}
|
|
*/
|
|
done: function(handleResult, handleError) {
|
|
this.then(handleResult, handleError).otherwise(crash);
|
|
},
|
|
|
|
/**
|
|
* Shortcut for .then(function() { return value; })
|
|
* @param {*} value
|
|
* @return {Promise} a promise that:
|
|
* - is fulfilled if value is not a promise, or
|
|
* - if value is a promise, will fulfill with its value, or reject
|
|
* with its reason.
|
|
*/
|
|
'yield': function(value) {
|
|
return this.then(function() {
|
|
return value;
|
|
});
|
|
},
|
|
|
|
/**
|
|
* Runs a side effect when this promise fulfills, without changing the
|
|
* fulfillment value.
|
|
* @param {function} onFulfilledSideEffect
|
|
* @returns {Promise}
|
|
*/
|
|
tap: function(onFulfilledSideEffect) {
|
|
return this.then(onFulfilledSideEffect)['yield'](this);
|
|
},
|
|
|
|
/**
|
|
* Assumes that this promise will fulfill with an array, and arranges
|
|
* for the onFulfilled to be called with the array as its argument list
|
|
* i.e. onFulfilled.apply(undefined, array).
|
|
* @param {function} onFulfilled function to receive spread arguments
|
|
* @return {Promise}
|
|
*/
|
|
spread: function(onFulfilled) {
|
|
return this.then(function(array) {
|
|
// array may contain promises, so resolve its contents.
|
|
return all(array, function(array) {
|
|
return onFulfilled.apply(undef, array);
|
|
});
|
|
});
|
|
},
|
|
|
|
/**
|
|
* Shortcut for .then(onFulfilledOrRejected, onFulfilledOrRejected)
|
|
* @deprecated
|
|
*/
|
|
always: function(onFulfilledOrRejected, onProgress) {
|
|
return this.then(onFulfilledOrRejected, onFulfilledOrRejected, onProgress);
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Returns a resolved promise. The returned promise will be
|
|
* - fulfilled with promiseOrValue if it is a value, or
|
|
* - if promiseOrValue is a promise
|
|
* - fulfilled with promiseOrValue's value after it is fulfilled
|
|
* - rejected with promiseOrValue's reason after it is rejected
|
|
* @param {*} value
|
|
* @return {Promise}
|
|
*/
|
|
function resolve(value) {
|
|
return promise(function(resolve) {
|
|
resolve(value);
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Returns a rejected promise for the supplied promiseOrValue. The returned
|
|
* promise will be rejected with:
|
|
* - promiseOrValue, if it is a value, or
|
|
* - if promiseOrValue is a promise
|
|
* - promiseOrValue's value after it is fulfilled
|
|
* - promiseOrValue's reason after it is rejected
|
|
* @param {*} promiseOrValue the rejected value of the returned {@link Promise}
|
|
* @return {Promise} rejected {@link Promise}
|
|
*/
|
|
function reject(promiseOrValue) {
|
|
return when(promiseOrValue, rejected);
|
|
}
|
|
|
|
/**
|
|
* Creates a {promise, resolver} pair, either or both of which
|
|
* may be given out safely to consumers.
|
|
* The resolver has resolve, reject, and progress. The promise
|
|
* has then plus extended promise API.
|
|
*
|
|
* @return {{
|
|
* promise: Promise,
|
|
* resolve: function:Promise,
|
|
* reject: function:Promise,
|
|
* notify: function:Promise
|
|
* resolver: {
|
|
* resolve: function:Promise,
|
|
* reject: function:Promise,
|
|
* notify: function:Promise
|
|
* }}}
|
|
*/
|
|
function defer() {
|
|
var deferred, pending, resolved;
|
|
|
|
// Optimize object shape
|
|
deferred = {
|
|
promise: undef, resolve: undef, reject: undef, notify: undef,
|
|
resolver: { resolve: undef, reject: undef, notify: undef }
|
|
};
|
|
|
|
deferred.promise = pending = promise(makeDeferred);
|
|
|
|
return deferred;
|
|
|
|
function makeDeferred(resolvePending, rejectPending, notifyPending) {
|
|
deferred.resolve = deferred.resolver.resolve = function(value) {
|
|
if(resolved) {
|
|
return resolve(value);
|
|
}
|
|
resolved = true;
|
|
resolvePending(value);
|
|
return pending;
|
|
};
|
|
|
|
deferred.reject = deferred.resolver.reject = function(reason) {
|
|
if(resolved) {
|
|
return resolve(rejected(reason));
|
|
}
|
|
resolved = true;
|
|
rejectPending(reason);
|
|
return pending;
|
|
};
|
|
|
|
deferred.notify = deferred.resolver.notify = function(update) {
|
|
notifyPending(update);
|
|
return update;
|
|
};
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Creates a new promise whose fate is determined by resolver.
|
|
* @param {function} resolver function(resolve, reject, notify)
|
|
* @returns {Promise} promise whose fate is determine by resolver
|
|
*/
|
|
function promise(resolver) {
|
|
return _promise(resolver, monitorApi.PromiseStatus && monitorApi.PromiseStatus());
|
|
}
|
|
|
|
/**
|
|
* Creates a new promise, linked to parent, whose fate is determined
|
|
* by resolver.
|
|
* @param {function} resolver function(resolve, reject, notify)
|
|
* @param {Promise?} status promise from which the new promise is begotten
|
|
* @returns {Promise} promise whose fate is determine by resolver
|
|
* @private
|
|
*/
|
|
function _promise(resolver, status) {
|
|
var self, value, consumers = [];
|
|
|
|
self = new Promise(_message, inspect);
|
|
self._status = status;
|
|
|
|
// Call the provider resolver to seal the promise's fate
|
|
try {
|
|
resolver(promiseResolve, promiseReject, promiseNotify);
|
|
} catch(e) {
|
|
promiseReject(e);
|
|
}
|
|
|
|
// Return the promise
|
|
return self;
|
|
|
|
/**
|
|
* Private message delivery. Queues and delivers messages to
|
|
* the promise's ultimate fulfillment value or rejection reason.
|
|
* @private
|
|
* @param {String} type
|
|
* @param {Array} args
|
|
* @param {Function} resolve
|
|
* @param {Function} notify
|
|
*/
|
|
function _message(type, args, resolve, notify) {
|
|
consumers ? consumers.push(deliver) : enqueue(function() { deliver(value); });
|
|
|
|
function deliver(p) {
|
|
p._message(type, args, resolve, notify);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Returns a snapshot of the promise's state at the instant inspect()
|
|
* is called. The returned object is not live and will not update as
|
|
* the promise's state changes.
|
|
* @returns {{ state:String, value?:*, reason?:* }} status snapshot
|
|
* of the promise.
|
|
*/
|
|
function inspect() {
|
|
return value ? value.inspect() : toPendingState();
|
|
}
|
|
|
|
/**
|
|
* Transition from pre-resolution state to post-resolution state, notifying
|
|
* all listeners of the ultimate fulfillment or rejection
|
|
* @param {*|Promise} val resolution value
|
|
*/
|
|
function promiseResolve(val) {
|
|
if(!consumers) {
|
|
return;
|
|
}
|
|
|
|
var queue = consumers;
|
|
consumers = undef;
|
|
|
|
enqueue(function () {
|
|
value = coerce(self, val);
|
|
if(status) {
|
|
updateStatus(value, status);
|
|
}
|
|
runHandlers(queue, value);
|
|
});
|
|
|
|
}
|
|
|
|
/**
|
|
* Reject this promise with the supplied reason, which will be used verbatim.
|
|
* @param {*} reason reason for the rejection
|
|
*/
|
|
function promiseReject(reason) {
|
|
promiseResolve(rejected(reason));
|
|
}
|
|
|
|
/**
|
|
* Issue a progress event, notifying all progress listeners
|
|
* @param {*} update progress event payload to pass to all listeners
|
|
*/
|
|
function promiseNotify(update) {
|
|
if(consumers) {
|
|
var queue = consumers;
|
|
enqueue(function () {
|
|
runHandlers(queue, progressed(update));
|
|
});
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Run a queue of functions as quickly as possible, passing
|
|
* value to each.
|
|
*/
|
|
function runHandlers(queue, value) {
|
|
for (var i = 0; i < queue.length; i++) {
|
|
queue[i](value);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Creates a fulfilled, local promise as a proxy for a value
|
|
* NOTE: must never be exposed
|
|
* @param {*} value fulfillment value
|
|
* @returns {Promise}
|
|
*/
|
|
function fulfilled(value) {
|
|
return near(
|
|
new NearFulfilledProxy(value),
|
|
function() { return toFulfilledState(value); }
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Creates a rejected, local promise with the supplied reason
|
|
* NOTE: must never be exposed
|
|
* @param {*} reason rejection reason
|
|
* @returns {Promise}
|
|
*/
|
|
function rejected(reason) {
|
|
return near(
|
|
new NearRejectedProxy(reason),
|
|
function() { return toRejectedState(reason); }
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Creates a near promise using the provided proxy
|
|
* NOTE: must never be exposed
|
|
* @param {object} proxy proxy for the promise's ultimate value or reason
|
|
* @param {function} inspect function that returns a snapshot of the
|
|
* returned near promise's state
|
|
* @returns {Promise}
|
|
*/
|
|
function near(proxy, inspect) {
|
|
return new Promise(function (type, args, resolve) {
|
|
try {
|
|
resolve(proxy[type].apply(proxy, args));
|
|
} catch(e) {
|
|
resolve(rejected(e));
|
|
}
|
|
}, inspect);
|
|
}
|
|
|
|
/**
|
|
* Create a progress promise with the supplied update.
|
|
* @private
|
|
* @param {*} update
|
|
* @return {Promise} progress promise
|
|
*/
|
|
function progressed(update) {
|
|
return new Promise(function (type, args, _, notify) {
|
|
var onProgress = args[2];
|
|
try {
|
|
notify(typeof onProgress === 'function' ? onProgress(update) : update);
|
|
} catch(e) {
|
|
notify(e);
|
|
}
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Coerces x to a trusted Promise
|
|
* @param {*} x thing to coerce
|
|
* @returns {*} Guaranteed to return a trusted Promise. If x
|
|
* is trusted, returns x, otherwise, returns a new, trusted, already-resolved
|
|
* Promise whose resolution value is:
|
|
* * the resolution value of x if it's a foreign promise, or
|
|
* * x if it's a value
|
|
*/
|
|
function coerce(self, x) {
|
|
if (x === self) {
|
|
return rejected(new TypeError());
|
|
}
|
|
|
|
if (x instanceof Promise) {
|
|
return x;
|
|
}
|
|
|
|
try {
|
|
var untrustedThen = x === Object(x) && x.then;
|
|
|
|
return typeof untrustedThen === 'function'
|
|
? assimilate(untrustedThen, x)
|
|
: fulfilled(x);
|
|
} catch(e) {
|
|
return rejected(e);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Safely assimilates a foreign thenable by wrapping it in a trusted promise
|
|
* @param {function} untrustedThen x's then() method
|
|
* @param {object|function} x thenable
|
|
* @returns {Promise}
|
|
*/
|
|
function assimilate(untrustedThen, x) {
|
|
return promise(function (resolve, reject) {
|
|
fcall(untrustedThen, x, resolve, reject);
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Proxy for a near, fulfilled value
|
|
* @param {*} value
|
|
* @constructor
|
|
*/
|
|
function NearFulfilledProxy(value) {
|
|
this.value = value;
|
|
}
|
|
|
|
NearFulfilledProxy.prototype.when = function(onResult) {
|
|
return typeof onResult === 'function' ? onResult(this.value) : this.value;
|
|
};
|
|
|
|
/**
|
|
* Proxy for a near rejection
|
|
* @param {*} reason
|
|
* @constructor
|
|
*/
|
|
function NearRejectedProxy(reason) {
|
|
this.reason = reason;
|
|
}
|
|
|
|
NearRejectedProxy.prototype.when = function(_, onError) {
|
|
if(typeof onError === 'function') {
|
|
return onError(this.reason);
|
|
} else {
|
|
throw this.reason;
|
|
}
|
|
};
|
|
|
|
function updateStatus(value, status) {
|
|
value.then(statusFulfilled, statusRejected);
|
|
|
|
function statusFulfilled() { status.fulfilled(); }
|
|
function statusRejected(r) { status.rejected(r); }
|
|
}
|
|
|
|
/**
|
|
* Determines if x is promise-like, i.e. a thenable object
|
|
* NOTE: Will return true for *any thenable object*, and isn't truly
|
|
* safe, since it may attempt to access the `then` property of x (i.e.
|
|
* clever/malicious getters may do weird things)
|
|
* @param {*} x anything
|
|
* @returns {boolean} true if x is promise-like
|
|
*/
|
|
function isPromiseLike(x) {
|
|
return x && typeof x.then === 'function';
|
|
}
|
|
|
|
/**
|
|
* Initiates a competitive race, returning a promise that will resolve when
|
|
* howMany of the supplied promisesOrValues have resolved, or will reject when
|
|
* it becomes impossible for howMany to resolve, for example, when
|
|
* (promisesOrValues.length - howMany) + 1 input promises reject.
|
|
*
|
|
* @param {Array} promisesOrValues array of anything, may contain a mix
|
|
* of promises and values
|
|
* @param howMany {number} number of promisesOrValues to resolve
|
|
* @param {function?} [onFulfilled] DEPRECATED, use returnedPromise.then()
|
|
* @param {function?} [onRejected] DEPRECATED, use returnedPromise.then()
|
|
* @param {function?} [onProgress] DEPRECATED, use returnedPromise.then()
|
|
* @returns {Promise} promise that will resolve to an array of howMany values that
|
|
* resolved first, or will reject with an array of
|
|
* (promisesOrValues.length - howMany) + 1 rejection reasons.
|
|
*/
|
|
function some(promisesOrValues, howMany, onFulfilled, onRejected, onProgress) {
|
|
|
|
return when(promisesOrValues, function(promisesOrValues) {
|
|
|
|
return promise(resolveSome).then(onFulfilled, onRejected, onProgress);
|
|
|
|
function resolveSome(resolve, reject, notify) {
|
|
var toResolve, toReject, values, reasons, fulfillOne, rejectOne, len, i;
|
|
|
|
len = promisesOrValues.length >>> 0;
|
|
|
|
toResolve = Math.max(0, Math.min(howMany, len));
|
|
values = [];
|
|
|
|
toReject = (len - toResolve) + 1;
|
|
reasons = [];
|
|
|
|
// No items in the input, resolve immediately
|
|
if (!toResolve) {
|
|
resolve(values);
|
|
|
|
} else {
|
|
rejectOne = function(reason) {
|
|
reasons.push(reason);
|
|
if(!--toReject) {
|
|
fulfillOne = rejectOne = identity;
|
|
reject(reasons);
|
|
}
|
|
};
|
|
|
|
fulfillOne = function(val) {
|
|
// This orders the values based on promise resolution order
|
|
values.push(val);
|
|
if (!--toResolve) {
|
|
fulfillOne = rejectOne = identity;
|
|
resolve(values);
|
|
}
|
|
};
|
|
|
|
for(i = 0; i < len; ++i) {
|
|
if(i in promisesOrValues) {
|
|
when(promisesOrValues[i], fulfiller, rejecter, notify);
|
|
}
|
|
}
|
|
}
|
|
|
|
function rejecter(reason) {
|
|
rejectOne(reason);
|
|
}
|
|
|
|
function fulfiller(val) {
|
|
fulfillOne(val);
|
|
}
|
|
}
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Initiates a competitive race, returning a promise that will resolve when
|
|
* any one of the supplied promisesOrValues has resolved or will reject when
|
|
* *all* promisesOrValues have rejected.
|
|
*
|
|
* @param {Array|Promise} promisesOrValues array of anything, may contain a mix
|
|
* of {@link Promise}s and values
|
|
* @param {function?} [onFulfilled] DEPRECATED, use returnedPromise.then()
|
|
* @param {function?} [onRejected] DEPRECATED, use returnedPromise.then()
|
|
* @param {function?} [onProgress] DEPRECATED, use returnedPromise.then()
|
|
* @returns {Promise} promise that will resolve to the value that resolved first, or
|
|
* will reject with an array of all rejected inputs.
|
|
*/
|
|
function any(promisesOrValues, onFulfilled, onRejected, onProgress) {
|
|
|
|
function unwrapSingleResult(val) {
|
|
return onFulfilled ? onFulfilled(val[0]) : val[0];
|
|
}
|
|
|
|
return some(promisesOrValues, 1, unwrapSingleResult, onRejected, onProgress);
|
|
}
|
|
|
|
/**
|
|
* Return a promise that will resolve only once all the supplied promisesOrValues
|
|
* have resolved. The resolution value of the returned promise will be an array
|
|
* containing the resolution values of each of the promisesOrValues.
|
|
* @memberOf when
|
|
*
|
|
* @param {Array|Promise} promisesOrValues array of anything, may contain a mix
|
|
* of {@link Promise}s and values
|
|
* @param {function?} [onFulfilled] DEPRECATED, use returnedPromise.then()
|
|
* @param {function?} [onRejected] DEPRECATED, use returnedPromise.then()
|
|
* @param {function?} [onProgress] DEPRECATED, use returnedPromise.then()
|
|
* @returns {Promise}
|
|
*/
|
|
function all(promisesOrValues, onFulfilled, onRejected, onProgress) {
|
|
return _map(promisesOrValues, identity).then(onFulfilled, onRejected, onProgress);
|
|
}
|
|
|
|
/**
|
|
* Joins multiple promises into a single returned promise.
|
|
* @return {Promise} a promise that will fulfill when *all* the input promises
|
|
* have fulfilled, or will reject when *any one* of the input promises rejects.
|
|
*/
|
|
function join(/* ...promises */) {
|
|
return _map(arguments, identity);
|
|
}
|
|
|
|
/**
|
|
* Settles all input promises such that they are guaranteed not to
|
|
* be pending once the returned promise fulfills. The returned promise
|
|
* will always fulfill, except in the case where `array` is a promise
|
|
* that rejects.
|
|
* @param {Array|Promise} array or promise for array of promises to settle
|
|
* @returns {Promise} promise that always fulfills with an array of
|
|
* outcome snapshots for each input promise.
|
|
*/
|
|
function settle(array) {
|
|
return _map(array, toFulfilledState, toRejectedState);
|
|
}
|
|
|
|
/**
|
|
* Promise-aware array map function, similar to `Array.prototype.map()`,
|
|
* but input array may contain promises or values.
|
|
* @param {Array|Promise} array array of anything, may contain promises and values
|
|
* @param {function} mapFunc map function which may return a promise or value
|
|
* @returns {Promise} promise that will fulfill with an array of mapped values
|
|
* or reject if any input promise rejects.
|
|
*/
|
|
function map(array, mapFunc) {
|
|
return _map(array, mapFunc);
|
|
}
|
|
|
|
/**
|
|
* Internal map that allows a fallback to handle rejections
|
|
* @param {Array|Promise} array array of anything, may contain promises and values
|
|
* @param {function} mapFunc map function which may return a promise or value
|
|
* @param {function?} fallback function to handle rejected promises
|
|
* @returns {Promise} promise that will fulfill with an array of mapped values
|
|
* or reject if any input promise rejects.
|
|
*/
|
|
function _map(array, mapFunc, fallback) {
|
|
return when(array, function(array) {
|
|
|
|
return _promise(resolveMap);
|
|
|
|
function resolveMap(resolve, reject, notify) {
|
|
var results, len, toResolve, i;
|
|
|
|
// Since we know the resulting length, we can preallocate the results
|
|
// array to avoid array expansions.
|
|
toResolve = len = array.length >>> 0;
|
|
results = [];
|
|
|
|
if(!toResolve) {
|
|
resolve(results);
|
|
return;
|
|
}
|
|
|
|
// Since mapFunc may be async, get all invocations of it into flight
|
|
for(i = 0; i < len; i++) {
|
|
if(i in array) {
|
|
resolveOne(array[i], i);
|
|
} else {
|
|
--toResolve;
|
|
}
|
|
}
|
|
|
|
function resolveOne(item, i) {
|
|
when(item, mapFunc, fallback).then(function(mapped) {
|
|
results[i] = mapped;
|
|
|
|
if(!--toResolve) {
|
|
resolve(results);
|
|
}
|
|
}, reject, notify);
|
|
}
|
|
}
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Traditional reduce function, similar to `Array.prototype.reduce()`, but
|
|
* input may contain promises and/or values, and reduceFunc
|
|
* may return either a value or a promise, *and* initialValue may
|
|
* be a promise for the starting value.
|
|
*
|
|
* @param {Array|Promise} promise array or promise for an array of anything,
|
|
* may contain a mix of promises and values.
|
|
* @param {function} reduceFunc reduce function reduce(currentValue, nextValue, index, total),
|
|
* where total is the total number of items being reduced, and will be the same
|
|
* in each call to reduceFunc.
|
|
* @returns {Promise} that will resolve to the final reduced value
|
|
*/
|
|
function reduce(promise, reduceFunc /*, initialValue */) {
|
|
var args = fcall(slice, arguments, 1);
|
|
|
|
return when(promise, function(array) {
|
|
var total;
|
|
|
|
total = array.length;
|
|
|
|
// Wrap the supplied reduceFunc with one that handles promises and then
|
|
// delegates to the supplied.
|
|
args[0] = function (current, val, i) {
|
|
return when(current, function (c) {
|
|
return when(val, function (value) {
|
|
return reduceFunc(c, value, i, total);
|
|
});
|
|
});
|
|
};
|
|
|
|
return reduceArray.apply(array, args);
|
|
});
|
|
}
|
|
|
|
// Snapshot states
|
|
|
|
/**
|
|
* Creates a fulfilled state snapshot
|
|
* @private
|
|
* @param {*} x any value
|
|
* @returns {{state:'fulfilled',value:*}}
|
|
*/
|
|
function toFulfilledState(x) {
|
|
return { state: 'fulfilled', value: x };
|
|
}
|
|
|
|
/**
|
|
* Creates a rejected state snapshot
|
|
* @private
|
|
* @param {*} x any reason
|
|
* @returns {{state:'rejected',reason:*}}
|
|
*/
|
|
function toRejectedState(x) {
|
|
return { state: 'rejected', reason: x };
|
|
}
|
|
|
|
/**
|
|
* Creates a pending state snapshot
|
|
* @private
|
|
* @returns {{state:'pending'}}
|
|
*/
|
|
function toPendingState() {
|
|
return { state: 'pending' };
|
|
}
|
|
|
|
//
|
|
// Internals, utilities, etc.
|
|
//
|
|
|
|
var reduceArray, slice, fcall, nextTick, handlerQueue,
|
|
setTimeout, funcProto, call, arrayProto, monitorApi,
|
|
cjsRequire, MutationObserver, undef;
|
|
|
|
cjsRequire = require;
|
|
|
|
//
|
|
// Shared handler queue processing
|
|
//
|
|
// Credit to Twisol (https://github.com/Twisol) for suggesting
|
|
// this type of extensible queue + trampoline approach for
|
|
// next-tick conflation.
|
|
|
|
handlerQueue = [];
|
|
|
|
/**
|
|
* Enqueue a task. If the queue is not currently scheduled to be
|
|
* drained, schedule it.
|
|
* @param {function} task
|
|
*/
|
|
function enqueue(task) {
|
|
if(handlerQueue.push(task) === 1) {
|
|
nextTick(drainQueue);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Drain the handler queue entirely, being careful to allow the
|
|
* queue to be extended while it is being processed, and to continue
|
|
* processing until it is truly empty.
|
|
*/
|
|
function drainQueue() {
|
|
runHandlers(handlerQueue);
|
|
handlerQueue = [];
|
|
}
|
|
|
|
// capture setTimeout to avoid being caught by fake timers
|
|
// used in time based tests
|
|
setTimeout = global.setTimeout;
|
|
|
|
// Allow attaching the monitor to when() if env has no console
|
|
monitorApi = typeof console !== 'undefined' ? console : when;
|
|
|
|
// Sniff "best" async scheduling option
|
|
// Prefer process.nextTick or MutationObserver, then check for
|
|
// vertx and finally fall back to setTimeout
|
|
/*global process*/
|
|
if (typeof process === 'object' && process.nextTick) {
|
|
nextTick = process.nextTick;
|
|
} else if(MutationObserver = global.MutationObserver || global.WebKitMutationObserver) {
|
|
nextTick = (function(document, MutationObserver, drainQueue) {
|
|
var el = document.createElement('div');
|
|
new MutationObserver(drainQueue).observe(el, { attributes: true });
|
|
|
|
return function() {
|
|
el.setAttribute('x', 'x');
|
|
};
|
|
}(document, MutationObserver, drainQueue));
|
|
} else {
|
|
try {
|
|
// vert.x 1.x || 2.x
|
|
nextTick = cjsRequire('vertx').runOnLoop || cjsRequire('vertx').runOnContext;
|
|
} catch(ignore) {
|
|
nextTick = function(t) { setTimeout(t, 0); };
|
|
}
|
|
}
|
|
|
|
//
|
|
// Capture/polyfill function and array utils
|
|
//
|
|
|
|
// Safe function calls
|
|
funcProto = Function.prototype;
|
|
call = funcProto.call;
|
|
fcall = funcProto.bind
|
|
? call.bind(call)
|
|
: function(f, context) {
|
|
return f.apply(context, slice.call(arguments, 2));
|
|
};
|
|
|
|
// Safe array ops
|
|
arrayProto = [];
|
|
slice = arrayProto.slice;
|
|
|
|
// ES5 reduce implementation if native not available
|
|
// See: http://es5.github.com/#x15.4.4.21 as there are many
|
|
// specifics and edge cases. ES5 dictates that reduce.length === 1
|
|
// This implementation deviates from ES5 spec in the following ways:
|
|
// 1. It does not check if reduceFunc is a Callable
|
|
reduceArray = arrayProto.reduce ||
|
|
function(reduceFunc /*, initialValue */) {
|
|
/*jshint maxcomplexity: 7*/
|
|
var arr, args, reduced, len, i;
|
|
|
|
i = 0;
|
|
arr = Object(this);
|
|
len = arr.length >>> 0;
|
|
args = arguments;
|
|
|
|
// If no initialValue, use first item of array (we know length !== 0 here)
|
|
// and adjust i to start at second item
|
|
if(args.length <= 1) {
|
|
// Skip to the first real element in the array
|
|
for(;;) {
|
|
if(i in arr) {
|
|
reduced = arr[i++];
|
|
break;
|
|
}
|
|
|
|
// If we reached the end of the array without finding any real
|
|
// elements, it's a TypeError
|
|
if(++i >= len) {
|
|
throw new TypeError();
|
|
}
|
|
}
|
|
} else {
|
|
// If initialValue provided, use it
|
|
reduced = args[1];
|
|
}
|
|
|
|
// Do the actual reduce
|
|
for(;i < len; ++i) {
|
|
if(i in arr) {
|
|
reduced = reduceFunc(reduced, arr[i], i, arr);
|
|
}
|
|
}
|
|
|
|
return reduced;
|
|
};
|
|
|
|
function identity(x) {
|
|
return x;
|
|
}
|
|
|
|
function crash(fatalError) {
|
|
if(typeof monitorApi.reportUnhandled === 'function') {
|
|
monitorApi.reportUnhandled();
|
|
} else {
|
|
enqueue(function() {
|
|
throw fatalError;
|
|
});
|
|
}
|
|
|
|
throw fatalError;
|
|
}
|
|
|
|
return when;
|
|
});
|
|
})(typeof define === 'function' && define.amd ? define : function (factory) { module.exports = factory(require); }, this);
|
|
|
|
},{"__browserify_process":15}],58:[function(require,module,exports){
|
|
var es = {
|
|
Client: require('./lib/client'),
|
|
errors: require('./lib/errors'),
|
|
ConnectionPool: require('./lib/connection_pool'),
|
|
Transport: require('./lib/transport')
|
|
};
|
|
|
|
module.exports = es;
|
|
|
|
},{"./lib/client":60,"./lib/connection_pool":63,"./lib/errors":68,"./lib/transport":78}],59:[function(require,module,exports){
|
|
/* jshint maxlen: false */
|
|
|
|
var ca = require('./client_action');
|
|
var api = module.exports = {};
|
|
|
|
api._namespaces = ['cluster', 'indices'];
|
|
|
|
/**
|
|
* Perform a [bulk](http://elasticsearch.org/guide/reference/api/bulk/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {String} params.consistency - Explicit write consistency setting for the operation
|
|
* @param {Boolean} params.refresh - Refresh the index after performing the operation
|
|
* @param {String} [params.replication=sync] - Explicitely set the replication type
|
|
* @param {String} params.type - Default document type for items which don't provide one
|
|
* @param {String} params.index - Default index for items which don't provide one
|
|
*/
|
|
api.bulk = ca({
|
|
params: {
|
|
consistency: {
|
|
type: 'enum',
|
|
options: [
|
|
'one',
|
|
'quorum',
|
|
'all'
|
|
]
|
|
},
|
|
refresh: {
|
|
type: 'boolean'
|
|
},
|
|
replication: {
|
|
type: 'enum',
|
|
'default': 'sync',
|
|
options: [
|
|
'sync',
|
|
'async'
|
|
]
|
|
},
|
|
type: {
|
|
type: 'string'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>/<%=type%>/_bulk',
|
|
req: {
|
|
index: {
|
|
type: 'string'
|
|
},
|
|
type: {
|
|
type: 'string'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/<%=index%>/_bulk',
|
|
req: {
|
|
index: {
|
|
type: 'string'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/_bulk'
|
|
}
|
|
],
|
|
bulkBody: true,
|
|
method: 'POST'
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [clearScroll](http://www.elasticsearch.org/guide/reference/api/search/scroll/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {String or String[] or Boolean} params.scrollId - A comma-separated list of scroll IDs to clear
|
|
*/
|
|
api.clearScroll = ca({
|
|
url: {
|
|
fmt: '/_search/scroll/<%=scrollId%>',
|
|
req: {
|
|
scrollId: {
|
|
type: 'list'
|
|
}
|
|
},
|
|
sortOrder: -1
|
|
},
|
|
method: 'DELETE'
|
|
});
|
|
|
|
|
|
api.cluster = function ClusterNS(transport) {
|
|
this.transport = transport;
|
|
};
|
|
|
|
/**
|
|
* Perform a [cluster.getSettings](http://elasticsearch.org/guide/reference/api/admin-cluster-update-settings/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
*/
|
|
api.cluster.prototype.getSettings = ca({
|
|
url: {
|
|
fmt: '/_cluster/settings'
|
|
}
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [cluster.health](http://elasticsearch.org/guide/reference/api/admin-cluster-health/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {String} [params.level=cluster] - Specify the level of detail for returned information
|
|
* @param {Boolean} params.local - Return local information, do not retrieve the state from master node (default: false)
|
|
* @param {Date or Number} params.masterTimeout - Explicit operation timeout for connection to master node
|
|
* @param {Date or Number} params.timeout - Explicit operation timeout
|
|
* @param {Number} params.waitForActiveShards - Wait until the specified number of shards is active
|
|
* @param {String} params.waitForNodes - Wait until the specified number of nodes is available
|
|
* @param {Number} params.waitForRelocatingShards - Wait until the specified number of relocating shards is finished
|
|
* @param {String} params.waitForStatus - Wait until cluster is in a specific state
|
|
* @param {String} params.index - Limit the information returned to a specific index
|
|
*/
|
|
api.cluster.prototype.health = ca({
|
|
params: {
|
|
level: {
|
|
type: 'enum',
|
|
'default': 'cluster',
|
|
options: [
|
|
'cluster',
|
|
'indices',
|
|
'shards'
|
|
]
|
|
},
|
|
local: {
|
|
type: 'boolean'
|
|
},
|
|
masterTimeout: {
|
|
type: 'time',
|
|
name: 'master_timeout'
|
|
},
|
|
timeout: {
|
|
type: 'time'
|
|
},
|
|
waitForActiveShards: {
|
|
type: 'number',
|
|
name: 'wait_for_active_shards'
|
|
},
|
|
waitForNodes: {
|
|
type: 'string',
|
|
name: 'wait_for_nodes'
|
|
},
|
|
waitForRelocatingShards: {
|
|
type: 'number',
|
|
name: 'wait_for_relocating_shards'
|
|
},
|
|
waitForStatus: {
|
|
type: 'enum',
|
|
'default': null,
|
|
options: [
|
|
'green',
|
|
'yellow',
|
|
'red'
|
|
],
|
|
name: 'wait_for_status'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/_cluster/health/<%=index%>',
|
|
req: {
|
|
index: {
|
|
type: 'string'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/_cluster/health'
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [cluster.nodeHotThreads](http://www.elasticsearch.org/guide/reference/api/admin-cluster-nodes-hot-threads/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {Date or Number} params.interval - The interval for the second sampling of threads
|
|
* @param {Number} params.snapshots - Number of samples of thread stacktrace (default: 10)
|
|
* @param {Number} params.threads - Specify the number of threads to provide information for (default: 3)
|
|
* @param {String} params.type - The type to sample (default: cpu)
|
|
* @param {String or String[] or Boolean} params.nodeId - A comma-separated list of node IDs or names to limit the returned information; use `_local` to return information from the node you're connecting to, leave empty to get information from all nodes
|
|
*/
|
|
api.cluster.prototype.nodeHotThreads = ca({
|
|
params: {
|
|
interval: {
|
|
type: 'time'
|
|
},
|
|
snapshots: {
|
|
type: 'number'
|
|
},
|
|
threads: {
|
|
type: 'number'
|
|
},
|
|
type: {
|
|
type: 'enum',
|
|
options: [
|
|
'cpu',
|
|
'wait',
|
|
'block'
|
|
]
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/_nodes/<%=nodeId%>/hotthreads',
|
|
req: {
|
|
nodeId: {
|
|
type: 'list'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/_nodes/hotthreads'
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [cluster.nodeInfo](http://elasticsearch.org/guide/reference/api/admin-cluster-nodes-info/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {Boolean} params.all - Return all available information
|
|
* @param {Boolean} params.clear - Reset the default settings
|
|
* @param {Boolean} params.http - Return information about HTTP
|
|
* @param {Boolean} params.jvm - Return information about the JVM
|
|
* @param {Boolean} params.network - Return information about network
|
|
* @param {Boolean} params.os - Return information about the operating system
|
|
* @param {Boolean} params.plugin - Return information about plugins
|
|
* @param {Boolean} params.process - Return information about the Elasticsearch process
|
|
* @param {Boolean} params.settings - Return information about node settings
|
|
* @param {Boolean} params.threadPool - Return information about the thread pool
|
|
* @param {Date or Number} params.timeout - Explicit operation timeout
|
|
* @param {Boolean} params.transport - Return information about transport
|
|
* @param {String or String[] or Boolean} params.nodeId - A comma-separated list of node IDs or names to limit the returned information; use `_local` to return information from the node you're connecting to, leave empty to get information from all nodes
|
|
*/
|
|
api.cluster.prototype.nodeInfo = ca({
|
|
params: {
|
|
all: {
|
|
type: 'boolean'
|
|
},
|
|
clear: {
|
|
type: 'boolean'
|
|
},
|
|
http: {
|
|
type: 'boolean'
|
|
},
|
|
jvm: {
|
|
type: 'boolean'
|
|
},
|
|
network: {
|
|
type: 'boolean'
|
|
},
|
|
os: {
|
|
type: 'boolean'
|
|
},
|
|
plugin: {
|
|
type: 'boolean'
|
|
},
|
|
process: {
|
|
type: 'boolean'
|
|
},
|
|
settings: {
|
|
type: 'boolean'
|
|
},
|
|
threadPool: {
|
|
type: 'boolean',
|
|
name: 'thread_pool'
|
|
},
|
|
timeout: {
|
|
type: 'time'
|
|
},
|
|
transport: {
|
|
type: 'boolean'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/_nodes/<%=nodeId%>',
|
|
req: {
|
|
nodeId: {
|
|
type: 'list'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/_nodes'
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [cluster.nodeShutdown](http://elasticsearch.org/guide/reference/api/admin-cluster-nodes-shutdown/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {Date or Number} params.delay - Set the delay for the operation (default: 1s)
|
|
* @param {Boolean} params.exit - Exit the JVM as well (default: true)
|
|
* @param {String or String[] or Boolean} params.nodeId - A comma-separated list of node IDs or names to perform the operation on; use `_local` to perform the operation on the node you're connected to, leave empty to perform the operation on all nodes
|
|
*/
|
|
api.cluster.prototype.nodeShutdown = ca({
|
|
params: {
|
|
delay: {
|
|
type: 'time'
|
|
},
|
|
exit: {
|
|
type: 'boolean'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/_cluster/nodes/<%=nodeId%>/_shutdown',
|
|
req: {
|
|
nodeId: {
|
|
type: 'list'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/_shutdown'
|
|
}
|
|
],
|
|
method: 'POST'
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [cluster.nodeStats](http://elasticsearch.org/guide/reference/api/admin-cluster-nodes-stats/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {Boolean} params.all - Return all available information
|
|
* @param {Boolean} params.clear - Reset the default level of detail
|
|
* @param {String or String[] or Boolean} params.fields - A comma-separated list of fields to return detailed information for, when returning the `indices` metric family (supports wildcards)
|
|
* @param {Boolean} params.fs - Return information about the filesystem
|
|
* @param {Boolean} params.http - Return information about HTTP
|
|
* @param {Boolean} params.indices - Return information about indices
|
|
* @param {Boolean} params.jvm - Return information about the JVM
|
|
* @param {Boolean} params.network - Return information about network
|
|
* @param {Boolean} params.os - Return information about the operating system
|
|
* @param {Boolean} params.process - Return information about the Elasticsearch process
|
|
* @param {Boolean} params.threadPool - Return information about the thread pool
|
|
* @param {Boolean} params.transport - Return information about transport
|
|
* @param {String} params.metricFamily - Limit the information returned to a certain metric family
|
|
* @param {String} params.metric - Limit the information returned for `indices` family to a specific metric
|
|
* @param {String or String[] or Boolean} params.nodeId - A comma-separated list of node IDs or names to limit the returned information; use `_local` to return information from the node you're connecting to, leave empty to get information from all nodes
|
|
*/
|
|
api.cluster.prototype.nodeStats = ca({
|
|
params: {
|
|
all: {
|
|
type: 'boolean'
|
|
},
|
|
clear: {
|
|
type: 'boolean'
|
|
},
|
|
fields: {
|
|
type: 'list'
|
|
},
|
|
fs: {
|
|
type: 'boolean'
|
|
},
|
|
http: {
|
|
type: 'boolean'
|
|
},
|
|
indices: {
|
|
type: 'boolean'
|
|
},
|
|
jvm: {
|
|
type: 'boolean'
|
|
},
|
|
network: {
|
|
type: 'boolean'
|
|
},
|
|
os: {
|
|
type: 'boolean'
|
|
},
|
|
process: {
|
|
type: 'boolean'
|
|
},
|
|
threadPool: {
|
|
type: 'boolean',
|
|
name: 'thread_pool'
|
|
},
|
|
transport: {
|
|
type: 'boolean'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/_nodes/<%=nodeId%>/stats',
|
|
req: {
|
|
nodeId: {
|
|
type: 'list'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/_nodes/stats'
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [cluster.putSettings](http://elasticsearch.org/guide/reference/api/admin-cluster-update-settings/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
*/
|
|
api.cluster.prototype.putSettings = ca({
|
|
url: {
|
|
fmt: '/_cluster/settings'
|
|
},
|
|
method: 'PUT'
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [cluster.reroute](http://elasticsearch.org/guide/reference/api/admin-cluster-reroute/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {Boolean} params.dryRun - Simulate the operation only and return the resulting state
|
|
* @param {Boolean} params.filterMetadata - Don't return cluster state metadata (default: false)
|
|
*/
|
|
api.cluster.prototype.reroute = ca({
|
|
params: {
|
|
dryRun: {
|
|
type: 'boolean',
|
|
name: 'dry_run'
|
|
},
|
|
filterMetadata: {
|
|
type: 'boolean',
|
|
name: 'filter_metadata'
|
|
}
|
|
},
|
|
url: {
|
|
fmt: '/_cluster/reroute'
|
|
},
|
|
method: 'POST'
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [cluster.state](http://elasticsearch.org/guide/reference/api/admin-cluster-state/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {Boolean} params.filterBlocks - Do not return information about blocks
|
|
* @param {Boolean} params.filterIndexTemplates - Do not return information about index templates
|
|
* @param {String or String[] or Boolean} params.filterIndices - Limit returned metadata information to specific indices
|
|
* @param {Boolean} params.filterMetadata - Do not return information about indices metadata
|
|
* @param {Boolean} params.filterNodes - Do not return information about nodes
|
|
* @param {Boolean} params.filterRoutingTable - Do not return information about shard allocation (`routing_table` and `routing_nodes`)
|
|
* @param {Boolean} params.local - Return local information, do not retrieve the state from master node (default: false)
|
|
* @param {Date or Number} params.masterTimeout - Specify timeout for connection to master
|
|
*/
|
|
api.cluster.prototype.state = ca({
|
|
params: {
|
|
filterBlocks: {
|
|
type: 'boolean',
|
|
name: 'filter_blocks'
|
|
},
|
|
filterIndexTemplates: {
|
|
type: 'boolean',
|
|
name: 'filter_index_templates'
|
|
},
|
|
filterIndices: {
|
|
type: 'list',
|
|
name: 'filter_indices'
|
|
},
|
|
filterMetadata: {
|
|
type: 'boolean',
|
|
name: 'filter_metadata'
|
|
},
|
|
filterNodes: {
|
|
type: 'boolean',
|
|
name: 'filter_nodes'
|
|
},
|
|
filterRoutingTable: {
|
|
type: 'boolean',
|
|
name: 'filter_routing_table'
|
|
},
|
|
local: {
|
|
type: 'boolean'
|
|
},
|
|
masterTimeout: {
|
|
type: 'time',
|
|
name: 'master_timeout'
|
|
}
|
|
},
|
|
url: {
|
|
fmt: '/_cluster/state'
|
|
}
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [count](http://elasticsearch.org/guide/reference/api/count/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {String} [params.ignoreIndices=none] - When performed on multiple indices, allows to ignore `missing` ones
|
|
* @param {Number} params.minScore - Include only documents with a specific `_score` value in the result
|
|
* @param {String} params.preference - Specify the node or shard the operation should be performed on (default: random)
|
|
* @param {String} params.routing - Specific routing value
|
|
* @param {String} params.source - The URL-encoded query definition (instead of using the request body)
|
|
* @param {String or String[] or Boolean} params.index - A comma-separated list of indices to restrict the results
|
|
* @param {String or String[] or Boolean} params.type - A comma-separated list of types to restrict the results
|
|
*/
|
|
api.count = ca({
|
|
params: {
|
|
ignoreIndices: {
|
|
type: 'enum',
|
|
'default': 'none',
|
|
options: [
|
|
'none',
|
|
'missing'
|
|
],
|
|
name: 'ignore_indices'
|
|
},
|
|
minScore: {
|
|
type: 'number',
|
|
name: 'min_score'
|
|
},
|
|
preference: {
|
|
type: 'string'
|
|
},
|
|
routing: {
|
|
type: 'string'
|
|
},
|
|
source: {
|
|
type: 'string'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>/<%=type%>/_count',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
},
|
|
type: {
|
|
type: 'list'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/<%=index%>/_count',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/_count'
|
|
}
|
|
],
|
|
method: 'POST'
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [delete](http://elasticsearch.org/guide/reference/api/delete/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {String} params.consistency - Specific write consistency setting for the operation
|
|
* @param {String} params.parent - ID of parent document
|
|
* @param {Boolean} params.refresh - Refresh the index after performing the operation
|
|
* @param {String} [params.replication=sync] - Specific replication type
|
|
* @param {String} params.routing - Specific routing value
|
|
* @param {Date or Number} params.timeout - Explicit operation timeout
|
|
* @param {Number} params.version - Explicit version number for concurrency control
|
|
* @param {String} params.versionType - Specific version type
|
|
* @param {String} params.id - The document ID
|
|
* @param {String} params.index - The name of the index
|
|
* @param {String} params.type - The type of the document
|
|
*/
|
|
api['delete'] = ca({
|
|
params: {
|
|
consistency: {
|
|
type: 'enum',
|
|
options: [
|
|
'one',
|
|
'quorum',
|
|
'all'
|
|
]
|
|
},
|
|
parent: {
|
|
type: 'string'
|
|
},
|
|
refresh: {
|
|
type: 'boolean'
|
|
},
|
|
replication: {
|
|
type: 'enum',
|
|
'default': 'sync',
|
|
options: [
|
|
'sync',
|
|
'async'
|
|
]
|
|
},
|
|
routing: {
|
|
type: 'string'
|
|
},
|
|
timeout: {
|
|
type: 'time'
|
|
},
|
|
version: {
|
|
type: 'number'
|
|
},
|
|
versionType: {
|
|
type: 'enum',
|
|
options: [
|
|
'internal',
|
|
'external'
|
|
],
|
|
name: 'version_type'
|
|
}
|
|
},
|
|
url: {
|
|
fmt: '/<%=index%>/<%=type%>/<%=id%>',
|
|
req: {
|
|
index: {
|
|
type: 'string'
|
|
},
|
|
type: {
|
|
type: 'string'
|
|
},
|
|
id: {
|
|
type: 'string'
|
|
}
|
|
},
|
|
sortOrder: -3
|
|
},
|
|
method: 'DELETE'
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [deleteByQuery](http://www.elasticsearch.org/guide/reference/api/delete-by-query/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {String} params.analyzer - The analyzer to use for the query string
|
|
* @param {String} params.consistency - Specific write consistency setting for the operation
|
|
* @param {String} [params.defaultOperator=OR] - The default operator for query string query (AND or OR)
|
|
* @param {String} params.df - The field to use as default where no field prefix is given in the query string
|
|
* @param {String} [params.ignoreIndices=none] - When performed on multiple indices, allows to ignore `missing` ones
|
|
* @param {String} [params.replication=sync] - Specific replication type
|
|
* @param {String} params.q - Query in the Lucene query string syntax
|
|
* @param {String} params.routing - Specific routing value
|
|
* @param {String} params.source - The URL-encoded query definition (instead of using the request body)
|
|
* @param {Date or Number} params.timeout - Explicit operation timeout
|
|
* @param {String or String[] or Boolean} params.index - A comma-separated list of indices to restrict the operation; use `_all` to perform the operation on all indices
|
|
* @param {String or String[] or Boolean} params.type - A comma-separated list of types to restrict the operation
|
|
*/
|
|
api.deleteByQuery = ca({
|
|
params: {
|
|
analyzer: {
|
|
type: 'string'
|
|
},
|
|
consistency: {
|
|
type: 'enum',
|
|
options: [
|
|
'one',
|
|
'quorum',
|
|
'all'
|
|
]
|
|
},
|
|
defaultOperator: {
|
|
type: 'enum',
|
|
'default': 'OR',
|
|
options: [
|
|
'AND',
|
|
'OR'
|
|
],
|
|
name: 'default_operator'
|
|
},
|
|
df: {
|
|
type: 'string'
|
|
},
|
|
ignoreIndices: {
|
|
type: 'enum',
|
|
'default': 'none',
|
|
options: [
|
|
'none',
|
|
'missing'
|
|
],
|
|
name: 'ignore_indices'
|
|
},
|
|
replication: {
|
|
type: 'enum',
|
|
'default': 'sync',
|
|
options: [
|
|
'sync',
|
|
'async'
|
|
]
|
|
},
|
|
q: {
|
|
type: 'string'
|
|
},
|
|
routing: {
|
|
type: 'string'
|
|
},
|
|
source: {
|
|
type: 'string'
|
|
},
|
|
timeout: {
|
|
type: 'time'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>/<%=type%>/_query',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
},
|
|
type: {
|
|
type: 'list'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/<%=index%>/_query',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
}
|
|
}
|
|
}
|
|
],
|
|
method: 'DELETE'
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [exists](http://elasticsearch.org/guide/reference/api/get/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {String} params.parent - The ID of the parent document
|
|
* @param {String} params.preference - Specify the node or shard the operation should be performed on (default: random)
|
|
* @param {Boolean} params.realtime - Specify whether to perform the operation in realtime or search mode
|
|
* @param {Boolean} params.refresh - Refresh the shard containing the document before performing the operation
|
|
* @param {String} params.routing - Specific routing value
|
|
* @param {String} params.id - The document ID
|
|
* @param {String} params.index - The name of the index
|
|
* @param {String} [params.type=_all] - The type of the document (use `_all` to fetch the first document matching the ID across all types)
|
|
*/
|
|
api.exists = ca({
|
|
params: {
|
|
parent: {
|
|
type: 'string'
|
|
},
|
|
preference: {
|
|
type: 'string'
|
|
},
|
|
realtime: {
|
|
type: 'boolean'
|
|
},
|
|
refresh: {
|
|
type: 'boolean'
|
|
},
|
|
routing: {
|
|
type: 'string'
|
|
}
|
|
},
|
|
url: {
|
|
fmt: '/<%=index%>/<%=type%>/<%=id%>',
|
|
opt: {
|
|
type: {
|
|
type: 'string',
|
|
'default': '_all'
|
|
}
|
|
},
|
|
req: {
|
|
index: {
|
|
type: 'string'
|
|
},
|
|
id: {
|
|
type: 'string'
|
|
}
|
|
},
|
|
sortOrder: -2
|
|
},
|
|
castExists: true,
|
|
method: 'HEAD'
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [explain](http://elasticsearch.org/guide/reference/api/explain/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {Boolean} params.analyzeWildcard - Specify whether wildcards and prefix queries in the query string query should be analyzed (default: false)
|
|
* @param {String} params.analyzer - The analyzer for the query string query
|
|
* @param {String} [params.defaultOperator=OR] - The default operator for query string query (AND or OR)
|
|
* @param {String} params.df - The default field for query string query (default: _all)
|
|
* @param {String or String[] or Boolean} params.fields - A comma-separated list of fields to return in the response
|
|
* @param {Boolean} params.lenient - Specify whether format-based query failures (such as providing text to a numeric field) should be ignored
|
|
* @param {Boolean} params.lowercaseExpandedTerms - Specify whether query terms should be lowercased
|
|
* @param {String} params.parent - The ID of the parent document
|
|
* @param {String} params.preference - Specify the node or shard the operation should be performed on (default: random)
|
|
* @param {String} params.q - Query in the Lucene query string syntax
|
|
* @param {String} params.routing - Specific routing value
|
|
* @param {String or String[] or Boolean} params.source - True or false to return the _source field or not, or a list of fields to return
|
|
* @param {String or String[] or Boolean} params.sourceExclude - A list of fields to exclude from the returned _source field
|
|
* @param {String or String[] or Boolean} params.sourceInclude - A list of fields to extract and return from the _source field
|
|
* @param {String} params.id - The document ID
|
|
* @param {String} params.index - The name of the index
|
|
* @param {String} params.type - The type of the document
|
|
*/
|
|
api.explain = ca({
|
|
params: {
|
|
analyzeWildcard: {
|
|
type: 'boolean',
|
|
name: 'analyze_wildcard'
|
|
},
|
|
analyzer: {
|
|
type: 'string'
|
|
},
|
|
defaultOperator: {
|
|
type: 'enum',
|
|
'default': 'OR',
|
|
options: [
|
|
'AND',
|
|
'OR'
|
|
],
|
|
name: 'default_operator'
|
|
},
|
|
df: {
|
|
type: 'string'
|
|
},
|
|
fields: {
|
|
type: 'list'
|
|
},
|
|
lenient: {
|
|
type: 'boolean'
|
|
},
|
|
lowercaseExpandedTerms: {
|
|
type: 'boolean',
|
|
name: 'lowercase_expanded_terms'
|
|
},
|
|
parent: {
|
|
type: 'string'
|
|
},
|
|
preference: {
|
|
type: 'string'
|
|
},
|
|
q: {
|
|
type: 'string'
|
|
},
|
|
routing: {
|
|
type: 'string'
|
|
},
|
|
source: {
|
|
type: 'list',
|
|
name: '_source'
|
|
},
|
|
sourceExclude: {
|
|
type: 'list',
|
|
name: '_source_exclude'
|
|
},
|
|
sourceInclude: {
|
|
type: 'list',
|
|
name: '_source_include'
|
|
}
|
|
},
|
|
url: {
|
|
fmt: '/<%=index%>/<%=type%>/<%=id%>/_explain',
|
|
req: {
|
|
index: {
|
|
type: 'string'
|
|
},
|
|
type: {
|
|
type: 'string'
|
|
},
|
|
id: {
|
|
type: 'string'
|
|
}
|
|
},
|
|
sortOrder: -3
|
|
},
|
|
method: 'POST'
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [get](http://elasticsearch.org/guide/reference/api/get/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {String or String[] or Boolean} params.fields - A comma-separated list of fields to return in the response
|
|
* @param {String} params.parent - The ID of the parent document
|
|
* @param {String} params.preference - Specify the node or shard the operation should be performed on (default: random)
|
|
* @param {Boolean} params.realtime - Specify whether to perform the operation in realtime or search mode
|
|
* @param {Boolean} params.refresh - Refresh the shard containing the document before performing the operation
|
|
* @param {String} params.routing - Specific routing value
|
|
* @param {String or String[] or Boolean} params.source - True or false to return the _source field or not, or a list of fields to return
|
|
* @param {String or String[] or Boolean} params.sourceExclude - A list of fields to exclude from the returned _source field
|
|
* @param {String or String[] or Boolean} params.sourceInclude - A list of fields to extract and return from the _source field
|
|
* @param {String} params.id - The document ID
|
|
* @param {String} params.index - The name of the index
|
|
* @param {String} [params.type=_all] - The type of the document (use `_all` to fetch the first document matching the ID across all types)
|
|
*/
|
|
api.get = ca({
|
|
params: {
|
|
fields: {
|
|
type: 'list'
|
|
},
|
|
parent: {
|
|
type: 'string'
|
|
},
|
|
preference: {
|
|
type: 'string'
|
|
},
|
|
realtime: {
|
|
type: 'boolean'
|
|
},
|
|
refresh: {
|
|
type: 'boolean'
|
|
},
|
|
routing: {
|
|
type: 'string'
|
|
},
|
|
source: {
|
|
type: 'list',
|
|
name: '_source'
|
|
},
|
|
sourceExclude: {
|
|
type: 'list',
|
|
name: '_source_exclude'
|
|
},
|
|
sourceInclude: {
|
|
type: 'list',
|
|
name: '_source_include'
|
|
}
|
|
},
|
|
url: {
|
|
fmt: '/<%=index%>/<%=type%>/<%=id%>',
|
|
opt: {
|
|
type: {
|
|
type: 'string',
|
|
'default': '_all'
|
|
}
|
|
},
|
|
req: {
|
|
index: {
|
|
type: 'string'
|
|
},
|
|
id: {
|
|
type: 'string'
|
|
}
|
|
},
|
|
sortOrder: -2
|
|
}
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [getSource](http://elasticsearch.org/guide/reference/api/get/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {String or String[] or Boolean} params.exclude - A list of fields to exclude from the returned _source field
|
|
* @param {String or String[] or Boolean} params.include - A list of fields to extract and return from the _source field
|
|
* @param {String} params.parent - The ID of the parent document
|
|
* @param {String} params.preference - Specify the node or shard the operation should be performed on (default: random)
|
|
* @param {Boolean} params.realtime - Specify whether to perform the operation in realtime or search mode
|
|
* @param {Boolean} params.refresh - Refresh the shard containing the document before performing the operation
|
|
* @param {String} params.routing - Specific routing value
|
|
* @param {String} params.id - The document ID
|
|
* @param {String} params.index - The name of the index
|
|
* @param {String} [params.type=_all] - The type of the document; use `_all` to fetch the first document matching the ID across all types
|
|
*/
|
|
api.getSource = ca({
|
|
params: {
|
|
exclude: {
|
|
type: 'list'
|
|
},
|
|
include: {
|
|
type: 'list'
|
|
},
|
|
parent: {
|
|
type: 'string'
|
|
},
|
|
preference: {
|
|
type: 'string'
|
|
},
|
|
realtime: {
|
|
type: 'boolean'
|
|
},
|
|
refresh: {
|
|
type: 'boolean'
|
|
},
|
|
routing: {
|
|
type: 'string'
|
|
}
|
|
},
|
|
url: {
|
|
fmt: '/<%=index%>/<%=type%>/<%=id%>/_source',
|
|
opt: {
|
|
type: {
|
|
type: 'string',
|
|
'default': '_all'
|
|
}
|
|
},
|
|
req: {
|
|
index: {
|
|
type: 'string'
|
|
},
|
|
id: {
|
|
type: 'string'
|
|
}
|
|
},
|
|
sortOrder: -2
|
|
}
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [index](http://elasticsearch.org/guide/reference/api/index_/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {String} params.consistency - Explicit write consistency setting for the operation
|
|
* @param {String} [params.opType=index] - Explicit operation type
|
|
* @param {String} params.parent - ID of the parent document
|
|
* @param {String} params.percolate - Percolator queries to execute while indexing the document
|
|
* @param {Boolean} params.refresh - Refresh the index after performing the operation
|
|
* @param {String} [params.replication=sync] - Specific replication type
|
|
* @param {String} params.routing - Specific routing value
|
|
* @param {Date or Number} params.timeout - Explicit operation timeout
|
|
* @param {Date or Number} params.timestamp - Explicit timestamp for the document
|
|
* @param {Duration} params.ttl - Expiration time for the document
|
|
* @param {Number} params.version - Explicit version number for concurrency control
|
|
* @param {String} params.versionType - Specific version type
|
|
* @param {String} params.id - Document ID
|
|
* @param {String} params.index - The name of the index
|
|
* @param {String} params.type - The type of the document
|
|
*/
|
|
api.index = ca({
|
|
params: {
|
|
consistency: {
|
|
type: 'enum',
|
|
options: [
|
|
'one',
|
|
'quorum',
|
|
'all'
|
|
]
|
|
},
|
|
opType: {
|
|
type: 'enum',
|
|
'default': 'index',
|
|
options: [
|
|
'index',
|
|
'create'
|
|
],
|
|
name: 'op_type'
|
|
},
|
|
parent: {
|
|
type: 'string'
|
|
},
|
|
percolate: {
|
|
type: 'string'
|
|
},
|
|
refresh: {
|
|
type: 'boolean'
|
|
},
|
|
replication: {
|
|
type: 'enum',
|
|
'default': 'sync',
|
|
options: [
|
|
'sync',
|
|
'async'
|
|
]
|
|
},
|
|
routing: {
|
|
type: 'string'
|
|
},
|
|
timeout: {
|
|
type: 'time'
|
|
},
|
|
timestamp: {
|
|
type: 'time'
|
|
},
|
|
ttl: {
|
|
type: 'duration'
|
|
},
|
|
version: {
|
|
type: 'number'
|
|
},
|
|
versionType: {
|
|
type: 'enum',
|
|
options: [
|
|
'internal',
|
|
'external'
|
|
],
|
|
name: 'version_type'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>/<%=type%>/<%=id%>',
|
|
req: {
|
|
index: {
|
|
type: 'string'
|
|
},
|
|
type: {
|
|
type: 'string'
|
|
},
|
|
id: {
|
|
type: 'string'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/<%=index%>/<%=type%>',
|
|
req: {
|
|
index: {
|
|
type: 'string'
|
|
},
|
|
type: {
|
|
type: 'string'
|
|
}
|
|
}
|
|
}
|
|
],
|
|
method: 'POST'
|
|
});
|
|
|
|
|
|
api.indices = function IndicesNS(transport) {
|
|
this.transport = transport;
|
|
};
|
|
|
|
/**
|
|
* Perform a [indices.analyze](http://www.elasticsearch.org/guide/reference/api/admin-indices-analyze/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {String} params.analyzer - The name of the analyzer to use
|
|
* @param {String} params.field - Use the analyzer configured for this field (instead of passing the analyzer name)
|
|
* @param {String or String[] or Boolean} params.filters - A comma-separated list of filters to use for the analysis
|
|
* @param {String} params.index - The name of the index to scope the operation
|
|
* @param {Boolean} params.preferLocal - With `true`, specify that a local shard should be used if available, with `false`, use a random shard (default: true)
|
|
* @param {String} params.text - The text on which the analysis should be performed (when request body is not used)
|
|
* @param {String} params.tokenizer - The name of the tokenizer to use for the analysis
|
|
* @param {String} [params.format=detailed] - Format of the output
|
|
*/
|
|
api.indices.prototype.analyze = ca({
|
|
params: {
|
|
analyzer: {
|
|
type: 'string'
|
|
},
|
|
field: {
|
|
type: 'string'
|
|
},
|
|
filters: {
|
|
type: 'list'
|
|
},
|
|
index: {
|
|
type: 'string'
|
|
},
|
|
preferLocal: {
|
|
type: 'boolean',
|
|
name: 'prefer_local'
|
|
},
|
|
text: {
|
|
type: 'string'
|
|
},
|
|
tokenizer: {
|
|
type: 'string'
|
|
},
|
|
format: {
|
|
type: 'enum',
|
|
'default': 'detailed',
|
|
options: [
|
|
'detailed',
|
|
'text'
|
|
]
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>/_analyze',
|
|
req: {
|
|
index: {
|
|
type: 'string'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/_analyze'
|
|
}
|
|
],
|
|
method: 'POST'
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [indices.clearCache](http://www.elasticsearch.org/guide/reference/api/admin-indices-clearcache/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {Boolean} params.fieldData - Clear field data
|
|
* @param {Boolean} params.fielddata - Clear field data
|
|
* @param {String or String[] or Boolean} params.fields - A comma-separated list of fields to clear when using the `field_data` parameter (default: all)
|
|
* @param {Boolean} params.filter - Clear filter caches
|
|
* @param {Boolean} params.filterCache - Clear filter caches
|
|
* @param {Boolean} params.filterKeys - A comma-separated list of keys to clear when using the `filter_cache` parameter (default: all)
|
|
* @param {Boolean} params.id - Clear ID caches for parent/child
|
|
* @param {Boolean} params.idCache - Clear ID caches for parent/child
|
|
* @param {String} [params.ignoreIndices=none] - When performed on multiple indices, allows to ignore `missing` ones
|
|
* @param {String or String[] or Boolean} params.index - A comma-separated list of index name to limit the operation
|
|
* @param {Boolean} params.recycler - Clear the recycler cache
|
|
*/
|
|
api.indices.prototype.clearCache = ca({
|
|
params: {
|
|
fieldData: {
|
|
type: 'boolean',
|
|
name: 'field_data'
|
|
},
|
|
fielddata: {
|
|
type: 'boolean'
|
|
},
|
|
fields: {
|
|
type: 'list'
|
|
},
|
|
filter: {
|
|
type: 'boolean'
|
|
},
|
|
filterCache: {
|
|
type: 'boolean',
|
|
name: 'filter_cache'
|
|
},
|
|
filterKeys: {
|
|
type: 'boolean',
|
|
name: 'filter_keys'
|
|
},
|
|
id: {
|
|
type: 'boolean'
|
|
},
|
|
idCache: {
|
|
type: 'boolean',
|
|
name: 'id_cache'
|
|
},
|
|
ignoreIndices: {
|
|
type: 'enum',
|
|
'default': 'none',
|
|
options: [
|
|
'none',
|
|
'missing'
|
|
],
|
|
name: 'ignore_indices'
|
|
},
|
|
index: {
|
|
type: 'list'
|
|
},
|
|
recycler: {
|
|
type: 'boolean'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>/_cache/clear',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/_cache/clear'
|
|
}
|
|
],
|
|
method: 'POST'
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [indices.close](http://www.elasticsearch.org/guide/reference/api/admin-indices-open-close/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {Date or Number} params.timeout - Explicit operation timeout
|
|
* @param {Date or Number} params.masterTimeout - Specify timeout for connection to master
|
|
* @param {String} params.index - The name of the index
|
|
*/
|
|
api.indices.prototype.close = ca({
|
|
params: {
|
|
timeout: {
|
|
type: 'time'
|
|
},
|
|
masterTimeout: {
|
|
type: 'time',
|
|
name: 'master_timeout'
|
|
}
|
|
},
|
|
url: {
|
|
fmt: '/<%=index%>/_close',
|
|
req: {
|
|
index: {
|
|
type: 'string'
|
|
}
|
|
},
|
|
sortOrder: -1
|
|
},
|
|
method: 'POST'
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [indices.create](http://www.elasticsearch.org/guide/reference/api/admin-indices-create-index/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {Date or Number} params.timeout - Explicit operation timeout
|
|
* @param {Date or Number} params.masterTimeout - Specify timeout for connection to master
|
|
* @param {String} params.index - The name of the index
|
|
*/
|
|
api.indices.prototype.create = ca({
|
|
params: {
|
|
timeout: {
|
|
type: 'time'
|
|
},
|
|
masterTimeout: {
|
|
type: 'time',
|
|
name: 'master_timeout'
|
|
}
|
|
},
|
|
url: {
|
|
fmt: '/<%=index%>',
|
|
req: {
|
|
index: {
|
|
type: 'string'
|
|
}
|
|
},
|
|
sortOrder: -1
|
|
},
|
|
method: 'POST'
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [indices.delete](http://www.elasticsearch.org/guide/reference/api/admin-indices-delete-index/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {Date or Number} params.timeout - Explicit operation timeout
|
|
* @param {Date or Number} params.masterTimeout - Specify timeout for connection to master
|
|
* @param {String or String[] or Boolean} params.index - A comma-separated list of indices to delete; use `_all` or empty string to delete all indices
|
|
*/
|
|
api.indices.prototype['delete'] = ca({
|
|
params: {
|
|
timeout: {
|
|
type: 'time'
|
|
},
|
|
masterTimeout: {
|
|
type: 'time',
|
|
name: 'master_timeout'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/'
|
|
}
|
|
],
|
|
method: 'DELETE'
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [indices.deleteAlias](http://www.elasticsearch.org/guide/reference/api/admin-indices-aliases/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {Date or Number} params.timeout - Explicit timestamp for the document
|
|
* @param {Date or Number} params.masterTimeout - Specify timeout for connection to master
|
|
* @param {String} params.index - The name of the index with an alias
|
|
* @param {String} params.name - The name of the alias to be deleted
|
|
*/
|
|
api.indices.prototype.deleteAlias = ca({
|
|
params: {
|
|
timeout: {
|
|
type: 'time'
|
|
},
|
|
masterTimeout: {
|
|
type: 'time',
|
|
name: 'master_timeout'
|
|
}
|
|
},
|
|
url: {
|
|
fmt: '/<%=index%>/_alias/<%=name%>',
|
|
req: {
|
|
index: {
|
|
type: 'string'
|
|
},
|
|
name: {
|
|
type: 'string'
|
|
}
|
|
},
|
|
sortOrder: -2
|
|
},
|
|
method: 'DELETE'
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [indices.deleteMapping](http://www.elasticsearch.org/guide/reference/api/admin-indices-delete-mapping/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {Date or Number} params.masterTimeout - Specify timeout for connection to master
|
|
* @param {String or String[] or Boolean} params.index - A comma-separated list of index names; use `_all` for all indices
|
|
* @param {String} params.type - The name of the document type to delete
|
|
*/
|
|
api.indices.prototype.deleteMapping = ca({
|
|
params: {
|
|
masterTimeout: {
|
|
type: 'time',
|
|
name: 'master_timeout'
|
|
}
|
|
},
|
|
url: {
|
|
fmt: '/<%=index%>/<%=type%>',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
},
|
|
type: {
|
|
type: 'string'
|
|
}
|
|
},
|
|
sortOrder: -2
|
|
},
|
|
method: 'DELETE'
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [indices.deleteTemplate](http://www.elasticsearch.org/guide/reference/api/admin-indices-templates/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {Date or Number} params.timeout - Explicit operation timeout
|
|
* @param {Date or Number} params.masterTimeout - Specify timeout for connection to master
|
|
* @param {String} params.name - The name of the template
|
|
*/
|
|
api.indices.prototype.deleteTemplate = ca({
|
|
params: {
|
|
timeout: {
|
|
type: 'time'
|
|
},
|
|
masterTimeout: {
|
|
type: 'time',
|
|
name: 'master_timeout'
|
|
}
|
|
},
|
|
url: {
|
|
fmt: '/_template/<%=name%>',
|
|
req: {
|
|
name: {
|
|
type: 'string'
|
|
}
|
|
},
|
|
sortOrder: -1
|
|
},
|
|
method: 'DELETE'
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [indices.deleteWarmer](http://www.elasticsearch.org/guide/reference/api/admin-indices-warmers/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {Date or Number} params.masterTimeout - Specify timeout for connection to master
|
|
* @param {String or String[] or Boolean} params.index - A comma-separated list of index names to register warmer for; use `_all` or empty string to perform the operation on all indices
|
|
* @param {String} params.name - The name of the warmer (supports wildcards); leave empty to delete all warmers
|
|
* @param {String or String[] or Boolean} params.type - A comma-separated list of document types to register warmer for; use `_all` or empty string to perform the operation on all types
|
|
*/
|
|
api.indices.prototype.deleteWarmer = ca({
|
|
params: {
|
|
masterTimeout: {
|
|
type: 'time',
|
|
name: 'master_timeout'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>/<%=type%>/_warmer/<%=name%>',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
},
|
|
type: {
|
|
type: 'list'
|
|
},
|
|
name: {
|
|
type: 'string'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/<%=index%>/_warmer/<%=name%>',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
},
|
|
name: {
|
|
type: 'string'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/<%=index%>/_warmer',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
}
|
|
}
|
|
}
|
|
],
|
|
method: 'DELETE'
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [indices.exists](http://www.elasticsearch.org/guide/reference/api/admin-indices-indices-exists/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {String or String[] or Boolean} params.index - A comma-separated list of indices to check
|
|
*/
|
|
api.indices.prototype.exists = ca({
|
|
url: {
|
|
fmt: '/<%=index%>',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
}
|
|
},
|
|
sortOrder: -1
|
|
},
|
|
castExists: true,
|
|
method: 'HEAD'
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [indices.existsAlias](http://www.elasticsearch.org/guide/reference/api/admin-indices-aliases/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {String} [params.ignoreIndices=none] - When performed on multiple indices, allows to ignore `missing` ones
|
|
* @param {String or String[] or Boolean} params.index - A comma-separated list of index names to filter aliases
|
|
* @param {String or String[] or Boolean} params.name - A comma-separated list of alias names to return
|
|
*/
|
|
api.indices.prototype.existsAlias = ca({
|
|
params: {
|
|
ignoreIndices: {
|
|
type: 'enum',
|
|
'default': 'none',
|
|
options: [
|
|
'none',
|
|
'missing'
|
|
],
|
|
name: 'ignore_indices'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>/_alias/<%=name%>',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
},
|
|
name: {
|
|
type: 'list'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/_alias/<%=name%>',
|
|
req: {
|
|
name: {
|
|
type: 'list'
|
|
}
|
|
}
|
|
}
|
|
],
|
|
castExists: true,
|
|
method: 'HEAD'
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [indices.existsType](http://www.elasticsearch.org/guide/reference/api/admin-indices-types-exists/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {String} [params.ignoreIndices=none] - When performed on multiple indices, allows to ignore `missing` ones
|
|
* @param {String or String[] or Boolean} params.index - A comma-separated list of index names; use `_all` to check the types across all indices
|
|
* @param {String or String[] or Boolean} params.type - A comma-separated list of document types to check
|
|
*/
|
|
api.indices.prototype.existsType = ca({
|
|
params: {
|
|
ignoreIndices: {
|
|
type: 'enum',
|
|
'default': 'none',
|
|
options: [
|
|
'none',
|
|
'missing'
|
|
],
|
|
name: 'ignore_indices'
|
|
}
|
|
},
|
|
url: {
|
|
fmt: '/<%=index%>/<%=type%>',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
},
|
|
type: {
|
|
type: 'list'
|
|
}
|
|
},
|
|
sortOrder: -2
|
|
},
|
|
castExists: true,
|
|
method: 'HEAD'
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [indices.flush](http://www.elasticsearch.org/guide/reference/api/admin-indices-flush/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {Boolean} params.force - TODO: ?
|
|
* @param {Boolean} params.full - TODO: ?
|
|
* @param {String} [params.ignoreIndices=none] - When performed on multiple indices, allows to ignore `missing` ones
|
|
* @param {Boolean} params.refresh - Refresh the index after performing the operation
|
|
* @param {String or String[] or Boolean} params.index - A comma-separated list of index names; use `_all` or empty string for all indices
|
|
*/
|
|
api.indices.prototype.flush = ca({
|
|
params: {
|
|
force: {
|
|
type: 'boolean'
|
|
},
|
|
full: {
|
|
type: 'boolean'
|
|
},
|
|
ignoreIndices: {
|
|
type: 'enum',
|
|
'default': 'none',
|
|
options: [
|
|
'none',
|
|
'missing'
|
|
],
|
|
name: 'ignore_indices'
|
|
},
|
|
refresh: {
|
|
type: 'boolean'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>/_flush',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/_flush'
|
|
}
|
|
],
|
|
method: 'POST'
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [indices.getAlias](http://www.elasticsearch.org/guide/reference/api/admin-indices-aliases/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {String} [params.ignoreIndices=none] - When performed on multiple indices, allows to ignore `missing` ones
|
|
* @param {String or String[] or Boolean} params.index - A comma-separated list of index names to filter aliases
|
|
* @param {String or String[] or Boolean} params.name - A comma-separated list of alias names to return
|
|
*/
|
|
api.indices.prototype.getAlias = ca({
|
|
params: {
|
|
ignoreIndices: {
|
|
type: 'enum',
|
|
'default': 'none',
|
|
options: [
|
|
'none',
|
|
'missing'
|
|
],
|
|
name: 'ignore_indices'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>/_alias/<%=name%>',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
},
|
|
name: {
|
|
type: 'list'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/_alias/<%=name%>',
|
|
req: {
|
|
name: {
|
|
type: 'list'
|
|
}
|
|
}
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [indices.getAliases](http://www.elasticsearch.org/guide/reference/api/admin-indices-aliases/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {Date or Number} params.timeout - Explicit operation timeout
|
|
* @param {String or String[] or Boolean} params.index - A comma-separated list of index names to filter aliases
|
|
*/
|
|
api.indices.prototype.getAliases = ca({
|
|
params: {
|
|
timeout: {
|
|
type: 'time'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>/_aliases',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/_aliases'
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [indices.getFieldMapping](http://www.elasticsearch.org/guide/en/elasticsearch/reference/current/indices-get-field-mapping.html) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {Boolean} params.includeDefaults - Whether the default mapping values should be returned as well
|
|
* @param {String or String[] or Boolean} params.index - A comma-separated list of index names
|
|
* @param {String or String[] or Boolean} params.type - A comma-separated list of document types
|
|
* @param {String or String[] or Boolean} params.field - A comma-separated list of fields
|
|
*/
|
|
api.indices.prototype.getFieldMapping = ca({
|
|
params: {
|
|
includeDefaults: {
|
|
type: 'boolean',
|
|
name: 'include_defaults'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>/<%=type%>/_mapping/field/<%=field%>',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
},
|
|
type: {
|
|
type: 'list'
|
|
},
|
|
field: {
|
|
type: 'list'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/<%=index%>/_mapping/field/<%=field%>',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
},
|
|
field: {
|
|
type: 'list'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/_mapping/field/<%=field%>',
|
|
req: {
|
|
field: {
|
|
type: 'list'
|
|
}
|
|
}
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [indices.getMapping](http://www.elasticsearch.org/guide/reference/api/admin-indices-get-mapping/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {String or String[] or Boolean} params.index - A comma-separated list of index names
|
|
* @param {String or String[] or Boolean} params.type - A comma-separated list of document types
|
|
*/
|
|
api.indices.prototype.getMapping = ca({
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>/<%=type%>/_mapping',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
},
|
|
type: {
|
|
type: 'list'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/<%=index%>/_mapping',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/_mapping'
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [indices.getSettings](http://www.elasticsearch.org/guide/reference/api/admin-indices-get-settings/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {String or String[] or Boolean} params.index - A comma-separated list of index names; use `_all` or empty string to perform the operation on all indices
|
|
*/
|
|
api.indices.prototype.getSettings = ca({
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>/_settings',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/_settings'
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [indices.getTemplate](http://www.elasticsearch.org/guide/reference/api/admin-indices-templates/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {String} params.name - The name of the template
|
|
*/
|
|
api.indices.prototype.getTemplate = ca({
|
|
urls: [
|
|
{
|
|
fmt: '/_template/<%=name%>',
|
|
req: {
|
|
name: {
|
|
type: 'string'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/_template'
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [indices.getWarmer](http://www.elasticsearch.org/guide/reference/api/admin-indices-warmers/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {String or String[] or Boolean} params.index - A comma-separated list of index names to restrict the operation; use `_all` to perform the operation on all indices
|
|
* @param {String} params.name - The name of the warmer (supports wildcards); leave empty to get all warmers
|
|
* @param {String or String[] or Boolean} params.type - A comma-separated list of document types to restrict the operation; leave empty to perform the operation on all types
|
|
*/
|
|
api.indices.prototype.getWarmer = ca({
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>/<%=type%>/_warmer/<%=name%>',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
},
|
|
type: {
|
|
type: 'list'
|
|
},
|
|
name: {
|
|
type: 'string'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/<%=index%>/_warmer/<%=name%>',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
},
|
|
name: {
|
|
type: 'string'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/<%=index%>/_warmer',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
}
|
|
}
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [indices.open](http://www.elasticsearch.org/guide/reference/api/admin-indices-open-close/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {Date or Number} params.timeout - Explicit operation timeout
|
|
* @param {Date or Number} params.masterTimeout - Specify timeout for connection to master
|
|
* @param {String} params.index - The name of the index
|
|
*/
|
|
api.indices.prototype.open = ca({
|
|
params: {
|
|
timeout: {
|
|
type: 'time'
|
|
},
|
|
masterTimeout: {
|
|
type: 'time',
|
|
name: 'master_timeout'
|
|
}
|
|
},
|
|
url: {
|
|
fmt: '/<%=index%>/_open',
|
|
req: {
|
|
index: {
|
|
type: 'string'
|
|
}
|
|
},
|
|
sortOrder: -1
|
|
},
|
|
method: 'POST'
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [indices.optimize](http://www.elasticsearch.org/guide/reference/api/admin-indices-optimize/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {Boolean} params.flush - Specify whether the index should be flushed after performing the operation (default: true)
|
|
* @param {String} [params.ignoreIndices=none] - When performed on multiple indices, allows to ignore `missing` ones
|
|
* @param {Number} params.maxNumSegments - The number of segments the index should be merged into (default: dynamic)
|
|
* @param {Boolean} params.onlyExpungeDeletes - Specify whether the operation should only expunge deleted documents
|
|
* @param {*} params.operationThreading - TODO: ?
|
|
* @param {Boolean} params.refresh - Specify whether the index should be refreshed after performing the operation (default: true)
|
|
* @param {Boolean} params.waitForMerge - Specify whether the request should block until the merge process is finished (default: true)
|
|
* @param {String or String[] or Boolean} params.index - A comma-separated list of index names; use `_all` or empty string to perform the operation on all indices
|
|
*/
|
|
api.indices.prototype.optimize = ca({
|
|
params: {
|
|
flush: {
|
|
type: 'boolean'
|
|
},
|
|
ignoreIndices: {
|
|
type: 'enum',
|
|
'default': 'none',
|
|
options: [
|
|
'none',
|
|
'missing'
|
|
],
|
|
name: 'ignore_indices'
|
|
},
|
|
maxNumSegments: {
|
|
type: 'number',
|
|
name: 'max_num_segments'
|
|
},
|
|
onlyExpungeDeletes: {
|
|
type: 'boolean',
|
|
name: 'only_expunge_deletes'
|
|
},
|
|
operationThreading: {
|
|
name: 'operation_threading'
|
|
},
|
|
refresh: {
|
|
type: 'boolean'
|
|
},
|
|
waitForMerge: {
|
|
type: 'boolean',
|
|
name: 'wait_for_merge'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>/_optimize',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/_optimize'
|
|
}
|
|
],
|
|
method: 'POST'
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [indices.putAlias](http://www.elasticsearch.org/guide/reference/api/admin-indices-aliases/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {Date or Number} params.timeout - Explicit timestamp for the document
|
|
* @param {Date or Number} params.masterTimeout - Specify timeout for connection to master
|
|
* @param {String} params.index - The name of the index with an alias
|
|
* @param {String} params.name - The name of the alias to be created or updated
|
|
*/
|
|
api.indices.prototype.putAlias = ca({
|
|
params: {
|
|
timeout: {
|
|
type: 'time'
|
|
},
|
|
masterTimeout: {
|
|
type: 'time',
|
|
name: 'master_timeout'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>/_alias/<%=name%>',
|
|
req: {
|
|
index: {
|
|
type: 'string'
|
|
},
|
|
name: {
|
|
type: 'string'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/_alias/<%=name%>',
|
|
req: {
|
|
name: {
|
|
type: 'string'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/<%=index%>/_alias',
|
|
req: {
|
|
index: {
|
|
type: 'string'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/_alias'
|
|
}
|
|
],
|
|
method: 'PUT'
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [indices.putMapping](http://www.elasticsearch.org/guide/reference/api/admin-indices-put-mapping/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {Boolean} params.ignoreConflicts - Specify whether to ignore conflicts while updating the mapping (default: false)
|
|
* @param {Date or Number} params.timeout - Explicit operation timeout
|
|
* @param {Date or Number} params.masterTimeout - Specify timeout for connection to master
|
|
* @param {String or String[] or Boolean} params.index - A comma-separated list of index names; use `_all` to perform the operation on all indices
|
|
* @param {String} params.type - The name of the document type
|
|
*/
|
|
api.indices.prototype.putMapping = ca({
|
|
params: {
|
|
ignoreConflicts: {
|
|
type: 'boolean',
|
|
name: 'ignore_conflicts'
|
|
},
|
|
timeout: {
|
|
type: 'time'
|
|
},
|
|
masterTimeout: {
|
|
type: 'time',
|
|
name: 'master_timeout'
|
|
}
|
|
},
|
|
url: {
|
|
fmt: '/<%=index%>/<%=type%>/_mapping',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
},
|
|
type: {
|
|
type: 'string'
|
|
}
|
|
},
|
|
sortOrder: -2
|
|
},
|
|
method: 'POST'
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [indices.putSettings](http://www.elasticsearch.org/guide/reference/api/admin-indices-update-settings/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {Date or Number} params.masterTimeout - Specify timeout for connection to master
|
|
* @param {String or String[] or Boolean} params.index - A comma-separated list of index names; use `_all` or empty string to perform the operation on all indices
|
|
*/
|
|
api.indices.prototype.putSettings = ca({
|
|
params: {
|
|
masterTimeout: {
|
|
type: 'time',
|
|
name: 'master_timeout'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>/_settings',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/_settings'
|
|
}
|
|
],
|
|
method: 'PUT'
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [indices.putTemplate](http://www.elasticsearch.org/guide/reference/api/admin-indices-templates/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {Number} params.order - The order for this template when merging multiple matching ones (higher numbers are merged later, overriding the lower numbers)
|
|
* @param {Date or Number} params.timeout - Explicit operation timeout
|
|
* @param {Date or Number} params.masterTimeout - Specify timeout for connection to master
|
|
* @param {String} params.name - The name of the template
|
|
*/
|
|
api.indices.prototype.putTemplate = ca({
|
|
params: {
|
|
order: {
|
|
type: 'number'
|
|
},
|
|
timeout: {
|
|
type: 'time'
|
|
},
|
|
masterTimeout: {
|
|
type: 'time',
|
|
name: 'master_timeout'
|
|
}
|
|
},
|
|
url: {
|
|
fmt: '/_template/<%=name%>',
|
|
req: {
|
|
name: {
|
|
type: 'string'
|
|
}
|
|
},
|
|
sortOrder: -1
|
|
},
|
|
method: 'POST'
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [indices.putWarmer](http://www.elasticsearch.org/guide/reference/api/admin-indices-warmers/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {Date or Number} params.masterTimeout - Specify timeout for connection to master
|
|
* @param {String or String[] or Boolean} params.index - A comma-separated list of index names to register the warmer for; use `_all` or empty string to perform the operation on all indices
|
|
* @param {String} params.name - The name of the warmer
|
|
* @param {String or String[] or Boolean} params.type - A comma-separated list of document types to register the warmer for; leave empty to perform the operation on all types
|
|
*/
|
|
api.indices.prototype.putWarmer = ca({
|
|
params: {
|
|
masterTimeout: {
|
|
type: 'time',
|
|
name: 'master_timeout'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>/<%=type%>/_warmer/<%=name%>',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
},
|
|
type: {
|
|
type: 'list'
|
|
},
|
|
name: {
|
|
type: 'string'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/<%=index%>/_warmer/<%=name%>',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
},
|
|
name: {
|
|
type: 'string'
|
|
}
|
|
}
|
|
}
|
|
],
|
|
method: 'PUT'
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [indices.refresh](http://www.elasticsearch.org/guide/reference/api/admin-indices-refresh/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {String} [params.ignoreIndices=none] - When performed on multiple indices, allows to ignore `missing` ones
|
|
* @param {*} params.operationThreading - TODO: ?
|
|
* @param {String or String[] or Boolean} params.index - A comma-separated list of index names; use `_all` or empty string to perform the operation on all indices
|
|
*/
|
|
api.indices.prototype.refresh = ca({
|
|
params: {
|
|
ignoreIndices: {
|
|
type: 'enum',
|
|
'default': 'none',
|
|
options: [
|
|
'none',
|
|
'missing'
|
|
],
|
|
name: 'ignore_indices'
|
|
},
|
|
operationThreading: {
|
|
name: 'operation_threading'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>/_refresh',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/_refresh'
|
|
}
|
|
],
|
|
method: 'POST'
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [indices.segments](http://elasticsearch.org/guide/reference/api/admin-indices-segments/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {String} [params.ignoreIndices=none] - When performed on multiple indices, allows to ignore `missing` ones
|
|
* @param {*} params.operationThreading - TODO: ?
|
|
* @param {String or String[] or Boolean} params.index - A comma-separated list of index names; use `_all` or empty string to perform the operation on all indices
|
|
*/
|
|
api.indices.prototype.segments = ca({
|
|
params: {
|
|
ignoreIndices: {
|
|
type: 'enum',
|
|
'default': 'none',
|
|
options: [
|
|
'none',
|
|
'missing'
|
|
],
|
|
name: 'ignore_indices'
|
|
},
|
|
operationThreading: {
|
|
name: 'operation_threading'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>/_segments',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/_segments'
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [indices.snapshotIndex](http://www.elasticsearch.org/guide/reference/api/admin-indices-gateway-snapshot/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {String} [params.ignoreIndices=none] - When performed on multiple indices, allows to ignore `missing` ones
|
|
* @param {String or String[] or Boolean} params.index - A comma-separated list of index names; use `_all` or empty string for all indices
|
|
*/
|
|
api.indices.prototype.snapshotIndex = ca({
|
|
params: {
|
|
ignoreIndices: {
|
|
type: 'enum',
|
|
'default': 'none',
|
|
options: [
|
|
'none',
|
|
'missing'
|
|
],
|
|
name: 'ignore_indices'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>/_gateway/snapshot',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/_gateway/snapshot'
|
|
}
|
|
],
|
|
method: 'POST'
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [indices.stats](http://elasticsearch.org/guide/reference/api/admin-indices-stats/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {Boolean} params.all - Return all available information
|
|
* @param {Boolean} params.clear - Reset the default level of detail
|
|
* @param {Boolean} params.completion - Return information about completion suggester stats
|
|
* @param {String or String[] or Boolean} params.completionFields - A comma-separated list of fields for `completion` metric (supports wildcards)
|
|
* @param {Boolean} params.docs - Return information about indexed and deleted documents
|
|
* @param {Boolean} params.fielddata - Return information about field data
|
|
* @param {String or String[] or Boolean} params.fielddataFields - A comma-separated list of fields for `fielddata` metric (supports wildcards)
|
|
* @param {String or String[] or Boolean} params.fields - A comma-separated list of fields to return detailed information for, when returning the `search` statistics
|
|
* @param {Boolean} params.filterCache - Return information about filter cache
|
|
* @param {Boolean} params.flush - Return information about flush operations
|
|
* @param {Boolean} params.get - Return information about get operations
|
|
* @param {Boolean} params.groups - A comma-separated list of search groups for `search` statistics
|
|
* @param {Boolean} params.idCache - Return information about ID cache
|
|
* @param {String} [params.ignoreIndices=none] - When performed on multiple indices, allows to ignore `missing` ones
|
|
* @param {Boolean} params.indexing - Return information about indexing operations
|
|
* @param {Boolean} params.merge - Return information about merge operations
|
|
* @param {Boolean} params.refresh - Return information about refresh operations
|
|
* @param {Boolean} params.search - Return information about search operations; use the `groups` parameter to include information for specific search groups
|
|
* @param {Boolean} params.store - Return information about the size of the index
|
|
* @param {Boolean} params.warmer - Return information about warmers
|
|
* @param {String or String[] or Boolean} params.index - A comma-separated list of index names; use `_all` or empty string to perform the operation on all indices
|
|
* @param {String or String[] or Boolean} params.indexingTypes - A comma-separated list of document types to include in the `indexing` statistics
|
|
* @param {String} params.metricFamily - Limit the information returned to a specific metric
|
|
* @param {String or String[] or Boolean} params.searchGroups - A comma-separated list of search groups to include in the `search` statistics
|
|
*/
|
|
api.indices.prototype.stats = ca({
|
|
params: {
|
|
all: {
|
|
type: 'boolean'
|
|
},
|
|
clear: {
|
|
type: 'boolean'
|
|
},
|
|
completion: {
|
|
type: 'boolean'
|
|
},
|
|
completionFields: {
|
|
type: 'list',
|
|
name: 'completion_fields'
|
|
},
|
|
docs: {
|
|
type: 'boolean'
|
|
},
|
|
fielddata: {
|
|
type: 'boolean'
|
|
},
|
|
fielddataFields: {
|
|
type: 'list',
|
|
name: 'fielddata_fields'
|
|
},
|
|
fields: {
|
|
type: 'list'
|
|
},
|
|
filterCache: {
|
|
type: 'boolean',
|
|
name: 'filter_cache'
|
|
},
|
|
flush: {
|
|
type: 'boolean'
|
|
},
|
|
get: {
|
|
type: 'boolean'
|
|
},
|
|
groups: {
|
|
type: 'boolean'
|
|
},
|
|
idCache: {
|
|
type: 'boolean',
|
|
name: 'id_cache'
|
|
},
|
|
ignoreIndices: {
|
|
type: 'enum',
|
|
'default': 'none',
|
|
options: [
|
|
'none',
|
|
'missing'
|
|
],
|
|
name: 'ignore_indices'
|
|
},
|
|
indexing: {
|
|
type: 'boolean'
|
|
},
|
|
merge: {
|
|
type: 'boolean'
|
|
},
|
|
refresh: {
|
|
type: 'boolean'
|
|
},
|
|
search: {
|
|
type: 'boolean'
|
|
},
|
|
store: {
|
|
type: 'boolean'
|
|
},
|
|
warmer: {
|
|
type: 'boolean'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>/_stats',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/_stats'
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [indices.status](http://elasticsearch.org/guide/reference/api/admin-indices-status/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {String} [params.ignoreIndices=none] - When performed on multiple indices, allows to ignore `missing` ones
|
|
* @param {*} params.operationThreading - TODO: ?
|
|
* @param {Boolean} params.recovery - Return information about shard recovery
|
|
* @param {Boolean} params.snapshot - TODO: ?
|
|
* @param {String or String[] or Boolean} params.index - A comma-separated list of index names; use `_all` or empty string to perform the operation on all indices
|
|
*/
|
|
api.indices.prototype.status = ca({
|
|
params: {
|
|
ignoreIndices: {
|
|
type: 'enum',
|
|
'default': 'none',
|
|
options: [
|
|
'none',
|
|
'missing'
|
|
],
|
|
name: 'ignore_indices'
|
|
},
|
|
operationThreading: {
|
|
name: 'operation_threading'
|
|
},
|
|
recovery: {
|
|
type: 'boolean'
|
|
},
|
|
snapshot: {
|
|
type: 'boolean'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>/_status',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/_status'
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [indices.updateAliases](http://www.elasticsearch.org/guide/reference/api/admin-indices-aliases/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {Date or Number} params.timeout - Request timeout
|
|
* @param {Date or Number} params.masterTimeout - Specify timeout for connection to master
|
|
* @param {String or String[] or Boolean} params.index - A comma-separated list of index names to filter aliases
|
|
*/
|
|
api.indices.prototype.updateAliases = ca({
|
|
params: {
|
|
timeout: {
|
|
type: 'time'
|
|
},
|
|
masterTimeout: {
|
|
type: 'time',
|
|
name: 'master_timeout'
|
|
}
|
|
},
|
|
url: {
|
|
fmt: '/_aliases'
|
|
},
|
|
method: 'POST'
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [indices.validateQuery](http://www.elasticsearch.org/guide/reference/api/validate/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {Boolean} params.explain - Return detailed information about the error
|
|
* @param {String} [params.ignoreIndices=none] - When performed on multiple indices, allows to ignore `missing` ones
|
|
* @param {*} params.operationThreading - TODO: ?
|
|
* @param {String} params.source - The URL-encoded query definition (instead of using the request body)
|
|
* @param {String} params.q - Query in the Lucene query string syntax
|
|
* @param {String or String[] or Boolean} params.index - A comma-separated list of index names to restrict the operation; use `_all` or empty string to perform the operation on all indices
|
|
* @param {String or String[] or Boolean} params.type - A comma-separated list of document types to restrict the operation; leave empty to perform the operation on all types
|
|
*/
|
|
api.indices.prototype.validateQuery = ca({
|
|
params: {
|
|
explain: {
|
|
type: 'boolean'
|
|
},
|
|
ignoreIndices: {
|
|
type: 'enum',
|
|
'default': 'none',
|
|
options: [
|
|
'none',
|
|
'missing'
|
|
],
|
|
name: 'ignore_indices'
|
|
},
|
|
operationThreading: {
|
|
name: 'operation_threading'
|
|
},
|
|
source: {
|
|
type: 'string'
|
|
},
|
|
q: {
|
|
type: 'string'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>/<%=type%>/_validate/query',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
},
|
|
type: {
|
|
type: 'list'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/<%=index%>/_validate/query',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/_validate/query'
|
|
}
|
|
],
|
|
method: 'POST'
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [info](http://elasticsearch.org/guide/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
*/
|
|
api.info = ca({
|
|
url: {
|
|
fmt: '/'
|
|
}
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [mget](http://elasticsearch.org/guide/reference/api/multi-get/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {String or String[] or Boolean} params.fields - A comma-separated list of fields to return in the response
|
|
* @param {String} params.preference - Specify the node or shard the operation should be performed on (default: random)
|
|
* @param {Boolean} params.realtime - Specify whether to perform the operation in realtime or search mode
|
|
* @param {Boolean} params.refresh - Refresh the shard containing the document before performing the operation
|
|
* @param {String or String[] or Boolean} params.source - True or false to return the _source field or not, or a list of fields to return
|
|
* @param {String or String[] or Boolean} params.sourceExclude - A list of fields to exclude from the returned _source field
|
|
* @param {String or String[] or Boolean} params.sourceInclude - A list of fields to extract and return from the _source field
|
|
* @param {String} params.index - The name of the index
|
|
* @param {String} params.type - The type of the document
|
|
*/
|
|
api.mget = ca({
|
|
params: {
|
|
fields: {
|
|
type: 'list'
|
|
},
|
|
preference: {
|
|
type: 'string'
|
|
},
|
|
realtime: {
|
|
type: 'boolean'
|
|
},
|
|
refresh: {
|
|
type: 'boolean'
|
|
},
|
|
source: {
|
|
type: 'list',
|
|
name: '_source'
|
|
},
|
|
sourceExclude: {
|
|
type: 'list',
|
|
name: '_source_exclude'
|
|
},
|
|
sourceInclude: {
|
|
type: 'list',
|
|
name: '_source_include'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>/<%=type%>/_mget',
|
|
req: {
|
|
index: {
|
|
type: 'string'
|
|
},
|
|
type: {
|
|
type: 'string'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/<%=index%>/_mget',
|
|
req: {
|
|
index: {
|
|
type: 'string'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/_mget'
|
|
}
|
|
],
|
|
method: 'POST'
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [mlt](http://elasticsearch.org/guide/reference/api/more-like-this/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {Number} params.boostTerms - The boost factor
|
|
* @param {Number} params.maxDocFreq - The word occurrence frequency as count: words with higher occurrence in the corpus will be ignored
|
|
* @param {Number} params.maxQueryTerms - The maximum query terms to be included in the generated query
|
|
* @param {Number} params.maxWordLen - The minimum length of the word: longer words will be ignored
|
|
* @param {Number} params.minDocFreq - The word occurrence frequency as count: words with lower occurrence in the corpus will be ignored
|
|
* @param {Number} params.minTermFreq - The term frequency as percent: terms with lower occurence in the source document will be ignored
|
|
* @param {Number} params.minWordLen - The minimum length of the word: shorter words will be ignored
|
|
* @param {String or String[] or Boolean} params.mltFields - Specific fields to perform the query against
|
|
* @param {Number} params.percentTermsToMatch - How many terms have to match in order to consider the document a match (default: 0.3)
|
|
* @param {String} params.routing - Specific routing value
|
|
* @param {Number} params.searchFrom - The offset from which to return results
|
|
* @param {String or String[] or Boolean} params.searchIndices - A comma-separated list of indices to perform the query against (default: the index containing the document)
|
|
* @param {String} params.searchQueryHint - The search query hint
|
|
* @param {String} params.searchScroll - A scroll search request definition
|
|
* @param {Number} params.searchSize - The number of documents to return (default: 10)
|
|
* @param {String} params.searchSource - A specific search request definition (instead of using the request body)
|
|
* @param {String} params.searchType - Specific search type (eg. `dfs_then_fetch`, `count`, etc)
|
|
* @param {String or String[] or Boolean} params.searchTypes - A comma-separated list of types to perform the query against (default: the same type as the document)
|
|
* @param {String or String[] or Boolean} params.stopWords - A list of stop words to be ignored
|
|
* @param {String} params.id - The document ID
|
|
* @param {String} params.index - The name of the index
|
|
* @param {String} params.type - The type of the document (use `_all` to fetch the first document matching the ID across all types)
|
|
*/
|
|
api.mlt = ca({
|
|
params: {
|
|
boostTerms: {
|
|
type: 'number',
|
|
name: 'boost_terms'
|
|
},
|
|
maxDocFreq: {
|
|
type: 'number',
|
|
name: 'max_doc_freq'
|
|
},
|
|
maxQueryTerms: {
|
|
type: 'number',
|
|
name: 'max_query_terms'
|
|
},
|
|
maxWordLen: {
|
|
type: 'number',
|
|
name: 'max_word_len'
|
|
},
|
|
minDocFreq: {
|
|
type: 'number',
|
|
name: 'min_doc_freq'
|
|
},
|
|
minTermFreq: {
|
|
type: 'number',
|
|
name: 'min_term_freq'
|
|
},
|
|
minWordLen: {
|
|
type: 'number',
|
|
name: 'min_word_len'
|
|
},
|
|
mltFields: {
|
|
type: 'list',
|
|
name: 'mlt_fields'
|
|
},
|
|
percentTermsToMatch: {
|
|
type: 'number',
|
|
name: 'percent_terms_to_match'
|
|
},
|
|
routing: {
|
|
type: 'string'
|
|
},
|
|
searchFrom: {
|
|
type: 'number',
|
|
name: 'search_from'
|
|
},
|
|
searchIndices: {
|
|
type: 'list',
|
|
name: 'search_indices'
|
|
},
|
|
searchQueryHint: {
|
|
type: 'string',
|
|
name: 'search_query_hint'
|
|
},
|
|
searchScroll: {
|
|
type: 'string',
|
|
name: 'search_scroll'
|
|
},
|
|
searchSize: {
|
|
type: 'number',
|
|
name: 'search_size'
|
|
},
|
|
searchSource: {
|
|
type: 'string',
|
|
name: 'search_source'
|
|
},
|
|
searchType: {
|
|
type: 'string',
|
|
name: 'search_type'
|
|
},
|
|
searchTypes: {
|
|
type: 'list',
|
|
name: 'search_types'
|
|
},
|
|
stopWords: {
|
|
type: 'list',
|
|
name: 'stop_words'
|
|
}
|
|
},
|
|
url: {
|
|
fmt: '/<%=index%>/<%=type%>/<%=id%>/_mlt',
|
|
req: {
|
|
index: {
|
|
type: 'string'
|
|
},
|
|
type: {
|
|
type: 'string'
|
|
},
|
|
id: {
|
|
type: 'string'
|
|
}
|
|
},
|
|
sortOrder: -3
|
|
},
|
|
method: 'POST'
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [msearch](http://www.elasticsearch.org/guide/reference/api/multi-search/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {String} params.searchType - Search operation type
|
|
* @param {String or String[] or Boolean} params.index - A comma-separated list of index names to use as default
|
|
* @param {String or String[] or Boolean} params.type - A comma-separated list of document types to use as default
|
|
*/
|
|
api.msearch = ca({
|
|
params: {
|
|
searchType: {
|
|
type: 'enum',
|
|
options: [
|
|
'query_then_fetch',
|
|
'query_and_fetch',
|
|
'dfs_query_then_fetch',
|
|
'dfs_query_and_fetch',
|
|
'count',
|
|
'scan'
|
|
],
|
|
name: 'search_type'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>/<%=type%>/_msearch',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
},
|
|
type: {
|
|
type: 'list'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/<%=index%>/_msearch',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/_msearch'
|
|
}
|
|
],
|
|
bulkBody: true,
|
|
method: 'POST'
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [percolate](http://elasticsearch.org/guide/reference/api/percolate/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {Boolean} params.preferLocal - With `true`, specify that a local shard should be used if available, with `false`, use a random shard (default: true)
|
|
* @param {String} params.index - The name of the index with a registered percolator query
|
|
* @param {String} params.type - The document type
|
|
*/
|
|
api.percolate = ca({
|
|
params: {
|
|
preferLocal: {
|
|
type: 'boolean',
|
|
name: 'prefer_local'
|
|
}
|
|
},
|
|
url: {
|
|
fmt: '/<%=index%>/<%=type%>/_percolate',
|
|
req: {
|
|
index: {
|
|
type: 'string'
|
|
},
|
|
type: {
|
|
type: 'string'
|
|
}
|
|
},
|
|
sortOrder: -2
|
|
},
|
|
method: 'POST'
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [scroll](http://www.elasticsearch.org/guide/reference/api/search/scroll/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {Duration} params.scroll - Specify how long a consistent view of the index should be maintained for scrolled search
|
|
* @param {String} params.scrollId - The scroll ID
|
|
*/
|
|
api.scroll = ca({
|
|
params: {
|
|
scroll: {
|
|
type: 'duration'
|
|
},
|
|
scrollId: {
|
|
type: 'string',
|
|
name: 'scroll_id'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/_search/scroll/<%=scrollId%>',
|
|
req: {
|
|
scrollId: {
|
|
type: 'string'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/_search/scroll'
|
|
}
|
|
],
|
|
method: 'POST'
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [search](http://www.elasticsearch.org/guide/reference/api/search/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {String} params.analyzer - The analyzer to use for the query string
|
|
* @param {Boolean} params.analyzeWildcard - Specify whether wildcard and prefix queries should be analyzed (default: false)
|
|
* @param {String} [params.defaultOperator=OR] - The default operator for query string query (AND or OR)
|
|
* @param {String} params.df - The field to use as default where no field prefix is given in the query string
|
|
* @param {Boolean} params.explain - Specify whether to return detailed information about score computation as part of a hit
|
|
* @param {String or String[] or Boolean} params.fields - A comma-separated list of fields to return as part of a hit
|
|
* @param {Number} params.from - Starting offset (default: 0)
|
|
* @param {String} [params.ignoreIndices=none] - When performed on multiple indices, allows to ignore `missing` ones
|
|
* @param {String or String[] or Boolean} params.indicesBoost - Comma-separated list of index boosts
|
|
* @param {Boolean} params.lenient - Specify whether format-based query failures (such as providing text to a numeric field) should be ignored
|
|
* @param {Boolean} params.lowercaseExpandedTerms - Specify whether query terms should be lowercased
|
|
* @param {String} params.preference - Specify the node or shard the operation should be performed on (default: random)
|
|
* @param {String} params.q - Query in the Lucene query string syntax
|
|
* @param {String or String[] or Boolean} params.routing - A comma-separated list of specific routing values
|
|
* @param {Duration} params.scroll - Specify how long a consistent view of the index should be maintained for scrolled search
|
|
* @param {String} params.searchType - Search operation type
|
|
* @param {Number} params.size - Number of hits to return (default: 10)
|
|
* @param {String or String[] or Boolean} params.sort - A comma-separated list of <field>:<direction> pairs
|
|
* @param {String or String[] or Boolean} params.source - True or false to return the _source field or not, or a list of fields to return
|
|
* @param {String or String[] or Boolean} params.sourceExclude - A list of fields to exclude from the returned _source field
|
|
* @param {String or String[] or Boolean} params.sourceInclude - A list of fields to extract and return from the _source field
|
|
* @param {String or String[] or Boolean} params.stats - Specific 'tag' of the request for logging and statistical purposes
|
|
* @param {String} params.suggestField - Specify which field to use for suggestions
|
|
* @param {String} [params.suggestMode=missing] - Specify suggest mode
|
|
* @param {Number} params.suggestSize - How many suggestions to return in response
|
|
* @param {Text} params.suggestText - The source text for which the suggestions should be returned
|
|
* @param {Date or Number} params.timeout - Explicit operation timeout
|
|
* @param {Boolean} params.version - Specify whether to return document version as part of a hit
|
|
* @param {String or String[] or Boolean} [params.index=_all] - A comma-separated list of index names to search; use `_all` or empty string to perform the operation on all indices
|
|
* @param {String or String[] or Boolean} params.type - A comma-separated list of document types to search; leave empty to perform the operation on all types
|
|
*/
|
|
api.search = ca({
|
|
params: {
|
|
analyzer: {
|
|
type: 'string'
|
|
},
|
|
analyzeWildcard: {
|
|
type: 'boolean',
|
|
name: 'analyze_wildcard'
|
|
},
|
|
defaultOperator: {
|
|
type: 'enum',
|
|
'default': 'OR',
|
|
options: [
|
|
'AND',
|
|
'OR'
|
|
],
|
|
name: 'default_operator'
|
|
},
|
|
df: {
|
|
type: 'string'
|
|
},
|
|
explain: {
|
|
type: 'boolean'
|
|
},
|
|
fields: {
|
|
type: 'list'
|
|
},
|
|
from: {
|
|
type: 'number'
|
|
},
|
|
ignoreIndices: {
|
|
type: 'enum',
|
|
'default': 'none',
|
|
options: [
|
|
'none',
|
|
'missing'
|
|
],
|
|
name: 'ignore_indices'
|
|
},
|
|
indicesBoost: {
|
|
type: 'list',
|
|
name: 'indices_boost'
|
|
},
|
|
lenient: {
|
|
type: 'boolean'
|
|
},
|
|
lowercaseExpandedTerms: {
|
|
type: 'boolean',
|
|
name: 'lowercase_expanded_terms'
|
|
},
|
|
preference: {
|
|
type: 'string'
|
|
},
|
|
q: {
|
|
type: 'string'
|
|
},
|
|
routing: {
|
|
type: 'list'
|
|
},
|
|
scroll: {
|
|
type: 'duration'
|
|
},
|
|
searchType: {
|
|
type: 'enum',
|
|
options: [
|
|
'query_then_fetch',
|
|
'query_and_fetch',
|
|
'dfs_query_then_fetch',
|
|
'dfs_query_and_fetch',
|
|
'count',
|
|
'scan'
|
|
],
|
|
name: 'search_type'
|
|
},
|
|
size: {
|
|
type: 'number'
|
|
},
|
|
sort: {
|
|
type: 'list'
|
|
},
|
|
source: {
|
|
type: 'list',
|
|
name: '_source'
|
|
},
|
|
sourceExclude: {
|
|
type: 'list',
|
|
name: '_source_exclude'
|
|
},
|
|
sourceInclude: {
|
|
type: 'list',
|
|
name: '_source_include'
|
|
},
|
|
stats: {
|
|
type: 'list'
|
|
},
|
|
suggestField: {
|
|
type: 'string',
|
|
name: 'suggest_field'
|
|
},
|
|
suggestMode: {
|
|
type: 'enum',
|
|
'default': 'missing',
|
|
options: [
|
|
'missing',
|
|
'popular',
|
|
'always'
|
|
],
|
|
name: 'suggest_mode'
|
|
},
|
|
suggestSize: {
|
|
type: 'number',
|
|
name: 'suggest_size'
|
|
},
|
|
suggestText: {
|
|
type: 'text',
|
|
name: 'suggest_text'
|
|
},
|
|
timeout: {
|
|
type: 'time'
|
|
},
|
|
version: {
|
|
type: 'boolean'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>/<%=type%>/_search',
|
|
opt: {
|
|
index: {
|
|
type: 'list',
|
|
'default': '_all'
|
|
}
|
|
},
|
|
req: {
|
|
type: {
|
|
type: 'list'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/<%=index%>/_search',
|
|
opt: {
|
|
index: {
|
|
type: 'list',
|
|
'default': '_all'
|
|
}
|
|
}
|
|
}
|
|
],
|
|
method: 'POST'
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [suggest](http://elasticsearch.org/guide/reference/api/search/suggest/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {String} [params.ignoreIndices=none] - When performed on multiple indices, allows to ignore `missing` ones
|
|
* @param {String} params.preference - Specify the node or shard the operation should be performed on (default: random)
|
|
* @param {String} params.routing - Specific routing value
|
|
* @param {String} params.source - The URL-encoded request definition (instead of using request body)
|
|
* @param {String or String[] or Boolean} params.index - A comma-separated list of index names to restrict the operation; use `_all` or empty string to perform the operation on all indices
|
|
*/
|
|
api.suggest = ca({
|
|
params: {
|
|
ignoreIndices: {
|
|
type: 'enum',
|
|
'default': 'none',
|
|
options: [
|
|
'none',
|
|
'missing'
|
|
],
|
|
name: 'ignore_indices'
|
|
},
|
|
preference: {
|
|
type: 'string'
|
|
},
|
|
routing: {
|
|
type: 'string'
|
|
},
|
|
source: {
|
|
type: 'string'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>/_suggest',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/_suggest'
|
|
}
|
|
],
|
|
method: 'POST'
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [update](http://elasticsearch.org/guide/reference/api/update/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {String} params.consistency - Explicit write consistency setting for the operation
|
|
* @param {String or String[] or Boolean} params.fields - A comma-separated list of fields to return in the response
|
|
* @param {String} params.lang - The script language (default: mvel)
|
|
* @param {String} params.parent - ID of the parent document
|
|
* @param {String} params.percolate - Perform percolation during the operation; use specific registered query name, attribute, or wildcard
|
|
* @param {Boolean} params.refresh - Refresh the index after performing the operation
|
|
* @param {String} [params.replication=sync] - Specific replication type
|
|
* @param {Number} params.retryOnConflict - Specify how many times should the operation be retried when a conflict occurs (default: 0)
|
|
* @param {String} params.routing - Specific routing value
|
|
* @param {*} params.script - The URL-encoded script definition (instead of using request body)
|
|
* @param {Date or Number} params.timeout - Explicit operation timeout
|
|
* @param {Date or Number} params.timestamp - Explicit timestamp for the document
|
|
* @param {Duration} params.ttl - Expiration time for the document
|
|
* @param {Number} params.version - Explicit version number for concurrency control
|
|
* @param {String} params.versionType - Specific version type
|
|
* @param {String} params.id - Document ID
|
|
* @param {String} params.index - The name of the index
|
|
* @param {String} params.type - The type of the document
|
|
*/
|
|
api.update = ca({
|
|
params: {
|
|
consistency: {
|
|
type: 'enum',
|
|
options: [
|
|
'one',
|
|
'quorum',
|
|
'all'
|
|
]
|
|
},
|
|
fields: {
|
|
type: 'list'
|
|
},
|
|
lang: {
|
|
type: 'string'
|
|
},
|
|
parent: {
|
|
type: 'string'
|
|
},
|
|
percolate: {
|
|
type: 'string'
|
|
},
|
|
refresh: {
|
|
type: 'boolean'
|
|
},
|
|
replication: {
|
|
type: 'enum',
|
|
'default': 'sync',
|
|
options: [
|
|
'sync',
|
|
'async'
|
|
]
|
|
},
|
|
retryOnConflict: {
|
|
type: 'number',
|
|
name: 'retry_on_conflict'
|
|
},
|
|
routing: {
|
|
type: 'string'
|
|
},
|
|
script: {},
|
|
timeout: {
|
|
type: 'time'
|
|
},
|
|
timestamp: {
|
|
type: 'time'
|
|
},
|
|
ttl: {
|
|
type: 'duration'
|
|
},
|
|
version: {
|
|
type: 'number'
|
|
},
|
|
versionType: {
|
|
type: 'enum',
|
|
options: [
|
|
'internal',
|
|
'external'
|
|
],
|
|
name: 'version_type'
|
|
}
|
|
},
|
|
url: {
|
|
fmt: '/<%=index%>/<%=type%>/<%=id%>/_update',
|
|
req: {
|
|
index: {
|
|
type: 'string'
|
|
},
|
|
type: {
|
|
type: 'string'
|
|
},
|
|
id: {
|
|
type: 'string'
|
|
}
|
|
},
|
|
sortOrder: -3
|
|
},
|
|
method: 'POST'
|
|
});
|
|
|
|
/**
|
|
* Perform a [create](http://elasticsearch.org/guide/reference/api/index_/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
* @param {String} params.consistency - Explicit write consistency setting for the operation
|
|
* @param {String} params.id - Document ID
|
|
* @param {String} params.parent - ID of the parent document
|
|
* @param {String} params.percolate - Percolator queries to execute while indexing the document
|
|
* @param {Boolean} params.refresh - Refresh the index after performing the operation
|
|
* @param {String} [params.replication=sync] - Specific replication type
|
|
* @param {String} params.routing - Specific routing value
|
|
* @param {Date or Number} params.timeout - Explicit operation timeout
|
|
* @param {Date or Number} params.timestamp - Explicit timestamp for the document
|
|
* @param {Duration} params.ttl - Expiration time for the document
|
|
* @param {Number} params.version - Explicit version number for concurrency control
|
|
* @param {String} params.versionType - Specific version type
|
|
* @param {String} params.index - The name of the index
|
|
* @param {String} params.type - The type of the document
|
|
*/
|
|
api.create = ca.proxy(api.index, {
|
|
transform: function (params) {
|
|
params.op_type = 'create';
|
|
}
|
|
});
|
|
|
|
|
|
},{"./client_action":61}],60:[function(require,module,exports){
|
|
/**
|
|
* A client that makes requests to Elasticsearch via a {{#crossLink "Transport"}}Transport{{/crossLink}}
|
|
*
|
|
* Initializing a client might look something like:
|
|
*
|
|
* ```
|
|
* var client = new es.Client({
|
|
* hosts: [
|
|
* 'es1.net:9200',
|
|
* {
|
|
* host: 'es2.net',
|
|
* port: 9200
|
|
* }
|
|
* ],
|
|
* sniffOnStart: true,
|
|
* log: {
|
|
* type: 'file',
|
|
* level: 'warning'
|
|
* }
|
|
* });
|
|
* ```
|
|
*
|
|
* @class Client
|
|
* @constructor
|
|
*/
|
|
|
|
module.exports = Client;
|
|
|
|
var api = require('./api.js');
|
|
var ca = require('./client_action');
|
|
var Transport = require('./transport');
|
|
|
|
function Client(config) {
|
|
this.transport = new Transport(config);
|
|
|
|
// instantiate the api's namespaces
|
|
for (var i = 0; i < this._namespaces.length; i++) {
|
|
this[this._namespaces[i]] = new this[this._namespaces[i]](this.transport);
|
|
}
|
|
}
|
|
|
|
Client.prototype = api;
|
|
|
|
/**
|
|
* Ping some node to ensure that the cluster is available in some respect
|
|
*
|
|
* @param {Object} params - Currently just a placeholder, no params used at this time
|
|
* @param {Function} cb - callback
|
|
*/
|
|
Client.prototype.ping = ca({
|
|
method: 'HEAD',
|
|
url: {
|
|
fmt: '/'
|
|
},
|
|
timeout: 100
|
|
});
|
|
|
|
Client.prototype.close = function () {
|
|
this.transport.close();
|
|
};
|
|
|
|
},{"./api.js":59,"./client_action":61,"./transport":78}],61:[function(require,module,exports){
|
|
/**
|
|
* Constructs a function that can be called to make a request to ES
|
|
* @type {[type]}
|
|
*/
|
|
module.exports = ClientAction;
|
|
|
|
var _ = require('./utils');
|
|
var when = require('when');
|
|
|
|
function ClientAction(spec) {
|
|
if (!_.isPlainObject(spec.params)) {
|
|
spec.params = {};
|
|
}
|
|
|
|
if (!spec.method) {
|
|
spec.method = 'GET';
|
|
}
|
|
|
|
return function (params, cb) {
|
|
if (typeof params === 'function') {
|
|
cb = params;
|
|
params = {};
|
|
} else {
|
|
params = params || {};
|
|
cb = typeof cb === 'function' ? cb : null;
|
|
}
|
|
|
|
try {
|
|
return exec(this.transport, spec, params, cb);
|
|
} catch (e) {
|
|
if (typeof cb === 'function') {
|
|
_.nextTick(cb, e);
|
|
} else {
|
|
return when.reject(e);
|
|
}
|
|
}
|
|
};
|
|
}
|
|
|
|
var castType = {
|
|
enum: function (param, val, name) {
|
|
/* jshint eqeqeq: false */
|
|
for (var i = 0; i < param.options.length; i++) {
|
|
if (param.options[i] == val) {
|
|
return param.options[i];
|
|
}
|
|
}
|
|
throw new TypeError('Invalid ' + name + ': expected one of ' + param.options.join(','));
|
|
},
|
|
duration: function (param, val, name) {
|
|
if (_.isNumeric(val) || _.isInterval(val)) {
|
|
return val;
|
|
} else {
|
|
throw new TypeError(
|
|
'Invalid ' + name + ': expected a number or interval ' +
|
|
'(an integer followed by one of Mwdhmsy).'
|
|
);
|
|
}
|
|
},
|
|
list: function (param, val, name) {
|
|
switch (typeof val) {
|
|
case 'number':
|
|
case 'string':
|
|
return val;
|
|
case 'object':
|
|
if (_.isArray(val)) {
|
|
return val.join(',');
|
|
}
|
|
/* falls through */
|
|
default:
|
|
throw new TypeError('Invalid ' + name + ': expected be a comma seperated list, array, number or string.');
|
|
}
|
|
},
|
|
boolean: function (param, val) {
|
|
val = _.isString(val) ? val.toLowerCase() : val;
|
|
return (val === 'no' || val === 'off') ? false : !!val;
|
|
},
|
|
number: function (param, val, name) {
|
|
if (_.isNumeric(val)) {
|
|
return val * 1;
|
|
} else {
|
|
throw new TypeError('Invalid ' + name + ': expected a number.');
|
|
}
|
|
},
|
|
string: function (param, val, name) {
|
|
switch (typeof val) {
|
|
case 'number':
|
|
case 'string':
|
|
return '' + val;
|
|
default:
|
|
throw new TypeError('Invalid ' + name + ': expected a string.');
|
|
}
|
|
},
|
|
time: function (param, val, name) {
|
|
if (typeof val === 'string') {
|
|
return val;
|
|
}
|
|
else if (_.isNumeric(val)) {
|
|
return '' + val;
|
|
}
|
|
else if (val instanceof Date) {
|
|
return '' + val.getTime();
|
|
}
|
|
else {
|
|
throw new TypeError('Invalid ' + name + ': expected some sort of time.');
|
|
}
|
|
}
|
|
};
|
|
|
|
function resolveUrl(url, params) {
|
|
var vars = {}, i, key;
|
|
|
|
if (url.req) {
|
|
// url has required params
|
|
if (!url.reqParamKeys) {
|
|
// create cached key list on demand
|
|
url.reqParamKeys = _.keys(url.req);
|
|
}
|
|
|
|
for (i = 0; i < url.reqParamKeys.length; i ++) {
|
|
key = url.reqParamKeys[i];
|
|
if (!params.hasOwnProperty(key)) {
|
|
// missing a required param
|
|
return false;
|
|
} else {
|
|
// cast of copy required param
|
|
if (castType[url.req[key].type]) {
|
|
vars[key] = castType[url.req[key].type](url.req[key], params[key], key);
|
|
} else {
|
|
vars[key] = params[key];
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (url.opt) {
|
|
// url has optional params
|
|
if (!url.optParamKeys) {
|
|
url.optParamKeys = _.keys(url.opt);
|
|
}
|
|
|
|
for (i = 0; i < url.optParamKeys.length; i ++) {
|
|
key = url.optParamKeys[i];
|
|
if (params[key]) {
|
|
if (castType[url.opt[key].type]) {
|
|
vars[key] = castType[url.opt[key].type](url.opt[key], params[key], key);
|
|
} else {
|
|
vars[key] = params[key];
|
|
}
|
|
} else {
|
|
vars[key] = url.opt[key]['default'];
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!url.template) {
|
|
// compile the template on demand
|
|
url.template = _.template(url.fmt);
|
|
}
|
|
|
|
return url.template(_.transform(vars, function (note, val, name) {
|
|
// encode each value
|
|
note[name] = encodeURIComponent(val);
|
|
// remove it from the params so that it isn't sent to the final request
|
|
delete params[name];
|
|
}, {}));
|
|
}
|
|
|
|
// export so that we can test this
|
|
ClientAction.resolveUrl = resolveUrl;
|
|
|
|
function exec(transport, spec, params, cb) {
|
|
var request = {
|
|
method: spec.method,
|
|
timeout: spec.timeout || 10000
|
|
};
|
|
var query = {};
|
|
var i;
|
|
|
|
// verify that we have the body if needed
|
|
if (spec.needsBody && !params.body) {
|
|
throw new TypeError('A request body is required.');
|
|
}
|
|
|
|
// control params
|
|
spec.bulkBody && (request.bulkBody = true);
|
|
spec.castExists && (request.castExists = true);
|
|
|
|
// pick the url
|
|
if (spec.url) {
|
|
// only one url option
|
|
request.path = resolveUrl(spec.url, params);
|
|
} else {
|
|
for (i = 0; i < spec.urls.length; i++) {
|
|
if (request.path = resolveUrl(spec.urls[i], params)) {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!request.path) {
|
|
// there must have been some mimimun requirements that were not met
|
|
throw new TypeError(
|
|
'Unable to build a path with those params. Supply at least ' +
|
|
_.keys(spec.urls[spec.urls.length - 1].req).join(', ')
|
|
);
|
|
}
|
|
|
|
// build the query string
|
|
if (!spec.paramKeys) {
|
|
// build a key list on demand
|
|
spec.paramKeys = _.keys(spec.params);
|
|
spec.requireParamKeys = _.transform(spec.params, function (req, param, key) {
|
|
if (param.required) {
|
|
req.push(key);
|
|
}
|
|
}, []);
|
|
}
|
|
|
|
var key, paramSpec;
|
|
|
|
for (key in params) {
|
|
if (params.hasOwnProperty(key) && params[key] != null) {
|
|
switch (key) {
|
|
case 'body':
|
|
request.body = params.body;
|
|
break;
|
|
case 'ignore':
|
|
request.ignore = _.isArray(params.ignore) ? params.ignore : [params.ignore];
|
|
break;
|
|
case 'timeout':
|
|
request.timeout = params.timeout;
|
|
break;
|
|
case 'method':
|
|
request.method = _.toUpperString(params.method);
|
|
break;
|
|
default:
|
|
paramSpec = spec.params[key];
|
|
if (paramSpec) {
|
|
// param keys don't always match the param name, in those cases it's stored in the param def as "name"
|
|
paramSpec.name = paramSpec.name || key;
|
|
if (params[key] != null) {
|
|
if (castType[paramSpec.type]) {
|
|
query[paramSpec.name] = castType[paramSpec.type](paramSpec, params[key], key);
|
|
} else {
|
|
query[paramSpec.name] = params[key];
|
|
}
|
|
|
|
if (paramSpec['default'] && query[paramSpec.name] === paramSpec['default']) {
|
|
delete query[paramSpec.name];
|
|
}
|
|
}
|
|
} else {
|
|
query[key] = params[key];
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
for (i = 0; i < spec.requireParamKeys.length; i ++) {
|
|
if (!query.hasOwnProperty(spec.requireParamKeys[i])) {
|
|
throw new TypeError('Missing required parameter ' + spec.requireParamKeys[i]);
|
|
}
|
|
}
|
|
|
|
request.query = query;
|
|
|
|
return transport.request(request, cb);
|
|
}
|
|
|
|
|
|
|
|
ClientAction.proxy = function (fn, spec) {
|
|
return function (params, cb) {
|
|
if (typeof params === 'function') {
|
|
cb = params;
|
|
params = {};
|
|
} else {
|
|
params = params || {};
|
|
cb = typeof cb === 'function' ? cb : null;
|
|
}
|
|
|
|
if (spec.transform) {
|
|
spec.transform(params);
|
|
}
|
|
|
|
return fn.call(this, params, cb);
|
|
};
|
|
};
|
|
|
|
},{"./utils":79,"when":57}],62:[function(require,module,exports){
|
|
module.exports = ConnectionAbstract;
|
|
|
|
var _ = require('./utils');
|
|
var EventEmitter = require('events').EventEmitter;
|
|
|
|
/**
|
|
* Abstract class used for Connection classes
|
|
* @class ConnectionAbstract
|
|
* @constructor
|
|
*/
|
|
function ConnectionAbstract(host, config) {
|
|
config = _.defaults(config || {}, {
|
|
deadTimeout: 30000
|
|
});
|
|
EventEmitter.call(this);
|
|
this.deadTimeout = config.deadTimeout;
|
|
this.requestCount = 0;
|
|
|
|
if (!host) {
|
|
throw new TypeError('Missing host');
|
|
} else if (host.makeUrl) {
|
|
this.host = host;
|
|
} else {
|
|
throw new TypeError('Invalid host');
|
|
}
|
|
|
|
_.makeBoundMethods(this);
|
|
}
|
|
_.inherits(ConnectionAbstract, EventEmitter);
|
|
|
|
/**
|
|
* Make a request using this connection. Must be overridden by Connection classes, which can add whatever keys to
|
|
* params that they like. These are just the basics.
|
|
*
|
|
* @param [params] {Object} - The parameters for the request
|
|
* @param params.path {String} - The path for which you are requesting
|
|
* @param params.method {String} - The HTTP method for the request (GET, HEAD, etc.)
|
|
* @param params.timeout {Integer} - The amount of time in milliseconds that this request should be allowed to run for.
|
|
* @param cb {Function} - A callback to be called once with `cb(err, responseBody, responseStatus)`
|
|
*/
|
|
ConnectionAbstract.prototype.request = function () {
|
|
throw new Error('Connection#request must be overwritten by the Connector');
|
|
};
|
|
|
|
ConnectionAbstract.prototype.ping = function (cb) {
|
|
if (typeof cb !== 'function') {
|
|
throw new TypeError('Callback must be a function');
|
|
}
|
|
|
|
return this.request({
|
|
path: '/',
|
|
method: 'HEAD',
|
|
timeout: 100
|
|
}, cb);
|
|
};
|
|
|
|
ConnectionAbstract.prototype.setStatus = function (status) {
|
|
var origStatus = this.status;
|
|
|
|
this.status = status;
|
|
|
|
if (this._deadTimeoutId) {
|
|
clearTimeout(this._deadTimeoutId);
|
|
this._deadTimeoutId = null;
|
|
}
|
|
|
|
if (status === 'dead') {
|
|
this._deadTimeoutId = setTimeout(this.bound.resuscitate, this.deadTimeout);
|
|
}
|
|
|
|
this.emit('status set', status, origStatus, this);
|
|
|
|
if (status === 'closed') {
|
|
this.removeAllListeners();
|
|
}
|
|
};
|
|
|
|
ConnectionAbstract.prototype.resuscitate = _.scheduled(function () {
|
|
var self = this;
|
|
|
|
if (self.status === 'dead') {
|
|
self.ping(function (err) {
|
|
if (!err) {
|
|
self.setStatus('alive');
|
|
} else {
|
|
self.setStatus('dead');
|
|
}
|
|
});
|
|
}
|
|
});
|
|
|
|
},{"./utils":79,"events":5}],63:[function(require,module,exports){
|
|
/**
|
|
* Manager of connections to a node(s), capable of ensuring that connections are clear and living
|
|
* before providing them to the application
|
|
*
|
|
* @class ConnectionPool
|
|
* @constructor
|
|
* @param {Object} config - The config object passed to the transport.
|
|
*/
|
|
|
|
module.exports = ConnectionPool;
|
|
|
|
var _ = require('./utils');
|
|
var Log = require('./log');
|
|
|
|
function ConnectionPool(config) {
|
|
_.makeBoundMethods(this);
|
|
|
|
this.log = config.log;
|
|
if (!this.log) {
|
|
this.log = new Log();
|
|
}
|
|
|
|
// get the selector config var
|
|
this.selector = _.funcEnum(config, 'selector', ConnectionPool.selectors, ConnectionPool.defaultSelectors);
|
|
// get the connection class
|
|
this.Connection = _.funcEnum(config, 'connectionClass', ConnectionPool.connectionClasses,
|
|
ConnectionPool.defaultConnectionClass);
|
|
|
|
// a map of connections to their "id" property, used when sniffing
|
|
this.index = {};
|
|
|
|
this.connections = {
|
|
alive: [],
|
|
dead: []
|
|
};
|
|
}
|
|
|
|
// selector options
|
|
ConnectionPool.selectors = require('./selectors');
|
|
ConnectionPool.defaultSelectors = 'round_robin';
|
|
|
|
// get the connection options
|
|
ConnectionPool.connectionClasses = require('./connectors');
|
|
ConnectionPool.defaultConnectionClass = ConnectionPool.connectionClasses._default;
|
|
delete ConnectionPool.connectionClasses._default;
|
|
|
|
/**
|
|
* Selects a connection from the list using the this.selector
|
|
* Features:
|
|
* - detects if the selector is async or not
|
|
* - sync selectors should still return asynchronously
|
|
* - catches errors in sync selectors
|
|
* - automatically selects the first dead connection when there no living connections
|
|
*
|
|
* @param {Function} cb [description]
|
|
* @return {[type]} [description]
|
|
*/
|
|
ConnectionPool.prototype.select = function (cb) {
|
|
if (this.connections.alive.length) {
|
|
if (this.selector.length > 1) {
|
|
this.selector(this.connections.alive, cb);
|
|
} else {
|
|
try {
|
|
_.nextTick(cb, null, this.selector(this.connections.alive));
|
|
} catch (e) {
|
|
cb(e);
|
|
}
|
|
}
|
|
} else {
|
|
_.nextTick(cb, null, this.connections.dead[0]);
|
|
}
|
|
};
|
|
|
|
ConnectionPool.prototype.onStatusSet = _.handler(function (status, oldStatus, connection) {
|
|
var from, to, index;
|
|
|
|
if (oldStatus === status) {
|
|
if (status === 'dead') {
|
|
// we want to remove the connection from it's current possition and move it to the end
|
|
status = 'redead';
|
|
} else {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
switch (status) {
|
|
case 'alive':
|
|
from = this.connections.dead;
|
|
to = this.connections.alive;
|
|
break;
|
|
case 'dead':
|
|
from = this.connections.alive;
|
|
to = this.connections.dead;
|
|
break;
|
|
case 'redead':
|
|
from = this.connections.dead;
|
|
to = this.connections.dead;
|
|
break;
|
|
case 'closed':
|
|
from = this.connections[oldStatus];
|
|
break;
|
|
}
|
|
|
|
if (from && from.indexOf) {
|
|
index = from.indexOf(connection);
|
|
if (~index) {
|
|
from.splice(index, 1);
|
|
}
|
|
}
|
|
|
|
if (to && to.indexOf) {
|
|
index = to.indexOf(connection);
|
|
if (!~index) {
|
|
to.push(connection);
|
|
}
|
|
}
|
|
});
|
|
|
|
ConnectionPool.prototype.addConnection = function (connection) {
|
|
if (!connection.id) {
|
|
connection.id = connection.host.toString();
|
|
}
|
|
|
|
if (!this.index[connection.id]) {
|
|
this.log.info('Adding connection to', connection.id);
|
|
this.index[connection.id] = connection;
|
|
connection.on('status set', this.bound.onStatusSet);
|
|
connection.setStatus('alive');
|
|
}
|
|
};
|
|
|
|
ConnectionPool.prototype.removeConnection = function (connection) {
|
|
if (!connection.id) {
|
|
connection.id = connection.host.toString();
|
|
}
|
|
|
|
if (this.index[connection.id]) {
|
|
delete this.index[connection.id];
|
|
connection.setStatus('closed');
|
|
connection.removeListener('status set', this.bound.onStatusSet);
|
|
}
|
|
};
|
|
|
|
ConnectionPool.prototype.setHosts = function (hosts) {
|
|
var connection;
|
|
var i;
|
|
var id;
|
|
var host;
|
|
var toRemove = _.clone(this.index);
|
|
|
|
for (i = 0; i < hosts.length; i++) {
|
|
host = hosts[i];
|
|
id = host.toString();
|
|
if (this.index[id]) {
|
|
delete toRemove[id];
|
|
} else {
|
|
connection = new this.Connection(host);
|
|
connection.id = id;
|
|
this.addConnection(connection);
|
|
}
|
|
}
|
|
|
|
var removeIds = _.keys(toRemove);
|
|
for (i = 0; i < removeIds.length; i++) {
|
|
this.removeConnection(this.index[removeIds[i]]);
|
|
}
|
|
};
|
|
|
|
ConnectionPool.prototype.close = function () {
|
|
this.setHosts([]);
|
|
};
|
|
ConnectionPool.prototype.empty = ConnectionPool.prototype.close;
|
|
|
|
},{"./connectors":65,"./log":70,"./selectors":74,"./utils":79}],64:[function(require,module,exports){
|
|
/**
|
|
* Connection that registers a module with angular, using angular's $http service
|
|
* to communicate with ES.
|
|
*
|
|
* @class connections.Angular
|
|
*/
|
|
module.exports = AngularConnector;
|
|
|
|
var _ = require('../utils');
|
|
var ConnectionAbstract = require('../connection');
|
|
var ConnectionFault = require('../errors').ConnectionFault;
|
|
|
|
function AngularConnector(host, config) {
|
|
ConnectionAbstract.call(this, host, config);
|
|
}
|
|
_.inherits(AngularConnector, ConnectionAbstract);
|
|
|
|
AngularConnector.prototype.request = function (params, cb) {
|
|
this.$http({
|
|
method: params.method,
|
|
url: this.host.makeUrl(params),
|
|
data: params.body,
|
|
cache: false,
|
|
timeout: params.timeout !== Infinity ? params.timeout : 0
|
|
}).then(function (response) {
|
|
cb(null, response.data, response.status);
|
|
}, function (err) {
|
|
cb(new ConnectionFault(err.message));
|
|
});
|
|
};
|
|
|
|
// must be overwritten before this connection can be used
|
|
AngularConnector.prototype.$http = null;
|
|
|
|
},{"../connection":62,"../errors":68,"../utils":79}],65:[function(require,module,exports){
|
|
var opts = {
|
|
xhr: require('./xhr'),
|
|
jquery: require('./jquery'),
|
|
angular: require('./angular')
|
|
};
|
|
var _ = require('../utils');
|
|
|
|
// remove modules that have been ignored by browserify
|
|
_.each(opts, function (conn, name) {
|
|
if (typeof conn !== 'function') {
|
|
delete opts[name];
|
|
}
|
|
});
|
|
|
|
// custom __default specification
|
|
if (opts.xhr) {
|
|
opts._default = 'xhr';
|
|
} else if (opts.angular) {
|
|
opts._default = 'angular';
|
|
} else {
|
|
opts._default = 'jquery';
|
|
}
|
|
|
|
module.exports = opts;
|
|
|
|
},{"../utils":79,"./angular":64,"./jquery":66,"./xhr":67}],66:[function(require,module,exports){
|
|
/* jshint browser: true, jquery: true */
|
|
|
|
/**
|
|
* Simple connection class for using the XHR object in browsers
|
|
*
|
|
* @class {XhrConnection}
|
|
*/
|
|
module.exports = JqueryConnector;
|
|
|
|
function JqueryConnector() {}
|
|
|
|
JqueryConnector.prototype.request = function (params, cb) {
|
|
jQuery.ajax(params).done(cb);
|
|
};
|
|
|
|
|
|
|
|
},{}],67:[function(require,module,exports){
|
|
/**
|
|
* Generic Transport for the browser, using the XmlHttpRequest object
|
|
*
|
|
* @class connections.Xhr
|
|
*/
|
|
module.exports = XhrConnector;
|
|
|
|
/* jshint browser:true */
|
|
|
|
var _ = require('../utils');
|
|
var ConnectionAbstract = require('../connection');
|
|
var ConnectionFault = require('../errors').ConnectionFault;
|
|
var TimeoutError = require('../errors').RequestTimeout;
|
|
var asyncDefault = !(navigator && /PhantomJS/i.test(navigator.userAgent));
|
|
|
|
function XhrConnector(host, config) {
|
|
ConnectionAbstract.call(this, host, config);
|
|
}
|
|
_.inherits(XhrConnector, ConnectionAbstract);
|
|
|
|
/**
|
|
* Simply returns an XHR object cross browser
|
|
* @type {Function}
|
|
*/
|
|
var getXhr = _.noop;
|
|
|
|
if (typeof XMLHttpRequest !== 'undefined') {
|
|
// rewrite the getXhr method to always return the native implementation
|
|
getXhr = function () {
|
|
return new XMLHttpRequest();
|
|
};
|
|
} else {
|
|
// find the first MS implementation available
|
|
getXhr = _.first(['Msxml2.XMLHTTP', 'Microsoft.XMLHTTP', 'Msxml2.XMLHTTP.4.0'], function (appName) {
|
|
/* jshint unused: false */
|
|
try {
|
|
var test = new window.ActiveXObject(appName);
|
|
return function () {
|
|
return new window.ActiveXObject(appName);
|
|
};
|
|
} catch (e) {
|
|
return null;
|
|
}
|
|
});
|
|
}
|
|
|
|
if (!getXhr) {
|
|
throw new Error('getXhr(): XMLHttpRequest not available');
|
|
}
|
|
|
|
XhrConnector.prototype.request = function (params, cb) {
|
|
var xhr = getXhr();
|
|
var timeout = params.timeout ? params.timeout : 10000;
|
|
var timeoutId;
|
|
var url = this.host.makeUrl(params);
|
|
var log = this.config.log;
|
|
var async = params.async === false ? false : asyncDefault;
|
|
|
|
if (params.auth) {
|
|
xhr.open(params.method, url, async, params.auth.user, params.auth.pass);
|
|
} else {
|
|
xhr.open(params.method, url, async);
|
|
}
|
|
|
|
xhr.onreadystatechange = function () {
|
|
if (xhr.readyState === 4) {
|
|
clearTimeout(timeoutId);
|
|
log.trace(params.method, url, params.body, xhr.responseText, xhr.status);
|
|
var err = xhr.status ? void 0 : new ConnectionFault(xhr.statusText || 'Request failed to complete.');
|
|
cb(err, xhr.responseText, xhr.status);
|
|
}
|
|
};
|
|
|
|
if (timeout !== Infinity) {
|
|
timeoutId = setTimeout(function () {
|
|
xhr.onreadystatechange = _.noop;
|
|
xhr.abort();
|
|
cb(new TimeoutError());
|
|
}, timeout);
|
|
}
|
|
|
|
xhr.send(params.body || void 0);
|
|
};
|
|
|
|
},{"../connection":62,"../errors":68,"../utils":79}],68:[function(require,module,exports){
|
|
var process=require("__browserify_process");var _ = require('./utils');
|
|
var errors = module.exports;
|
|
|
|
function ErrorAbstract(msg, constructor) {
|
|
this.message = msg;
|
|
|
|
Error.call(this, this.message);
|
|
if (process.browser) {
|
|
this.stack = '';
|
|
} else {
|
|
Error.captureStackTrace(this, constructor);
|
|
}
|
|
}
|
|
errors._Abstract = ErrorAbstract;
|
|
_.inherits(ErrorAbstract, Error);
|
|
|
|
/**
|
|
* Connection Error
|
|
* @param {String} [msg] - An error message that will probably end up in a log.
|
|
*/
|
|
errors.ConnectionFault = function ConnectionFault(msg) {
|
|
ErrorAbstract.call(this, msg || 'Connection Failure', errors.ConnectionFault);
|
|
};
|
|
_.inherits(errors.ConnectionFault, ErrorAbstract);
|
|
|
|
/**
|
|
* No Living Connections
|
|
* @param {String} [msg] - An error message that will probably end up in a log.
|
|
*/
|
|
errors.NoConnections = function NoConnections(msg) {
|
|
ErrorAbstract.call(this, msg || 'No Living connections', errors.NoConnections);
|
|
};
|
|
_.inherits(errors.NoConnections, ErrorAbstract);
|
|
|
|
/**
|
|
* Generic Error
|
|
* @param {String} [msg] - An error message that will probably end up in a log.
|
|
*/
|
|
errors.Generic = function Generic(msg) {
|
|
ErrorAbstract.call(this, msg || 'Generic Error', errors.Generic);
|
|
};
|
|
_.inherits(errors.Generic, ErrorAbstract);
|
|
|
|
/**
|
|
* Request Timeout Error
|
|
* @param {String} [msg] - An error message that will probably end up in a log.
|
|
*/
|
|
errors.RequestTimeout = function RequestTimeout(msg) {
|
|
ErrorAbstract.call(this, msg || 'Request Timeout', errors.RequestTimeout);
|
|
};
|
|
_.inherits(errors.RequestTimeout, ErrorAbstract);
|
|
|
|
|
|
/**
|
|
* Request Body could not be parsed
|
|
* @param {String} [msg] - An error message that will probably end up in a log.
|
|
*/
|
|
errors.Serialization = function Serialization(msg) {
|
|
ErrorAbstract.call(this, msg || 'Unable to parse/serialize body', errors.Serialization);
|
|
};
|
|
_.inherits(errors.Serialization, ErrorAbstract);
|
|
|
|
|
|
var statusCodes = {
|
|
|
|
/**
|
|
* Service Unavailable
|
|
* @param {String} [msg] - An error message that will probably end up in a log.
|
|
*/
|
|
503: 'Service Unavailable',
|
|
|
|
/**
|
|
* Internal Server Error
|
|
* @param {String} [msg] - An error message that will probably end up in a log.
|
|
*/
|
|
500: 'Internal Server Error',
|
|
|
|
/**
|
|
* Precondition Failed
|
|
* @param {String} [msg] - An error message that will probably end up in a log.
|
|
*/
|
|
412: 'Precondition Failed',
|
|
|
|
/**
|
|
* Conflict
|
|
* @param {String} [msg] - An error message that will probably end up in a log.
|
|
*/
|
|
409: 'Conflict',
|
|
|
|
/**
|
|
* Forbidden
|
|
* @param {String} [msg] - An error message that will probably end up in a log.
|
|
*/
|
|
403: 'Forbidden',
|
|
|
|
/**
|
|
* Not Found
|
|
* @param {String} [msg] - An error message that will probably end up in a log.
|
|
*/
|
|
404: 'Not Found',
|
|
|
|
/**
|
|
* Bad Request
|
|
* @param {String} [msg] - An error message that will probably end up in a log.
|
|
*/
|
|
400: 'Bad Request',
|
|
|
|
/**
|
|
* Moved Permanently
|
|
* @param {String} [msg] - An error message that will probably end up in a log.
|
|
*/
|
|
301: 'Moved Permanently'
|
|
};
|
|
|
|
_.each(statusCodes, function (name, status) {
|
|
var className = _.studlyCase(name);
|
|
|
|
function StatusCodeError(msg) {
|
|
ErrorAbstract.call(this, msg || name, errors[className]);
|
|
}
|
|
|
|
_.inherits(StatusCodeError, ErrorAbstract);
|
|
errors[className] = StatusCodeError;
|
|
errors[status] = StatusCodeError;
|
|
});
|
|
|
|
},{"./utils":79,"__browserify_process":15}],69:[function(require,module,exports){
|
|
/**
|
|
* Class to wrap URLS, formatting them and maintaining their separate details
|
|
* @type {[type]}
|
|
*/
|
|
module.exports = Host;
|
|
|
|
var url = require('url');
|
|
var qs = require('querystring');
|
|
var _ = require('./utils');
|
|
|
|
var startsWithProtocolRE = /^([a-z]+:)?\/\//;
|
|
|
|
// simple reference used when formatting as a url
|
|
var defaultPort = {
|
|
http: 80,
|
|
https: 443
|
|
};
|
|
|
|
var urlParseFields = [
|
|
'protocol', 'hostname', 'pathname', 'port', 'auth', 'query'
|
|
];
|
|
|
|
var simplify = ['host', 'path'];
|
|
|
|
function Host(config) {
|
|
config = config || {};
|
|
|
|
// defaults
|
|
this.protocol = 'http';
|
|
this.host = 'localhost';
|
|
this.port = 9200;
|
|
this.auth = null;
|
|
this.query = null;
|
|
|
|
if (typeof config === 'string') {
|
|
if (!startsWithProtocolRE.test(config)) {
|
|
config = 'http://' + config;
|
|
}
|
|
config = _.pick(url.parse(config, false, true), urlParseFields);
|
|
}
|
|
|
|
if (_.isObject(config)) {
|
|
// move hostname/portname to host/port semi-intelligently.
|
|
_.each(simplify, function (to) {
|
|
var from = to + 'name';
|
|
if (config[from] && config[to]) {
|
|
if (config[to].indexOf(config[from]) === 0) {
|
|
config[to] = config[from];
|
|
}
|
|
} else if (config[from]) {
|
|
config[to] = config[from];
|
|
}
|
|
delete config[from];
|
|
});
|
|
} else {
|
|
config = {};
|
|
}
|
|
|
|
_.assign(this, config);
|
|
|
|
// make sure the query string is parsed
|
|
if (this.query === null) {
|
|
// majority case
|
|
this.query = {};
|
|
} else if (!_.isPlainObject(this.query)) {
|
|
this.query = qs.parse(this.query);
|
|
}
|
|
|
|
// make sure that the port is a number
|
|
if (typeof this.port !== 'number') {
|
|
this.port = parseInt(this.port, 10);
|
|
if (isNaN(this.port)) {
|
|
this.port = 9200;
|
|
}
|
|
}
|
|
|
|
// make sure the path starts with a leading slash
|
|
// and that empty paths convert to '/'
|
|
if (!this.path || this.path.charAt(0) !== '/') {
|
|
this.path = '/' + (this.path || '');
|
|
}
|
|
|
|
// strip trailing ':' on the protocol (when config comes from url.parse)
|
|
if (this.protocol.substr(-1) === ':') {
|
|
this.protocol = this.protocol.substring(0, this.protocol.length - 1);
|
|
}
|
|
}
|
|
|
|
Host.prototype.makeUrl = function (params) {
|
|
params = params || {};
|
|
// build the port
|
|
var port = '';
|
|
if (this.port !== defaultPort[this.protocol]) {
|
|
// add an actual port
|
|
port = ':' + this.port;
|
|
}
|
|
|
|
// build the path
|
|
var path = '' + (this.path || '') + (params.path || '');
|
|
|
|
// if path doesn't start with '/' add it.
|
|
if (path.charAt(0) !== '/') {
|
|
path = '/' + path;
|
|
}
|
|
|
|
// build the query string
|
|
var query = '';
|
|
if (params.query) {
|
|
// if the user passed in a query, merge it with the defaults from the host
|
|
query = qs.stringify(
|
|
_.defaults(typeof params.query === 'string' ? qs.parse(params.query) : params.query, this.query)
|
|
);
|
|
} else if (this.query) {
|
|
// just stringify the hosts query
|
|
query = qs.stringify(this.query);
|
|
}
|
|
|
|
return this.protocol + '://' + this.host + port + path + (query ? '?' + query : '');
|
|
};
|
|
|
|
Host.prototype.toString = function () {
|
|
return this.makeUrl();
|
|
};
|
|
|
|
},{"./utils":79,"querystring":8,"url":9}],70:[function(require,module,exports){
|
|
var process=require("__browserify_process");var _ = require('./utils');
|
|
var url = require('url');
|
|
var EventEmitter = require('events').EventEmitter;
|
|
|
|
/**
|
|
* Log bridge, which is an [EventEmitter](http://nodejs.org/api/events.html#events_class_events_eventemitter)
|
|
* that sends events to one or more outputs/loggers. Setup these loggers by
|
|
* specifying their config as the first argument, or by passing it to addOutput().
|
|
*
|
|
* @class Log
|
|
* @uses Loggers.Stdio
|
|
* @constructor
|
|
* @param {string|Object|ArrayOfStrings|ArrayOfObjects} output - Either the level
|
|
* to setup a single logger, a full config object for alogger, or an array of
|
|
* config objects to use for creating log outputs.
|
|
* @param {string} output.level - One of the keys in Log.levels (error, warning, etc.)
|
|
* @param {string} output.type - The name of the logger to use for this output
|
|
*/
|
|
function Log(config) {
|
|
config = config || {};
|
|
|
|
var i;
|
|
var outputs;
|
|
|
|
if (config.log) {
|
|
if (_.isArrayOfStrings(config.log)) {
|
|
outputs = [{
|
|
levels: config.log
|
|
}];
|
|
} else {
|
|
outputs = _.createArray(config.log, function (val) {
|
|
if (_.isPlainObject(val)) {
|
|
return val;
|
|
}
|
|
if (typeof val === 'string') {
|
|
return {
|
|
level: val
|
|
};
|
|
}
|
|
});
|
|
}
|
|
|
|
if (!outputs) {
|
|
throw new TypeError('Invalid logging output config. Expected either a log level, array of log levels, ' +
|
|
'a logger config object, or an array of logger config objects.');
|
|
}
|
|
|
|
for (i = 0; i < outputs.length; i++) {
|
|
this.addOutput(outputs[i]);
|
|
}
|
|
}
|
|
}
|
|
_.inherits(Log, EventEmitter);
|
|
|
|
Log.loggers = require('./loggers');
|
|
|
|
Log.prototype.close = function () {
|
|
this.emit('closing');
|
|
if (this.listenerCount()) {
|
|
console.error('Something is still listening for log events, but the logger is closing.');
|
|
this.clearAllListeners();
|
|
}
|
|
};
|
|
|
|
Log.prototype.listenerCount = function (event) {
|
|
// compatability for node < 0.10
|
|
if (EventEmitter.listenerCount) {
|
|
return EventEmitter.listenerCount(this, event);
|
|
} else {
|
|
return this.listeners(event).length;
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Levels observed by the loggers, ordered by rank
|
|
*
|
|
* @property levels
|
|
* @type Array
|
|
* @static
|
|
*/
|
|
Log.levels = [
|
|
/**
|
|
* Event fired for error level log entries
|
|
* @event error
|
|
* @param {Error} error - The error object to log
|
|
*/
|
|
'error',
|
|
/**
|
|
* Event fired for "warning" level log entries, which usually represent things
|
|
* like correctly formatted error responses from ES (400, ...) and recoverable
|
|
* errors (one node unresponsive)
|
|
*
|
|
* @event warning
|
|
* @param {String} message - A message to be logged
|
|
*/
|
|
'warning',
|
|
/**
|
|
* Event fired for "info" level log entries, which usually describe what a
|
|
* client is doing (sniffing etc)
|
|
*
|
|
* @event info
|
|
* @param {String} message - A message to be logged
|
|
*/
|
|
'info',
|
|
/**
|
|
* Event fired for "debug" level log entries, which will describe requests sent,
|
|
* including their url (no data, response codes, or exec times)
|
|
*
|
|
* @event debug
|
|
* @param {String} message - A message to be logged
|
|
*/
|
|
'debug',
|
|
/**
|
|
* Event fired for "trace" level log entries, which provide detailed information
|
|
* about each request made from a client, including reponse codes, execution times,
|
|
* and a full curl command that can be copied and pasted into a terminal
|
|
*
|
|
* @event trace
|
|
* @param {String} method method, , body, responseStatus, responseBody
|
|
* @param {String} url - The url the request was made to
|
|
* @param {String} body - The body of the request
|
|
* @param {Integer} responseStatus - The status code returned from the response
|
|
* @param {String} responseBody - The body of the response
|
|
*/
|
|
'trace'
|
|
];
|
|
|
|
/**
|
|
* Converts a log config value (string or array) to an array of level names which
|
|
* it represents
|
|
*
|
|
* @method parseLevels
|
|
* @static
|
|
* @private
|
|
* @param {String|ArrayOfStrings} input - Cound be a string to specify the max
|
|
* level, or an array of exact levels
|
|
* @return {Array} -
|
|
*/
|
|
Log.parseLevels = function (input) {
|
|
switch (typeof input) {
|
|
case 'string':
|
|
var i = _.indexOf(Log.levels, input);
|
|
if (i >= 0) {
|
|
return Log.levels.slice(0, i + 1);
|
|
}
|
|
/* fall through */
|
|
case 'object':
|
|
if (_.isArray(input)) {
|
|
var valid = _.intersection(input, Log.levels);
|
|
if (valid.length === input.length) {
|
|
return valid;
|
|
}
|
|
}
|
|
/* fall through */
|
|
default:
|
|
throw new TypeError('invalid logging level ' + input + '. Expected zero or more of these options: ' +
|
|
Log.levels.join(', '));
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Combine the array-like param into a simple string
|
|
*
|
|
* @method join
|
|
* @static
|
|
* @private
|
|
* @param {*} arrayish - An array like object that can be itterated by _.each
|
|
* @return {String} - The final string.
|
|
*/
|
|
Log.join = function (arrayish) {
|
|
return _.map(arrayish, function (item) {
|
|
if (_.isPlainObject(item)) {
|
|
return _.inspect(item) + '\n';
|
|
} else {
|
|
return item.toString();
|
|
}
|
|
}).join(' ');
|
|
};
|
|
|
|
/**
|
|
* Create a new logger, based on the config.
|
|
*
|
|
* @method addOutput
|
|
* @param {object} config - An object with config options for the logger.
|
|
* @param {String} [config.type=stdio] - The name of an output/logger. Options
|
|
* can be found in the `src/loggers` directory.
|
|
* @param {String|ArrayOfStrings} [config.levels=warning] - The levels to output
|
|
* to this logger, when an array is specified no levels other than the ones
|
|
* specified will be listened to. When a string is specified, that and all lower
|
|
* levels will be logged.
|
|
* @return {Logger}
|
|
*/
|
|
Log.prototype.addOutput = function (config) {
|
|
config = config || {};
|
|
|
|
// force "levels" key
|
|
config.levels = Log.parseLevels(config.levels || config.level || 'warning');
|
|
delete config.level;
|
|
|
|
var Logger = _.funcEnum(config, 'type', Log.loggers, process.browser ? 'console' : 'stdio');
|
|
return new Logger(this, config);
|
|
};
|
|
|
|
/**
|
|
* Log an error
|
|
*
|
|
* @method error
|
|
* @param {Error|String} error The Error to log
|
|
* @return {Boolean} - True if any outputs accepted the message
|
|
*/
|
|
Log.prototype.error = function (e) {
|
|
if (this.listenerCount('error')) {
|
|
return this.emit('error', e instanceof Error ? e : new Error(e));
|
|
}
|
|
};
|
|
|
|
|
|
/**
|
|
* Log a warning message
|
|
*
|
|
* @method warning
|
|
* @param {*} msg* - Any amount of messages that will be joined before logged
|
|
* @return {Boolean} - True if any outputs accepted the message
|
|
*/
|
|
Log.prototype.warning = function (/* ...msg */) {
|
|
if (this.listenerCount('warning')) {
|
|
return this.emit('warning', Log.join(arguments));
|
|
}
|
|
};
|
|
|
|
|
|
/**
|
|
* Log useful info about what's going on
|
|
*
|
|
* @method info
|
|
* @param {*} msg* - Any amount of messages that will be joined before logged
|
|
* @return {Boolean} - True if any outputs accepted the message
|
|
*/
|
|
Log.prototype.info = function (/* ...msg */) {
|
|
if (this.listenerCount('info')) {
|
|
return this.emit('info', Log.join(arguments));
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Log a debug level message
|
|
*
|
|
* @method debug
|
|
* @param {*} msg* - Any amount of messages that will be joined before logged
|
|
* @return {Boolean} - True if any outputs accepted the message
|
|
*/
|
|
Log.prototype.debug = function (/* ...msg */) {
|
|
if (this.listenerCount('debug')) {
|
|
return this.emit('debug', Log.join(arguments) /*+ _.getStackTrace(Log.prototype.debug)*/);
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Log a trace level message
|
|
*
|
|
* @method trace
|
|
* @param {String} method - HTTP request method
|
|
* @param {String|Object} requestUrl - URL requested. If the value is an object,
|
|
* it is expected to be the return value of Node's url.parse()
|
|
* @param {String} body - The request's body
|
|
* @param {String} responseBody - body returned from ES
|
|
* @param {String} responseStatus - HTTP status code
|
|
* @return {Boolean} - True if any outputs accepted the message
|
|
*/
|
|
Log.prototype.trace = function (method, requestUrl, body, responseBody, responseStatus) {
|
|
if (this.listenerCount('trace')) {
|
|
if (typeof requestUrl === 'string') {
|
|
requestUrl = url.parse(requestUrl, true, true);
|
|
}
|
|
requestUrl = _.defaults({
|
|
host: 'localhost:9200',
|
|
query: _.defaults({
|
|
pretty: true
|
|
}, requestUrl.query)
|
|
}, requestUrl);
|
|
delete requestUrl.auth;
|
|
|
|
if (!requestUrl.pathname && requestUrl.path) {
|
|
requestUrl.pathname = requestUrl.path.split('?').shift();
|
|
}
|
|
|
|
requestUrl = url.format(requestUrl);
|
|
|
|
var message = '<- ' + responseStatus + '\n' + prettyJSON(responseBody);
|
|
|
|
/* jshint quotmark: double */
|
|
var curlCall = "curl '" + requestUrl.replace(/'/g, "\\'") + "' -X" + method.toUpperCase();
|
|
if (body) {
|
|
curlCall += " -d '" + prettyJSON(body) + "'";
|
|
}
|
|
/* jshint quotmark: single */
|
|
|
|
return this.emit('trace', message, curlCall);
|
|
}
|
|
};
|
|
|
|
function prettyJSON(body) {
|
|
try {
|
|
return JSON.stringify(JSON.parse(body), null, ' ').replace(/'/g, '\\\'');
|
|
} catch (e) {
|
|
return body || '';
|
|
}
|
|
}
|
|
|
|
module.exports = Log;
|
|
|
|
},{"./loggers":72,"./utils":79,"__browserify_process":15,"events":5,"url":9}],71:[function(require,module,exports){
|
|
var _ = require('./utils');
|
|
|
|
/**
|
|
* Abstract class providing common functionality to loggers
|
|
* @param {[type]} log [description]
|
|
* @param {[type]} config [description]
|
|
*/
|
|
function LoggerAbstract(log, config) {
|
|
this.log = log;
|
|
this.listeningLevels = [];
|
|
|
|
_.makeBoundMethods(this);
|
|
|
|
// when the log closes, remove our event listeners
|
|
this.log.on('closing', this.bound.cleanUpListeners);
|
|
|
|
this.setupListeners(config.levels);
|
|
}
|
|
|
|
function padNumToTen(n) {
|
|
return n < 10 ? '0' + n.toString(10) : n.toString(10);
|
|
}
|
|
|
|
/**
|
|
* Create a timestamp string used in the format function. Defers to Log.timestamp if it is defined,
|
|
* Also, feel free to override this at the logger level.
|
|
* @return {String} - Timestamp in ISO 8601 UTC
|
|
*/
|
|
LoggerAbstract.prototype.timestamp = function () {
|
|
var d = new Date();
|
|
return d.getUTCFullYear() + '-' +
|
|
padNumToTen(d.getUTCMonth() + 1) + '-' +
|
|
padNumToTen(d.getUTCDate()) + 'T' +
|
|
padNumToTen(d.getUTCHours()) + ':' +
|
|
padNumToTen(d.getUTCMinutes()) + ':' +
|
|
padNumToTen(d.getUTCSeconds()) + 'Z';
|
|
};
|
|
|
|
function indent(text, spaces) {
|
|
var space = _.repeat(' ', spaces || 2);
|
|
return text.split(/\r?\n/).map(function (line) {
|
|
return space + line;
|
|
}).join('\n');
|
|
}
|
|
|
|
LoggerAbstract.prototype.format = function (label, message) {
|
|
return label + ': ' + this.timestamp() + '\n' + indent(message) + '\n\n';
|
|
};
|
|
|
|
LoggerAbstract.prototype.write = function () {
|
|
throw new Error('This should be overwritten by the logger');
|
|
};
|
|
|
|
/**
|
|
* Clear the current event listeners and then re-listen for events based on the level specified
|
|
*
|
|
* @method setupListeners
|
|
* @private
|
|
* @param {Integer} level - The max log level that this logger should listen to
|
|
* @return {undefined}
|
|
*/
|
|
LoggerAbstract.prototype.setupListeners = function (levels) {
|
|
this.cleanUpListeners();
|
|
|
|
this.listeningLevels = levels;
|
|
|
|
_.each(this.listeningLevels, function (level) {
|
|
var fnName = 'on' + _.ucfirst(level);
|
|
if (this.bound[fnName]) {
|
|
this.log.on(level, this.bound[fnName]);
|
|
} else {
|
|
throw new Error(fnName + ' is not a function');
|
|
}
|
|
}, this);
|
|
};
|
|
|
|
/**
|
|
* Clear the current event listeners
|
|
*
|
|
* @method cleanUpListeners
|
|
* @private
|
|
* @return {undefined}
|
|
*/
|
|
LoggerAbstract.prototype.cleanUpListeners = _.handler(function () {
|
|
_.each(this.listeningLevels, function (level) {
|
|
this.log.removeListener(level, this.bound['on' + _.ucfirst(level)]);
|
|
}, this);
|
|
});
|
|
|
|
/**
|
|
* Handler for the logs "error" event
|
|
*
|
|
* @method onError
|
|
* @private
|
|
* @param {Error} e - The Error object to log
|
|
* @return {undefined}
|
|
*/
|
|
LoggerAbstract.prototype.onError = _.handler(function (e) {
|
|
this.write((e.name === 'Error' ? 'ERROR' : e.name), e.stack);
|
|
});
|
|
|
|
/**
|
|
* Handler for the logs "warning" event
|
|
*
|
|
* @method onWarning
|
|
* @private
|
|
* @param {String} msg - The message to be logged
|
|
* @return {undefined}
|
|
*/
|
|
LoggerAbstract.prototype.onWarning = _.handler(function (msg) {
|
|
this.write('WARNING', msg);
|
|
});
|
|
|
|
/**
|
|
* Handler for the logs "info" event
|
|
*
|
|
* @method onInfo
|
|
* @private
|
|
* @param {String} msg - The message to be logged
|
|
* @return {undefined}
|
|
*/
|
|
LoggerAbstract.prototype.onInfo = _.handler(function (msg) {
|
|
this.write('INFO', msg);
|
|
});
|
|
|
|
/**
|
|
* Handler for the logs "debug" event
|
|
*
|
|
* @method onDebug
|
|
* @private
|
|
* @param {String} msg - The message to be logged
|
|
* @return {undefined}
|
|
*/
|
|
LoggerAbstract.prototype.onDebug = _.handler(function (msg) {
|
|
this.write('DEBUG', msg);
|
|
});
|
|
|
|
/**
|
|
* Handler for the logs "trace" event
|
|
*
|
|
* @method onTrace
|
|
* @private
|
|
* @param {String} msg - The message to be logged
|
|
* @return {undefined}
|
|
*/
|
|
LoggerAbstract.prototype.onTrace = _.handler(function (message) {
|
|
this.write('TRACE', message);
|
|
});
|
|
|
|
|
|
module.exports = LoggerAbstract;
|
|
|
|
},{"./utils":79}],72:[function(require,module,exports){
|
|
module.exports = {
|
|
console: require('./console')
|
|
};
|
|
|
|
},{"./console":73}],73:[function(require,module,exports){
|
|
/**
|
|
* Special version of the Stream logger, which logs errors and warnings to stderr and all other
|
|
* levels to stdout.
|
|
*
|
|
* @class Loggers.Console
|
|
* @extends LoggerAbstract
|
|
* @constructor
|
|
* @param {Object} config - The configuration for the Logger
|
|
* @param {string} config.level - The highest log level for this logger to output.
|
|
* @param {Log} bridge - The object that triggers logging events, which we will record
|
|
*/
|
|
|
|
module.exports = Console;
|
|
|
|
var LoggerAbstract = require('../logger');
|
|
var _ = require('../utils');
|
|
|
|
function Console(log, config) {
|
|
// call my super
|
|
LoggerAbstract.call(this, log, config);
|
|
|
|
// config/state
|
|
this.color = _.has(config, 'color') ? !!config.color : true;
|
|
}
|
|
_.inherits(Console, LoggerAbstract);
|
|
|
|
/**
|
|
* Override the LoggerAbstract's setup listeners to do a little extra setup
|
|
*
|
|
* @param {Array} levels - The levels that we should be listeneing for
|
|
*/
|
|
Console.prototype.setupListeners = function (levels) {
|
|
// since some of our functions are bound a bit differently (to the console)
|
|
// create some of the bound properties manually
|
|
this.bound.onError = this.onError;
|
|
this.bound.onWarning = this.onWarning;
|
|
this.bound.onInfo = this.onInfo;
|
|
this.bound.onDebug = this.onDebug;
|
|
this.bound.onTrace = this.onTrace;
|
|
|
|
// call the super method
|
|
LoggerAbstract.prototype.setupListeners.call(this, levels);
|
|
};
|
|
|
|
/**
|
|
* Handler for the bridges "error" event
|
|
*
|
|
* @method onError
|
|
* @private
|
|
* @param {Error} e - The Error object to log
|
|
* @return {undefined}
|
|
*/
|
|
Console.prototype.onError = function (e) {
|
|
if (console.error && console.trace) {
|
|
console.error(e.name === 'Error' ? 'ERROR' : e.name, e.stack || e.message);
|
|
} else {
|
|
console.log(e.name === 'Error' ? 'ERROR' : e.name, e.stack || e.message);
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Handler for the bridges "warning" event
|
|
*
|
|
* @method onWarning
|
|
* @private
|
|
* @param {String} msg - The message to be logged
|
|
* @return {undefined}
|
|
*/
|
|
Console.prototype.onWarning = function (msg) {
|
|
console[console.warn ? 'warn' : 'log']('WARNING', msg);
|
|
};
|
|
|
|
/**
|
|
* Handler for the bridges "info" event
|
|
*
|
|
* @method onInfo
|
|
* @private
|
|
* @param {String} msg - The message to be logged
|
|
* @return {undefined}
|
|
*/
|
|
Console.prototype.onInfo = function (msg) {
|
|
console[console.warn ? 'info' : 'log']('INFO', msg);
|
|
};
|
|
|
|
/**
|
|
* Handler for the bridges "debug" event
|
|
*
|
|
* @method onDebug
|
|
* @private
|
|
* @param {String} msg - The message to be logged
|
|
* @return {undefined}
|
|
*/
|
|
Console.prototype.onDebug = function (msg) {
|
|
console[console.debug ? 'debug' : 'log']('DEBUG', msg);
|
|
};
|
|
/**
|
|
* Handler for the bridges "trace" event
|
|
*
|
|
* @method onTrace
|
|
* @private
|
|
* @return {undefined}
|
|
*/
|
|
Console.prototype.onTrace = function (message, curlCall) {
|
|
console.log('TRACE:\n' + curlCall + '\n' + message);
|
|
};
|
|
|
|
},{"../logger":71,"../utils":79}],74:[function(require,module,exports){
|
|
module.exports = {
|
|
random: require('./random'),
|
|
roundRobin: require('./round_robin')
|
|
};
|
|
|
|
},{"./random":75,"./round_robin":76}],75:[function(require,module,exports){
|
|
/**
|
|
* Selects a connection randomly
|
|
*
|
|
* @module selectors
|
|
* @type {Function}
|
|
* @param {Array} connection - The list of connections to choose from
|
|
* @return {Connection} - The selected connection
|
|
*/
|
|
module.exports = function RandomSelector(connections) {
|
|
return connections[Math.floor(Math.random() * connections.length)];
|
|
};
|
|
|
|
},{}],76:[function(require,module,exports){
|
|
/**
|
|
* Selects a connection the simplest way possible, Round Robin
|
|
*
|
|
* @module selectors
|
|
* @type {Function}
|
|
* @param {Array} connections - The list of connections that this selector needs to choose from
|
|
* @return {Connection} - The selected connection
|
|
*/
|
|
module.exports = function (connections) {
|
|
var connection = connections[0];
|
|
connections.push(connections.shift());
|
|
return connection;
|
|
};
|
|
|
|
},{}],77:[function(require,module,exports){
|
|
/**
|
|
* Simple JSON serializer
|
|
* @type {[type]}
|
|
*/
|
|
module.exports = Json;
|
|
|
|
var _ = require('../utils');
|
|
|
|
function Json() {}
|
|
|
|
Json.prototype.serialize = function (val, replacer, spaces) {
|
|
if (val == null) {
|
|
return;
|
|
}
|
|
else if (typeof val === 'string') {
|
|
return val;
|
|
} else {
|
|
return JSON.stringify(val, replacer, spaces);
|
|
}
|
|
};
|
|
|
|
Json.prototype.unserialize = function (str) {
|
|
if (typeof str === 'string') {
|
|
try {
|
|
return JSON.parse(str);
|
|
} catch (e) {
|
|
return;
|
|
}
|
|
} else {
|
|
return str;
|
|
}
|
|
};
|
|
|
|
Json.prototype.bulkBody = function (val) {
|
|
var body = '', i;
|
|
|
|
if (_.isArray(val)) {
|
|
for (i = 0; i < val.length; i++) {
|
|
body += this.serialize(val[i]) + '\n';
|
|
}
|
|
} else if (typeof val === 'string') {
|
|
// make sure the string ends in a new line
|
|
body = val + (val[val.length - 1] === '\n' ? '' : '\n');
|
|
} else {
|
|
throw new TypeError('Bulk body should either be an Array of commands/string, or a String');
|
|
}
|
|
|
|
return body;
|
|
};
|
|
|
|
},{"../utils":79}],78:[function(require,module,exports){
|
|
/**
|
|
* Class that manages making request, called by all of the API methods.
|
|
* @type {[type]}
|
|
*/
|
|
module.exports = Transport;
|
|
|
|
var _ = require('./utils');
|
|
var errors = require('./errors');
|
|
var Host = require('./host');
|
|
var Log = require('./log');
|
|
var when = require('when');
|
|
|
|
function Transport(config) {
|
|
config = config || {};
|
|
|
|
var LogClass;
|
|
// setup the log
|
|
switch (typeof config.log) {
|
|
case 'function':
|
|
LogClass = config.log;
|
|
break;
|
|
case 'undefined':
|
|
config.log = 'warning';
|
|
/* fall through */
|
|
default:
|
|
LogClass = Log;
|
|
}
|
|
|
|
config.log = this.log = new LogClass(config);
|
|
|
|
// overwrite the createDefer method if a new implementation is provided
|
|
if (typeof config.createDefer === 'function') {
|
|
this.createDefer = config.createDefer;
|
|
}
|
|
|
|
// setup the connection pool
|
|
var ConnectionPool = _.funcEnum(config, 'connectionPool', Transport.connectionPools, 'main');
|
|
this.connectionPool = new ConnectionPool(config);
|
|
|
|
if (config.hosts) {
|
|
var hosts = _.createArray(config.hosts, function (val) {
|
|
if (_.isPlainObject(val) || _.isString(val)) {
|
|
return val;
|
|
}
|
|
});
|
|
if (!hosts) {
|
|
throw new Error('Invalid hosts config. Expected a URL, an array of urls, a host config object, or an array of ' +
|
|
'host config objects.');
|
|
}
|
|
|
|
this.connectionPool.setHosts(_.map(hosts, function (conf) {
|
|
return new Host(conf);
|
|
}));
|
|
}
|
|
|
|
// setup the serializer
|
|
var Serializer = _.funcEnum(config, 'serializer', Transport.serializers, 'json');
|
|
this.serializer = new Serializer(config);
|
|
}
|
|
|
|
Transport.connectionPools = {
|
|
main: require('./connection_pool')
|
|
};
|
|
|
|
Transport.serializers = {
|
|
json: require('./serializers/json')
|
|
};
|
|
|
|
/**
|
|
* Perform a request with the client's transport
|
|
*
|
|
* @method request
|
|
* @todo async body writing
|
|
* @todo abort
|
|
* @todo access to custom headers, modifying of request in general
|
|
* @param {object} params
|
|
* @param {String} params.url - The url for the request
|
|
* @param {String} params.method - The HTTP method for the request
|
|
* @param {String} params.body - The body of the HTTP request
|
|
* @param {Function} cb - A function to call back with (error, responseBody, responseStatus)
|
|
*/
|
|
Transport.prototype.request = function (params, cb) {
|
|
|
|
var self = this;
|
|
var remainingRetries = this.maxRetries;
|
|
var connection; // set in sendReqWithConnection
|
|
var connectionReq; // an object with an abort method, set in sendReqWithConnection
|
|
var request; // the object returned to the user, might be a deferred
|
|
|
|
self.log.debug('starting request', params);
|
|
|
|
if (params.body && params.method === 'GET') {
|
|
_.nextTick(respond, new TypeError('Body can not be sent with method "GET"'));
|
|
return;
|
|
}
|
|
|
|
// serialize the body
|
|
if (params.body) {
|
|
params.body = self.serializer[params.bulkBody ? 'bulkBody' : 'serialize'](params.body);
|
|
}
|
|
|
|
params.req = {
|
|
timeout: params.timeout,
|
|
method: params.method,
|
|
path: params.path,
|
|
query: params.query,
|
|
body: params.body,
|
|
};
|
|
|
|
self.connectionPool.select(sendReqWithConnection);
|
|
|
|
function abortRequest() {
|
|
remainingRetries = 0;
|
|
connectionReq.abort();
|
|
}
|
|
|
|
function sendReqWithConnection(err, _connection) {
|
|
if (err) {
|
|
respond(err);
|
|
} else if (_connection) {
|
|
connection = _connection;
|
|
connectionReq = connection.request(params.req, checkRespForFailure);
|
|
} else {
|
|
self.log.warning('No living connections');
|
|
respond(new errors.NoConnections());
|
|
}
|
|
}
|
|
|
|
function checkRespForFailure(err, body, status) {
|
|
if (err && remainingRetries) {
|
|
remainingRetries--;
|
|
self.log.error(err.message, '-- retrying');
|
|
self.connectionPool.select(sendReqWithConnection);
|
|
} else {
|
|
self.log.info('Request complete');
|
|
respond(err, body, status);
|
|
}
|
|
}
|
|
|
|
function respond(err, body, status) {
|
|
var parsedBody;
|
|
|
|
if (!err && body) {
|
|
parsedBody = self.serializer.unserialize(body);
|
|
if (parsedBody == null) {
|
|
err = new errors.Serialization();
|
|
}
|
|
}
|
|
|
|
if (!err) {
|
|
if ((status < 200 || status >= 300)
|
|
&& (!params.ignore || !_.contains(params.ignore, status))
|
|
) {
|
|
if (errors[status]) {
|
|
err = new errors[status](parsedBody && parsedBody.error);
|
|
} else {
|
|
err = new errors.Generic('unknown error');
|
|
}
|
|
}
|
|
}
|
|
|
|
if (params.castExists) {
|
|
if (err && err instanceof errors.NotFound) {
|
|
parsedBody = false;
|
|
err = void 0;
|
|
} else {
|
|
parsedBody = !err;
|
|
}
|
|
}
|
|
|
|
if (typeof cb === 'function') {
|
|
cb(err, parsedBody, status);
|
|
} else if (err) {
|
|
request.reject(err);
|
|
} else {
|
|
request.resolve({
|
|
body: parsedBody,
|
|
status: status
|
|
});
|
|
}
|
|
}
|
|
|
|
// determine the API based on the presense of a callback
|
|
if (typeof cb === 'function') {
|
|
request = {
|
|
abort: abortRequest
|
|
};
|
|
} else {
|
|
var defer = this.createDefer();
|
|
defer.promise.abort = abortRequest;
|
|
request = defer.promise;
|
|
}
|
|
|
|
return request;
|
|
};
|
|
|
|
Transport.prototype.createDefer = function () {
|
|
return when.defer();
|
|
};
|
|
|
|
/**
|
|
* Ask an ES node for a list of all the nodes, add/remove nodes from the connection
|
|
* pool as appropriate
|
|
*
|
|
* @param {Function} cb - Function to call back once complete
|
|
*/
|
|
Transport.prototype.sniff = function (cb) {
|
|
var self = this;
|
|
|
|
// make cb a function if it isn't
|
|
cb = typeof cb === 'function' ? cb : _.noop;
|
|
|
|
this.request({
|
|
path: '/_cluster/nodes',
|
|
method: 'GET'
|
|
}, function (err, resp) {
|
|
if (!err && resp && resp.nodes) {
|
|
var hosts = _.map(self.nodesToHostCallback(resp.nodes), function (hostConfig) {
|
|
return new Host(hostConfig);
|
|
});
|
|
this.connectionPool.setHosts(hosts);
|
|
}
|
|
cb(err, resp);
|
|
});
|
|
};
|
|
|
|
Transport.prototype.close = function () {
|
|
this.log.close();
|
|
this.connectionPool.close();
|
|
};
|
|
|
|
},{"./connection_pool":63,"./errors":68,"./host":69,"./log":70,"./serializers/json":77,"./utils":79,"when":57}],79:[function(require,module,exports){
|
|
var process=require("__browserify_process");var path = require('path');
|
|
var _ = require('lodash');
|
|
var nodeUtils = require('util'); // not included in the browser build
|
|
|
|
/**
|
|
* Custom utils library, basically a modified version of [lodash](http://lodash.com/docs) +
|
|
* [node.utils](http://nodejs.org/api/util.html#util_util) that doesn't use mixins to prevent
|
|
* confusion when requiring lodash itself.
|
|
*
|
|
* @class utils
|
|
* @static
|
|
*/
|
|
var utils = _.extend({}, _, nodeUtils);
|
|
_ = utils;
|
|
|
|
utils.inspect = function (thing, opts) {
|
|
if (process.browser) {
|
|
try {
|
|
return JSON.stringify(thing, null, ' ') + '\n';
|
|
} catch (e) {
|
|
return thing ? thing.toString() : '' + thing;
|
|
}
|
|
} else {
|
|
return nodeUtils.inspect(thing, _.defaults(opts || {}, {
|
|
showHidden: true,
|
|
depth: null,
|
|
color: true
|
|
}));
|
|
}
|
|
};
|
|
|
|
|
|
/**
|
|
* Link to [path.join](http://nodejs.org/api/path.html#path_path_join_path1_path2)
|
|
*
|
|
* @method utils.joinPath
|
|
* @type {function}
|
|
*/
|
|
utils.joinPath = path.join;
|
|
|
|
/**
|
|
* Recursively re-key an object, applying "transform" to each key
|
|
* @param {Object} obj - The object to re-key
|
|
* @param {Function} transform - The transformation function to apply to each key
|
|
* @param {Boolean} [recursive=true] - Should this act recursively?
|
|
* @param {Object} out - used primarily for recursion, allows you to specify the object which new keys will be written to
|
|
* @return {Object}
|
|
*/
|
|
utils.reKey = function (obj, transform, recursive) {
|
|
// defaults
|
|
if (recursive === void 0) { recursive = true; }
|
|
if (typeof transform !== 'function') { throw new TypeError('invalid transform function'); }
|
|
|
|
var out = {};
|
|
|
|
_.each(obj, function (prop, name) {
|
|
if (recursive && _.isPlainObject(prop)) {
|
|
out[transform(name)] = utils.reKey(prop, transform, recursive);
|
|
} else {
|
|
out[transform(name)] = prop;
|
|
}
|
|
});
|
|
|
|
return out;
|
|
};
|
|
|
|
/**
|
|
* Recursively merge two objects, walking into each object and concating arrays. If both to and from have a value at a
|
|
* key, but the values' types don't match to's value is left unmodified. Only Array and Object values are merged - that
|
|
* it to say values with a typeof "object"
|
|
*
|
|
* @param {Object} to - Object to merge into (no cloning, the original object
|
|
* is modified)
|
|
* @param {Object} from - Object to pull changed from
|
|
* @return {Object} - returns the modified to value
|
|
*/
|
|
utils.deepMerge = function (to, from) {
|
|
Object.keys(from).forEach(function (key) {
|
|
switch (typeof to[key]) {
|
|
case 'undefined':
|
|
to[key] = from[key];
|
|
break;
|
|
case 'object':
|
|
if (_.isArray(to[key]) && _.isArray(from[key])) {
|
|
to[key] = to[key].concat(from[key]);
|
|
}
|
|
else if (_.isPlainObject(to[key]) && _.isPlainObject(from[key])) {
|
|
utils.deepMerge(to[key], from[key]);
|
|
}
|
|
}
|
|
});
|
|
return to;
|
|
};
|
|
|
|
/**
|
|
* Test if a value is an array and it's contents are of a specific type
|
|
*
|
|
* @method isArrayOf<Strings|Object|Array|Finite|Function|RegExp>s
|
|
* @param {Array} arr - An array to check
|
|
* @return {Boolean}
|
|
*/
|
|
'String Object PlainObject Array Finite Function RegExp'.split(' ').forEach(function (type) {
|
|
var check = _.bindKey(_, 'is' + type);
|
|
|
|
utils['isArrayOf' + type + 's'] = function (arr) {
|
|
// quick shallow check of arrays
|
|
return _.isArray(arr) && _.every(arr.slice(0, 10), check);
|
|
};
|
|
});
|
|
|
|
|
|
/**
|
|
* Capitalize the first letter of a word
|
|
*
|
|
* @method ucfirst
|
|
* @param {string} word - The word to transform
|
|
* @return {string}
|
|
*/
|
|
utils.ucfirst = function (word) {
|
|
return word[0].toUpperCase() + word.substring(1).toLowerCase();
|
|
};
|
|
|
|
/**
|
|
* Base algo for studlyCase and camelCase
|
|
* @param {boolean} firstWordCap - Should the first character of the first word be capitalized
|
|
* @return {Function}
|
|
*/
|
|
function adjustWordCase(firstWordCap, otherWordsCap, sep) {
|
|
return function (string) {
|
|
var i = 0;
|
|
var words = [];
|
|
var word = '';
|
|
var code, c, upper, lower;
|
|
|
|
for (; i < string.length; i++) {
|
|
code = string.charCodeAt(i);
|
|
c = string.charAt(i);
|
|
lower = code >= 97 && code <= 122;
|
|
upper = code >= 65 && code <= 90;
|
|
|
|
if (upper || !lower) {
|
|
// new word
|
|
if (word.length) {
|
|
words.push(word);
|
|
}
|
|
word = '';
|
|
}
|
|
|
|
if (upper || lower) {
|
|
if (lower && word.length) {
|
|
word += c;
|
|
} else {
|
|
if ((!words.length && firstWordCap) || (words.length && otherWordsCap)) {
|
|
word = c.toUpperCase();
|
|
}
|
|
else {
|
|
word = c.toLowerCase();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if (word.length) {
|
|
words.push(word);
|
|
}
|
|
// add the leading underscore back to strings the had it originally
|
|
if (words.lenth && string.charAt(0) === '_') {
|
|
words[0] = '_' + words[0];
|
|
}
|
|
return words.join(sep);
|
|
};
|
|
}
|
|
|
|
/**
|
|
* Transform a string into StudlyCase
|
|
*
|
|
* @method studlyCase
|
|
* @param {String} string
|
|
* @return {String}
|
|
*/
|
|
utils.studlyCase = adjustWordCase(true, true, '');
|
|
|
|
/**
|
|
* Transform a string into camelCase
|
|
*
|
|
* @method camelCase
|
|
* @param {String} string
|
|
* @return {String}
|
|
*/
|
|
utils.camelCase = adjustWordCase(false, true, '');
|
|
|
|
/**
|
|
* Transform a string into snakeCase
|
|
*
|
|
* @method snakeCase
|
|
* @param {String} string
|
|
* @return {String}
|
|
*/
|
|
utils.snakeCase = adjustWordCase(false, false, '_');
|
|
|
|
/**
|
|
* Lower-case a string, and return an empty string if any is not a string
|
|
*
|
|
* @param any {*} - Something or nothing
|
|
* @returns {string}
|
|
*/
|
|
utils.toLowerString = function (any) {
|
|
if (any) {
|
|
if (typeof any !== 'string') {
|
|
any = any.toString();
|
|
}
|
|
} else {
|
|
any = '';
|
|
}
|
|
return any.toLowerCase();
|
|
};
|
|
|
|
/**
|
|
* Upper-case the string, return an empty string if any is not a string
|
|
*
|
|
* @param any {*} - Something or nothing
|
|
* @returns {string}
|
|
*/
|
|
utils.toUpperString = function (any) {
|
|
if (any) {
|
|
if (typeof any !== 'string') {
|
|
any = any.toString();
|
|
}
|
|
} else {
|
|
any = '';
|
|
}
|
|
return any.toUpperCase();
|
|
};
|
|
|
|
/**
|
|
* Test if a value is "numeric" meaning that it can be transformed into something besides NaN
|
|
*
|
|
* @method isNumeric
|
|
* @param {*} val
|
|
* @return {Boolean}
|
|
*/
|
|
utils.isNumeric = function (val) {
|
|
return typeof val !== 'object' && val - parseFloat(val) >= 0;
|
|
};
|
|
|
|
// regexp to test for intervals
|
|
var intervalRE = /^(\d+(?:\.\d+)?)([Mwdhmsy])$/;
|
|
|
|
/**
|
|
* Test if a string represents an interval (eg. 1m, 2Y)
|
|
*
|
|
* @method isInterval
|
|
* @param {String} val
|
|
* @return {Boolean}
|
|
*/
|
|
utils.isInterval = function (val) {
|
|
return !!(val.match && val.match(intervalRE));
|
|
};
|
|
|
|
/**
|
|
* Repeat a string n times
|
|
*
|
|
* @todo TestPerformance
|
|
* @method repeat
|
|
* @param {String} what - The string to repeat
|
|
* @param {Number} times - Times the string should be repeated
|
|
* @return {String}
|
|
*/
|
|
utils.repeat = function (what, times) {
|
|
return (new Array(times + 1)).join(what);
|
|
};
|
|
|
|
/**
|
|
* Override node's util.inherits function, providing a browser safe version thanks to lodash
|
|
*
|
|
* @param constructor {Function} - the constructor that should subClass superConstructor
|
|
* @param superConstructor {Function} - The parent constructor
|
|
*/
|
|
utils.inherits = require('inherits');
|
|
|
|
/**
|
|
* Remove leading/trailing spaces from a string
|
|
*
|
|
* @param str {String} - Any string
|
|
* @returns {String}
|
|
*/
|
|
utils.trim = function (str) {
|
|
return typeof str === 'string' ? str.replace(/^\s+|\s+$/g, '') : '';
|
|
};
|
|
|
|
utils.collectMatches = function (text, regExp) {
|
|
var matches = [], match;
|
|
while (match = regExp.exec(text)) {
|
|
matches.push(match);
|
|
if (regExp.global !== true) {
|
|
// would loop forever if not true
|
|
break;
|
|
}
|
|
}
|
|
return matches;
|
|
};
|
|
|
|
/**
|
|
* Call a function, applying the arguments object to it in an optimized way, rather than always turning it into an array
|
|
*
|
|
* @param func {Function} - The function to execute
|
|
* @param context {*} - The context the function will be executed with
|
|
* @param args {Arguments} - The arguments to send to func
|
|
* @param [sliceIndex=0] {Integer} - The index that args should be sliced at, before feeding args to func
|
|
* @returns {*} - the return value of func
|
|
*/
|
|
utils.applyArgs = function (func, context, args, sliceIndex) {
|
|
sliceIndex = sliceIndex || 0;
|
|
switch (args.length - sliceIndex) {
|
|
case 0:
|
|
return func.call(context);
|
|
case 1:
|
|
return func.call(context, args[0 + sliceIndex]);
|
|
case 2:
|
|
return func.call(context, args[0 + sliceIndex], args[1 + sliceIndex]);
|
|
case 3:
|
|
return func.call(context, args[0 + sliceIndex], args[1 + sliceIndex], args[2 + sliceIndex]);
|
|
case 4:
|
|
return func.call(context, args[0 + sliceIndex], args[1 + sliceIndex], args[2 + sliceIndex], args[3 + sliceIndex]);
|
|
case 5:
|
|
return func.call(context, args[0 + sliceIndex], args[1 + sliceIndex],
|
|
args[2 + sliceIndex], args[3 + sliceIndex], args[4 + sliceIndex]);
|
|
default:
|
|
return func.apply(context, Array.prototype.slice.call(args, sliceIndex));
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Schedule a function to be called on the next tick, and supply it with these arguments
|
|
* when it is called.
|
|
* @return {[type]} [description]
|
|
*/
|
|
_.nextTick = function (cb) {
|
|
// bind the function and schedule it
|
|
process.nextTick(_.bindKey(_, 'applyArgs', cb, null, arguments, 1));
|
|
};
|
|
|
|
/**
|
|
* Marks a method as a handler. Currently this just makes a property on the method
|
|
* flagging it to be bound to the object at object creation when "makeBoundMethods" is called
|
|
*
|
|
* ```
|
|
* ClassName.prototype.methodName = _.handler(function () {
|
|
* // this will always be bound when called via classInstance.bound.methodName
|
|
* this === classInstance
|
|
* });
|
|
* ```
|
|
*
|
|
* @alias _.scheduled
|
|
* @param {Function} func - The method that is being defined
|
|
* @return {Function}
|
|
*/
|
|
_.handler = function (func) {
|
|
func._provideBound = true;
|
|
return func;
|
|
};
|
|
_.scheduled = _.handler;
|
|
|
|
/**
|
|
* Creates an "bound" property on an object, which all or a subset of methods from
|
|
* the object which are bound to the original object.
|
|
*
|
|
* ```
|
|
* var obj = {
|
|
* onEvent: function () {}
|
|
* };
|
|
*
|
|
* _.makeBoundMethods(obj);
|
|
*
|
|
* obj.bound.onEvent() // is bound to obj, and can safely be used as an event handler.
|
|
* ```
|
|
*
|
|
* @param {Object} obj - The object to bind the methods to
|
|
* @param {Array} [methods] - The methods to bind, false values === bind all flagged with _provideBound
|
|
*/
|
|
_.makeBoundMethods = function (obj, methods) {
|
|
obj.bound = {};
|
|
if (!methods) {
|
|
methods = [];
|
|
for (var prop in obj) {
|
|
// dearest maintainer, we want to look through the prototype
|
|
if (typeof obj[prop] === 'function' && obj[prop]._provideBound === true) {
|
|
obj.bound[prop] = _.bind(obj[prop], obj);
|
|
}
|
|
}
|
|
} else {
|
|
_.each(methods, function (method) {
|
|
obj.bound[method] = _.bindKey(obj, method);
|
|
});
|
|
}
|
|
};
|
|
|
|
_.noop = function () {};
|
|
|
|
/**
|
|
* Implements the standard "string or constructor" check that I was copy/pasting everywhere
|
|
* @param {String|Function} val - the value that the user passed in
|
|
* @param {Object} opts - a map of the options
|
|
* @return {Function|undefined} - If a valid option was specified, then the constructor is returned
|
|
*/
|
|
_.funcEnum = function (config, name, opts, def) {
|
|
var val = config[name];
|
|
switch (typeof val) {
|
|
case 'undefined':
|
|
return opts[def];
|
|
case 'function':
|
|
return val;
|
|
case 'string':
|
|
if (opts[val]) {
|
|
return opts[val];
|
|
}
|
|
/* falls through */
|
|
default:
|
|
throw new TypeError('Invalid ' + name + ' "' + val + '", expected a function or one of ' +
|
|
_.keys(opts).join(', '));
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Accepts any object and attempts to convert it into an array. If the object passed in is not
|
|
* an array it will be wrapped in one. Then the transform/map function will be called for each element
|
|
* and create a new array that is returned. If the map function fails to return something, the loop is
|
|
* halted and false is returned instead of an array.
|
|
*
|
|
* @param {*} input - The value to convert
|
|
* @param {Function} transform - A function called for each element of the resulting array
|
|
* @return {Array|false} - an array on success, or false on failure.
|
|
*/
|
|
_.createArray = function (input, transform) {
|
|
transform = typeof transform === 'function' ? transform : _.identity;
|
|
var output = [];
|
|
var item;
|
|
var i;
|
|
|
|
if (!_.isArray(input)) {
|
|
input = [input];
|
|
}
|
|
|
|
for (i = 0; i < input.length; i++) {
|
|
item = transform(input[i]);
|
|
if (item === void 0) {
|
|
return false;
|
|
} else {
|
|
output.push(item);
|
|
}
|
|
}
|
|
return output;
|
|
};
|
|
|
|
module.exports = utils;
|
|
|
|
},{"__browserify_process":15,"inherits":17,"lodash":50,"path":7,"util":10}],80:[function(require,module,exports){
|
|
var process=require("__browserify_process");var path = require('path');
|
|
var _ = require('../../../src/lib/utils');
|
|
|
|
var defaults = {
|
|
executable: process.env.ES_HOME ? path.join(process.env.ES_HOME, './bin/elasticsearch') : null,
|
|
clusterName: 'yaml-test-runner',
|
|
dataPath: '/tmp/yaml-test-runner',
|
|
host: 'localhost',
|
|
port: '9200',
|
|
match: '**'
|
|
};
|
|
|
|
if (process.browser) {
|
|
/* jshint browser:true */
|
|
if (window.ES_HOST) {
|
|
defaults.host = window.ES_HOST;
|
|
}
|
|
|
|
if (window.ES_PORT) {
|
|
defaults.port = window.ES_PORT;
|
|
}
|
|
|
|
module.exports = defaults;
|
|
} else {
|
|
module.exports = require('optimist')
|
|
.default(defaults)
|
|
.boolean('createServer')
|
|
.argv;
|
|
}
|
|
|
|
},{"../../../src/lib/utils":79,"__browserify_process":15,"optimist":54,"path":7}],81:[function(require,module,exports){
|
|
var process=require("__browserify_process");if (process.browser) {
|
|
/* jshint browser: true */
|
|
var es = window.elasticsearch;
|
|
} else {
|
|
var es = require('../../../src/elasticsearch');
|
|
}
|
|
var argv = require('./argv');
|
|
var server = require('./server');
|
|
var path = require('path');
|
|
var fs = require('fs');
|
|
var _ = require('../../../src/lib/utils');
|
|
|
|
// current client
|
|
var client = null;
|
|
|
|
// when set to a boolean, hold the test of a ping
|
|
var externalExists;
|
|
|
|
// a reference to a personal instance of ES Server
|
|
var esServer = null;
|
|
|
|
module.exports = {
|
|
create: function create(cb) {
|
|
if (argv.createServer || externalExists === false) {
|
|
if (!esServer) {
|
|
server.start(function (err, _server) {
|
|
esServer = _server;
|
|
if (err) {
|
|
done(err);
|
|
} else {
|
|
doCreateClient(done);
|
|
}
|
|
});
|
|
} else {
|
|
doCreateClient(done);
|
|
}
|
|
} else if (externalExists === void 0) {
|
|
doCreateClient(function () {
|
|
client.ping(function (err) {
|
|
if (err instanceof es.errors.ConnectionFault) {
|
|
externalExists = !err;
|
|
create(done);
|
|
} else {
|
|
done(err);
|
|
}
|
|
});
|
|
});
|
|
} else {
|
|
doCreateClient(done);
|
|
}
|
|
|
|
function done(err) {
|
|
cb(err, client);
|
|
}
|
|
|
|
function doCreateClient(cb) {
|
|
// close existing client
|
|
if (client) {
|
|
client.close();
|
|
}
|
|
|
|
client = new es.Client({
|
|
hosts: [
|
|
{
|
|
host: esServer ? esServer.__hostname : argv.host,
|
|
port: esServer ? esServer.__port : argv.port
|
|
}
|
|
],
|
|
log: {
|
|
type: process.browser ? 'console' : 'stdio',
|
|
level: 'trace',
|
|
color: false
|
|
}
|
|
});
|
|
|
|
_.nextTick(cb);
|
|
}
|
|
},
|
|
get: function () {
|
|
return client;
|
|
}
|
|
};
|
|
|
|
},{"../../../src/elasticsearch":58,"../../../src/lib/utils":79,"./argv":80,"./server":83,"__browserify_process":15,"fs":6,"path":7}],82:[function(require,module,exports){
|
|
var __dirname="/";var path = require('path');
|
|
var async = require('async');
|
|
var jsYaml = require('js-yaml');
|
|
var expect = require('expect.js');
|
|
var YamlFile = require('./yaml_file');
|
|
var _ = require('../../../src/lib/utils');
|
|
var es = require('../../../src/elasticsearch');
|
|
var clientManager = require('./client_manager');
|
|
var Minimatch = require('minimatch').Minimatch;
|
|
var argv = require('./argv');
|
|
var testDir = path.resolve(__dirname, './tests');
|
|
var doPattern = new Minimatch(argv.match);
|
|
|
|
// before running any tests...
|
|
before(function (done) {
|
|
// start our personal ES Server
|
|
this.timeout(null);
|
|
clientManager.create(done);
|
|
});
|
|
|
|
before(function (done) {
|
|
// make sure ES is empty
|
|
clientManager.get().indices.delete({
|
|
index: '*',
|
|
ignore: 404
|
|
}, done);
|
|
});
|
|
|
|
var files = _.map(require('./yaml_tests.json'), function (docs, filename) {
|
|
if (doPattern.match(filename)) {
|
|
return new YamlFile(filename, docs);
|
|
}
|
|
});
|
|
|
|
},{"../../../src/elasticsearch":58,"../../../src/lib/utils":79,"./argv":80,"./client_manager":81,"./yaml_file":85,"./yaml_tests.json":86,"async":1,"expect.js":16,"js-yaml":18,"minimatch":51,"path":7}],83:[function(require,module,exports){
|
|
var process=require("__browserify_process");var childProc = require('child_process');
|
|
var events = require('events');
|
|
var path = require('path');
|
|
var fs = require('fs');
|
|
var _ = require('../../../src/lib/utils');
|
|
var argv = require('./argv');
|
|
|
|
exports.start = function (cb) {
|
|
|
|
if (!argv.executable || !fs.existsSync(argv.executable)) {
|
|
return cb(new Error('unable to find elasticsearch executable, ' +
|
|
'set ES_HOME env var to the instalation path of elasticsearch'));
|
|
}
|
|
|
|
var server = childProc.spawn(
|
|
argv.executable,
|
|
[
|
|
'-f',
|
|
'-Des.cluster.name=' + argv.clusterName,
|
|
'-Des.path.data=' + argv.dataPath,
|
|
// '-Des.logger.level=DEBUG',
|
|
'-Des.discovery.zen.ping.multicast.enabled=false',
|
|
],
|
|
{
|
|
cwd: void 0,
|
|
env: process.env,
|
|
stdio: [
|
|
'ignore',
|
|
'pipe',
|
|
'pipe'
|
|
]
|
|
}
|
|
);
|
|
|
|
server.stdout.on('data', function onProcessData(line) {
|
|
line = _.trim(line.toString());
|
|
var match;
|
|
if (match = line.match(/\{inet\[\/?([^:]+):(\d+)\]\}/)) {
|
|
server.__hostname = match[1];
|
|
server.__port = match[2];
|
|
}
|
|
|
|
if (line.match(/started\s*$/m)) {
|
|
console.log('Personal ES Server started at', server.__hostname + ':' + server.__port);
|
|
server.stdout.removeListener('data', onProcessData);
|
|
server.stdout.resume();
|
|
cb(null, server);
|
|
}
|
|
});
|
|
|
|
server.stderr.on('data', function (line) {
|
|
console.error(_.trim(line.toString()));
|
|
});
|
|
|
|
server.on('close', function (code) {
|
|
server.stdout.removeAllListeners();
|
|
server.stderr.removeAllListeners();
|
|
console.log('Personal ES Server Shutdown with exit code', code);
|
|
});
|
|
|
|
process.on('exit', function () {
|
|
server.kill();
|
|
});
|
|
|
|
};
|
|
|
|
},{"../../../src/lib/utils":79,"./argv":80,"__browserify_process":15,"child_process":4,"events":5,"fs":6,"path":7}],84:[function(require,module,exports){
|
|
var process=require("__browserify_process");/**
|
|
* Class to wrap a single document from a yaml test file
|
|
*
|
|
* @constructor
|
|
* @class YamlDoc
|
|
* @param actions {Array} - The array of actions directly from the Yaml file
|
|
*/
|
|
module.exports = YamlDoc;
|
|
|
|
var _ = require('../../../src/lib/utils');
|
|
var clientManager = require('./client_manager');
|
|
var expect = require('expect.js');
|
|
|
|
/**
|
|
* The version that ES is running, in comparable string form XXX-XXX-XXX, fetched when needed
|
|
* @type {String}
|
|
*/
|
|
var ES_VERSION = null;
|
|
|
|
// core expression for finding a version
|
|
var versionExp = '([\\d\\.]*\\d)(?:\\.\\w+)?';
|
|
|
|
/**
|
|
* Regular Expression to extract a version number from a string
|
|
* @type {RegExp}
|
|
*/
|
|
var versionRE = new RegExp(versionExp);
|
|
|
|
/**
|
|
* Regular Expression to extract a version range from a string
|
|
* @type {RegExp}
|
|
*/
|
|
var versionRangeRE = new RegExp(versionExp + '\\s*\\-\\s*' + versionExp);
|
|
|
|
|
|
/**
|
|
* Fetches the client.info, and parses out the version number to a comparable string
|
|
* @param done {Function} - callback
|
|
*/
|
|
function getVersionFromES(done) {
|
|
clientManager.get().info({}, function (err, resp) {
|
|
if (err) {
|
|
throw new Error('unable to get info about ES');
|
|
}
|
|
expect(resp.version.number).to.match(versionRE);
|
|
ES_VERSION = versionToComparableString(versionRE.exec(resp.version.number)[1]);
|
|
done();
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Transform x.x.x into xxx.xxx.xxx, striping off any text at the end like beta or pre-alpha35
|
|
*
|
|
* @param {String} version - Version number represented as a string
|
|
* @return {String} - Version number represented as three numbers, seperated by -, all numbers are
|
|
* padded with 0 and will be three characters long so the strings can be compared.
|
|
*/
|
|
function versionToComparableString(version) {
|
|
var parts = _.map(version.split('.'), function (part) {
|
|
part = '' + _.parseInt(part);
|
|
return (new Array(4 - part.length)).join('0') + part;
|
|
});
|
|
|
|
while (parts.length < 3) {
|
|
parts.push('000');
|
|
}
|
|
|
|
return parts.join('-');
|
|
}
|
|
|
|
/**
|
|
* Compare a version range to the ES_VERSION, determining if the current version
|
|
* falls within the range.
|
|
*
|
|
* @param {String} rangeString - a string representing two version numbers seperated by a "-"
|
|
* @return {Boolean} - is the current version within the range (inclusive)
|
|
*/
|
|
function rangeMatchesCurrentVersion(rangeString, done) {
|
|
function doWork() {
|
|
expect(rangeString).to.match(versionRangeRE);
|
|
|
|
var range = versionRangeRE.exec(rangeString);
|
|
range = _.map(_.last(range, 2), versionToComparableString);
|
|
|
|
done(ES_VERSION >= range[0] && ES_VERSION <= range[1]);
|
|
}
|
|
|
|
if (!ES_VERSION) {
|
|
getVersionFromES(doWork);
|
|
} else {
|
|
doWork();
|
|
}
|
|
}
|
|
|
|
// empty all of the indices in ES please
|
|
function clearIndices(done) {
|
|
clientManager.get().indices.delete({
|
|
index: '*',
|
|
ignore: 404
|
|
}, done);
|
|
}
|
|
|
|
function YamlDoc(doc, file) {
|
|
var self = this;
|
|
|
|
self.file = file;
|
|
self.description = _.keys(doc).shift();
|
|
self._stash = {};
|
|
self._last_requests_response = null;
|
|
|
|
// setup the actions, creating a bound and testable method for each
|
|
self._actions = _.map(self.flattenTestActions(doc[self.description]), function (action, i) {
|
|
// get the method that will do the action
|
|
var method = self['do_' + action.name];
|
|
|
|
// check that it's a function
|
|
expect(method).to.be.a('function');
|
|
|
|
if (_.isPlainObject(action.args)) {
|
|
action.name += ' ' + _.keys(action.args).join(', ');
|
|
} else if (action.args) {
|
|
action.name += ' ' + action.args;
|
|
}
|
|
|
|
// wrap in a check for skipping
|
|
action.bound = _.bind(method, self, action.args);
|
|
|
|
// create a function that can be passed to
|
|
action.testable = function (done) {
|
|
if (self.skipping || self.file.skipping) {
|
|
return done();
|
|
}
|
|
if (method.length > 1) {
|
|
action.bound(done);
|
|
} else {
|
|
action.bound();
|
|
done();
|
|
}
|
|
};
|
|
|
|
return action;
|
|
});
|
|
}
|
|
|
|
YamlDoc.prototype = {
|
|
|
|
/**
|
|
* convert tests actions
|
|
* from: [ {name:args, name:args}, {name:args}, ... ]
|
|
* to: [ {name:'', args:'' }, {name:'', args:''} ]
|
|
* so it's easier to work with
|
|
* @param {ArrayOfObjects} config - Actions to be taken as defined in the yaml specs
|
|
*/
|
|
flattenTestActions: function (config) {
|
|
// creates [ [ {name:"", args:"" }, ... ], ... ]
|
|
// from [ {name:args, name:args}, {name:args} ]
|
|
var actionSets = _.map(config, function (set) {
|
|
return _.map(_.pairs(set), function (pair) {
|
|
return { name: pair[0], args: pair[1] };
|
|
});
|
|
});
|
|
|
|
// do a single level flatten, merge=ing the nested arrays from step one
|
|
// into a master array, creating an array of action objects
|
|
return _.reduce(actionSets, function (note, set) {
|
|
return note.concat(set);
|
|
}, []);
|
|
},
|
|
|
|
/**
|
|
* Itterate over each of the actions, provides the testable function, and a name/description.
|
|
* return a litteral false to stop itterating
|
|
* @param {Function} ittr - The function to call for each action.
|
|
* @return {undefined}
|
|
*/
|
|
each: function (ittr) {
|
|
for (var i = 0; i < this._actions.length; i++) {
|
|
if (ittr(this._actions[i].testable, this._actions[i].name) === false) {
|
|
break;
|
|
}
|
|
}
|
|
},
|
|
|
|
/**
|
|
* Get a value from the last response, using dot-notation
|
|
*
|
|
* Example
|
|
* ===
|
|
*
|
|
* get '_source.tags.1'
|
|
*
|
|
* from {
|
|
* _source: {
|
|
* tags: [
|
|
* 'one',
|
|
* 'two'
|
|
* ]
|
|
* }
|
|
* }
|
|
*
|
|
* returns 'two'
|
|
*
|
|
* @param {string} path - The dot-notation path to the value needed.
|
|
* @return {*} - The value requested, or undefined if it was not found
|
|
*/
|
|
get: function (path, from) {
|
|
|
|
var log = process.env.LOG_GETS && !from ? console.log.bind(console) : function () {};
|
|
var i;
|
|
|
|
if (!from) {
|
|
if (path[0] === '$') {
|
|
from = this._stash;
|
|
path = path.substring(1);
|
|
} else {
|
|
from = this._last_requests_response;
|
|
}
|
|
}
|
|
|
|
log('getting', path, 'from', from);
|
|
|
|
var steps = path ? path.split('.') : [];
|
|
var remainingSteps;
|
|
|
|
for (i = 0; from != null && i < steps.length; i++) {
|
|
if (from[steps[i]] === void 0) {
|
|
remainingSteps = steps.slice(i).join('.').replace(/\\\./g, '.');
|
|
from = from[remainingSteps];
|
|
break;
|
|
} else {
|
|
from = from[steps[i]];
|
|
}
|
|
}
|
|
|
|
log('found', typeof from !== 'function' ? from : 'function');
|
|
return from;
|
|
},
|
|
|
|
/**
|
|
* Do a skip operation, setting the skipping flag to true if the version matches
|
|
* the range defined in args.version
|
|
*
|
|
* @param args
|
|
* @param done
|
|
*/
|
|
do_skip: function (args, done) {
|
|
rangeMatchesCurrentVersion(args.version, _.bind(function (match) {
|
|
if (match) {
|
|
if (this.description === 'setup') {
|
|
this.file.skipping = true;
|
|
// console.log('skipping this file' + (args.reason ? ' because ' + args.reason : ''));
|
|
} else {
|
|
this.skipping = true;
|
|
// console.log('skipping the rest of this doc' + (args.reason ? ' because ' + args.reason : ''));
|
|
}
|
|
} else {
|
|
this.skipping = false;
|
|
this.file.skipping = false;
|
|
}
|
|
done();
|
|
}, this));
|
|
},
|
|
|
|
/**
|
|
* Do a request, as outlined in the args
|
|
*
|
|
* @param {[type]} args [description]
|
|
* @param {Function} done [description]
|
|
* @return {[type]} [description]
|
|
*/
|
|
do_do: function (args, done) {
|
|
var catcher;
|
|
|
|
// resolve the catch arg to a value used for matching once the request is complete
|
|
switch (args.catch) {
|
|
case void 0:
|
|
catcher = null;
|
|
break;
|
|
case 'missing':
|
|
catcher = 404;
|
|
break;
|
|
case 'conflict':
|
|
catcher = 409;
|
|
break;
|
|
case 'forbidden':
|
|
catcher = 403;
|
|
break;
|
|
case 'request':
|
|
catcher = /.*/;
|
|
break;
|
|
case 'param':
|
|
catcher = TypeError;
|
|
break;
|
|
default:
|
|
catcher = args.catch.match(/^\/(.*)\/$/);
|
|
if (catcher) {
|
|
catcher = new RegExp(catcher[1]);
|
|
}
|
|
}
|
|
|
|
delete args.catch;
|
|
|
|
var client = clientManager.get();
|
|
var action = Object.keys(args).pop();
|
|
var clientActionName = _.map(action.split('.'), _.camelCase).join('.');
|
|
var clientAction = this.get(clientActionName, client);
|
|
var params = _.transform(args[action], function (note, val, name) {
|
|
note[_.camelCase(name)] = (typeof val === 'string' && val[0] === '$') ? this.get(val) : val;
|
|
}, {}, this);
|
|
|
|
expect(clientAction || clientActionName).to.be.a('function');
|
|
|
|
if (typeof clientAction === 'function') {
|
|
if (_.isNumeric(catcher)) {
|
|
params.ignore = _.union(params.ignore || [], [catcher]);
|
|
catcher = null;
|
|
}
|
|
|
|
var cb = _.bind(function (error, body, status) {
|
|
this._last_requests_response = body;
|
|
|
|
if (error) {
|
|
if (catcher) {
|
|
if (catcher instanceof RegExp) {
|
|
// error message should match the regexp
|
|
expect(error.message).to.match(catcher);
|
|
error = null;
|
|
} else if (typeof catcher === 'function') {
|
|
// error should be an instance of
|
|
expect(error).to.be.a(catcher);
|
|
error = null;
|
|
} else {
|
|
return done(new Error('Invalid catcher ' + catcher));
|
|
}
|
|
} else {
|
|
return done(error);
|
|
}
|
|
}
|
|
|
|
done(error);
|
|
}, this);
|
|
|
|
clientAction.call(client, params, cb);
|
|
} else {
|
|
done(new Error('stepped in do_do, did not find a function'));
|
|
}
|
|
|
|
},
|
|
|
|
/**
|
|
* Set a value from the respose into the stash
|
|
*
|
|
* Example
|
|
* ====
|
|
* { _id: id } # stash the value of `response._id` as `id`
|
|
*
|
|
* @param {Object} args - The object set to the "set" key in the test
|
|
* @return {undefined}
|
|
*/
|
|
do_set: function (args) {
|
|
_.forOwn(args, function (name, path) {
|
|
this._stash[name] = this.get(path);
|
|
}, this);
|
|
},
|
|
|
|
/**
|
|
* Test that the specified path exists in the response and has a
|
|
* true value (eg. not 0, false, undefined, null or the empty string)
|
|
*
|
|
* @param {string} path - Path to the response value to test
|
|
* @return {undefined}
|
|
*/
|
|
do_is_true: function (path) {
|
|
expect(this.get(path)).to.be.ok;
|
|
},
|
|
|
|
/**
|
|
* Test that the specified path exists in the response and has a
|
|
* false value (eg. 0, false, undefined, null or the empty string)
|
|
*
|
|
* @param {string} path - Path to the response value to test
|
|
* @return {undefined}
|
|
*/
|
|
do_is_false: function (path) {
|
|
expect(this.get(path)).to.not.be.ok;
|
|
},
|
|
|
|
/**
|
|
* Test that the response field (arg key) matches the value specified
|
|
*
|
|
* @param {Object} args - Hash of fields->values that need to be checked
|
|
* @return {undefined}
|
|
*/
|
|
do_match: function (args) {
|
|
_.forOwn(args, function (val, path) {
|
|
if (val[0] === '$') {
|
|
val = this.get(val);
|
|
}
|
|
expect(this.get(path)).to.eql(val);
|
|
}, this);
|
|
},
|
|
|
|
/**
|
|
* Test that the response field (arg key) is less than the value specified
|
|
*
|
|
* @param {Object} args - Hash of fields->values that need to be checked
|
|
* @return {undefined}
|
|
*/
|
|
do_lt: function (args) {
|
|
_.forOwn(args, function (num, path) {
|
|
expect(this.get(path)).to.be.below(num);
|
|
}, this);
|
|
},
|
|
|
|
/**
|
|
* Test that the response field (arg key) is greater than the value specified
|
|
*
|
|
* @param {Object} args - Hash of fields->values that need to be checked
|
|
* @return {undefined}
|
|
*/
|
|
do_gt: function (args) {
|
|
_.forOwn(args, function (num, path) {
|
|
expect(this.get(path)).to.be.above(num);
|
|
}, this);
|
|
},
|
|
|
|
/**
|
|
* Test that the response field (arg key) has a length equal to that specified.
|
|
* For object values, checks the length of the keys.
|
|
*
|
|
* @param {Object} args - Hash of fields->values that need to be checked
|
|
* @return {undefined}
|
|
*/
|
|
do_length: function (args) {
|
|
_.forOwn(args, function (len, path) {
|
|
expect(_.size(this.get(path))).to.be(len);
|
|
}, this);
|
|
}
|
|
};
|
|
|
|
},{"../../../src/lib/utils":79,"./client_manager":81,"__browserify_process":15,"expect.js":16}],85:[function(require,module,exports){
|
|
/**
|
|
* Class representing a YAML file
|
|
* @type {[type]}
|
|
*/
|
|
module.exports = YamlFile;
|
|
|
|
var YamlDoc = require('./yaml_doc');
|
|
var clientManager = require('./client_manager');
|
|
var _ = require('../../../src/lib/utils');
|
|
var async = require('async');
|
|
|
|
function YamlFile(filename, docs) {
|
|
var file = this;
|
|
|
|
// file level skipping flag
|
|
file.skipping = false;
|
|
|
|
describe(filename, function () {
|
|
file.docs = _.map(docs, function (doc) {
|
|
doc = new YamlDoc(doc, file);
|
|
if (doc.description === 'setup') {
|
|
beforeEach(/* doc */function (done) {
|
|
// console.log('setting up:', filename);
|
|
async.series(_.pluck(doc._actions, 'testable'), done);
|
|
});
|
|
} else {
|
|
it(doc.description, function (done) {
|
|
// console.log('test doc');
|
|
async.series(_.pluck(doc._actions, 'testable'), done);
|
|
});
|
|
}
|
|
});
|
|
|
|
afterEach(/* doc */function (done) {
|
|
// console.log('clearing indices');
|
|
clientManager.get().indices.delete({
|
|
index: '*',
|
|
ignore: 404
|
|
}, done);
|
|
});
|
|
});
|
|
|
|
}
|
|
|
|
},{"../../../src/lib/utils":79,"./client_manager":81,"./yaml_doc":84,"async":1}],86:[function(require,module,exports){
|
|
module.exports={
|
|
"bulk/10_basic.yaml": [
|
|
{
|
|
"Array of objects": [
|
|
{
|
|
"do": {
|
|
"bulk": {
|
|
"refresh": true,
|
|
"body": [
|
|
{
|
|
"index": {
|
|
"_index": "test_index",
|
|
"_type": "test_type",
|
|
"_id": "test_id"
|
|
}
|
|
},
|
|
{
|
|
"f1": "v1",
|
|
"f2": 42
|
|
},
|
|
{
|
|
"index": {
|
|
"_index": "test_index",
|
|
"_type": "test_type",
|
|
"_id": "test_id2"
|
|
}
|
|
},
|
|
{
|
|
"f1": "v2",
|
|
"f2": 47
|
|
}
|
|
]
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"count": {
|
|
"index": "test_index"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"count": 2
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"bulk/20_list_of_strings.yaml": [
|
|
{
|
|
"List of strings": [
|
|
{
|
|
"do": {
|
|
"bulk": {
|
|
"refresh": true,
|
|
"body": [
|
|
"{\"index\": {\"_index\": \"test_index\", \"_type\": \"test_type\", \"_id\": \"test_id\"}}",
|
|
"{\"f1\": \"v1\", \"f2\": 42}",
|
|
"{\"index\": {\"_index\": \"test_index\", \"_type\": \"test_type\", \"_id\": \"test_id2\"}}",
|
|
"{\"f1\": \"v2\", \"f2\": 47}"
|
|
]
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"count": {
|
|
"index": "test_index"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"count": 2
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"bulk/30_big_string.yaml": [
|
|
{
|
|
"One big string": [
|
|
{
|
|
"do": {
|
|
"bulk": {
|
|
"refresh": true,
|
|
"body": "\n{\"index\": {\"_index\": \"test_index\", \"_type\": \"test_type\", \"_id\": \"test_id\"}}\n{\"f1\": \"v1\", \"f2\": 42}\n{\"index\": {\"_index\": \"test_index\", \"_type\": \"test_type\", \"_id\": \"test_id2\"}}\n{\"f1\": \"v2\", \"f2\": 47}\n"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"count": {
|
|
"index": "test_index"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"count": 2
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"cluster.node_info/10_basic.yaml": [
|
|
{
|
|
"node_info test": [
|
|
{
|
|
"do": {
|
|
"cluster.node_info": {}
|
|
}
|
|
},
|
|
{
|
|
"is_true": "ok"
|
|
},
|
|
{
|
|
"is_true": "nodes"
|
|
},
|
|
{
|
|
"is_true": "cluster_name"
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"cluster.node_stats/10_basic.yaml": [
|
|
{
|
|
"Nodes stats": [
|
|
{
|
|
"do": {
|
|
"cluster.node_stats": {
|
|
"indices": true,
|
|
"transport": true
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_true": "cluster_name"
|
|
},
|
|
{
|
|
"is_true": "nodes"
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"cluster.put_settings/10_basic.yaml": [
|
|
{
|
|
"Test put settings": [
|
|
{
|
|
"do": {
|
|
"cluster.put_settings": {
|
|
"body": {
|
|
"transient": {
|
|
"discovery.zen.minimum_master_nodes": 1
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"transient": {
|
|
"discovery.zen.minimum_master_nodes": "1"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.get_settings": {}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"transient": {
|
|
"discovery.zen.minimum_master_nodes": "1"
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"cluster.reroute/10_basic.yaml": [
|
|
{
|
|
"Basic sanity check": [
|
|
{
|
|
"do": {
|
|
"cluster.reroute": {}
|
|
}
|
|
},
|
|
{
|
|
"is_true": "ok"
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"cluster.state/10_basic.yaml": [
|
|
{
|
|
"cluster state test": [
|
|
{
|
|
"do": {
|
|
"cluster.state": {}
|
|
}
|
|
},
|
|
{
|
|
"is_true": "master_node"
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"create/10_with_id.yaml": [
|
|
{
|
|
"Create with ID": [
|
|
{
|
|
"do": {
|
|
"create": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_true": "ok"
|
|
},
|
|
{
|
|
"match": {
|
|
"_index": "test_1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_type": "test"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_id": "1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_version": 1
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_index": "test_1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_type": "test"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_id": "1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_version": 1
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_source": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "conflict",
|
|
"create": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"create/15_without_id.yaml": [
|
|
{
|
|
"Create without ID": [
|
|
{
|
|
"do": {
|
|
"create": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_true": "ok"
|
|
},
|
|
{
|
|
"is_true": "_id"
|
|
},
|
|
{
|
|
"match": {
|
|
"_index": "test_1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_type": "test"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_version": 1
|
|
}
|
|
},
|
|
{
|
|
"set": {
|
|
"_id": "id"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": "$id"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_index": "test_1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_type": "test"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_id": "$id"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_version": 1
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_source": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"create/30_internal_version.yaml": [
|
|
{
|
|
"Internal version": [
|
|
{
|
|
"do": {
|
|
"create": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_version": 1
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "conflict",
|
|
"create": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"create/35_external_version.yaml": [
|
|
{
|
|
"External version": [
|
|
{
|
|
"do": {
|
|
"create": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
},
|
|
"version_type": "external",
|
|
"version": 5
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_version": 5
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "conflict",
|
|
"create": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
},
|
|
"version_type": "external",
|
|
"version": 5
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "conflict",
|
|
"create": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
},
|
|
"version_type": "external",
|
|
"version": 6
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"create/40_routing.yaml": [
|
|
{
|
|
"Routing": [
|
|
{
|
|
"do": {
|
|
"create": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"routing": 5,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"routing": 5,
|
|
"fields": [
|
|
"_routing"
|
|
]
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_id": "1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"fields._routing": "5"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "missing",
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"create/50_parent.yaml": [
|
|
{
|
|
"Parent": [
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_1",
|
|
"body": {
|
|
"mappings": {
|
|
"test": {
|
|
"_parent": {
|
|
"type": "foo"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "/RoutingMissingException/",
|
|
"create": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"create": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"parent": 5,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"parent": 5,
|
|
"fields": [
|
|
"_parent",
|
|
"_routing"
|
|
]
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_id": "1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"fields._parent": "5"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"fields._routing": "5"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "missing",
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"create/55_parent_with_routing.yaml": [
|
|
{
|
|
"Parent with routing": [
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_1",
|
|
"body": {
|
|
"mappings": {
|
|
"test": {
|
|
"_parent": {
|
|
"type": "foo"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"create": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"parent": 5,
|
|
"routing": 4,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"parent": 5,
|
|
"routing": 4,
|
|
"fields": [
|
|
"_parent",
|
|
"_routing"
|
|
]
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_id": "1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"fields._parent": "5"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"fields._routing": "4"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "missing",
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"parent": 5
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"routing": 4
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"create/60_refresh.yaml": [
|
|
{
|
|
"Refresh": [
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_1",
|
|
"body": {
|
|
"settings": {
|
|
"index.refresh_interval": -1
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"create": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"search": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"body": {
|
|
"query": {
|
|
"term": {
|
|
"_id": 1
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"hits.total": 0
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"create": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 2,
|
|
"refresh": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"search": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"body": {
|
|
"query": {
|
|
"term": {
|
|
"_id": 2
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"hits.total": 1
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"create/70_timestamp.yaml": [
|
|
{
|
|
"Timestamp": [
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_1",
|
|
"body": {
|
|
"mappings": {
|
|
"test": {
|
|
"_timestamp": {
|
|
"enabled": 1,
|
|
"store": "yes"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"create": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"fields": "_timestamp"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_true": "fields._timestamp"
|
|
},
|
|
{
|
|
"do": {
|
|
"delete": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"create": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
},
|
|
"timestamp": 1372011280000
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"fields": "_timestamp"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"fields._timestamp": 1372011280000
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"delete": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"create": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
},
|
|
"timestamp": "2013-06-23T18:14:40.000Z"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"fields": "_timestamp"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"fields._timestamp": 1372011280000
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"create/75_ttl.yaml": [
|
|
{
|
|
"TTL": [
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_1",
|
|
"body": {
|
|
"mappings": {
|
|
"test": {
|
|
"_ttl": {
|
|
"enabled": 1,
|
|
"store": "yes",
|
|
"default": "10s"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"create": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"fields": "_ttl"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"lt": {
|
|
"fields._ttl": 10000
|
|
}
|
|
},
|
|
{
|
|
"gt": {
|
|
"fields._ttl": 0
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"delete": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"create": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
},
|
|
"ttl": 100000
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"fields": "_ttl"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"lt": {
|
|
"fields._ttl": 100000
|
|
}
|
|
},
|
|
{
|
|
"gt": {
|
|
"fields._ttl": 10000
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"delete": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"create": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
},
|
|
"ttl": "20s"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"fields": "_ttl"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"lt": {
|
|
"fields._ttl": 20000
|
|
}
|
|
},
|
|
{
|
|
"gt": {
|
|
"fields._ttl": 10000
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"delete": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "/AlreadyExpiredException/",
|
|
"create": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
},
|
|
"ttl": "20s",
|
|
"timestamp": "2013-06-23T18:14:40.000Z"
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"delete/10_basic.yaml": [
|
|
{
|
|
"Basic": [
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_true": "ok"
|
|
},
|
|
{
|
|
"match": {
|
|
"_version": 1
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"delete": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_version": 2
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"delete/20_internal_version.yaml": [
|
|
{
|
|
"Internal version": [
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_version": 1
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "conflict",
|
|
"delete": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"version": 2
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"delete": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"version": 1
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_version": 2
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"delete/25_external_version.yaml": [
|
|
{
|
|
"External version": [
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
},
|
|
"version_type": "external",
|
|
"version": 5
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_version": 5
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "conflict",
|
|
"delete": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"version_type": "external",
|
|
"version": 4
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"delete": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"version_type": "external",
|
|
"version": 6
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_version": 6
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"delete/30_routing.yaml": [
|
|
{
|
|
"Routing": [
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"routing": 5,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "missing",
|
|
"delete": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"routing": 4
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"delete": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"routing": 5
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"delete/40_parent.yaml": [
|
|
{
|
|
"Parent": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.2",
|
|
"reason": "Delete ignores the parent param"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_1",
|
|
"body": {
|
|
"mappings": {
|
|
"test": {
|
|
"_parent": {
|
|
"type": "foo"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"parent": 5,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "missing",
|
|
"delete": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"parent": 1
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"delete": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"parent": 5
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"delete/45_parent_with_routing.yaml": [
|
|
{
|
|
"Parent with routing": [
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_1",
|
|
"body": {
|
|
"mappings": {
|
|
"test": {
|
|
"_parent": {
|
|
"type": "foo"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"parent": 5,
|
|
"routing": 4,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "missing",
|
|
"delete": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"parent": 5,
|
|
"routing": 1
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"delete": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"parent": 5,
|
|
"routing": 4
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"delete/50_refresh.yaml": [
|
|
{
|
|
"Refresh": [
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_1",
|
|
"body": {
|
|
"settings": {
|
|
"index.refresh_interval": -1
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
},
|
|
"refresh": 1
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 2,
|
|
"body": {
|
|
"foo": "bar"
|
|
},
|
|
"refresh": 1
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"search": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"body": {
|
|
"query": {
|
|
"terms": {
|
|
"_id": [
|
|
1,
|
|
2
|
|
]
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"hits.total": 2
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"delete": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"search": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"body": {
|
|
"query": {
|
|
"terms": {
|
|
"_id": [
|
|
1,
|
|
2
|
|
]
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"hits.total": 2
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"delete": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 2,
|
|
"refresh": 1
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"search": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"body": {
|
|
"query": {
|
|
"terms": {
|
|
"_id": [
|
|
1,
|
|
2
|
|
]
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"hits.total": 1
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"delete/60_missing.yaml": [
|
|
{
|
|
"Missing document with catch": [
|
|
{
|
|
"do": {
|
|
"catch": "missing",
|
|
"delete": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1
|
|
}
|
|
}
|
|
}
|
|
]
|
|
},
|
|
{
|
|
"Missing document with ignore": [
|
|
{
|
|
"do": {
|
|
"delete": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"ignore": 404
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"delete_by_query/10_basic.yaml": [
|
|
{
|
|
"Basic delete_by_query": [
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 2,
|
|
"body": {
|
|
"foo": "baz"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 3,
|
|
"body": {
|
|
"foo": "foo"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.refresh": {}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"delete_by_query": {
|
|
"index": "test_1",
|
|
"body": {
|
|
"match": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_true": "ok"
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.refresh": {}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"count": {
|
|
"index": "test_1"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"count": 2
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"exists/10_basic.yaml": [
|
|
{
|
|
"Basic": [
|
|
{
|
|
"do": {
|
|
"exists": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_false": ""
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_true": ""
|
|
},
|
|
{
|
|
"do": {
|
|
"exists": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_true": ""
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"exists/30_parent.yaml": [
|
|
{
|
|
"Parent": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.2",
|
|
"reason": "Parent not supported in exists https://github.com/elasticsearch/elasticsearch/issues/3276"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_1",
|
|
"body": {
|
|
"mappings": {
|
|
"test": {
|
|
"_parent": {
|
|
"type": "foo"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"parent": 5,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"exists": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"parent": 5
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_true": ""
|
|
},
|
|
{
|
|
"do": {
|
|
"exists": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_false": ""
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"exists/40_routing.yaml": [
|
|
{
|
|
"Routing": [
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"routing": 5,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"exists": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"routing": 5
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_true": ""
|
|
},
|
|
{
|
|
"do": {
|
|
"exists": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_false": ""
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"exists/55_parent_with_routing.yaml": [
|
|
{
|
|
"Parent with routing": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.2",
|
|
"reason": "Parent not supported in exists https://github.com/elasticsearch/elasticsearch/issues/3276"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_1",
|
|
"body": {
|
|
"mappings": {
|
|
"test": {
|
|
"_parent": {
|
|
"type": "foo"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"parent": 5,
|
|
"routing": 4,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_true": ""
|
|
},
|
|
{
|
|
"do": {
|
|
"exists": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"parent": 5,
|
|
"routing": 4
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_true": ""
|
|
},
|
|
{
|
|
"do": {
|
|
"exists": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"parent": 5
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_false": ""
|
|
},
|
|
{
|
|
"do": {
|
|
"exists": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"routing": 4
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_true": ""
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"exists/60_realtime_refresh.yaml": [
|
|
{
|
|
"Realtime Refresh": [
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_1",
|
|
"body": {
|
|
"settings": {
|
|
"index": {
|
|
"refresh_interval": -1,
|
|
"number_of_replicas": 0
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"exists": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"realtime": 1
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_true": ""
|
|
},
|
|
{
|
|
"do": {
|
|
"exists": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"realtime": 0
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_false": ""
|
|
},
|
|
{
|
|
"do": {
|
|
"exists": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"realtime": 0,
|
|
"refresh": 1
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_true": ""
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"exists/70_defaults.yaml": [
|
|
{
|
|
"Client-side default type": [
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"exists": {
|
|
"index": "test_1",
|
|
"id": 1
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_true": ""
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"explain/10_basic.yaml": [
|
|
{
|
|
"Basic mlt": [
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar",
|
|
"title": "howdy"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.refresh": {}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"explain": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"query": {
|
|
"match_all": {}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_true": "matched"
|
|
},
|
|
{
|
|
"is_true": "ok"
|
|
},
|
|
{
|
|
"match": {
|
|
"explanation.value": 1
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"explain/20_source_filtering.yaml": [
|
|
{
|
|
"Source filtering": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.999",
|
|
"reason": "source filtering is not supported in pre 1.0 versions."
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"include": {
|
|
"field1": "v1",
|
|
"field2": "v2"
|
|
},
|
|
"count": 1
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.refresh": {
|
|
"index": "test_1"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"explain": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"_source": false,
|
|
"body": {
|
|
"query": {
|
|
"match_all": {}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_index": "test_1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_type": "test"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_id": "1"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "get._source"
|
|
},
|
|
{
|
|
"do": {
|
|
"explain": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"_source": true,
|
|
"body": {
|
|
"query": {
|
|
"match_all": {}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"get._source.include.field1": "v1"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"explain": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"_source": "include.field1",
|
|
"body": {
|
|
"query": {
|
|
"match_all": {}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"get._source.include.field1": "v1"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "get._source.include.field2"
|
|
},
|
|
{
|
|
"do": {
|
|
"explain": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"_source_include": "include.field1",
|
|
"body": {
|
|
"query": {
|
|
"match_all": {}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"get._source.include.field1": "v1"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "get._source.include.field2"
|
|
},
|
|
{
|
|
"do": {
|
|
"explain": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"_source_include": "include.field1,include.field2",
|
|
"body": {
|
|
"query": {
|
|
"match_all": {}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"get._source.include.field1": "v1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"get._source.include.field2": "v2"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "get._source.count"
|
|
},
|
|
{
|
|
"do": {
|
|
"explain": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"_source_include": "include",
|
|
"_source_exclude": "*.field2",
|
|
"body": {
|
|
"query": {
|
|
"match_all": {}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"get._source.include.field1": "v1"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "get._source.include.field2"
|
|
},
|
|
{
|
|
"is_false": "get._source.count"
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"get/10_basic.yaml": [
|
|
{
|
|
"Basic": [
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": "中文",
|
|
"body": {
|
|
"foo": "Hello: 中文"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": "中文"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_index": "test_1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_type": "test"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_id": "中文"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_source": {
|
|
"foo": "Hello: 中文"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"id": "中文"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_index": "test_1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_type": "test"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_id": "中文"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_source": {
|
|
"foo": "Hello: 中文"
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"get/15_default_values.yaml": [
|
|
{
|
|
"Default values": [
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"id": 1
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_index": "test_1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_type": "test"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_id": "1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_source": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"get/20_fields.yaml": [
|
|
{
|
|
"Fields": [
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar",
|
|
"count": 1
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"fields": "foo"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_index": "test_1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_type": "test"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_id": "1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"fields.foo": "bar"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "_source"
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"fields": [
|
|
"foo",
|
|
"count"
|
|
]
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"fields.foo": "bar"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"fields.count": 1
|
|
}
|
|
},
|
|
{
|
|
"is_false": "_source"
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"fields": [
|
|
"foo",
|
|
"count",
|
|
"_source"
|
|
]
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"fields.foo": "bar"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"fields.count": 1
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_source.foo": "bar"
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"get/30_parent.yaml": [
|
|
{
|
|
"Parent": [
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_1",
|
|
"body": {
|
|
"mappings": {
|
|
"test": {
|
|
"_parent": {
|
|
"type": "foo"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"parent": "中文",
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"parent": "中文",
|
|
"fields": [
|
|
"_parent",
|
|
"_routing"
|
|
]
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_id": "1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"fields._parent": "中文"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"fields._routing": "中文"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "missing",
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"get/40_routing.yaml": [
|
|
{
|
|
"Routing": [
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"routing": 5,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"routing": 5,
|
|
"fields": [
|
|
"_routing"
|
|
]
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_id": "1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"fields._routing": "5"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "missing",
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"get/55_parent_with_routing.yaml": [
|
|
{
|
|
"Parent with routing": [
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_1",
|
|
"body": {
|
|
"mappings": {
|
|
"test": {
|
|
"_parent": {
|
|
"type": "foo"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"parent": 5,
|
|
"routing": 4,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"parent": 5,
|
|
"routing": 4,
|
|
"fields": [
|
|
"_parent",
|
|
"_routing"
|
|
]
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_id": "1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"fields._parent": "5"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"fields._routing": "4"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "missing",
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"parent": 5
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"routing": 4
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"get/60_realtime_refresh.yaml": [
|
|
{
|
|
"Realtime Refresh": [
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_1",
|
|
"body": {
|
|
"settings": {
|
|
"index": {
|
|
"refresh_interval": -1,
|
|
"number_of_replicas": 0
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"realtime": 1
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_true": "exists"
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "missing",
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"realtime": 0
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"realtime": 0,
|
|
"refresh": 1
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_true": "exists"
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"get/70_source_filtering.yaml": [
|
|
{
|
|
"Source filtering": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.999",
|
|
"reason": "source filtering is not supported in pre 1.0 versions."
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"include": {
|
|
"field1": "v1",
|
|
"field2": "v2"
|
|
},
|
|
"count": 1
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"_source": false
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_index": "test_1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_type": "test"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_id": "1"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "_source"
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"_source": true
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_source.include.field1": "v1"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"_source": "include.field1"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_source.include.field1": "v1"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "_source.include.field2"
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"_source_include": "include.field1"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_source.include.field1": "v1"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "_source.include.field2"
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"_source_include": "include.field1,include.field2"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_source.include.field1": "v1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_source.include.field2": "v2"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "_source.count"
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"_source_include": "include",
|
|
"_source_exclude": "*.field2"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_source.include.field1": "v1"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "_source.include.field2"
|
|
},
|
|
{
|
|
"is_false": "_source.count"
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"fields": "count",
|
|
"_source": true
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_index": "test_1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_type": "test"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_id": "1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"fields.count": 1
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_source.include.field1": "v1"
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"get/80_missing.yaml": [
|
|
{
|
|
"Missing document with catch": [
|
|
{
|
|
"do": {
|
|
"catch": "missing",
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1
|
|
}
|
|
}
|
|
}
|
|
]
|
|
},
|
|
{
|
|
"Missing document with ignore": [
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"ignore": 404
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"get_source/10_basic.yaml": [
|
|
{
|
|
"Basic": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.0",
|
|
"reason": "Get source not supported in pre 0.90.1 versions."
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get_source": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get_source": {
|
|
"index": "test_1",
|
|
"id": 1
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"get_source/15_default_values.yaml": [
|
|
{
|
|
"Default values": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.0",
|
|
"reason": "Get source not supported in pre 0.90.1 versions."
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get_source": {
|
|
"index": "test_1",
|
|
"id": 1
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"get_source/30_parent.yaml": [
|
|
{
|
|
"Parent": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.0",
|
|
"reason": "Get source not supported in pre 0.90.1 versions."
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_1",
|
|
"body": {
|
|
"mappings": {
|
|
"test": {
|
|
"_parent": {
|
|
"type": "foo"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"parent": 5,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get_source": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"parent": 5
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "missing",
|
|
"get_source": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"get_source/40_routing.yaml": [
|
|
{
|
|
"Routing": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.0",
|
|
"reason": "Get source not supported in pre 0.90.1 versions."
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"routing": 5,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get_source": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"routing": 5
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "missing",
|
|
"get_source": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"get_source/55_parent_with_routing.yaml": [
|
|
{
|
|
"Parent with routing": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.0",
|
|
"reason": "Get source not supported in pre 0.90.1 versions."
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_1",
|
|
"body": {
|
|
"mappings": {
|
|
"test": {
|
|
"_parent": {
|
|
"type": "foo"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"parent": 5,
|
|
"routing": 4,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get_source": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"parent": 5,
|
|
"routing": 4
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "missing",
|
|
"get_source": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"parent": 5
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get_source": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"routing": 4
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"get_source/60_realtime_refresh.yaml": [
|
|
{
|
|
"Realtime": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.0",
|
|
"reason": "Get source not supported in pre 0.90.1 versions."
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get_source": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"realtime": 1
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "missing",
|
|
"get_source": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"realtime": 0
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get_source": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"realtime": 0,
|
|
"refresh": 1
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"get_source/70_source_filtering.yaml": [
|
|
{
|
|
"Source filtering": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.999",
|
|
"reason": "source filtering is not supported in pre 1.0 versions."
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"include": {
|
|
"field1": "v1",
|
|
"field2": "v2"
|
|
},
|
|
"count": 1
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get_source": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"_source_include": "include.field1"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"include.field1": "v1"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "include.field2"
|
|
},
|
|
{
|
|
"do": {
|
|
"get_source": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"_source_include": "include.field1,include.field2"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"include.field1": "v1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"include.field2": "v2"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "count"
|
|
},
|
|
{
|
|
"do": {
|
|
"get_source": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"_source_include": "include",
|
|
"_source_exclude": "*.field2"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"include.field1": "v1"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "include.field2"
|
|
},
|
|
{
|
|
"is_false": "count"
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"get_source/80_missing.yaml": [
|
|
{
|
|
"Missing document with catch": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.0",
|
|
"reason": "Get source not supported in pre 0.90.1 versions."
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "missing",
|
|
"get_source": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1
|
|
}
|
|
}
|
|
}
|
|
]
|
|
},
|
|
{
|
|
"Missing document with ignore": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.0",
|
|
"reason": "Get source not supported in pre 0.90.1 versions."
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get_source": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"ignore": 404
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"index/10_with_id.yaml": [
|
|
{
|
|
"Index with ID": [
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test-weird-index-中文",
|
|
"type": "weird.type",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_true": "ok"
|
|
},
|
|
{
|
|
"match": {
|
|
"_index": "test-weird-index-中文"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_type": "weird.type"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_id": "1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_version": 1
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test-weird-index-中文",
|
|
"type": "weird.type",
|
|
"id": 1
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_index": "test-weird-index-中文"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_type": "weird.type"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_id": "1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_version": 1
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_source": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"index/15_without_id.yaml": [
|
|
{
|
|
"Index without ID": [
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_true": "ok"
|
|
},
|
|
{
|
|
"is_true": "_id"
|
|
},
|
|
{
|
|
"match": {
|
|
"_index": "test_1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_type": "test"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_version": 1
|
|
}
|
|
},
|
|
{
|
|
"set": {
|
|
"_id": "id"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": "$id"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_index": "test_1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_type": "test"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_id": "$id"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_version": 1
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_source": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"index/20_optype.yaml": [
|
|
{
|
|
"Optype": [
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"op_type": "create",
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "conflict",
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"op_type": "create",
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"op_type": "index",
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_version": 2
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"index/30_internal_version.yaml": [
|
|
{
|
|
"Internal version": [
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_version": 1
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_version": 2
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "conflict",
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
},
|
|
"version": 1
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
},
|
|
"version": 2
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_version": 3
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"index/35_external_version.yaml": [
|
|
{
|
|
"External version": [
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
},
|
|
"version_type": "external",
|
|
"version": 5
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_version": 5
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "conflict",
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
},
|
|
"version_type": "external",
|
|
"version": 5
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
},
|
|
"version_type": "external",
|
|
"version": 6
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_version": 6
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"index/40_routing.yaml": [
|
|
{
|
|
"Routing": [
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"routing": 5,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"routing": 5,
|
|
"fields": [
|
|
"_routing"
|
|
]
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_id": "1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"fields._routing": "5"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "missing",
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"index/50_parent.yaml": [
|
|
{
|
|
"Parent": [
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_1",
|
|
"body": {
|
|
"mappings": {
|
|
"test": {
|
|
"_parent": {
|
|
"type": "foo"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "/RoutingMissingException/",
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"parent": 5,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"parent": 5,
|
|
"fields": [
|
|
"_parent",
|
|
"_routing"
|
|
]
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_id": "1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"fields._parent": "5"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"fields._routing": "5"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "missing",
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"index/55_parent_with_routing.yaml": [
|
|
{
|
|
"Parent with routing": [
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_1",
|
|
"body": {
|
|
"mappings": {
|
|
"test": {
|
|
"_parent": {
|
|
"type": "foo"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"parent": 5,
|
|
"routing": 4,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"parent": 5,
|
|
"routing": 4,
|
|
"fields": [
|
|
"_parent",
|
|
"_routing"
|
|
]
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_id": "1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"fields._parent": "5"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"fields._routing": "4"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "missing",
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"parent": 5
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"routing": 4
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"index/60_refresh.yaml": [
|
|
{
|
|
"Refresh": [
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_1",
|
|
"body": {
|
|
"settings": {
|
|
"index.refresh_interval": -1
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"search": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"body": {
|
|
"query": {
|
|
"term": {
|
|
"_id": 1
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"hits.total": 0
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 2,
|
|
"refresh": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"search": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"body": {
|
|
"query": {
|
|
"term": {
|
|
"_id": 2
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"hits.total": 1
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"index/70_timestamp.yaml": [
|
|
{
|
|
"Timestamp": [
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_1",
|
|
"body": {
|
|
"mappings": {
|
|
"test": {
|
|
"_timestamp": {
|
|
"enabled": 1,
|
|
"store": "yes"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"fields": "_timestamp"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_true": "fields._timestamp"
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
},
|
|
"timestamp": 1372011280000
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"fields": "_timestamp"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"fields._timestamp": 1372011280000
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
},
|
|
"timestamp": "2013-06-23T18:14:40.000Z"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"fields": "_timestamp"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"fields._timestamp": 1372011280000
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"index/75_ttl.yaml": [
|
|
{
|
|
"TTL": [
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_1",
|
|
"body": {
|
|
"mappings": {
|
|
"test": {
|
|
"_ttl": {
|
|
"enabled": 1,
|
|
"store": "yes",
|
|
"default": "10s"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"fields": "_ttl"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"lt": {
|
|
"fields._ttl": 10000
|
|
}
|
|
},
|
|
{
|
|
"gt": {
|
|
"fields._ttl": 0
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
},
|
|
"ttl": 100000
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"fields": "_ttl"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"lt": {
|
|
"fields._ttl": 100000
|
|
}
|
|
},
|
|
{
|
|
"gt": {
|
|
"fields._ttl": 10000
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
},
|
|
"ttl": "20s"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"fields": "_ttl"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"lt": {
|
|
"fields._ttl": 20000
|
|
}
|
|
},
|
|
{
|
|
"gt": {
|
|
"fields._ttl": 10000
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "/AlreadyExpiredException/",
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
},
|
|
"ttl": "20s",
|
|
"timestamp": "2013-06-23T18:14:40.000Z"
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"indices.analyze/10_analyze.yaml": [
|
|
{
|
|
"setup": [
|
|
{
|
|
"do": {
|
|
"ping": {}
|
|
}
|
|
}
|
|
]
|
|
},
|
|
{
|
|
"Basic test": [
|
|
{
|
|
"do": {
|
|
"indices.analyze": {
|
|
"text": "Foo Bar"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"length": {
|
|
"tokens": 2
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"tokens.0.token": "foo"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"tokens.1.token": "bar"
|
|
}
|
|
}
|
|
]
|
|
},
|
|
{
|
|
"Tokenizer and filter": [
|
|
{
|
|
"do": {
|
|
"indices.analyze": {
|
|
"filters": "lowercase",
|
|
"text": "Foo Bar",
|
|
"tokenizer": "keyword"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"length": {
|
|
"tokens": 1
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"tokens.0.token": "foo bar"
|
|
}
|
|
}
|
|
]
|
|
},
|
|
{
|
|
"Index and field": [
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test",
|
|
"body": {
|
|
"mappings": {
|
|
"test": {
|
|
"properties": {
|
|
"text": {
|
|
"type": "string",
|
|
"analyzer": "whitespace"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.analyze": {
|
|
"field": "text",
|
|
"index": "test",
|
|
"text": "Foo Bar!"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"length": {
|
|
"tokens": 2
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"tokens.0.token": "Foo"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"tokens.1.token": "Bar!"
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"indices.analyze/20_analyze_text_format.yaml": [
|
|
{
|
|
"Text format": [
|
|
{
|
|
"do": {
|
|
"indices.analyze": {
|
|
"format": "text",
|
|
"text": "tHE BLACK and white! AND red"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"tokens": "[black:4->9:<ALPHANUM>]\n\n4: \n[white:14->19:<ALPHANUM>]\n\n6: \n[red:25->28:<ALPHANUM>]\n"
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"indices.clear_cache/10_basic.yaml": [
|
|
{
|
|
"clear_cache test": [
|
|
{
|
|
"do": {
|
|
"indices.clear_cache": {}
|
|
}
|
|
},
|
|
{
|
|
"is_true": "ok"
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"indices.delete_alias/10_basic.yaml": [
|
|
{
|
|
"Basic test for delete alias": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.0",
|
|
"reason": "Delete alias not supported before 0.90.1"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "testind"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.put_alias": {
|
|
"index": "testind",
|
|
"name": "testali",
|
|
"body": {
|
|
"routing": "routing value"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.get_alias": {
|
|
"name": "testali"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"testind.aliases.testali.search_routing": "routing value"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"testind.aliases.testali.index_routing": "routing value"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.delete_alias": {
|
|
"index": "testind",
|
|
"name": "testali"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "missing",
|
|
"indices.get_alias": {
|
|
"index": "testind",
|
|
"name": "testali"
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"indices.delete_mapping/10_basic.yaml": [
|
|
{
|
|
"delete mapping tests": [
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_index",
|
|
"body": {
|
|
"mappings": {
|
|
"test_type": {
|
|
"properties": {
|
|
"text": {
|
|
"type": "string",
|
|
"analyzer": "whitespace"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.exists_type": {
|
|
"index": "test_index",
|
|
"type": "test_type"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_true": ""
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.delete_mapping": {
|
|
"index": "test_index",
|
|
"type": "test_type"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.exists_type": {
|
|
"index": "test_index",
|
|
"type": "test_type"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_false": ""
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"indices.exists/10_basic.yaml": [
|
|
{
|
|
"Test indices.exists": [
|
|
{
|
|
"do": {
|
|
"indices.exists": {
|
|
"index": "test_index"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_false": ""
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_index"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.exists": {
|
|
"index": "test_index"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_true": ""
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"indices.get_field_mapping/10_basic.yaml": [
|
|
{
|
|
"setup": [
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_index",
|
|
"body": {
|
|
"mappings": {
|
|
"test_type": {
|
|
"properties": {
|
|
"text": {
|
|
"type": "string"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
]
|
|
},
|
|
{
|
|
"Get field mapping with no index and type": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.5",
|
|
"reason": "The API was added in 0.90.6"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.get_field_mapping": {
|
|
"field": "text"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"test_index.test_type.text.mapping.text.type": "string"
|
|
}
|
|
}
|
|
]
|
|
},
|
|
{
|
|
"Get field mapping by index only": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.5",
|
|
"reason": "The API was added in 0.90.6"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.get_field_mapping": {
|
|
"index": "test_index",
|
|
"field": "text"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"test_index.test_type.text.mapping.text.type": "string"
|
|
}
|
|
}
|
|
]
|
|
},
|
|
{
|
|
"Get field mapping by type & field": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.5",
|
|
"reason": "The API was added in 0.90.6"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.get_field_mapping": {
|
|
"index": "test_index",
|
|
"type": "test_type",
|
|
"field": "text"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"test_index.test_type.text.mapping.text.type": "string"
|
|
}
|
|
}
|
|
]
|
|
},
|
|
{
|
|
"Get field mapping by type & field, with another field that doesn't exist": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.5",
|
|
"reason": "The API was added in 0.90.6"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.get_field_mapping": {
|
|
"index": "test_index",
|
|
"type": "test_type",
|
|
"field": [
|
|
"text",
|
|
"text1"
|
|
]
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"test_index.test_type.text.mapping.text.type": "string"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "test_index.test_type.text1"
|
|
}
|
|
]
|
|
},
|
|
{
|
|
"Get field mapping with include_defaults": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.5",
|
|
"reason": "The API was added in 0.90.6"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.get_field_mapping": {
|
|
"index": "test_index",
|
|
"type": "test_type",
|
|
"field": "text",
|
|
"include_defaults": true
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"test_index.test_type.text.mapping.text.type": "string"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"test_index.test_type.text.mapping.text.analyzer": "default"
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"indices.get_field_mapping/20_missing_field.yaml": [
|
|
{
|
|
"Raise 404 when field doesn't exist": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.5",
|
|
"reason": "get field mapping was added in 0.90.6"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_index",
|
|
"body": {
|
|
"mappings": {
|
|
"test_type": {
|
|
"properties": {
|
|
"text": {
|
|
"type": "string",
|
|
"analyzer": "whitespace"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "missing",
|
|
"indices.get_field_mapping": {
|
|
"index": "test_index",
|
|
"type": "test_type",
|
|
"field": "not_text"
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"indices.get_field_mapping/30_missing_type.yaml": [
|
|
{
|
|
"Raise 404 when type doesn't exist": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.5",
|
|
"reason": "get field mapping was added in 0.90.6"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_index",
|
|
"body": {
|
|
"mappings": {
|
|
"test_type": {
|
|
"properties": {
|
|
"text": {
|
|
"type": "string",
|
|
"analyzer": "whitespace"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "missing",
|
|
"indices.get_field_mapping": {
|
|
"index": "test_index",
|
|
"type": "not_test_type",
|
|
"field": "text"
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"indices.get_field_mapping/40_missing_index.yaml": [
|
|
{
|
|
"Raise 404 when index doesn't exist": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.5",
|
|
"reason": "get field mapping was added in 0.90.6"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "missing",
|
|
"indices.get_field_mapping": {
|
|
"index": "test_index",
|
|
"type": "type",
|
|
"field": "field"
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"indices.get_mapping/10_basic.yaml": [
|
|
{
|
|
"setup": [
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_index",
|
|
"body": {
|
|
"mappings": {
|
|
"test_type": {
|
|
"properties": {
|
|
"text": {
|
|
"type": "string",
|
|
"analyzer": "whitespace"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
]
|
|
},
|
|
{
|
|
"Get index mapping": [
|
|
{
|
|
"do": {
|
|
"indices.get_mapping": {
|
|
"index": "test_index"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"test_index.test_type.properties.text.type": "string"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"test_index.test_type.properties.text.analyzer": "whitespace"
|
|
}
|
|
}
|
|
]
|
|
},
|
|
{
|
|
"Get type mapping - pre 1.0": [
|
|
{
|
|
"skip": {
|
|
"version": "0.90.9 - 999",
|
|
"reason": "for newer versions the index name is always returned"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.get_mapping": {
|
|
"index": "test_index",
|
|
"type": "test_type"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"test_type.properties.text.type": "string"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"test_type.properties.text.analyzer": "whitespace"
|
|
}
|
|
}
|
|
]
|
|
},
|
|
{
|
|
"Get type mapping - post 1.0": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.9",
|
|
"reason": "for older versions, just the type name is returned"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.get_mapping": {
|
|
"index": "test_index",
|
|
"type": "test_type"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"test_index.test_type.properties.text.type": "string"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"test_index.test_type.properties.text.analyzer": "whitespace"
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"indices.get_mapping/20_missing_type.yaml": [
|
|
{
|
|
"Raise 404 when type doesn't exist": [
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_index",
|
|
"body": {
|
|
"mappings": {
|
|
"test_type": {
|
|
"properties": {
|
|
"text": {
|
|
"type": "string",
|
|
"analyzer": "whitespace"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "missing",
|
|
"indices.get_mapping": {
|
|
"index": "test_index",
|
|
"type": "not_test_type"
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"indices.get_mapping/30_missing_index.yaml": [
|
|
{
|
|
"Raise 404 when index doesn't exist": [
|
|
{
|
|
"do": {
|
|
"catch": "missing",
|
|
"indices.get_mapping": {
|
|
"index": "test_index",
|
|
"type": "not_test_type"
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"indices.get_template/10_basic.yaml": [
|
|
{
|
|
"Get template": [
|
|
{
|
|
"do": {
|
|
"indices.put_template": {
|
|
"name": "test",
|
|
"body": {
|
|
"template": "test-*",
|
|
"settings": {
|
|
"number_of_shards": 1,
|
|
"number_of_replicas": 0
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.get_template": {
|
|
"name": "test"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"test.template": "test-*"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"test.settings": {
|
|
"index.number_of_shards": "1",
|
|
"index.number_of_replicas": "0"
|
|
}
|
|
}
|
|
}
|
|
]
|
|
},
|
|
{
|
|
"Get all templates": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.3",
|
|
"reason": "Get all templates not available before 0.90.4"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.put_template": {
|
|
"name": "test",
|
|
"body": {
|
|
"template": "test-*",
|
|
"settings": {
|
|
"number_of_shards": 1
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.put_template": {
|
|
"name": "test2",
|
|
"body": {
|
|
"template": "test2-*",
|
|
"settings": {
|
|
"number_of_shards": 1
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.get_template": {}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"test.template": "test-*"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"test2.template": "test2-*"
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"indices.get_template/20_get_missing.yaml": [
|
|
{
|
|
"setup": [
|
|
{
|
|
"do": {
|
|
"indices.delete_template": {
|
|
"name": "*",
|
|
"ignore": 404
|
|
}
|
|
}
|
|
}
|
|
]
|
|
},
|
|
{
|
|
"Get missing template - post 0.90.3": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.2",
|
|
"reason": "Missing templates throw 404 from 0.90.3"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "missing",
|
|
"indices.get_template": {
|
|
"name": "test"
|
|
}
|
|
}
|
|
}
|
|
]
|
|
},
|
|
{
|
|
"Get missing template - pre 0.90.3": [
|
|
{
|
|
"skip": {
|
|
"version": "0.90.3 - 999",
|
|
"reason": "Missing templates didn't throw 404 before 0.90.3"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.delete_template": {
|
|
"name": "test",
|
|
"ignore": 404
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.get_template": {
|
|
"name": "test"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"": {}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"indices.open/10_basic.yaml": [
|
|
{
|
|
"Basic test for index open/close": [
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_index"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.close": {
|
|
"index": "test_index"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "forbidden",
|
|
"search": {
|
|
"index": "test_index"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.open": {
|
|
"index": "test_index"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"search": {
|
|
"index": "test_index"
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"indices.optimize/10_basic.yaml": [
|
|
{
|
|
"Optimize index tests": [
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "testing"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.optimize": {
|
|
"index": "testing",
|
|
"max_num_segments": 1
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_true": "ok"
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"indices.put_alias/10_basic.yaml": [
|
|
{
|
|
"Basic test for put alias": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.0",
|
|
"reason": "Put alias not supported before 0.90.1"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_index"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.exists_alias": {
|
|
"name": "test_alias"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_false": ""
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.put_alias": {
|
|
"index": "test_index",
|
|
"name": "test_alias"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_true": "ok"
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.exists_alias": {
|
|
"name": "test_alias"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_true": ""
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.get_alias": {
|
|
"name": "test_alias"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"test_index.aliases.test_alias": {}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"indices.put_mapping/10_basic.yaml": [
|
|
{
|
|
"Test Create and update mapping": [
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_index"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.put_mapping": {
|
|
"index": "test_index",
|
|
"type": "test_type",
|
|
"body": {
|
|
"test_type": {
|
|
"properties": {
|
|
"text": {
|
|
"type": "string",
|
|
"analyzer": "whitespace"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.get_mapping": {
|
|
"index": "test_index"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"test_index.test_type.properties.text.type": "string"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"test_index.test_type.properties.text.analyzer": "whitespace"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.put_mapping": {
|
|
"index": "test_index",
|
|
"type": "test_type",
|
|
"body": {
|
|
"test_type": {
|
|
"properties": {
|
|
"text": {
|
|
"type": "multi_field",
|
|
"fields": {
|
|
"text": {
|
|
"type": "string",
|
|
"analyzer": "whitespace"
|
|
},
|
|
"text_raw": {
|
|
"type": "string",
|
|
"index": "not_analyzed"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.get_mapping": {
|
|
"index": "test_index"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"test_index.test_type.properties.text.type": "multi_field"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"test_index.test_type.properties.text.fields.text_raw.index": "not_analyzed"
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"indices.put_settings/10_basic.yaml": [
|
|
{
|
|
"Test indices settings": [
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test-index",
|
|
"body": {
|
|
"settings": {
|
|
"index": {
|
|
"number_of_replicas": 0
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.get_settings": {
|
|
"index": "test-index"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"test-index.settings.index\\.number_of_replicas": "0"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.put_settings": {
|
|
"body": {
|
|
"number_of_replicas": 1
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.get_settings": {}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"test-index.settings.index\\.number_of_replicas": "1"
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"indices.put_template/10_basic.yaml": [
|
|
{
|
|
"Put template": [
|
|
{
|
|
"do": {
|
|
"indices.put_template": {
|
|
"name": "test",
|
|
"body": {
|
|
"template": "test-*",
|
|
"settings": {
|
|
"number_of_shards": 1,
|
|
"number_of_replicas": 0
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_true": "ok"
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.get_template": {
|
|
"name": "test"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"test.template": "test-*"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"test.settings": {
|
|
"index.number_of_shards": "1",
|
|
"index.number_of_replicas": "0"
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"indices.put_warmer/10_basic.yaml": [
|
|
{
|
|
"Basic test for warmers": [
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_index"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "missing",
|
|
"indices.get_warmer": {
|
|
"index": "test_index",
|
|
"name": "test_warmer"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.put_warmer": {
|
|
"index": "test_index",
|
|
"name": "test_warmer",
|
|
"body": {
|
|
"query": {
|
|
"match_all": {}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_true": "ok"
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.get_warmer": {
|
|
"index": "test_index",
|
|
"name": "test_warmer"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"test_index.warmers.test_warmer.source.query.match_all": {}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.delete_warmer": {
|
|
"index": "test_index"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_true": "ok"
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "missing",
|
|
"indices.get_warmer": {
|
|
"index": "test_index",
|
|
"name": "test_warmer"
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"indices.segments/10_basic.yaml": [
|
|
{
|
|
"segments test": [
|
|
{
|
|
"do": {
|
|
"indices.segments": {}
|
|
}
|
|
},
|
|
{
|
|
"is_true": "ok"
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"indices.snapshot_index/10_basic.yaml": [
|
|
{
|
|
"snapshot_index test": [
|
|
{
|
|
"do": {
|
|
"indices.snapshot_index": {}
|
|
}
|
|
},
|
|
{
|
|
"is_true": "ok"
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"indices.stats/10_basic.yaml": [
|
|
{
|
|
"stats test": [
|
|
{
|
|
"do": {
|
|
"indices.stats": {}
|
|
}
|
|
},
|
|
{
|
|
"is_true": "ok"
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"indices.status/10_basic.yaml": [
|
|
{
|
|
"Indices status test": [
|
|
{
|
|
"do": {
|
|
"indices.status": {}
|
|
}
|
|
},
|
|
{
|
|
"is_true": "ok"
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "missing",
|
|
"indices.status": {
|
|
"index": "not_here"
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"indices.update_aliases/10_basic.yaml": [
|
|
{
|
|
"Basic test for aliases": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.0",
|
|
"reason": "Exists alias not supported before 0.90.1"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_index"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.exists_alias": {
|
|
"name": "test_alias"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_false": ""
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.update_aliases": {
|
|
"body": {
|
|
"actions": [
|
|
{
|
|
"add": {
|
|
"index": "test_index",
|
|
"alias": "test_alias",
|
|
"routing": "routing_value"
|
|
}
|
|
}
|
|
]
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_true": "ok"
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.exists_alias": {
|
|
"name": "test_alias"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_true": ""
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.get_aliases": {
|
|
"index": "test_index"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"test_index.aliases.test_alias": {
|
|
"index_routing": "routing_value",
|
|
"search_routing": "routing_value"
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"indices.validate_query/10_basic.yaml": [
|
|
{
|
|
"Validate query api": [
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "testing"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.validate_query": {
|
|
"q": "query string"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_true": "valid"
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.validate_query": {
|
|
"body": {
|
|
"query": {
|
|
"invalid_query": {}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_false": "valid"
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"info/10_info.yaml": [
|
|
{
|
|
"Info": [
|
|
{
|
|
"do": {
|
|
"info": {}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"status": 200
|
|
}
|
|
},
|
|
{
|
|
"is_true": "ok"
|
|
},
|
|
{
|
|
"is_true": "name"
|
|
},
|
|
{
|
|
"is_true": "tagline"
|
|
},
|
|
{
|
|
"is_true": "version"
|
|
},
|
|
{
|
|
"is_true": "version.number"
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"info/20_lucene_version.yaml": [
|
|
{
|
|
"Lucene Version": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.0",
|
|
"reason": "Lucene version not included in info before 0.90.1: https://github.com/elasticsearch/elasticsearch/issues/2988"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"info": {}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"status": 200
|
|
}
|
|
},
|
|
{
|
|
"is_true": "version.lucene_version"
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"mget/10_basic.yaml": [
|
|
{
|
|
"Basic multi-get": [
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_2"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.flush": {
|
|
"refresh": true
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"mget": {
|
|
"body": {
|
|
"docs": [
|
|
{
|
|
"_index": "test_2",
|
|
"_type": "test",
|
|
"_id": 1
|
|
},
|
|
{
|
|
"_index": "test_1",
|
|
"_type": "none",
|
|
"_id": 1
|
|
},
|
|
{
|
|
"_index": "test_1",
|
|
"_type": "test",
|
|
"_id": 2
|
|
},
|
|
{
|
|
"_index": "test_1",
|
|
"_type": "test",
|
|
"_id": 1
|
|
}
|
|
]
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_false": "docs.0.exists"
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.0._index": "test_2"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.0._type": "test"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.0._id": "1"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "docs.1.exists"
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.1._index": "test_1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.1._type": "none"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.1._id": "1"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "docs.2.exists"
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.2._index": "test_1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.2._type": "test"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.2._id": "2"
|
|
}
|
|
},
|
|
{
|
|
"is_true": "docs.3.exists"
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.3._index": "test_1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.3._type": "test"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.3._id": "1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.3._version": 1
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.3._source": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"mget/11_default_index_type.yaml": [
|
|
{
|
|
"Default index/type": [
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_2"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"mget": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"body": {
|
|
"docs": [
|
|
{
|
|
"_index": "test_2",
|
|
"_id": 1
|
|
},
|
|
{
|
|
"_type": "none",
|
|
"_id": 1
|
|
},
|
|
{
|
|
"_id": 2
|
|
},
|
|
{
|
|
"_id": 1
|
|
}
|
|
]
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_false": "docs.0.exists"
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.0._index": "test_2"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.0._type": "test"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.0._id": "1"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "docs.1.exists"
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.1._index": "test_1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.1._type": "none"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.1._id": "1"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "docs.2.exists"
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.2._index": "test_1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.2._type": "test"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.2._id": "2"
|
|
}
|
|
},
|
|
{
|
|
"is_true": "docs.3.exists"
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.3._index": "test_1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.3._type": "test"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.3._id": "1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.3._version": 1
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.3._source": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"mget/12_non_existent_index.yaml": [
|
|
{
|
|
"Non-existent index": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.2",
|
|
"reason": "See https://github.com/elasticsearch/elasticsearch/issues/3267"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"mget": {
|
|
"body": {
|
|
"docs": [
|
|
{
|
|
"_index": "test_2",
|
|
"_type": "test",
|
|
"_id": 1
|
|
}
|
|
]
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_false": "docs.0.exists"
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.0._index": "test_2"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.0._type": "test"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.0._id": "1"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"mget": {
|
|
"body": {
|
|
"index": "test_2",
|
|
"docs": [
|
|
{
|
|
"_index": "test_1",
|
|
"_type": "test",
|
|
"_id": 1
|
|
}
|
|
]
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_true": "docs.0.exists"
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.0._index": "test_1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.0._type": "test"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.0._id": "1"
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"mget/13_missing_metadata.yaml": [
|
|
{
|
|
"Missing metadata": [
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "/ActionRequestValidationException.+ id is missing/",
|
|
"mget": {
|
|
"body": {
|
|
"docs": [
|
|
{
|
|
"_index": "test_1",
|
|
"_type": "test"
|
|
}
|
|
]
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "/ActionRequestValidationException.+ index is missing/",
|
|
"mget": {
|
|
"body": {
|
|
"docs": [
|
|
{
|
|
"_type": "test",
|
|
"_id": 1
|
|
}
|
|
]
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "/ActionRequestValidationException.+ no documents to get/",
|
|
"mget": {
|
|
"body": {
|
|
"docs": []
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "/ActionRequestValidationException.+ no documents to get/",
|
|
"mget": {
|
|
"body": {}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"mget": {
|
|
"body": {
|
|
"docs": [
|
|
{
|
|
"_index": "test_1",
|
|
"_id": 1
|
|
}
|
|
]
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_true": "docs.0.exists"
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.0._index": "test_1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.0._type": "test"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.0._id": "1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.0._version": 1
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.0._source": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"mget/15_ids.yaml": [
|
|
{
|
|
"IDs": [
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test_2",
|
|
"id": 2,
|
|
"body": {
|
|
"foo": "baz"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"mget": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"body": {
|
|
"ids": [
|
|
1,
|
|
2
|
|
]
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_true": "docs.0.exists"
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.0._index": "test_1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.0._type": "test"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.0._id": "1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.0._version": 1
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.0._source": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_false": "docs.1.exists"
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.1._index": "test_1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.1._type": "test"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.1._id": "2"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"mget": {
|
|
"index": "test_1",
|
|
"body": {
|
|
"ids": [
|
|
1,
|
|
2
|
|
]
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_true": "docs.0.exists"
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.0._index": "test_1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.0._type": "test"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.0._id": "1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.0._version": 1
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.0._source": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_true": "docs.1.exists"
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.1._index": "test_1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.1._type": "test_2"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.1._id": "2"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.1._version": 1
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.1._source": {
|
|
"foo": "baz"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "/ActionRequestValidationException.+ no documents to get/",
|
|
"mget": {
|
|
"index": "test_1",
|
|
"body": {
|
|
"ids": []
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "/ActionRequestValidationException.+ no documents to get/",
|
|
"mget": {
|
|
"index": "test_1",
|
|
"body": {}
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"mget/20_fields.yaml": [
|
|
{
|
|
"Fields": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.2",
|
|
"reason": "Single string fields not supported pre 0.90.3. See https://github.com/elasticsearch/elasticsearch/issues/3270"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"mget": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"body": {
|
|
"docs": [
|
|
{
|
|
"_id": 1
|
|
},
|
|
{
|
|
"_id": 1,
|
|
"fields": "foo"
|
|
},
|
|
{
|
|
"_id": 1,
|
|
"fields": [
|
|
"foo"
|
|
]
|
|
},
|
|
{
|
|
"_id": 1,
|
|
"fields": [
|
|
"foo",
|
|
"_source"
|
|
]
|
|
}
|
|
]
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_false": "docs.0.fields"
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.0._source": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.1.fields.foo": "bar"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "docs.1._source"
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.2.fields.foo": "bar"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "docs.2._source"
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.3.fields.foo": "bar"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.3._source": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"mget": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"fields": "foo",
|
|
"body": {
|
|
"docs": [
|
|
{
|
|
"_id": 1
|
|
},
|
|
{
|
|
"_id": 1,
|
|
"fields": "foo"
|
|
},
|
|
{
|
|
"_id": 1,
|
|
"fields": [
|
|
"foo"
|
|
]
|
|
},
|
|
{
|
|
"_id": 1,
|
|
"fields": [
|
|
"foo",
|
|
"_source"
|
|
]
|
|
}
|
|
]
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.0.fields.foo": "bar"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "docs.0._source"
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.1.fields.foo": "bar"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "docs.1._source"
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.2.fields.foo": "bar"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "docs.2._source"
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.3.fields.foo": "bar"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.3._source": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"mget": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"fields": [
|
|
"foo"
|
|
],
|
|
"body": {
|
|
"docs": [
|
|
{
|
|
"_id": 1
|
|
},
|
|
{
|
|
"_id": 1,
|
|
"fields": "foo"
|
|
},
|
|
{
|
|
"_id": 1,
|
|
"fields": [
|
|
"foo"
|
|
]
|
|
},
|
|
{
|
|
"_id": 1,
|
|
"fields": [
|
|
"foo",
|
|
"_source"
|
|
]
|
|
}
|
|
]
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.0.fields.foo": "bar"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "docs.0._source"
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.1.fields.foo": "bar"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "docs.1._source"
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.2.fields.foo": "bar"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "docs.2._source"
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.3.fields.foo": "bar"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.3._source": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"mget": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"fields": [
|
|
"foo",
|
|
"_source"
|
|
],
|
|
"body": {
|
|
"docs": [
|
|
{
|
|
"_id": 1
|
|
},
|
|
{
|
|
"_id": 1,
|
|
"fields": "foo"
|
|
},
|
|
{
|
|
"_id": 1,
|
|
"fields": [
|
|
"foo"
|
|
]
|
|
},
|
|
{
|
|
"_id": 1,
|
|
"fields": [
|
|
"foo",
|
|
"_source"
|
|
]
|
|
}
|
|
]
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.0.fields.foo": "bar"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.0._source": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.1.fields.foo": "bar"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "docs.1._source"
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.2.fields.foo": "bar"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "docs.2._source"
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.3.fields.foo": "bar"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.3._source": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"mget/20_fields_pre_0.90.3.yaml": [
|
|
{
|
|
"Fields": [
|
|
{
|
|
"skip": {
|
|
"version": "0.90.3 - 999",
|
|
"reason": "Tests pre 0.90.2 for single string fields"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"mget": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"body": {
|
|
"docs": [
|
|
{
|
|
"_id": 1
|
|
},
|
|
{
|
|
"_id": 1,
|
|
"fields": "foo"
|
|
},
|
|
{
|
|
"_id": 1,
|
|
"fields": [
|
|
"foo"
|
|
]
|
|
},
|
|
{
|
|
"_id": 1,
|
|
"fields": [
|
|
"foo",
|
|
"_source"
|
|
]
|
|
}
|
|
]
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_false": "docs.0.fields"
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.0._source": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.2.fields.foo": "bar"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "docs.2._source"
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.3.fields.foo": "bar"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.3._source": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"mget": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"fields": "foo",
|
|
"body": {
|
|
"docs": [
|
|
{
|
|
"_id": 1
|
|
},
|
|
{
|
|
"_id": 1,
|
|
"fields": "foo"
|
|
},
|
|
{
|
|
"_id": 1,
|
|
"fields": [
|
|
"foo"
|
|
]
|
|
},
|
|
{
|
|
"_id": 1,
|
|
"fields": [
|
|
"foo",
|
|
"_source"
|
|
]
|
|
}
|
|
]
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.0.fields.foo": "bar"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "docs.0._source"
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.1.fields.foo": "bar"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "docs.1._source"
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.2.fields.foo": "bar"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "docs.2._source"
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.3.fields.foo": "bar"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.3._source": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"mget": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"fields": [
|
|
"foo"
|
|
],
|
|
"body": {
|
|
"docs": [
|
|
{
|
|
"_id": 1
|
|
},
|
|
{
|
|
"_id": 1,
|
|
"fields": "foo"
|
|
},
|
|
{
|
|
"_id": 1,
|
|
"fields": [
|
|
"foo"
|
|
]
|
|
},
|
|
{
|
|
"_id": 1,
|
|
"fields": [
|
|
"foo",
|
|
"_source"
|
|
]
|
|
}
|
|
]
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.0.fields.foo": "bar"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "docs.0._source"
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.1.fields.foo": "bar"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "docs.1._source"
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.2.fields.foo": "bar"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "docs.2._source"
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.3.fields.foo": "bar"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.3._source": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"mget": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"fields": [
|
|
"foo",
|
|
"_source"
|
|
],
|
|
"body": {
|
|
"docs": [
|
|
{
|
|
"_id": 1
|
|
},
|
|
{
|
|
"_id": 1,
|
|
"fields": "foo"
|
|
},
|
|
{
|
|
"_id": 1,
|
|
"fields": [
|
|
"foo"
|
|
]
|
|
},
|
|
{
|
|
"_id": 1,
|
|
"fields": [
|
|
"foo",
|
|
"_source"
|
|
]
|
|
}
|
|
]
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.0.fields.foo": "bar"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.0._source": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.2.fields.foo": "bar"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "docs.2._source"
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.3.fields.foo": "bar"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.3._source": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"mget/30_parent.yaml": [
|
|
{
|
|
"Parent": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.2",
|
|
"reason": "Parent not supported in mget. https://github.com/elasticsearch/elasticsearch/issues/3274"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_1",
|
|
"body": {
|
|
"mappings": {
|
|
"test": {
|
|
"_parent": {
|
|
"type": "foo"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"parent": 4,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 2,
|
|
"parent": 5,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"mget": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"body": {
|
|
"docs": [
|
|
{
|
|
"_id": 1
|
|
},
|
|
{
|
|
"_id": 1,
|
|
"parent": 5,
|
|
"fields": [
|
|
"_parent",
|
|
"_routing"
|
|
]
|
|
},
|
|
{
|
|
"_id": 1,
|
|
"parent": 4,
|
|
"fields": [
|
|
"_parent",
|
|
"_routing"
|
|
]
|
|
},
|
|
{
|
|
"_id": 2,
|
|
"parent": 5,
|
|
"fields": [
|
|
"_parent",
|
|
"_routing"
|
|
]
|
|
}
|
|
]
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_false": "docs.0.exists"
|
|
},
|
|
{
|
|
"is_false": "docs.1.exists"
|
|
},
|
|
{
|
|
"is_true": "docs.2.exists"
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.2._index": "test_1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.2._type": "test"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.2._id": "1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.2.fields._parent": "4"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.2.fields._routing": "4"
|
|
}
|
|
},
|
|
{
|
|
"is_true": "docs.3.exists"
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.3._index": "test_1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.3._type": "test"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.3._id": "2"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.3.fields._parent": "5"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.3.fields._routing": "5"
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"mget/40_routing.yaml": [
|
|
{
|
|
"Routing": [
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"routing": 5,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"mget": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"fields": [
|
|
"_routing"
|
|
],
|
|
"body": {
|
|
"docs": [
|
|
{
|
|
"_id": 1
|
|
},
|
|
{
|
|
"_id": 1,
|
|
"routing": 4
|
|
},
|
|
{
|
|
"_id": 1,
|
|
"routing": 5
|
|
}
|
|
]
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_false": "docs.0.exists"
|
|
},
|
|
{
|
|
"is_false": "docs.1.exists"
|
|
},
|
|
{
|
|
"is_true": "docs.2.exists"
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.2._index": "test_1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.2._type": "test"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.2._id": "1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.2.fields._routing": "5"
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"mget/55_parent_with_routing.yaml": [
|
|
{
|
|
"Parent": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.2",
|
|
"reason": "Parent not supported in mget. https://github.com/elasticsearch/elasticsearch/issues/3274"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_1",
|
|
"body": {
|
|
"mappings": {
|
|
"test": {
|
|
"_parent": {
|
|
"type": "foo"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"parent": 4,
|
|
"routing": 5,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"mget": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"fields": [
|
|
"_routing",
|
|
"_parent"
|
|
],
|
|
"body": {
|
|
"docs": [
|
|
{
|
|
"_id": 1
|
|
},
|
|
{
|
|
"_id": 1,
|
|
"parent": 4
|
|
},
|
|
{
|
|
"_id": 1,
|
|
"parent": 4,
|
|
"routing": 5
|
|
}
|
|
]
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_false": "docs.0.exists"
|
|
},
|
|
{
|
|
"is_false": "docs.1.exists"
|
|
},
|
|
{
|
|
"is_true": "docs.2.exists"
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.2._index": "test_1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.2._type": "test"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.2._id": "1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.2.fields._parent": "4"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.2.fields._routing": "5"
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"mget/60_realtime_refresh.yaml": [
|
|
{
|
|
"Realtime Refresh": [
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_1",
|
|
"body": {
|
|
"settings": {
|
|
"index": {
|
|
"refresh_interval": -1,
|
|
"number_of_replicas": 0
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"mget": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"realtime": 0,
|
|
"body": {
|
|
"ids": [
|
|
1
|
|
]
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_false": "docs.0.exists"
|
|
},
|
|
{
|
|
"do": {
|
|
"mget": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"realtime": 1,
|
|
"body": {
|
|
"ids": [
|
|
1
|
|
]
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_true": "docs.0.exists"
|
|
},
|
|
{
|
|
"do": {
|
|
"mget": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"realtime": 0,
|
|
"refresh": 1,
|
|
"body": {
|
|
"ids": [
|
|
1
|
|
]
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_true": "docs.0.exists"
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"mget/70_source_filtering.yaml": [
|
|
{
|
|
"setup": [
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"include": {
|
|
"field1": "v1",
|
|
"field2": "v2"
|
|
},
|
|
"count": 1
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 2,
|
|
"body": {
|
|
"include": {
|
|
"field1": "v1",
|
|
"field2": "v2"
|
|
},
|
|
"count": 1
|
|
}
|
|
}
|
|
}
|
|
}
|
|
]
|
|
},
|
|
{
|
|
"Source filtering - true/false": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.999",
|
|
"reason": "source filtering is not supported in pre 1.0 versions."
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"mget": {
|
|
"body": {
|
|
"docs": [
|
|
{
|
|
"_index": "test_1",
|
|
"_type": "test",
|
|
"_id": "1",
|
|
"_source": false
|
|
},
|
|
{
|
|
"_index": "test_1",
|
|
"_type": "test",
|
|
"_id": "2",
|
|
"_source": true
|
|
}
|
|
]
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.0._id": "1"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "docs.0._source"
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.1._id": "2"
|
|
}
|
|
},
|
|
{
|
|
"is_true": "docs.1._source"
|
|
}
|
|
]
|
|
},
|
|
{
|
|
"Source filtering - include field": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.999",
|
|
"reason": "source filtering is not supported in pre 1.0 versions."
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"mget": {
|
|
"body": {
|
|
"docs": [
|
|
{
|
|
"_index": "test_1",
|
|
"_type": "test",
|
|
"_id": "1",
|
|
"_source": "include.field1"
|
|
},
|
|
{
|
|
"_index": "test_1",
|
|
"_type": "test",
|
|
"_id": "2",
|
|
"_source": [
|
|
"include.field1"
|
|
]
|
|
}
|
|
]
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.0._source": {
|
|
"include": {
|
|
"field1": "v1"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.1._source": {
|
|
"include": {
|
|
"field1": "v1"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
]
|
|
},
|
|
{
|
|
"Source filtering - include nested field": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.999",
|
|
"reason": "source filtering is not supported in pre 1.0 versions."
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"mget": {
|
|
"body": {
|
|
"docs": [
|
|
{
|
|
"_index": "test_1",
|
|
"_type": "test",
|
|
"_id": "1",
|
|
"_source": {
|
|
"include": "include.field1"
|
|
}
|
|
},
|
|
{
|
|
"_index": "test_1",
|
|
"_type": "test",
|
|
"_id": "2",
|
|
"_source": {
|
|
"include": [
|
|
"include.field1"
|
|
]
|
|
}
|
|
}
|
|
]
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.0._source": {
|
|
"include": {
|
|
"field1": "v1"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.1._source": {
|
|
"include": {
|
|
"field1": "v1"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
]
|
|
},
|
|
{
|
|
"Source filtering - exclude field": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.999",
|
|
"reason": "source filtering is not supported in pre 1.0 versions."
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"mget": {
|
|
"body": {
|
|
"docs": [
|
|
{
|
|
"_index": "test_1",
|
|
"_type": "test",
|
|
"_id": "1",
|
|
"_source": {
|
|
"include": [
|
|
"include"
|
|
],
|
|
"exclude": [
|
|
"*.field2"
|
|
]
|
|
}
|
|
}
|
|
]
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.0._source": {
|
|
"include": {
|
|
"field1": "v1"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
]
|
|
},
|
|
{
|
|
"Source filtering - ids and true/false": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.999",
|
|
"reason": "source filtering is not supported in pre 1.0 versions."
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"mget": {
|
|
"_source": false,
|
|
"index": "test_1",
|
|
"body": {
|
|
"ids": [
|
|
1,
|
|
2
|
|
]
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_false": "docs.0._source"
|
|
},
|
|
{
|
|
"is_false": "docs.1._source"
|
|
},
|
|
{
|
|
"do": {
|
|
"mget": {
|
|
"_source": true,
|
|
"index": "test_1",
|
|
"body": {
|
|
"ids": [
|
|
1,
|
|
2
|
|
]
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_true": "docs.0._source"
|
|
},
|
|
{
|
|
"is_true": "docs.1._source"
|
|
}
|
|
]
|
|
},
|
|
{
|
|
"Source filtering - ids and include field": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.999",
|
|
"reason": "source filtering is not supported in pre 1.0 versions."
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"mget": {
|
|
"_source": "include.field1",
|
|
"index": "test_1",
|
|
"body": {
|
|
"ids": [
|
|
1,
|
|
2
|
|
]
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.0._source": {
|
|
"include": {
|
|
"field1": "v1"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.1._source": {
|
|
"include": {
|
|
"field1": "v1"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
]
|
|
},
|
|
{
|
|
"Source filtering - ids and include nested field": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.999",
|
|
"reason": "source filtering is not supported in pre 1.0 versions."
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"mget": {
|
|
"_source_include": "include.field1,count",
|
|
"index": "test_1",
|
|
"body": {
|
|
"ids": [
|
|
1,
|
|
2
|
|
]
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.0._source": {
|
|
"include": {
|
|
"field1": "v1"
|
|
},
|
|
"count": 1
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.1._source": {
|
|
"include": {
|
|
"field1": "v1"
|
|
},
|
|
"count": 1
|
|
}
|
|
}
|
|
}
|
|
]
|
|
},
|
|
{
|
|
"Source filtering - ids and exclude field": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.999",
|
|
"reason": "source filtering is not supported in pre 1.0 versions."
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"mget": {
|
|
"_source_include": "include",
|
|
"_source_exclude": "*.field2",
|
|
"index": "test_1",
|
|
"body": {
|
|
"ids": [
|
|
1,
|
|
2
|
|
]
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.0._source": {
|
|
"include": {
|
|
"field1": "v1"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"docs.1._source": {
|
|
"include": {
|
|
"field1": "v1"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"mlt/10_basic.yaml": [
|
|
{
|
|
"Basic mlt": [
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar",
|
|
"title": "howdy"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.refresh": {}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"mlt": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"mlt_fields": "title"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"hits.total": 0
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"msearch/10_basic.yaml": [
|
|
{
|
|
"Basic multi-search": [
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 2,
|
|
"body": {
|
|
"foo": "baz"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 3,
|
|
"body": {
|
|
"foo": "foo"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.refresh": {}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"msearch": {
|
|
"body": [
|
|
{
|
|
"index": "test_1"
|
|
},
|
|
{
|
|
"query": {
|
|
"match_all": {}
|
|
}
|
|
},
|
|
{
|
|
"index": "test_2"
|
|
},
|
|
{
|
|
"query": {
|
|
"match_all": {}
|
|
}
|
|
},
|
|
{
|
|
"search_type": "count",
|
|
"index": "test_1"
|
|
},
|
|
{
|
|
"query": {
|
|
"match": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"responses.0.hits.total": 3
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"responses.1.error": "IndexMissingException[[test_2] missing]"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"responses.2.hits.total": 1
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"percolate/10_old.yaml": [
|
|
{
|
|
"Basic percolation tests": [
|
|
{
|
|
"skip": {
|
|
"version": "0.90.9 - 999",
|
|
"reason": "Percolator redesign"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_index"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "_percolator",
|
|
"type": "test_index",
|
|
"id": "test_percolator",
|
|
"body": {
|
|
"query": {
|
|
"match_all": {}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.refresh": {}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"percolate": {
|
|
"index": "test_index",
|
|
"type": "test_type",
|
|
"body": {
|
|
"doc": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_true": "ok"
|
|
},
|
|
{
|
|
"match": {
|
|
"matches": [
|
|
"test_percolator"
|
|
]
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"percolate/15_new.yaml": [
|
|
{
|
|
"Basic percolation tests": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.9",
|
|
"reason": "Percolator redesign"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_index"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_index",
|
|
"type": ".percolator",
|
|
"id": "test_percolator",
|
|
"body": {
|
|
"query": {
|
|
"match_all": {}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.refresh": {}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"percolate": {
|
|
"index": "test_index",
|
|
"type": "test_type",
|
|
"body": {
|
|
"doc": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"matches": [
|
|
{
|
|
"_index": "test_index",
|
|
"_id": "test_percolator"
|
|
}
|
|
]
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"ping/10_ping.yaml": [
|
|
{
|
|
"Ping": [
|
|
{
|
|
"do": {
|
|
"ping": {}
|
|
}
|
|
},
|
|
{
|
|
"is_true": ""
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"scroll/10_basic.yaml": [
|
|
{
|
|
"Basic scroll": [
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_scroll"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_scroll",
|
|
"type": "test",
|
|
"id": 42,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.refresh": {}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"search": {
|
|
"index": "test_scroll",
|
|
"search_type": "scan",
|
|
"scroll": "1m",
|
|
"body": {
|
|
"query": {
|
|
"match_all": {}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"set": {
|
|
"_scroll_id": "scroll_id"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"scroll": {
|
|
"scroll_id": "$scroll_id"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"hits.total": 1
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"hits.hits.0._id": "42"
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"search/10_source_filtering.yaml": [
|
|
{
|
|
"Source filtering": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.999",
|
|
"reason": "source filtering is not supported in pre 1.0 versions."
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"include": {
|
|
"field1": "v1",
|
|
"field2": "v2"
|
|
},
|
|
"count": 1
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.refresh": {}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"search": {
|
|
"body": "{ _source: true, query: { match_all: {} } }"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"length": {
|
|
"hits.hits": 1
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"hits.hits.0._source.count": 1
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"search": {
|
|
"body": "{ _source: false, query: { match_all: {} } }"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"length": {
|
|
"hits.hits": 1
|
|
}
|
|
},
|
|
{
|
|
"is_false": "hits.hits.0._source"
|
|
},
|
|
{
|
|
"do": {
|
|
"search": {
|
|
"body": {
|
|
"query": {
|
|
"match_all": {}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"length": {
|
|
"hits.hits": 1
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"hits.hits.0._source.count": 1
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"search": {
|
|
"body": {
|
|
"_source": "include.field1",
|
|
"query": {
|
|
"match_all": {}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"hits.hits.0._source.include.field1": "v1"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "hits.hits.0._source.include.field2"
|
|
},
|
|
{
|
|
"do": {
|
|
"search": {
|
|
"_source_include": "include.field1",
|
|
"body": {
|
|
"_source": "include.field2",
|
|
"query": {
|
|
"match_all": {}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"hits.hits.0._source.include.field1": "v1"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "hits.hits.0._source.include.field2"
|
|
},
|
|
{
|
|
"do": {
|
|
"search": {
|
|
"_source_include": "include.field1",
|
|
"body": {
|
|
"query": {
|
|
"match_all": {}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"hits.hits.0._source.include.field1": "v1"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "hits.hits.0._source.include.field2"
|
|
},
|
|
{
|
|
"do": {
|
|
"search": {
|
|
"_source_exclude": "count",
|
|
"body": {
|
|
"query": {
|
|
"match_all": {}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"hits.hits.0._source.include": {
|
|
"field1": "v1",
|
|
"field2": "v2"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_false": "hits.hits.0._source.count"
|
|
},
|
|
{
|
|
"do": {
|
|
"search": {
|
|
"body": {
|
|
"_source": [
|
|
"include.field1",
|
|
"include.field2"
|
|
],
|
|
"query": {
|
|
"match_all": {}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"hits.hits.0._source.include.field1": "v1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"hits.hits.0._source.include.field2": "v2"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "hits.hits.0._source.count"
|
|
},
|
|
{
|
|
"do": {
|
|
"search": {
|
|
"body": {
|
|
"_source": {
|
|
"include": [
|
|
"include.field1",
|
|
"include.field2"
|
|
]
|
|
},
|
|
"query": {
|
|
"match_all": {}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"hits.hits.0._source.include.field1": "v1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"hits.hits.0._source.include.field2": "v2"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "hits.hits.0._source.count"
|
|
},
|
|
{
|
|
"do": {
|
|
"search": {
|
|
"body": {
|
|
"_source": {
|
|
"includes": "include",
|
|
"excludes": "*.field2"
|
|
},
|
|
"query": {
|
|
"match_all": {}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"hits.hits.0._source.include.field1": "v1"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "hits.hits.0._source.include.field2"
|
|
},
|
|
{
|
|
"do": {
|
|
"search": {
|
|
"body": {
|
|
"fields": [
|
|
"include.field2"
|
|
],
|
|
"query": {
|
|
"match_all": {}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"hits.hits.0.fields": {
|
|
"include.field2": "v2"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_false": "hits.hits.0._source"
|
|
},
|
|
{
|
|
"do": {
|
|
"search": {
|
|
"body": {
|
|
"fields": [
|
|
"include.field2",
|
|
"_source"
|
|
],
|
|
"query": {
|
|
"match_all": {}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"hits.hits.0.fields": {
|
|
"include.field2": "v2"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_true": "hits.hits.0._source"
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"search/20_default_values.yaml": [
|
|
{
|
|
"Default index": [
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_2"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_1"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_2",
|
|
"type": "test",
|
|
"id": 42,
|
|
"body": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.refresh": {
|
|
"index": [
|
|
"test_1",
|
|
"test_2"
|
|
]
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"search": {
|
|
"type": "test",
|
|
"body": {
|
|
"query": {
|
|
"match": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"hits.total": 2
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"hits.hits.0._index": "test_1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"hits.hits.0._type": "test"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"hits.hits.0._id": "1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"hits.hits.1._index": "test_2"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"hits.hits.1._type": "test"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"hits.hits.1._id": "42"
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"suggest/10_basic.yaml": [
|
|
{
|
|
"setup": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.2",
|
|
"reason": "Suggest is broken on 0.90.2 - see #3246"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test",
|
|
"type": "test",
|
|
"id": "testing_document",
|
|
"body": {
|
|
"body": "Amsterdam meetup"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.refresh": {}
|
|
}
|
|
}
|
|
]
|
|
},
|
|
{
|
|
"Basic tests for suggest API - pre v1": [
|
|
{
|
|
"skip": {
|
|
"version": "1 - 99",
|
|
"reason": "Standard analyzer uses stopwords"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"suggest": {
|
|
"body": {
|
|
"test_suggestion": {
|
|
"text": "The Amsterdma meetpu",
|
|
"term": {
|
|
"field": "body"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"test_suggestion.0.options.0.text": "amsterdam"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"test_suggestion.1.options.0.text": "meetup"
|
|
}
|
|
}
|
|
]
|
|
},
|
|
{
|
|
"Basic tests for suggest API - post v1": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.9",
|
|
"reason": "Standard analyzer ignores stopwords"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"suggest": {
|
|
"body": {
|
|
"test_suggestion": {
|
|
"text": "The Amsterdma meetpu",
|
|
"term": {
|
|
"field": "body"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"test_suggestion.1.options.0.text": "amsterdam"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"test_suggestion.2.options.0.text": "meetup"
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"update/10_doc.yaml": [
|
|
{
|
|
"Partial document": [
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar",
|
|
"count": 1,
|
|
"nested": {
|
|
"one": 1,
|
|
"two": 2
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"update": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"doc": {
|
|
"foo": "baz",
|
|
"nested": {
|
|
"one": 3
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_index": "test_1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_type": "test"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_id": "1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_version": 2
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_source.foo": "baz"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_source.count": 1
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_source.nested.one": 3
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_source.nested.two": 2
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"update/15_script.yaml": [
|
|
{
|
|
"Script": [
|
|
{
|
|
"do": {
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar",
|
|
"count": 1
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"update": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"script": "1",
|
|
"body": {
|
|
"lang": "mvel",
|
|
"script": "ctx._source.foo = bar",
|
|
"params": {
|
|
"bar": "xxx"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_index": "test_1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_type": "test"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_id": "1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_version": 2
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_source.foo": "xxx"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_source.count": 1
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"update": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"lang": "mvel",
|
|
"script": "ctx._source.foo = 'yyy'"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_index": "test_1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_type": "test"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_id": "1"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_version": 3
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_source.foo": "yyy"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_source.count": 1
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "/script_lang not supported \\[doesnotexist\\]/",
|
|
"update": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"script": "1",
|
|
"lang": "doesnotexist",
|
|
"params": {
|
|
"bar": "xxx"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "/script_lang not supported \\[doesnotexist\\]/",
|
|
"update": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"lang": "doesnotexist",
|
|
"script": "1"
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"update/20_doc_upsert.yaml": [
|
|
{
|
|
"Doc upsert": [
|
|
{
|
|
"do": {
|
|
"update": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"doc": {
|
|
"foo": "bar",
|
|
"count": 1
|
|
},
|
|
"upsert": {
|
|
"foo": "baz"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_source.foo": "baz"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "_source.count"
|
|
},
|
|
{
|
|
"do": {
|
|
"update": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"doc": {
|
|
"foo": "bar",
|
|
"count": 1
|
|
},
|
|
"upsert": {
|
|
"foo": "baz"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_source.foo": "bar"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_source.count": 1
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"update/22_doc_as_upsert.yaml": [
|
|
{
|
|
"Doc as upsert": [
|
|
{
|
|
"skip": {
|
|
"version": "0-0.90.1",
|
|
"reason": "doc_as_upsert added in 0.90.2"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"update": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"doc": {
|
|
"foo": "bar",
|
|
"count": 1
|
|
},
|
|
"doc_as_upsert": 1
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_source.foo": "bar"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_source.count": 1
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"update": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"doc": {
|
|
"count": 2
|
|
},
|
|
"doc_as_upsert": 1
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_source.foo": "bar"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_source.count": 2
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"update/25_script_upsert.yaml": [
|
|
{
|
|
"Script upsert": [
|
|
{
|
|
"do": {
|
|
"update": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"script": "ctx._source.foo = bar",
|
|
"params": {
|
|
"bar": "xxx"
|
|
},
|
|
"upsert": {
|
|
"foo": "baz"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_source.foo": "baz"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"update": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"script": "ctx._source.foo = bar",
|
|
"params": {
|
|
"bar": "xxx"
|
|
},
|
|
"upsert": {
|
|
"foo": "baz"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_source.foo": "xxx"
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"update/30_internal_version.yaml": [
|
|
{
|
|
"Internal version": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.9",
|
|
"reason": "Versions not supported by update API before 1.0.0"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "conflict",
|
|
"update": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"version": 1,
|
|
"body": {
|
|
"doc": {
|
|
"foo": "baz"
|
|
},
|
|
"upsert": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"update": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"doc": {
|
|
"foo": "baz"
|
|
},
|
|
"upsert": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_version": 1
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "conflict",
|
|
"update": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"version": 2,
|
|
"body": {
|
|
"doc": {
|
|
"foo": "baz"
|
|
},
|
|
"upsert": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"update": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"version": 1,
|
|
"body": {
|
|
"doc": {
|
|
"foo": "baz"
|
|
},
|
|
"upsert": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_version": 2
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"update/35_external_version.yaml": [
|
|
{
|
|
"External version": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 0.90.9",
|
|
"reason": "Versions not supported by update API before 1.0.0"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"update": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"version": 2,
|
|
"version_type": "external",
|
|
"body": {
|
|
"doc": {
|
|
"foo": "baz"
|
|
},
|
|
"upsert": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_version": 2
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "conflict",
|
|
"update": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"version": 2,
|
|
"version_type": "external",
|
|
"body": {
|
|
"doc": {
|
|
"foo": "baz"
|
|
},
|
|
"upsert": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"update": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"version": 3,
|
|
"version_type": "external",
|
|
"body": {
|
|
"doc": {
|
|
"foo": "baz"
|
|
},
|
|
"upsert": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"_version": 3
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"update/40_routing.yaml": [
|
|
{
|
|
"Routing": [
|
|
{
|
|
"do": {
|
|
"update": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"routing": 5,
|
|
"body": {
|
|
"doc": {
|
|
"foo": "baz"
|
|
},
|
|
"upsert": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"routing": 5,
|
|
"fields": "_routing"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"fields._routing": "5"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "missing",
|
|
"update": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"doc": {
|
|
"foo": "baz"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"update": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"routing": 5,
|
|
"fields": "foo",
|
|
"body": {
|
|
"doc": {
|
|
"foo": "baz"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"get.fields.foo": "baz"
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"update/50_parent.yaml": [
|
|
{
|
|
"Parent": [
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_1",
|
|
"body": {
|
|
"mappings": {
|
|
"test": {
|
|
"_parent": {
|
|
"type": "foo"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "/RoutingMissingException/",
|
|
"update": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"doc": {
|
|
"foo": "baz"
|
|
},
|
|
"upsert": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"update": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"parent": 5,
|
|
"body": {
|
|
"doc": {
|
|
"foo": "baz"
|
|
},
|
|
"upsert": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"parent": 5,
|
|
"fields": [
|
|
"_parent",
|
|
"_routing"
|
|
]
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"fields._parent": "5"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"fields._routing": "5"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "missing",
|
|
"update": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"doc": {
|
|
"foo": "baz"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"update": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"parent": 5,
|
|
"fields": "foo",
|
|
"body": {
|
|
"doc": {
|
|
"foo": "baz"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"get.fields.foo": "baz"
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"update/55_parent_with_routing.yaml": [
|
|
{
|
|
"Parent with routing": [
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_1",
|
|
"body": {
|
|
"mappings": {
|
|
"test": {
|
|
"_parent": {
|
|
"type": "foo"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"update": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"parent": 5,
|
|
"routing": 4,
|
|
"body": {
|
|
"doc": {
|
|
"foo": "baz"
|
|
},
|
|
"upsert": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"routing": 4,
|
|
"parent": 5,
|
|
"fields": [
|
|
"_parent",
|
|
"_routing"
|
|
]
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"fields._parent": "5"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"fields._routing": "4"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "missing",
|
|
"update": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"parent": 5,
|
|
"body": {
|
|
"doc": {
|
|
"foo": "baz"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"update": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"parent": 5,
|
|
"routing": 4,
|
|
"fields": "foo",
|
|
"body": {
|
|
"doc": {
|
|
"foo": "baz"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"get.fields.foo": "baz"
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"update/60_refresh.yaml": [
|
|
{
|
|
"Refresh": [
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_1",
|
|
"body": {
|
|
"settings": {
|
|
"index.refresh_interval": -1
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"update": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"doc": {
|
|
"foo": "baz"
|
|
},
|
|
"upsert": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"search": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"body": {
|
|
"query": {
|
|
"term": {
|
|
"_id": 1
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"hits.total": 0
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"update": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 2,
|
|
"refresh": 1,
|
|
"body": {
|
|
"doc": {
|
|
"foo": "baz"
|
|
},
|
|
"upsert": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"search": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"body": {
|
|
"query": {
|
|
"term": {
|
|
"_id": 2
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"hits.total": 1
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"update/70_timestamp.yaml": [
|
|
{
|
|
"Timestamp": [
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_1",
|
|
"body": {
|
|
"mappings": {
|
|
"test": {
|
|
"_timestamp": {
|
|
"enabled": 1,
|
|
"store": "yes"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"update": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"doc": {
|
|
"foo": "baz"
|
|
},
|
|
"upsert": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"fields": "_timestamp"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"is_true": "fields._timestamp"
|
|
},
|
|
{
|
|
"do": {
|
|
"update": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"doc": {
|
|
"foo": "baz"
|
|
},
|
|
"upsert": {
|
|
"foo": "bar"
|
|
}
|
|
},
|
|
"timestamp": 1372011280000
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"fields": "_timestamp"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"fields._timestamp": 1372011280000
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"update": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"doc": {
|
|
"foo": "baz"
|
|
},
|
|
"upsert": {
|
|
"foo": "bar"
|
|
}
|
|
},
|
|
"timestamp": "2013-06-23T18:14:40.000Z"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"fields": "_timestamp"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"fields._timestamp": 1372011280000
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"update/75_ttl.yaml": [
|
|
{
|
|
"TTL": [
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_1",
|
|
"body": {
|
|
"mappings": {
|
|
"test": {
|
|
"_ttl": {
|
|
"enabled": 1,
|
|
"store": "yes",
|
|
"default": "10s"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"update": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"doc": {
|
|
"foo": "baz"
|
|
},
|
|
"upsert": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"fields": "_ttl"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"lt": {
|
|
"fields._ttl": 10000
|
|
}
|
|
},
|
|
{
|
|
"gt": {
|
|
"fields._ttl": 0
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"update": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"doc": {
|
|
"foo": "baz"
|
|
},
|
|
"upsert": {
|
|
"foo": "bar"
|
|
}
|
|
},
|
|
"ttl": 100000
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"fields": "_ttl"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"lt": {
|
|
"fields._ttl": 100000
|
|
}
|
|
},
|
|
{
|
|
"gt": {
|
|
"fields._ttl": 10000
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"update": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"doc": {
|
|
"foo": "baz"
|
|
},
|
|
"upsert": {
|
|
"foo": "bar"
|
|
}
|
|
},
|
|
"ttl": "20s"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"fields": "_ttl"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"lt": {
|
|
"fields._ttl": 20000
|
|
}
|
|
},
|
|
{
|
|
"gt": {
|
|
"fields._ttl": 10000
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"catch": "/AlreadyExpiredException/",
|
|
"index": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"foo": "bar"
|
|
},
|
|
"ttl": "20s",
|
|
"timestamp": "2013-06-23T18:14:40.000Z"
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"update/80_fields.yaml": [
|
|
{
|
|
"Fields": [
|
|
{
|
|
"do": {
|
|
"update": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"fields": "foo,bar,_source",
|
|
"body": {
|
|
"doc": {
|
|
"foo": "baz"
|
|
},
|
|
"upsert": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"get._source.foo": "bar"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"get.fields.foo": "bar"
|
|
}
|
|
},
|
|
{
|
|
"is_false": "get.fields.bar"
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"update/85_fields_meta.yaml": [
|
|
{
|
|
"Metadata Fields": [
|
|
{
|
|
"skip": {
|
|
"version": "0 - 999",
|
|
"reason": "Update doesn't return metadata fields, waiting for #3259"
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"indices.create": {
|
|
"index": "test_1",
|
|
"body": {
|
|
"mappings": {
|
|
"test": {
|
|
"_parent": {
|
|
"type": "foo"
|
|
},
|
|
"_timestamp": {
|
|
"enabled": 1,
|
|
"store": "yes"
|
|
},
|
|
"_ttl": {
|
|
"enabled": 1,
|
|
"store": "yes",
|
|
"default": "10s"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"cluster.health": {
|
|
"wait_for_status": "yellow"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"update": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"parent": 5,
|
|
"fields": [
|
|
"_parent",
|
|
"_routing",
|
|
"_timestamp",
|
|
"_ttl"
|
|
],
|
|
"body": {
|
|
"doc": {
|
|
"foo": "baz"
|
|
},
|
|
"upsert": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"get.fields._parent": "5"
|
|
}
|
|
},
|
|
{
|
|
"match": {
|
|
"get.fields._routing": "5"
|
|
}
|
|
},
|
|
{
|
|
"is_true": "get.fields._timestamp"
|
|
},
|
|
{
|
|
"is_true": "get.fields._ttl"
|
|
},
|
|
{
|
|
"do": {
|
|
"get": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"parent": 5,
|
|
"fields": [
|
|
"_parent",
|
|
"_routing",
|
|
"_timestamp",
|
|
"_ttl"
|
|
]
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"update/90_missing.yaml": [
|
|
{
|
|
"Missing document (partial doc)": [
|
|
{
|
|
"do": {
|
|
"catch": "missing",
|
|
"update": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"doc": {
|
|
"foo": "bar"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"update": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"doc": {
|
|
"foo": "bar"
|
|
}
|
|
},
|
|
"ignore": 404
|
|
}
|
|
}
|
|
}
|
|
]
|
|
},
|
|
{
|
|
"Missing document (script)": [
|
|
{
|
|
"do": {
|
|
"catch": "missing",
|
|
"update": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"body": {
|
|
"script": "ctx._source.foo = bar",
|
|
"params": {
|
|
"bar": "xxx"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"do": {
|
|
"update": {
|
|
"index": "test_1",
|
|
"type": "test",
|
|
"id": 1,
|
|
"ignore": 404,
|
|
"body": {
|
|
"script": "ctx._source.foo = bar",
|
|
"params": {
|
|
"bar": "xxx"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
]
|
|
}
|
|
},{}]},{},[82])
|
|
; |