45998 lines
1.1 MiB
45998 lines
1.1 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");/*!
|
|
* agentkeepalive - lib/agent.js
|
|
*
|
|
* refer:
|
|
* * @atimb "Real keep-alive HTTP agent": https://gist.github.com/2963672
|
|
* * https://github.com/joyent/node/blob/master/lib/http.js
|
|
*
|
|
* Copyright(c) 2012 fengmk2 <fengmk2@gmail.com>
|
|
* MIT Licensed
|
|
*/
|
|
|
|
"use strict";
|
|
|
|
/**
|
|
* Module dependencies.
|
|
*/
|
|
|
|
var http = require('http');
|
|
var https = require('https');
|
|
var util = require('util');
|
|
|
|
var debug;
|
|
if (process.env.NODE_DEBUG && /agentkeepalive/.test(process.env.NODE_DEBUG)) {
|
|
debug = function (x) {
|
|
console.error('agentkeepalive:', x);
|
|
};
|
|
} else {
|
|
debug = function () { };
|
|
}
|
|
|
|
function Agent(options) {
|
|
options = options || {};
|
|
http.Agent.call(this, options);
|
|
|
|
var self = this;
|
|
// max requests per keepalive socket, default is 0, no limit.
|
|
self.maxKeepAliveRequests = parseInt(options.maxKeepAliveRequests, 10) || 0;
|
|
// max keep alive time, default 60 seconds.
|
|
// if set `maxKeepAliveTime = 0`, will disable keepalive feature.
|
|
self.maxKeepAliveTime = parseInt(options.maxKeepAliveTime, 10);
|
|
if (isNaN(self.maxKeepAliveTime)) {
|
|
self.maxKeepAliveTime = 60000;
|
|
}
|
|
self.unusedSockets = {};
|
|
self.createSocketCount = 0;
|
|
self.timeoutSocketCount = 0;
|
|
self.requestFinishedCount = 0;
|
|
|
|
// override the `free` event listener
|
|
self.removeAllListeners('free');
|
|
self.on('free', function (socket, host, port, localAddress) {
|
|
self.requestFinishedCount++;
|
|
socket._requestCount++;
|
|
var name = host + ':' + port;
|
|
if (localAddress) {
|
|
name += ':' + localAddress;
|
|
}
|
|
if (self.requests[name] && self.requests[name].length > 0) {
|
|
self.requests[name].shift().onSocket(socket);
|
|
if (self.requests[name].length === 0) {
|
|
// don't leak
|
|
delete self.requests[name];
|
|
}
|
|
} else {
|
|
// If there are no pending requests just destroy the
|
|
// socket and it will get removed from the pool. This
|
|
// gets us out of timeout issues and allows us to
|
|
// default to Connection:keep-alive.
|
|
// socket.destroy();
|
|
if (self.maxKeepAliveTime === 0 ||
|
|
(self.maxKeepAliveRequests && socket._requestCount >= self.maxKeepAliveRequests)) {
|
|
socket.destroy();
|
|
return;
|
|
}
|
|
|
|
// Avoid duplicitive timeout events by removing timeout listeners set on
|
|
// socket by previous requests. node does not do this normally because it
|
|
// assumes sockets are too short-lived for it to matter. It becomes a
|
|
// problem when sockets are being reused. Steps are being taken to fix
|
|
// this issue upstream in node v0.10.0.
|
|
//
|
|
// See https://github.com/joyent/node/commit/451ff1540ab536237e8d751d241d7fc3391a4087
|
|
if (self.maxKeepAliveTime && socket._events && Array.isArray(socket._events.timeout)) {
|
|
socket.removeAllListeners('timeout');
|
|
// Restore the socket's setTimeout() that was remove as collateral
|
|
// damage.
|
|
socket.setTimeout(self.maxKeepAliveTime, socket._maxKeepAliveTimeout);
|
|
}
|
|
// keepalive
|
|
if (!self.unusedSockets[name]) {
|
|
self.unusedSockets[name] = [];
|
|
}
|
|
self.unusedSockets[name].push(socket);
|
|
}
|
|
});
|
|
}
|
|
|
|
util.inherits(Agent, http.Agent);
|
|
module.exports = Agent;
|
|
|
|
Agent.prototype.addRequest = function (req, host, port, localAddress) {
|
|
var name = host + ':' + port;
|
|
if (localAddress) {
|
|
name += ':' + localAddress;
|
|
}
|
|
if (this.unusedSockets[name] && this.unusedSockets[name].length > 0) {
|
|
return req.onSocket(this.unusedSockets[name].shift());
|
|
}
|
|
return http.Agent.prototype.addRequest.call(this, req, host, port, localAddress);
|
|
};
|
|
|
|
Agent.prototype.createSocket = function (name, host, port, localAddress, req) {
|
|
var self = this;
|
|
var socket = http.Agent.prototype.createSocket.call(this, name, host, port, localAddress, req);
|
|
socket._requestCount = 0;
|
|
if (self.maxKeepAliveTime) {
|
|
socket._maxKeepAliveTimeout = function () {
|
|
socket.destroy();
|
|
self.timeoutSocketCount++;
|
|
};
|
|
socket.setTimeout(self.maxKeepAliveTime, socket._maxKeepAliveTimeout);
|
|
// Disable Nagle's algorithm: http://blog.caustik.com/2012/04/08/scaling-node-js-to-100k-concurrent-connections/
|
|
socket.setNoDelay(true);
|
|
}
|
|
this.createSocketCount++;
|
|
return socket;
|
|
};
|
|
|
|
Agent.prototype.removeSocket = function (socket, name, host, port, localAddress) {
|
|
if (this.unusedSockets[name]) {
|
|
var unusedIndex = this.unusedSockets[name].indexOf(socket);
|
|
if (unusedIndex !== -1) {
|
|
this.unusedSockets[name].splice(unusedIndex, 1);
|
|
if (this.unusedSockets[name].length === 0) {
|
|
// don't leak
|
|
delete this.unusedSockets[name];
|
|
}
|
|
}
|
|
}
|
|
return http.Agent.prototype.removeSocket.call(this, socket, name, host, port, localAddress);
|
|
};
|
|
|
|
function HttpsAgent(options) {
|
|
Agent.call(this, options);
|
|
this.createConnection = https.globalAgent.createConnection;
|
|
}
|
|
util.inherits(HttpsAgent, Agent);
|
|
HttpsAgent.prototype.defaultPort = 443;
|
|
|
|
Agent.HttpsAgent = HttpsAgent;
|
|
|
|
},{"__browserify_process":91,"http":72,"https":60,"util":68}],2:[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":91}],3:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
module.exports = [
|
|
"000000", "800000", "008000", "808000", "000080", "800080", "008080", "c0c0c0",
|
|
"808080", "ff0000", "00ff00", "ffff00", "0000ff", "ff00ff", "00ffff", "ffffff",
|
|
|
|
"000000", "00005f", "000087", "0000af", "0000d7", "0000ff",
|
|
"005f00", "005f5f", "005f87", "005faf", "005fd7", "005fff",
|
|
"008700", "00875f", "008787", "0087af", "0087d7", "0087ff",
|
|
"00af00", "00af5f", "00af87", "00afaf", "00afd7", "00afff",
|
|
"00d700", "00d75f", "00d787", "00d7af", "00d7d7", "00d7ff",
|
|
"00ff00", "00ff5f", "00ff87", "00ffaf", "00ffd7", "00ffff",
|
|
|
|
"5f0000", "5f005f", "5f0087", "5f00af", "5f00d7", "5f00ff",
|
|
"5f5f00", "5f5f5f", "5f5f87", "5f5faf", "5f5fd7", "5f5fff",
|
|
"5f8700", "5f875f", "5f8787", "5f87af", "5f87d7", "5f87ff",
|
|
"5faf00", "5faf5f", "5faf87", "5fafaf", "5fafd7", "5fafff",
|
|
"5fd700", "5fd75f", "5fd787", "5fd7af", "5fd7d7", "5fd7ff",
|
|
"5fff00", "5fff5f", "5fff87", "5fffaf", "5fffd7", "5fffff",
|
|
|
|
"870000", "87005f", "870087", "8700af", "8700d7", "8700ff",
|
|
"875f00", "875f5f", "875f87", "875faf", "875fd7", "875fff",
|
|
"878700", "87875f", "878787", "8787af", "8787d7", "8787ff",
|
|
"87af00", "87af5f", "87af87", "87afaf", "87afd7", "87afff",
|
|
"87d700", "87d75f", "87d787", "87d7af", "87d7d7", "87d7ff",
|
|
"87ff00", "87ff5f", "87ff87", "87ffaf", "87ffd7", "87ffff",
|
|
|
|
"af0000", "af005f", "af0087", "af00af", "af00d7", "af00ff",
|
|
"af5f00", "af5f5f", "af5f87", "af5faf", "af5fd7", "af5fff",
|
|
"af8700", "af875f", "af8787", "af87af", "af87d7", "af87ff",
|
|
"afaf00", "afaf5f", "afaf87", "afafaf", "afafd7", "afafff",
|
|
"afd700", "afd75f", "afd787", "afd7af", "afd7d7", "afd7ff",
|
|
"afff00", "afff5f", "afff87", "afffaf", "afffd7", "afffff",
|
|
|
|
"d70000", "d7005f", "d70087", "d700af", "d700d7", "d700ff",
|
|
"d75f00", "d75f5f", "d75f87", "d75faf", "d75fd7", "d75fff",
|
|
"d78700", "d7875f", "d78787", "d787af", "d787d7", "d787ff",
|
|
"d7af00", "d7af5f", "d7af87", "d7afaf", "d7afd7", "d7afff",
|
|
"d7d700", "d7d75f", "d7d787", "d7d7af", "d7d7d7", "d7d7ff",
|
|
"d7ff00", "d7ff5f", "d7ff87", "d7ffaf", "d7ffd7", "d7ffff",
|
|
|
|
"ff0000", "ff005f", "ff0087", "ff00af", "ff00d7", "ff00ff",
|
|
"ff5f00", "ff5f5f", "ff5f87", "ff5faf", "ff5fd7", "ff5fff",
|
|
"ff8700", "ff875f", "ff8787", "ff87af", "ff87d7", "ff87ff",
|
|
"ffaf00", "ffaf5f", "ffaf87", "ffafaf", "ffafd7", "ffafff",
|
|
"ffd700", "ffd75f", "ffd787", "ffd7af", "ffd7d7", "ffd7ff",
|
|
"ffff00", "ffff5f", "ffff87", "ffffaf", "ffffd7", "ffffff",
|
|
|
|
"080808", "121212", "1c1c1c", "262626", "303030", "3a3a3a",
|
|
"444444", "4e4e4e", "585858", "626262", "6c6c6c", "767676",
|
|
"808080", "8a8a8a", "949494", "9e9e9e", "a8a8a8", "b2b2b2",
|
|
"bcbcbc", "c6c6c6", "d0d0d0", "dadada", "e4e4e4", "eeeeee"
|
|
];
|
|
|
|
},{}],4:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
var push = Array.prototype.push, reduce = Array.prototype.reduce, abs = Math.abs
|
|
, colors, match, result, i;
|
|
|
|
colors = require('./_xterm-colors').map(function (color) {
|
|
return {
|
|
r: parseInt(color.slice(0, 2), 16),
|
|
g: parseInt(color.slice(2, 4), 16),
|
|
b: parseInt(color.slice(4), 16)
|
|
};
|
|
});
|
|
|
|
match = colors.slice(0, 16);
|
|
|
|
module.exports = result = [];
|
|
|
|
i = 0;
|
|
while (i < 8) {
|
|
result.push(30 + i++);
|
|
}
|
|
i = 0;
|
|
while (i < 8) {
|
|
result.push(90 + i++);
|
|
}
|
|
push.apply(result, colors.slice(16).map(function (data) {
|
|
var index, diff = Infinity;
|
|
match.every(function (match, i) {
|
|
var ndiff = reduce.call('rgb', function (diff, channel) {
|
|
return (diff += abs(match[channel] - data[channel]));
|
|
}, 0);
|
|
if (ndiff < diff) {
|
|
index = i;
|
|
diff = ndiff;
|
|
}
|
|
return ndiff;
|
|
});
|
|
return result[index];
|
|
}));
|
|
|
|
},{"./_xterm-colors":3}],5:[function(require,module,exports){
|
|
var process=require("__browserify_process");'use strict';
|
|
|
|
var d = require('es5-ext/lib/Object/descriptor')
|
|
, extend = require('es5-ext/lib/Object/extend')
|
|
, map = require('es5-ext/lib/Object/map')
|
|
, reduce = require('es5-ext/lib/Object/reduce')
|
|
, repeat = require('es5-ext/lib/String/prototype/repeat')
|
|
, memoize = require('memoizee')
|
|
, tty = require('tty')
|
|
|
|
, join = Array.prototype.join, defineProperty = Object.defineProperty
|
|
, defineProperties = Object.defineProperties, abs = Math.abs
|
|
, floor = Math.floor, max = Math.max, min = Math.min
|
|
|
|
, mods, proto, getFn, getMove, xtermMatch
|
|
, up, down, right, left, getHeight, memoized;
|
|
|
|
mods = extend({
|
|
// Style
|
|
bold: { _bold: [1, 22] },
|
|
italic: { _italic: [3, 23] },
|
|
underline: { _underline: [4, 24] },
|
|
blink: { _blink: [5, 25] },
|
|
inverse: { _inverse: [7, 27] },
|
|
strike: { _strike: [9, 29] }
|
|
},
|
|
|
|
// Color
|
|
['black', 'red', 'green', 'yellow', 'blue', 'magenta', 'cyan', 'white']
|
|
.reduce(function (obj, color, index) {
|
|
// foreground
|
|
obj[color] = { _fg: [30 + index, 39] };
|
|
obj[color + 'Bright'] = { _fg: [90 + index, 39] };
|
|
|
|
// background
|
|
obj['bg' + color[0].toUpperCase() + color.slice(1)] =
|
|
{ _bg: [40 + index, 49] };
|
|
obj['bg' + color[0].toUpperCase() + color.slice(1) + 'Bright'] =
|
|
{ _bg: [100 + index, 49] };
|
|
|
|
return obj;
|
|
}, {}));
|
|
|
|
// Some use cli-color as: console.log(clc.red('Error!'));
|
|
// Which is inefficient as on each call it configures new clc object
|
|
// with memoization we reuse once created object
|
|
memoized = memoize(function (scope, mod) {
|
|
return defineProperty(getFn(), '_cliColorData',
|
|
d(extend({}, scope._cliColorData, mod)));
|
|
});
|
|
|
|
proto = Object.create(Function.prototype, extend(map(mods, function (mod) {
|
|
return d.gs(function () { return memoized(this, mod); });
|
|
}), {
|
|
// xterm (255) color
|
|
xterm: d(memoize(function (code) {
|
|
code = isNaN(code) ? 255 : min(max(code, 0), 255);
|
|
return defineProperty(getFn(), '_cliColorData',
|
|
d(extend({}, this._cliColorData, {
|
|
_fg: [xtermMatch ? xtermMatch[code] : ('38;5;' + code), 39]
|
|
})));
|
|
}, { method: 'xterm' })),
|
|
bgXterm: d(memoize(function (code) {
|
|
code = isNaN(code) ? 255 : min(max(code, 0), 255);
|
|
return defineProperty(getFn(), '_cliColorData',
|
|
d(extend({}, this._cliColorData, {
|
|
_bg: [xtermMatch ? (xtermMatch[code] + 10) : ('48;5;' + code), 49]
|
|
})));
|
|
}, { method: 'bgXterm' }))
|
|
}));
|
|
|
|
if (process.platform === 'win32') {
|
|
xtermMatch = require('./_xterm-match');
|
|
}
|
|
|
|
getFn = function () {
|
|
var fn = function (/*…msg*/) {
|
|
var data = fn._cliColorData, close = '';
|
|
return reduce(data, function (str, mod) {
|
|
close = '\x1b[' + mod[1] + 'm' + close;
|
|
return str + '\x1b[' + mod[0] + 'm';
|
|
}, '', true) + join.call(arguments, ' ') + close;
|
|
};
|
|
fn.__proto__ = proto;
|
|
return fn;
|
|
};
|
|
|
|
getMove = function (control) {
|
|
return function (num) {
|
|
num = isNaN(num) ? 0 : max(floor(num), 0);
|
|
return num ? ('\x1b[' + num + control) : '';
|
|
};
|
|
};
|
|
|
|
module.exports = defineProperties(getFn(), {
|
|
width: d.gs(process.stdout.getWindowSize ? function () {
|
|
return process.stdout.getWindowSize()[0];
|
|
} : function () {
|
|
return tty.getWindowSize ? tty.getWindowSize()[1] : 0;
|
|
}),
|
|
height: d.gs(getHeight = process.stdout.getWindowSize ? function () {
|
|
return process.stdout.getWindowSize()[1];
|
|
} : function () {
|
|
return tty.getWindowSize ? tty.getWindowSize()[0] : 0;
|
|
}),
|
|
reset: d.gs(function () {
|
|
return repeat.call('\n', getHeight() - 1) + '\x1bc';
|
|
}),
|
|
up: d(up = getMove('A')),
|
|
down: d(down = getMove('B')),
|
|
right: d(right = getMove('C')),
|
|
left: d(left = getMove('D')),
|
|
move: d(function (x, y) {
|
|
x = isNaN(x) ? 0 : floor(x);
|
|
y = isNaN(y) ? 0 : floor(y);
|
|
return ((x > 0) ? right(x) : left(-x)) + ((y > 0) ? down(y) : up(-y));
|
|
}),
|
|
moveTo: d(function (x, y) {
|
|
x = isNaN(x) ? 1 : (max(floor(x), 0) + 1);
|
|
y = isNaN(y) ? 1 : (max(floor(y), 0) + 1);
|
|
return '\x1b[' + y + ';' + x + 'H';
|
|
}),
|
|
bol: d(function (n/*, erase*/) {
|
|
var dir;
|
|
n = isNaN(n) ? 0 : Number(n);
|
|
dir = (n >= 0) ? 'E' : 'F';
|
|
n = floor(abs(n));
|
|
return arguments[1] ?
|
|
(((!n || (dir === 'F')) ? '\x1b[0E\x1bK' : '') +
|
|
repeat.call('\x1b[1' + dir + '\x1b[K', n)) : '\x1b[' + n + dir;
|
|
}),
|
|
beep: d('\x07'),
|
|
xtermSupported: d(!xtermMatch),
|
|
_cliColorData: d({})
|
|
});
|
|
|
|
},{"./_xterm-match":4,"__browserify_process":91,"es5-ext/lib/Object/descriptor":19,"es5-ext/lib/Object/extend":20,"es5-ext/lib/Object/map":26,"es5-ext/lib/Object/reduce":27,"es5-ext/lib/String/prototype/repeat":32,"memoizee":42,"tty":66}],6:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
var isArguments = require('../Function/is-arguments')
|
|
|
|
, isArray = Array.isArray, slice = Array.prototype.slice;
|
|
|
|
module.exports = function (obj) {
|
|
if (isArray(obj)) {
|
|
return obj;
|
|
} else if (isArguments(obj)) {
|
|
return (obj.length === 1) ? [obj[0]] : Array.apply(null, obj);
|
|
} else {
|
|
return slice.call(obj);
|
|
}
|
|
};
|
|
|
|
},{"../Function/is-arguments":11}],7:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
var numIsNaN = require('../../Number/is-nan')
|
|
, ois = require('../../Object/is')
|
|
, value = require('../../Object/valid-value')
|
|
|
|
, indexOf = Array.prototype.indexOf;
|
|
|
|
module.exports = function (searchElement/*, fromIndex*/) {
|
|
var i;
|
|
if (!numIsNaN(searchElement) && (searchElement !== 0)) {
|
|
return indexOf.apply(this, arguments);
|
|
}
|
|
|
|
for (i = (arguments[1] >>> 0); i < (value(this).length >>> 0); ++i) {
|
|
if (this.hasOwnProperty(i) && ois(searchElement, this[i])) {
|
|
return i;
|
|
}
|
|
}
|
|
return -1;
|
|
};
|
|
|
|
},{"../../Number/is-nan":13,"../../Object/is":25,"../../Object/valid-value":29}],8:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
var value = require('../../Object/valid-value');
|
|
|
|
module.exports = function () {
|
|
var i, l;
|
|
if (!(l = (value(this).length >>> 0))) {
|
|
return null;
|
|
}
|
|
i = l - 1;
|
|
while (!this.hasOwnProperty(i)) {
|
|
if (--i === -1) {
|
|
return null;
|
|
}
|
|
}
|
|
return i;
|
|
};
|
|
|
|
},{"../../Object/valid-value":29}],9:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
var lastIndex = require('./last-index');
|
|
|
|
module.exports = function () {
|
|
var i;
|
|
if ((i = lastIndex.call(this)) !== null) {
|
|
return this[i];
|
|
}
|
|
return undefined;
|
|
};
|
|
|
|
},{"./last-index":8}],10:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
var d = require('../Object/descriptor')
|
|
, extend = require('../Object/extend')
|
|
|
|
, captureStackTrace = Error.captureStackTrace
|
|
, CustomError;
|
|
|
|
CustomError = module.exports = function CustomError(message, code/*, ext*/) {
|
|
var ext = arguments[2];
|
|
if (ext != null) extend(this, ext);
|
|
this.message = String(message);
|
|
if (code != null) this.code = String(code);
|
|
if (captureStackTrace) captureStackTrace(this, CustomError);
|
|
};
|
|
|
|
CustomError.prototype = Object.create(Error.prototype, {
|
|
constructor: d(CustomError),
|
|
name: d('CustomError')
|
|
});
|
|
|
|
},{"../Object/descriptor":19,"../Object/extend":20}],11:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
var toString = Object.prototype.toString
|
|
|
|
, id = toString.call((function () { return arguments; }()));
|
|
|
|
module.exports = function (x) {
|
|
return toString.call(x) === id;
|
|
};
|
|
|
|
},{}],12:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
module.exports = function (value) {
|
|
value = Number(value);
|
|
if (isNaN(value) || (value === 0)) {
|
|
return value;
|
|
}
|
|
return (value > 0) ? 1 : -1;
|
|
};
|
|
|
|
},{}],13:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
module.exports = function (value) {
|
|
return (value !== value); //jslint: skip
|
|
};
|
|
|
|
},{}],14:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
var toString = Object.prototype.toString
|
|
|
|
, id = toString.call(1);
|
|
|
|
module.exports = function (x) {
|
|
return ((typeof x === 'number') ||
|
|
((x instanceof Number) ||
|
|
((typeof x === 'object') && (toString.call(x) === id))));
|
|
};
|
|
|
|
},{}],15:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
var sign = require('../Math/sign')
|
|
|
|
, abs = Math.abs, floor = Math.floor;
|
|
|
|
module.exports = function (value) {
|
|
if (isNaN(value)) {
|
|
return 0;
|
|
}
|
|
value = Number(value);
|
|
if ((value === 0) || !isFinite(value)) {
|
|
return value;
|
|
}
|
|
|
|
return sign(value) * floor(abs(value));
|
|
};
|
|
|
|
},{"../Math/sign":12}],16:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
var toInt = require('./to-int')
|
|
|
|
, max = Math.max;
|
|
|
|
module.exports = function (value) { return max(0, toInt(value)); };
|
|
|
|
},{"./to-int":15}],17:[function(require,module,exports){
|
|
// Internal method, used by iteration functions.
|
|
// Calls a function for each key-value pair found in object
|
|
// Optionally takes compareFn to iterate object in specific order
|
|
|
|
'use strict';
|
|
|
|
var isCallable = require('./is-callable')
|
|
, callable = require('./valid-callable')
|
|
, value = require('./valid-value')
|
|
|
|
, call = Function.prototype.call, keys = Object.keys;
|
|
|
|
module.exports = function (method) {
|
|
return function (obj, cb/*, thisArg, compareFn*/) {
|
|
var list, thisArg = arguments[2], compareFn = arguments[3];
|
|
obj = Object(value(obj));
|
|
callable(cb);
|
|
|
|
list = keys(obj);
|
|
if (compareFn) {
|
|
list.sort(isCallable(compareFn) ? compareFn.bind(obj) : undefined);
|
|
}
|
|
return list[method](function (key, index) {
|
|
return call.call(cb, thisArg, obj[key], key, obj, index);
|
|
});
|
|
};
|
|
};
|
|
|
|
},{"./is-callable":22,"./valid-callable":28,"./valid-value":29}],18:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
var isPlainObject = require('./is-plain-object')
|
|
, forEach = require('./for-each')
|
|
, extend = require('./extend')
|
|
, value = require('./valid-value')
|
|
|
|
, recursive;
|
|
|
|
recursive = function (to, from, cloned) {
|
|
forEach(from, function (value, key) {
|
|
var index;
|
|
if (isPlainObject(value)) {
|
|
if ((index = cloned[0].indexOf(value)) === -1) {
|
|
cloned[0].push(value);
|
|
cloned[1].push(to[key] = extend({}, value));
|
|
recursive(to[key], value, cloned);
|
|
} else {
|
|
to[key] = cloned[1][index];
|
|
}
|
|
}
|
|
}, from);
|
|
};
|
|
|
|
module.exports = function (obj/*, deep*/) {
|
|
var copy;
|
|
if ((copy = Object(value(obj))) === obj) {
|
|
copy = extend({}, obj);
|
|
if (arguments[1]) {
|
|
recursive(copy, obj, [[obj], [copy]]);
|
|
}
|
|
}
|
|
return copy;
|
|
};
|
|
|
|
},{"./extend":20,"./for-each":21,"./is-plain-object":24,"./valid-value":29}],19:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
var isCallable = require('./is-callable')
|
|
, callable = require('./valid-callable')
|
|
, validValue = require('./valid-value')
|
|
, copy = require('./copy')
|
|
, map = require('./map')
|
|
, isString = require('../String/is-string')
|
|
, contains = require('../String/prototype/contains')
|
|
|
|
, bind = Function.prototype.bind
|
|
, defineProperty = Object.defineProperty
|
|
, d;
|
|
|
|
d = module.exports = function (dscr, value) {
|
|
var c, e, w;
|
|
if (arguments.length < 2) {
|
|
value = dscr;
|
|
dscr = null;
|
|
}
|
|
if (dscr == null) {
|
|
c = w = true;
|
|
e = false;
|
|
} else {
|
|
c = contains.call(dscr, 'c');
|
|
e = contains.call(dscr, 'e');
|
|
w = contains.call(dscr, 'w');
|
|
}
|
|
|
|
return { value: value, configurable: c, enumerable: e, writable: w };
|
|
};
|
|
|
|
d.gs = function (dscr, get, set) {
|
|
var c, e;
|
|
if (isCallable(dscr)) {
|
|
set = (get == null) ? undefined : callable(get);
|
|
get = dscr;
|
|
dscr = null;
|
|
} else {
|
|
get = (get == null) ? undefined : callable(get);
|
|
set = (set == null) ? undefined : callable(set);
|
|
}
|
|
if (dscr == null) {
|
|
c = true;
|
|
e = false;
|
|
} else {
|
|
c = contains.call(dscr, 'c');
|
|
e = contains.call(dscr, 'e');
|
|
}
|
|
|
|
return { get: get, set: set, configurable: c, enumerable: e };
|
|
};
|
|
|
|
d.binder = function self(name, dv) {
|
|
var value, dgs;
|
|
if (!isString(name)) {
|
|
return map(name, function (dv, name) { return self(name, dv); });
|
|
}
|
|
value = validValue(dv) && callable(dv.value);
|
|
dgs = copy(dv);
|
|
delete dgs.writable;
|
|
delete dgs.value;
|
|
dgs.get = function () {
|
|
dv.value = bind.call(value, this);
|
|
defineProperty(this, name, dv);
|
|
return this[name];
|
|
};
|
|
return dgs;
|
|
};
|
|
|
|
},{"../String/is-string":30,"../String/prototype/contains":31,"./copy":18,"./is-callable":22,"./map":26,"./valid-callable":28,"./valid-value":29}],20:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
var value = require('./valid-value')
|
|
|
|
, forEach = Array.prototype.forEach, slice = Array.prototype.slice
|
|
, keys = Object.keys
|
|
|
|
, extend;
|
|
|
|
extend = function (src) {
|
|
keys(Object(src)).forEach(function (key) {
|
|
this[key] = src[key];
|
|
}, this);
|
|
};
|
|
|
|
module.exports = function (dest/*, …src*/) {
|
|
forEach.call(arguments, value);
|
|
slice.call(arguments, 1).forEach(extend, dest);
|
|
return dest;
|
|
};
|
|
|
|
},{"./valid-value":29}],21:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
module.exports = require('./_iterate')('forEach');
|
|
|
|
},{"./_iterate":17}],22:[function(require,module,exports){
|
|
// Inspired by: http://www.davidflanagan.com/2009/08/typeof-isfuncti.html
|
|
|
|
'use strict';
|
|
|
|
var forEach = Array.prototype.forEach.bind([]);
|
|
|
|
module.exports = function (obj) {
|
|
var type;
|
|
if (!obj) {
|
|
return false;
|
|
}
|
|
type = typeof obj;
|
|
if (type === 'function') {
|
|
return true;
|
|
}
|
|
if (type !== 'object') {
|
|
return false;
|
|
}
|
|
|
|
try {
|
|
forEach(obj);
|
|
return true;
|
|
} catch (e) {
|
|
if (e instanceof TypeError) {
|
|
return false;
|
|
}
|
|
throw e;
|
|
}
|
|
};
|
|
|
|
},{}],23:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
var value = require('./valid-value');
|
|
|
|
module.exports = function (obj) {
|
|
var i;
|
|
value(obj);
|
|
for (i in obj) { //jslint: skip
|
|
if (obj.propertyIsEnumerable(i)) return false;
|
|
}
|
|
return true;
|
|
};
|
|
|
|
},{"./valid-value":29}],24:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
var getPrototypeOf = Object.getPrototypeOf, prototype = Object.prototype
|
|
, toString = prototype.toString
|
|
|
|
, id = {}.toString();
|
|
|
|
module.exports = function (value) {
|
|
var proto;
|
|
if (!value || (typeof value !== 'object') || (toString.call(value) !== id)) {
|
|
return false;
|
|
}
|
|
proto = getPrototypeOf(value);
|
|
return (proto === prototype) || (getPrototypeOf(proto) === null);
|
|
};
|
|
|
|
},{}],25:[function(require,module,exports){
|
|
// Implementation credits go to:
|
|
// http://wiki.ecmascript.org/doku.php?id=harmony:egal
|
|
|
|
'use strict';
|
|
|
|
module.exports = function (x, y) {
|
|
return (x === y) ?
|
|
((x !== 0) || ((1 / x) === (1 / y))) :
|
|
((x !== x) && (y !== y)); //jslint: skip
|
|
};
|
|
|
|
},{}],26:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
var callable = require('./valid-callable')
|
|
, forEach = require('./for-each')
|
|
|
|
, call = Function.prototype.call;
|
|
|
|
module.exports = function (obj, cb/*, thisArg*/) {
|
|
var o = {}, thisArg = arguments[2];
|
|
callable(cb);
|
|
forEach(obj, function (value, key, obj, index) {
|
|
o[key] = call.call(cb, thisArg, value, key, obj, index);
|
|
});
|
|
return o;
|
|
};
|
|
|
|
},{"./for-each":21,"./valid-callable":28}],27:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
var isCallable = require('./is-callable')
|
|
, callable = require('./valid-callable')
|
|
, value = require('./valid-value')
|
|
|
|
, call = Function.prototype.call, keys = Object.keys;
|
|
|
|
module.exports = exports = function self(obj, cb/*, initial, compareFn*/) {
|
|
var list, fn, initial, compareFn, initialized;
|
|
value(obj) && callable(cb);
|
|
|
|
obj = Object(obj);
|
|
initial = arguments[2];
|
|
compareFn = arguments[3];
|
|
|
|
list = keys(obj);
|
|
if (compareFn) {
|
|
list.sort(isCallable(compareFn) ? compareFn.bind(obj) : undefined);
|
|
}
|
|
|
|
fn = function (value, key, index) {
|
|
if (initialized) {
|
|
return call.call(cb, undefined, value, obj[key], key, obj, index);
|
|
} else {
|
|
initialized = true;
|
|
return call.call(cb, undefined, obj[value], obj[key], key, obj, index,
|
|
value);
|
|
}
|
|
};
|
|
|
|
if ((arguments.length < 3) || (initial === self.NO_INITIAL)) {
|
|
return list.reduce(fn);
|
|
} else {
|
|
initialized = true;
|
|
return list.reduce(fn, initial);
|
|
}
|
|
};
|
|
exports.NO_INITIAL = {};
|
|
|
|
},{"./is-callable":22,"./valid-callable":28,"./valid-value":29}],28:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
var isCallable = require('./is-callable');
|
|
|
|
module.exports = function (fn) {
|
|
if (!isCallable(fn)) {
|
|
throw new TypeError(fn + " is not a function");
|
|
}
|
|
return fn;
|
|
};
|
|
|
|
},{"./is-callable":22}],29:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
module.exports = function (value) {
|
|
if (value == null) {
|
|
throw new TypeError("Cannot use null or undefined");
|
|
}
|
|
return value;
|
|
};
|
|
|
|
},{}],30:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
var toString = Object.prototype.toString
|
|
|
|
, id = toString.call('');
|
|
|
|
module.exports = function (x) {
|
|
return (typeof x === 'string') || (x && (typeof x === 'object') &&
|
|
((x instanceof String) || (toString.call(x) === id))) || false;
|
|
};
|
|
|
|
},{}],31:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
var indexOf = String.prototype.indexOf;
|
|
|
|
module.exports = function (searchString/*, position*/) {
|
|
return indexOf.call(this, searchString, arguments[1]) > -1;
|
|
};
|
|
|
|
},{}],32:[function(require,module,exports){
|
|
// Not rocket science but taken from:
|
|
// http://closure-library.googlecode.com/svn/trunk/closure/goog/string/string.js
|
|
|
|
'use strict';
|
|
|
|
var value = require('../../Object/valid-value')
|
|
, toUint = require('../../Number/to-uint');
|
|
|
|
module.exports = function (n) {
|
|
return new Array((isNaN(n) ? 1 : toUint(n)) + 1).join(String(value(this)));
|
|
};
|
|
|
|
},{"../../Number/to-uint":16,"../../Object/valid-value":29}],33:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
module.exports = new Function("return this")();
|
|
|
|
},{}],34:[function(require,module,exports){
|
|
// To be used internally, memoize factory
|
|
|
|
'use strict';
|
|
|
|
var callable = require('es5-ext/lib/Object/valid-callable')
|
|
, forEach = require('es5-ext/lib/Object/for-each')
|
|
, ee = require('event-emitter/lib/core')
|
|
|
|
, ext;
|
|
|
|
module.exports = exports = function (core) {
|
|
return function self(fn/*, options */) {
|
|
var options, length, get, clear, conf;
|
|
|
|
callable(fn);
|
|
if (fn.memoized) {
|
|
// Do not memoize already memoized function
|
|
return fn;
|
|
}
|
|
|
|
options = Object(arguments[1]);
|
|
conf = ee({ memoize: self, fn: fn });
|
|
|
|
// Normalize length
|
|
if (isNaN(options.length)) {
|
|
length = fn.length;
|
|
// Special case
|
|
if (options.async && ext.async) {
|
|
--length;
|
|
}
|
|
} else {
|
|
length = (options.length === false) ? false : (options.length >>> 0);
|
|
}
|
|
|
|
core(conf, length);
|
|
|
|
forEach(ext, function (fn, name) {
|
|
if (fn.force) {
|
|
fn(conf, options);
|
|
} else if (options[name]) {
|
|
fn(options[name], conf, options);
|
|
}
|
|
});
|
|
|
|
fn = conf.fn;
|
|
get = conf.get;
|
|
clear = conf.clear;
|
|
|
|
conf.memoized.clear = function () { clear(get(arguments)); };
|
|
conf.memoized.clearAll = function () {
|
|
conf.emit('purgeall');
|
|
conf.clearAll();
|
|
};
|
|
conf.memoized.memoized = true;
|
|
conf.emit('ready');
|
|
return conf.memoized;
|
|
};
|
|
};
|
|
ext = exports.ext = {};
|
|
|
|
},{"es5-ext/lib/Object/for-each":21,"es5-ext/lib/Object/valid-callable":28,"event-emitter/lib/core":46}],35:[function(require,module,exports){
|
|
// Support for asynchronous functions
|
|
|
|
'use strict';
|
|
|
|
var toArray = require('es5-ext/lib/Array/from')
|
|
, last = require('es5-ext/lib/Array/prototype/last')
|
|
, isArguments = require('es5-ext/lib/Function/is-arguments')
|
|
, forEach = require('es5-ext/lib/Object/for-each')
|
|
, isCallable = require('es5-ext/lib/Object/is-callable')
|
|
, nextTick = require('next-tick')
|
|
|
|
, isArray = Array.isArray, slice = Array.prototype.slice
|
|
, apply = Function.prototype.apply;
|
|
|
|
require('../_base').ext.async = function (ignore, conf) {
|
|
var cache, purge;
|
|
|
|
cache = conf.async = {};
|
|
|
|
(function (org) {
|
|
var value, cb, initContext, initArgs, fn, resolver;
|
|
|
|
conf.on('init', function (id) {
|
|
value.id = id;
|
|
cache[id] = cb ? [cb] : [];
|
|
});
|
|
|
|
conf.on('hit', function (id, syncArgs, syncCtx) {
|
|
if (!cb) {
|
|
return;
|
|
}
|
|
|
|
if (isArray(cache[id])) {
|
|
cache[id].push(cb);
|
|
} else {
|
|
nextTick(function (cb, id, ctx, args) {
|
|
if (cache[id]) {
|
|
conf.emit('hitasync', id, syncArgs, syncCtx);
|
|
apply.call(cb, this.context, this);
|
|
} else {
|
|
// Purged in a meantime, we shouldn't rely on cached value, recall
|
|
fn.apply(ctx, args);
|
|
}
|
|
}.bind(cache[id], cb, id, initContext, initArgs));
|
|
initContext = initArgs = null;
|
|
}
|
|
});
|
|
conf.fn = function () {
|
|
var args, asyncArgs;
|
|
args = arguments;
|
|
asyncArgs = toArray(args);
|
|
asyncArgs.push(value = function self(err) {
|
|
var i, cb, waiting, res;
|
|
if (self.id == null) {
|
|
// Shouldn't happen, means async callback was called sync way
|
|
nextTick(apply.bind(self, this, arguments));
|
|
return;
|
|
}
|
|
waiting = cache[self.id];
|
|
if (conf.cache.hasOwnProperty(self.id)) {
|
|
if (err) {
|
|
delete cache[self.id];
|
|
conf.clear(self.id);
|
|
} else {
|
|
arguments.context = this;
|
|
cache[self.id] = arguments;
|
|
conf.emit('initasync', self.id, waiting.length);
|
|
}
|
|
} else {
|
|
delete cache[self.id];
|
|
}
|
|
for (i = 0; (cb = waiting[i]); ++i) {
|
|
res = apply.call(cb, this, arguments);
|
|
}
|
|
return res;
|
|
});
|
|
return apply.call(org, this, asyncArgs);
|
|
};
|
|
|
|
fn = conf.memoized;
|
|
resolver = function (args) {
|
|
cb = last.call(args);
|
|
if (isCallable(cb)) {
|
|
return slice.call(args, 0, -1);
|
|
} else {
|
|
cb = null;
|
|
return args;
|
|
}
|
|
};
|
|
conf.memoized = function () {
|
|
return fn.apply(initContext = this, initArgs = resolver(arguments));
|
|
};
|
|
forEach(fn, function (value, name) {
|
|
conf.memoized[name] = function () {
|
|
return fn[name].apply(this, resolver(arguments));
|
|
};
|
|
});
|
|
|
|
}(conf.fn));
|
|
|
|
conf.on('purge', purge = function (id) {
|
|
// If false, we don't have value yet, so we assume that intention is not
|
|
// to memoize this call. After value is obtained we don't cache it but
|
|
// gracefully pass to callback
|
|
if (isArguments(cache[id])) {
|
|
conf.emit('purgeasync', id);
|
|
delete cache[id];
|
|
}
|
|
});
|
|
|
|
conf.on('purgeall', function () {
|
|
forEach(conf.async, function (value, id) { purge(id); });
|
|
});
|
|
};
|
|
|
|
},{"../_base":34,"es5-ext/lib/Array/from":6,"es5-ext/lib/Array/prototype/last":9,"es5-ext/lib/Function/is-arguments":11,"es5-ext/lib/Object/for-each":21,"es5-ext/lib/Object/is-callable":22,"next-tick":48}],36:[function(require,module,exports){
|
|
// Call dispose callback on each cache purge
|
|
|
|
'use strict';
|
|
|
|
var callable = require('es5-ext/lib/Object/valid-callable')
|
|
, forEach = require('es5-ext/lib/Object/for-each')
|
|
, ext = require('../_base').ext
|
|
|
|
, slice = Array.prototype.slice;
|
|
|
|
ext.dispose = function (dispose, conf, options) {
|
|
var clear, async;
|
|
|
|
callable(dispose);
|
|
|
|
async = (options.async && ext.async);
|
|
conf.on('purge' + (async ? 'async' : ''), clear = async ? function (id) {
|
|
var value = conf.async[id];
|
|
delete conf.cache[id];
|
|
dispose.apply(conf.memoized['_memoize:context_'], slice.call(value, 1));
|
|
} : function (id) {
|
|
var value = conf.cache[id];
|
|
delete conf.cache[id];
|
|
dispose.call(conf.memoized['_memoize:context_'], value);
|
|
});
|
|
|
|
if (!async) {
|
|
conf.on('purgeall', function () {
|
|
forEach(conf.cache, function (value, id) { clear(id); });
|
|
});
|
|
}
|
|
};
|
|
|
|
},{"../_base":34,"es5-ext/lib/Object/for-each":21,"es5-ext/lib/Object/valid-callable":28}],37:[function(require,module,exports){
|
|
// Timeout cached values
|
|
|
|
'use strict';
|
|
|
|
var isNumber = require('es5-ext/lib/Number/is-number')
|
|
, forEach = require('es5-ext/lib/Object/for-each')
|
|
, nextTick = require('next-tick')
|
|
, ext = require('../_base').ext
|
|
|
|
, max = Math.max, min = Math.min;
|
|
|
|
ext.maxAge = function (maxAge, conf, options) {
|
|
var cache, async, preFetchAge, preFetchCache;
|
|
|
|
maxAge = maxAge >>> 0;
|
|
if (!maxAge) {
|
|
return;
|
|
}
|
|
|
|
cache = {};
|
|
async = options.async && ext.async;
|
|
conf.on('init' + (async ? 'async' : ''), function (id) {
|
|
cache[id] = setTimeout(function () { conf.clear(id); }, maxAge);
|
|
if (preFetchCache) {
|
|
preFetchCache[id] = setTimeout(function () { delete preFetchCache[id]; },
|
|
preFetchAge);
|
|
}
|
|
});
|
|
conf.on('purge' + (async ? 'async' : ''), function (id) {
|
|
clearTimeout(cache[id]);
|
|
if (preFetchCache && preFetchCache[id]) {
|
|
clearTimeout(preFetchCache[id]);
|
|
delete preFetchCache[id];
|
|
}
|
|
delete cache[id];
|
|
});
|
|
|
|
if (options.preFetch) {
|
|
if (isNumber(options.preFetch)) {
|
|
preFetchAge = max(min(Number(options.preFetch), 1), 0);
|
|
} else {
|
|
preFetchAge = 0.333;
|
|
}
|
|
if (preFetchAge) {
|
|
preFetchCache = {};
|
|
preFetchAge = (1 - preFetchAge) * maxAge;
|
|
conf.on('hit' + (async ? 'async' : ''), function (id, args, ctx) {
|
|
if (!preFetchCache[id]) {
|
|
preFetchCache[id] = true;
|
|
nextTick(function () {
|
|
if (preFetchCache[id] === true) {
|
|
delete preFetchCache[id];
|
|
conf.clear(id);
|
|
conf.memoized.apply(ctx, args);
|
|
}
|
|
});
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
if (!async) {
|
|
conf.on('purgeall', function () {
|
|
forEach(cache, function (id) {
|
|
clearTimeout(id);
|
|
});
|
|
cache = {};
|
|
if (preFetchCache) {
|
|
forEach(preFetchCache, function (id) {
|
|
clearTimeout(id);
|
|
});
|
|
preFetchCache = {};
|
|
}
|
|
});
|
|
}
|
|
};
|
|
|
|
},{"../_base":34,"es5-ext/lib/Number/is-number":14,"es5-ext/lib/Object/for-each":21,"next-tick":48}],38:[function(require,module,exports){
|
|
// Limit cache size, LRU (least recently used) algorithm.
|
|
|
|
'use strict';
|
|
|
|
var ext = require('../_base').ext;
|
|
|
|
ext.max = function (max, conf, options) {
|
|
var index, base, size, queue, map, async;
|
|
|
|
max = max >>> 0;
|
|
if (!max) {
|
|
return;
|
|
}
|
|
|
|
index = -1;
|
|
base = size = 0;
|
|
queue = {};
|
|
map = {};
|
|
async = options.async && ext.async;
|
|
|
|
conf.on('init' + (async ? 'async' : ''), function (id) {
|
|
queue[++index] = id;
|
|
map[id] = index;
|
|
++size;
|
|
if (size > max) {
|
|
conf.clear(queue[base]);
|
|
}
|
|
});
|
|
|
|
conf.on('hit' + (async ? 'async' : ''), function (id) {
|
|
var oldIndex = map[id];
|
|
queue[++index] = id;
|
|
map[id] = index;
|
|
delete queue[oldIndex];
|
|
if (base === oldIndex) {
|
|
while (!queue.hasOwnProperty(++base)) continue; //jslint: skip
|
|
}
|
|
});
|
|
|
|
conf.on('purge' + (async ? 'async' : ''), function (id) {
|
|
var oldIndex = map[id];
|
|
delete queue[oldIndex];
|
|
--size;
|
|
if (base === oldIndex) {
|
|
if (!size) {
|
|
index = -1;
|
|
base = 0;
|
|
} else {
|
|
while (!queue.hasOwnProperty(++base)) continue; //jslint: skip
|
|
}
|
|
}
|
|
});
|
|
|
|
if (!async) {
|
|
conf.on('purgeall', function () {
|
|
index = -1;
|
|
base = size = 0;
|
|
queue = {};
|
|
map = {};
|
|
});
|
|
}
|
|
};
|
|
|
|
},{"../_base":34}],39:[function(require,module,exports){
|
|
// Memoized methods factory
|
|
|
|
'use strict';
|
|
|
|
var d = require('es5-ext/lib/Object/descriptor')
|
|
, global = require('es5-ext/lib/global')
|
|
, extend = require('es5-ext/lib/Object/extend')
|
|
, isString = require('es5-ext/lib/String/is-string')
|
|
|
|
, create = Object.create, defineProperty = Object.defineProperty;
|
|
|
|
require('../_base').ext.method = function (method, conf, options) {
|
|
if (isString(options.method)) {
|
|
method = { name: String(options.method),
|
|
descriptor: { configurable: true, writable: true } };
|
|
} else {
|
|
method = options.method;
|
|
method.name = String(method.name);
|
|
method.descriptor = (method.descriptor == null) ?
|
|
{ configurable: true, writable: true } : Object(method.descriptor);
|
|
}
|
|
options = create(options);
|
|
options.method = undefined;
|
|
|
|
(function (fn) {
|
|
conf.memoized = function () {
|
|
var memoized;
|
|
if (this && (this !== global)) {
|
|
memoized = method.descriptor.value =
|
|
conf.memoize(conf.fn.bind(this), options);
|
|
defineProperty(this, method.name, method.descriptor);
|
|
defineProperty(memoized, '_memoize:context_', d(this));
|
|
return memoized.apply(this, arguments);
|
|
}
|
|
return fn.apply(this, arguments);
|
|
};
|
|
extend(conf.memoized, fn);
|
|
}(conf.memoized));
|
|
};
|
|
|
|
},{"../_base":34,"es5-ext/lib/Object/descriptor":19,"es5-ext/lib/Object/extend":20,"es5-ext/lib/String/is-string":30,"es5-ext/lib/global":33}],40:[function(require,module,exports){
|
|
// Reference counter, useful for garbage collector like functionality
|
|
|
|
'use strict';
|
|
|
|
var ext = require('../_base').ext;
|
|
|
|
ext.refCounter = function (ignore, conf, options) {
|
|
var cache, async;
|
|
|
|
cache = {};
|
|
async = options.async && ext.async;
|
|
|
|
conf.on('init' + (async ? 'async' : ''), async ? function (id, length) {
|
|
cache[id] = length;
|
|
} : function (id) { cache[id] = 1; });
|
|
conf.on('hit' + (async ? 'async' : ''), function (id) { ++cache[id]; });
|
|
conf.on('purge' + (async ? 'async' : ''), function (id) {
|
|
delete cache[id];
|
|
});
|
|
if (!async) {
|
|
conf.on('purgeall', function () { cache = {}; });
|
|
}
|
|
|
|
conf.memoized.clearRef = function () {
|
|
var id = conf.get(arguments);
|
|
if (cache.hasOwnProperty(id)) {
|
|
if (!--cache[id]) {
|
|
conf.clear(id);
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
return null;
|
|
};
|
|
};
|
|
|
|
},{"../_base":34}],41:[function(require,module,exports){
|
|
// Normalize arguments before passing them to underlying function
|
|
|
|
'use strict';
|
|
|
|
var toArray = require('es5-ext/lib/Array/from')
|
|
, forEach = require('es5-ext/lib/Object/for-each')
|
|
, callable = require('es5-ext/lib/Object/valid-callable')
|
|
|
|
, slice = Array.prototype.slice
|
|
|
|
, resolve;
|
|
|
|
resolve = function (args) {
|
|
return this.map(function (r, i) {
|
|
return r ? r(args[i]) : args[i];
|
|
}).concat(slice.call(args, this.length));
|
|
};
|
|
|
|
require('../_base').ext.resolvers = function (resolvers, conf) {
|
|
var resolver;
|
|
|
|
resolver = toArray(resolvers);
|
|
resolver.forEach(function (r) { (r == null) || callable(r); });
|
|
resolver = resolve.bind(resolver);
|
|
|
|
(function (fn) {
|
|
conf.memoized = function () {
|
|
var value;
|
|
conf.memoized.args = arguments;
|
|
value = fn.apply(this, resolver(arguments));
|
|
delete conf.memoized.args;
|
|
return value;
|
|
};
|
|
forEach(fn, function (value, name) {
|
|
conf.memoized[name] = function () {
|
|
return fn[name].apply(this, resolver(arguments));
|
|
};
|
|
});
|
|
}(conf.memoized));
|
|
};
|
|
|
|
},{"../_base":34,"es5-ext/lib/Array/from":6,"es5-ext/lib/Object/for-each":21,"es5-ext/lib/Object/valid-callable":28}],42:[function(require,module,exports){
|
|
// Provides memoize with all options
|
|
|
|
'use strict';
|
|
|
|
var regular = require('./regular')
|
|
, primitive = require('./primitive')
|
|
|
|
, call = Function.prototype.call;
|
|
|
|
// Order is significant!
|
|
require('./ext/dispose');
|
|
require('./ext/resolvers');
|
|
require('./ext/async');
|
|
require('./ext/ref-counter');
|
|
require('./ext/method');
|
|
require('./ext/max-age');
|
|
require('./ext/max');
|
|
|
|
module.exports = function (fn/* options */) {
|
|
var options = Object(arguments[1]);
|
|
return call.call(options.primitive ? primitive : regular, this, fn, options);
|
|
};
|
|
|
|
},{"./ext/async":35,"./ext/dispose":36,"./ext/max":38,"./ext/max-age":37,"./ext/method":39,"./ext/ref-counter":40,"./ext/resolvers":41,"./primitive":43,"./regular":44}],43:[function(require,module,exports){
|
|
// Memoize working in primitive mode
|
|
|
|
'use strict';
|
|
|
|
var CustomError = require('es5-ext/lib/Error/custom')
|
|
, hasListeners = require('event-emitter/lib/has-listeners')
|
|
|
|
, getId0 = function () { return ''; }
|
|
, getId1 = function (args) { return args[0]; }
|
|
|
|
, apply = Function.prototype.apply, call = Function.prototype.call;
|
|
|
|
module.exports = require('./_base')(function (conf, length) {
|
|
var get, cache = conf.cache = {}, fn
|
|
, hitListeners, initListeners, purgeListeners;
|
|
|
|
if (length === 1) {
|
|
get = conf.get = getId1;
|
|
} else if (length === false) {
|
|
get = conf.get = function (args) {
|
|
var id = '', i, length = args.length;
|
|
if (length) {
|
|
id += args[i = 0];
|
|
while (--length) {
|
|
id += '\u0001' + args[++i];
|
|
}
|
|
} else {
|
|
id = '\u0002';
|
|
}
|
|
return id;
|
|
};
|
|
} else if (length) {
|
|
get = conf.get = function (args) {
|
|
var id = String(args[0]), i = 0, l = length;
|
|
while (--l) { id += '\u0001' + args[++i]; }
|
|
return id;
|
|
};
|
|
} else {
|
|
get = conf.get = getId0;
|
|
}
|
|
|
|
conf.memoized = (length === 1) ? function (id) {
|
|
var value;
|
|
if (cache.hasOwnProperty(id)) {
|
|
hitListeners && conf.emit('hit', id, arguments, this);
|
|
return cache[id];
|
|
} else {
|
|
if (arguments.length === 1) {
|
|
value = call.call(fn, this, id);
|
|
} else {
|
|
value = apply.call(fn, this, arguments);
|
|
}
|
|
if (cache.hasOwnProperty(id)) {
|
|
throw new CustomError("Circular invocation", 'CIRCULAR_INVOCATION');
|
|
}
|
|
cache[id] = value;
|
|
initListeners && conf.emit('init', id);
|
|
return value;
|
|
}
|
|
} : function () {
|
|
var id = get(arguments), value;
|
|
if (cache.hasOwnProperty(id)) {
|
|
hitListeners && conf.emit('hit', id, arguments, this);
|
|
return cache[id];
|
|
} else {
|
|
value = apply.call(conf.fn, this, arguments);
|
|
if (cache.hasOwnProperty(id)) {
|
|
throw new CustomError("Circular invocation", 'CIRCULAR_INVOCATION');
|
|
}
|
|
cache[id] = value;
|
|
initListeners && conf.emit('init', id);
|
|
return value;
|
|
}
|
|
};
|
|
|
|
conf.clear = function (id) {
|
|
if (cache.hasOwnProperty(id)) {
|
|
purgeListeners && conf.emit('purge', id);
|
|
delete cache[id];
|
|
}
|
|
};
|
|
conf.clearAll = function () { cache = conf.cache = {}; };
|
|
|
|
conf.once('ready', function () {
|
|
fn = conf.fn;
|
|
hitListeners = hasListeners(conf, 'hit');
|
|
initListeners = hasListeners(conf, 'init');
|
|
purgeListeners = hasListeners(conf, 'purge');
|
|
});
|
|
});
|
|
|
|
},{"./_base":34,"es5-ext/lib/Error/custom":10,"event-emitter/lib/has-listeners":47}],44:[function(require,module,exports){
|
|
// Memoize working in object mode (supports any type of arguments)
|
|
|
|
'use strict';
|
|
|
|
var CustomError = require('es5-ext/lib/Error/custom')
|
|
, indexOf = require('es5-ext/lib/Array/prototype/e-index-of')
|
|
, hasListeners = require('event-emitter/lib/has-listeners')
|
|
|
|
, apply = Function.prototype.apply;
|
|
|
|
// Results are saved internally within array matrix:
|
|
// [0] -> Result of calling function with no arguments
|
|
// [1] -> Matrix that keeps results when function is called with one argument
|
|
// [1][0] -> Array of arguments with which
|
|
// function have been called
|
|
// [1][1] -> Array of results that matches [1][0] array
|
|
// [2] -> Matrix that keeps results when function is called with two arguments
|
|
// [2][0] -> Array of first (of two) arguments with which
|
|
// function have been called
|
|
// [2][1] -> Matrixes that keeps results for two arguments function calls
|
|
// Each matrix matches first argument found in [2][0]
|
|
// [2][1][x][0] -> Array of second arguments with which
|
|
// function have been called.
|
|
// [2][1][x][1] -> Array of results that matches [2][1][x][0]
|
|
// arguments array
|
|
// ...and so on
|
|
module.exports = require('./_base')(function (conf, length) {
|
|
var map, map1, map2, get, set, clear, count, fn
|
|
, hitListeners, initListeners, purgeListeners
|
|
, cache = conf.cache = {}, argsCache;
|
|
|
|
if (length === 0) {
|
|
map = null;
|
|
get = conf.get = function () { return map; };
|
|
set = function () { return ((map = 1)); };
|
|
clear = function () { map = null; };
|
|
conf.clearAll = function () {
|
|
map = null;
|
|
cache = conf.cache = {};
|
|
};
|
|
} else {
|
|
count = 0;
|
|
if (length === 1) {
|
|
map1 = [];
|
|
map2 = [];
|
|
get = conf.get = function (args) {
|
|
var index = indexOf.call(map1, args[0]);
|
|
return (index === -1) ? null : map2[index];
|
|
};
|
|
set = function (args) {
|
|
map1.push(args[0]);
|
|
map2.push(++count);
|
|
return count;
|
|
};
|
|
clear = function (id) {
|
|
var index = indexOf.call(map2, id);
|
|
if (index !== -1) {
|
|
map1.splice(index, 1);
|
|
map2.splice(index, 1);
|
|
}
|
|
};
|
|
conf.clearAll = function () {
|
|
map1 = [];
|
|
map2 = [];
|
|
cache = conf.cache = {};
|
|
};
|
|
} else if (length === false) {
|
|
map = [];
|
|
argsCache = {};
|
|
get = conf.get = function (args) {
|
|
var index = 0, set = map, i, length = args.length;
|
|
if (length === 0) {
|
|
return set[length] || null;
|
|
} else if ((set = set[length])) {
|
|
while (index < (length - 1)) {
|
|
i = indexOf.call(set[0], args[index]);
|
|
if (i === -1) return null;
|
|
set = set[1][i];
|
|
++index;
|
|
}
|
|
i = indexOf.call(set[0], args[index]);
|
|
if (i === -1) return null;
|
|
return set[1][i] || null;
|
|
}
|
|
return null;
|
|
};
|
|
set = function (args) {
|
|
var index = 0, set = map, i, length = args.length;
|
|
if (length === 0) {
|
|
set[length] = ++count;
|
|
} else {
|
|
if (!set[length]) {
|
|
set[length] = [[], []];
|
|
}
|
|
set = set[length];
|
|
while (index < (length - 1)) {
|
|
i = indexOf.call(set[0], args[index]);
|
|
if (i === -1) {
|
|
i = set[0].push(args[index]) - 1;
|
|
set[1].push([[], []]);
|
|
}
|
|
set = set[1][i];
|
|
++index;
|
|
}
|
|
i = indexOf.call(set[0], args[index]);
|
|
if (i === -1) {
|
|
i = set[0].push(args[index]) - 1;
|
|
}
|
|
set[1][i] = ++count;
|
|
}
|
|
argsCache[count] = args;
|
|
return count;
|
|
};
|
|
clear = function (id) {
|
|
var index = 0, set = map, i, args = argsCache[id], length = args.length
|
|
, path = [];
|
|
if (length === 0) {
|
|
delete set[length];
|
|
} else if ((set = set[length])) {
|
|
while (index < (length - 1)) {
|
|
i = indexOf.call(set[0], args[index]);
|
|
if (i === -1) {
|
|
return;
|
|
}
|
|
path.push(set, i);
|
|
set = set[1][i];
|
|
++index;
|
|
}
|
|
i = indexOf.call(set[0], args[index]);
|
|
if (i === -1) {
|
|
return;
|
|
}
|
|
id = set[1][i];
|
|
set[0].splice(i, 1);
|
|
set[1].splice(i, 1);
|
|
while (!set[0].length && path.length) {
|
|
i = path.pop();
|
|
set = path.pop();
|
|
set[0].splice(i, 1);
|
|
set[1].splice(i, 1);
|
|
}
|
|
}
|
|
delete argsCache[id];
|
|
};
|
|
conf.clearAll = function () {
|
|
map = [];
|
|
cache = conf.cache = {};
|
|
argsCache = {};
|
|
};
|
|
} else {
|
|
map = [[], []];
|
|
argsCache = {};
|
|
get = conf.get = function (args) {
|
|
var index = 0, set = map, i;
|
|
while (index < (length - 1)) {
|
|
i = indexOf.call(set[0], args[index]);
|
|
if (i === -1) return null;
|
|
set = set[1][i];
|
|
++index;
|
|
}
|
|
i = indexOf.call(set[0], args[index]);
|
|
if (i === -1) return null;
|
|
return set[1][i] || null;
|
|
};
|
|
set = function (args) {
|
|
var index = 0, set = map, i;
|
|
while (index < (length - 1)) {
|
|
i = indexOf.call(set[0], args[index]);
|
|
if (i === -1) {
|
|
i = set[0].push(args[index]) - 1;
|
|
set[1].push([[], []]);
|
|
}
|
|
set = set[1][i];
|
|
++index;
|
|
}
|
|
i = indexOf.call(set[0], args[index]);
|
|
if (i === -1) {
|
|
i = set[0].push(args[index]) - 1;
|
|
}
|
|
set[1][i] = ++count;
|
|
argsCache[count] = args;
|
|
return count;
|
|
};
|
|
clear = function (id) {
|
|
var index = 0, set = map, i, path = [], args = argsCache[id];
|
|
while (index < (length - 1)) {
|
|
i = indexOf.call(set[0], args[index]);
|
|
if (i === -1) {
|
|
return;
|
|
}
|
|
path.push(set, i);
|
|
set = set[1][i];
|
|
++index;
|
|
}
|
|
i = indexOf.call(set[0], args[index]);
|
|
if (i === -1) {
|
|
return;
|
|
}
|
|
id = set[1][i];
|
|
set[0].splice(i, 1);
|
|
set[1].splice(i, 1);
|
|
while (!set[0].length && path.length) {
|
|
i = path.pop();
|
|
set = path.pop();
|
|
set[0].splice(i, 1);
|
|
set[1].splice(i, 1);
|
|
}
|
|
delete argsCache[id];
|
|
};
|
|
conf.clearAll = function () {
|
|
map = [[], []];
|
|
cache = conf.cache = {};
|
|
argsCache = {};
|
|
};
|
|
}
|
|
}
|
|
conf.memoized = function () {
|
|
var id = get(arguments), value;
|
|
if (id != null) {
|
|
hitListeners && conf.emit('hit', id, arguments, this);
|
|
return cache[id];
|
|
} else {
|
|
value = apply.call(fn, this, arguments);
|
|
id = get(arguments);
|
|
if (id != null) {
|
|
throw new CustomError("Circular invocation", 'CIRCULAR_INVOCATION');
|
|
}
|
|
id = set(arguments);
|
|
cache[id] = value;
|
|
initListeners && conf.emit('init', id);
|
|
return value;
|
|
}
|
|
};
|
|
conf.clear = function (id) {
|
|
if (cache.hasOwnProperty(id)) {
|
|
purgeListeners && conf.emit('purge', id);
|
|
clear(id);
|
|
delete cache[id];
|
|
}
|
|
};
|
|
|
|
conf.once('ready', function () {
|
|
fn = conf.fn;
|
|
hitListeners = hasListeners(conf, 'hit');
|
|
initListeners = hasListeners(conf, 'init');
|
|
purgeListeners = hasListeners(conf, 'purge');
|
|
});
|
|
});
|
|
|
|
},{"./_base":34,"es5-ext/lib/Array/prototype/e-index-of":7,"es5-ext/lib/Error/custom":10,"event-emitter/lib/has-listeners":47}],45:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
module.exports = '_ee2_';
|
|
|
|
},{}],46:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
var d = require('es5-ext/lib/Object/descriptor')
|
|
, callable = require('es5-ext/lib/Object/valid-callable')
|
|
, id = require('./_id')
|
|
|
|
, apply = Function.prototype.apply, call = Function.prototype.call
|
|
, create = Object.create, defineProperty = Object.defineProperty
|
|
, defineProperties = Object.defineProperties
|
|
, descriptor = { configurable: true, enumerable: false, writable: true }
|
|
|
|
, on, once, off, emit
|
|
, colId, methods, descriptors, base;
|
|
|
|
colId = id + 'l_';
|
|
|
|
on = function (type, listener) {
|
|
var data;
|
|
|
|
callable(listener);
|
|
|
|
if (!this.hasOwnProperty(id)) {
|
|
data = descriptor.value = {};
|
|
defineProperty(this, id, descriptor);
|
|
descriptor.value = null;
|
|
} else {
|
|
data = this[id];
|
|
}
|
|
if (!data.hasOwnProperty(type)) data[type] = listener;
|
|
else if (data[type].hasOwnProperty(colId)) data[type].push(listener);
|
|
else (data[type] = [data[type], listener])[colId] = true;
|
|
|
|
return this;
|
|
};
|
|
|
|
once = function (type, listener) {
|
|
var once, self;
|
|
|
|
callable(listener);
|
|
self = this;
|
|
on.call(this, type, once = function () {
|
|
off.call(self, type, once);
|
|
apply.call(listener, this, arguments);
|
|
});
|
|
|
|
once._listener = listener;
|
|
return this;
|
|
};
|
|
|
|
off = function (type, listener) {
|
|
var data, listeners, candidate, i;
|
|
|
|
callable(listener);
|
|
|
|
if (!this.hasOwnProperty(id)) return this;
|
|
data = this[id];
|
|
if (!data.hasOwnProperty(type)) return this;
|
|
listeners = data[type];
|
|
|
|
if (listeners.hasOwnProperty(colId)) {
|
|
for (i = 0; (candidate = listeners[i]); ++i) {
|
|
if ((candidate === listener) || (candidate._listener === listener)) {
|
|
if (listeners.length === 2) data[type] = listeners[i ? 0 : 1];
|
|
else listeners.splice(i, 1);
|
|
}
|
|
}
|
|
} else {
|
|
if ((listeners === listener) || (listeners._listener === listener)) {
|
|
delete data[type];
|
|
}
|
|
}
|
|
|
|
return this;
|
|
};
|
|
|
|
emit = function (type) {
|
|
var data, i, l, listener, listeners, args;
|
|
|
|
if (!this.hasOwnProperty(id)) return;
|
|
data = this[id];
|
|
if (!data.hasOwnProperty(type)) return;
|
|
listeners = data[type];
|
|
|
|
if (listeners.hasOwnProperty(colId)) {
|
|
l = arguments.length;
|
|
args = new Array(l - 1);
|
|
for (i = 1; i < l; ++i) {
|
|
args[i - 1] = arguments[i];
|
|
}
|
|
|
|
listeners = listeners.slice();
|
|
for (i = 0; (listener = listeners[i]); ++i) {
|
|
apply.call(listener, this, args);
|
|
}
|
|
} else {
|
|
switch (arguments.length) {
|
|
case 1:
|
|
call.call(listeners, this);
|
|
break;
|
|
case 2:
|
|
call.call(listeners, this, arguments[1]);
|
|
break;
|
|
case 3:
|
|
call.call(listeners, this, arguments[1], arguments[2]);
|
|
break;
|
|
default:
|
|
l = arguments.length;
|
|
args = new Array(l - 1);
|
|
for (i = 1; i < l; ++i) {
|
|
args[i - 1] = arguments[i];
|
|
}
|
|
apply.call(listeners, this, args);
|
|
}
|
|
}
|
|
};
|
|
|
|
methods = {
|
|
on: on,
|
|
once: once,
|
|
off: off,
|
|
emit: emit
|
|
};
|
|
|
|
descriptors = {
|
|
on: d(on),
|
|
once: d(once),
|
|
off: d(off),
|
|
emit: d(emit)
|
|
};
|
|
|
|
base = defineProperties({}, descriptors);
|
|
|
|
module.exports = exports = function (o) {
|
|
return (o == null) ? create(base) : defineProperties(Object(o), descriptors);
|
|
};
|
|
exports.methods = methods;
|
|
|
|
},{"./_id":45,"es5-ext/lib/Object/descriptor":19,"es5-ext/lib/Object/valid-callable":28}],47:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
var isEmpty = require('es5-ext/lib/Object/is-empty')
|
|
, value = require('es5-ext/lib/Object/valid-value')
|
|
, id = require('./_id');
|
|
|
|
module.exports = function (obj/*, type*/) {
|
|
var type;
|
|
value(obj);
|
|
type = arguments[1];
|
|
if (arguments.length > 1) {
|
|
return obj.hasOwnProperty(id) && obj[id].hasOwnProperty(type);
|
|
} else {
|
|
return obj.hasOwnProperty(id) && !isEmpty(obj[id]);
|
|
}
|
|
};
|
|
|
|
},{"./_id":45,"es5-ext/lib/Object/is-empty":23,"es5-ext/lib/Object/valid-value":29}],48:[function(require,module,exports){
|
|
var process=require("__browserify_process");'use strict';
|
|
|
|
if ((typeof process !== 'undefined') && process &&
|
|
(typeof process.nextTick === 'function')) {
|
|
|
|
// Node.js
|
|
module.exports = process.nextTick;
|
|
|
|
} else if (typeof setImmediate === 'function') {
|
|
|
|
// W3C Draft
|
|
// https://dvcs.w3.org/hg/webperf/raw-file/tip/specs/setImmediate/Overview.html
|
|
module.exports = function (cb) { setImmediate(cb); };
|
|
|
|
} else {
|
|
|
|
// Wide available standard
|
|
module.exports = function (cb) { setTimeout(cb, 0); };
|
|
}
|
|
|
|
},{"__browserify_process":91}],49:[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":90}],50:[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;
|
|
}
|
|
|
|
},{}],51:[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.
|
|
|
|
// a duplex stream is just a stream that is both readable and writable.
|
|
// Since JS doesn't have multiple prototypal inheritance, this class
|
|
// prototypally inherits from Readable, and then parasitically from
|
|
// Writable.
|
|
|
|
module.exports = Duplex;
|
|
var util = require('util');
|
|
var shims = require('_shims');
|
|
var timers = require('timers');
|
|
var Readable = require('_stream_readable');
|
|
var Writable = require('_stream_writable');
|
|
|
|
util.inherits(Duplex, Readable);
|
|
|
|
shims.forEach(shims.keys(Writable.prototype), function(method) {
|
|
if (!Duplex.prototype[method])
|
|
Duplex.prototype[method] = Writable.prototype[method];
|
|
});
|
|
|
|
function Duplex(options) {
|
|
if (!(this instanceof Duplex))
|
|
return new Duplex(options);
|
|
|
|
Readable.call(this, options);
|
|
Writable.call(this, options);
|
|
|
|
if (options && options.readable === false)
|
|
this.readable = false;
|
|
|
|
if (options && options.writable === false)
|
|
this.writable = false;
|
|
|
|
this.allowHalfOpen = true;
|
|
if (options && options.allowHalfOpen === false)
|
|
this.allowHalfOpen = false;
|
|
|
|
this.once('end', onend);
|
|
}
|
|
|
|
// the no-half-open enforcer
|
|
function onend() {
|
|
// if we allow half-open state, or if the writable side ended,
|
|
// then we're ok.
|
|
if (this.allowHalfOpen || this._writableState.ended)
|
|
return;
|
|
|
|
// no more data can be written.
|
|
// But allow more writes to happen in this tick.
|
|
timers.setImmediate(shims.bind(this.end, this));
|
|
}
|
|
|
|
},{"_shims":50,"_stream_readable":53,"_stream_writable":55,"timers":65,"util":68}],52:[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.
|
|
|
|
// a passthrough stream.
|
|
// basically just the most minimal sort of Transform stream.
|
|
// Every written chunk gets output as-is.
|
|
|
|
module.exports = PassThrough;
|
|
|
|
var Transform = require('_stream_transform');
|
|
var util = require('util');
|
|
util.inherits(PassThrough, Transform);
|
|
|
|
function PassThrough(options) {
|
|
if (!(this instanceof PassThrough))
|
|
return new PassThrough(options);
|
|
|
|
Transform.call(this, options);
|
|
}
|
|
|
|
PassThrough.prototype._transform = function(chunk, encoding, cb) {
|
|
cb(null, chunk);
|
|
};
|
|
|
|
},{"_stream_transform":54,"util":68}],53:[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.
|
|
|
|
module.exports = Readable;
|
|
Readable.ReadableState = ReadableState;
|
|
|
|
var EE = require('events').EventEmitter;
|
|
var Stream = require('stream');
|
|
var shims = require('_shims');
|
|
var Buffer = require('buffer').Buffer;
|
|
var timers = require('timers');
|
|
var util = require('util');
|
|
var StringDecoder;
|
|
|
|
util.inherits(Readable, Stream);
|
|
|
|
function ReadableState(options, stream) {
|
|
options = options || {};
|
|
|
|
// the point at which it stops calling _read() to fill the buffer
|
|
// Note: 0 is a valid value, means "don't call _read preemptively ever"
|
|
var hwm = options.highWaterMark;
|
|
this.highWaterMark = (hwm || hwm === 0) ? hwm : 16 * 1024;
|
|
|
|
// cast to ints.
|
|
this.highWaterMark = ~~this.highWaterMark;
|
|
|
|
this.buffer = [];
|
|
this.length = 0;
|
|
this.pipes = null;
|
|
this.pipesCount = 0;
|
|
this.flowing = false;
|
|
this.ended = false;
|
|
this.endEmitted = false;
|
|
this.reading = false;
|
|
|
|
// In streams that never have any data, and do push(null) right away,
|
|
// the consumer can miss the 'end' event if they do some I/O before
|
|
// consuming the stream. So, we don't emit('end') until some reading
|
|
// happens.
|
|
this.calledRead = false;
|
|
|
|
// a flag to be able to tell if the onwrite cb is called immediately,
|
|
// or on a later tick. We set this to true at first, becuase any
|
|
// actions that shouldn't happen until "later" should generally also
|
|
// not happen before the first write call.
|
|
this.sync = true;
|
|
|
|
// whenever we return null, then we set a flag to say
|
|
// that we're awaiting a 'readable' event emission.
|
|
this.needReadable = false;
|
|
this.emittedReadable = false;
|
|
this.readableListening = false;
|
|
|
|
|
|
// object stream flag. Used to make read(n) ignore n and to
|
|
// make all the buffer merging and length checks go away
|
|
this.objectMode = !!options.objectMode;
|
|
|
|
// Crypto is kind of old and crusty. Historically, its default string
|
|
// encoding is 'binary' so we have to make this configurable.
|
|
// Everything else in the universe uses 'utf8', though.
|
|
this.defaultEncoding = options.defaultEncoding || 'utf8';
|
|
|
|
// when piping, we only care about 'readable' events that happen
|
|
// after read()ing all the bytes and not getting any pushback.
|
|
this.ranOut = false;
|
|
|
|
// the number of writers that are awaiting a drain event in .pipe()s
|
|
this.awaitDrain = 0;
|
|
|
|
// if true, a maybeReadMore has been scheduled
|
|
this.readingMore = false;
|
|
|
|
this.decoder = null;
|
|
this.encoding = null;
|
|
if (options.encoding) {
|
|
if (!StringDecoder)
|
|
StringDecoder = require('string_decoder').StringDecoder;
|
|
this.decoder = new StringDecoder(options.encoding);
|
|
this.encoding = options.encoding;
|
|
}
|
|
}
|
|
|
|
function Readable(options) {
|
|
if (!(this instanceof Readable))
|
|
return new Readable(options);
|
|
|
|
this._readableState = new ReadableState(options, this);
|
|
|
|
// legacy
|
|
this.readable = true;
|
|
|
|
Stream.call(this);
|
|
}
|
|
|
|
// Manually shove something into the read() buffer.
|
|
// This returns true if the highWaterMark has not been hit yet,
|
|
// similar to how Writable.write() returns true if you should
|
|
// write() some more.
|
|
Readable.prototype.push = function(chunk, encoding) {
|
|
var state = this._readableState;
|
|
|
|
if (typeof chunk === 'string' && !state.objectMode) {
|
|
encoding = encoding || state.defaultEncoding;
|
|
if (encoding !== state.encoding) {
|
|
chunk = new Buffer(chunk, encoding);
|
|
encoding = '';
|
|
}
|
|
}
|
|
|
|
return readableAddChunk(this, state, chunk, encoding, false);
|
|
};
|
|
|
|
// Unshift should *always* be something directly out of read()
|
|
Readable.prototype.unshift = function(chunk) {
|
|
var state = this._readableState;
|
|
return readableAddChunk(this, state, chunk, '', true);
|
|
};
|
|
|
|
function readableAddChunk(stream, state, chunk, encoding, addToFront) {
|
|
var er = chunkInvalid(state, chunk);
|
|
if (er) {
|
|
stream.emit('error', er);
|
|
} else if (chunk === null || chunk === undefined) {
|
|
state.reading = false;
|
|
if (!state.ended)
|
|
onEofChunk(stream, state);
|
|
} else if (state.objectMode || chunk && chunk.length > 0) {
|
|
if (state.ended && !addToFront) {
|
|
var e = new Error('stream.push() after EOF');
|
|
stream.emit('error', e);
|
|
} else if (state.endEmitted && addToFront) {
|
|
var e = new Error('stream.unshift() after end event');
|
|
stream.emit('error', e);
|
|
} else {
|
|
if (state.decoder && !addToFront && !encoding)
|
|
chunk = state.decoder.write(chunk);
|
|
|
|
// update the buffer info.
|
|
state.length += state.objectMode ? 1 : chunk.length;
|
|
if (addToFront) {
|
|
state.buffer.unshift(chunk);
|
|
} else {
|
|
state.reading = false;
|
|
state.buffer.push(chunk);
|
|
}
|
|
|
|
if (state.needReadable)
|
|
emitReadable(stream);
|
|
|
|
maybeReadMore(stream, state);
|
|
}
|
|
} else if (!addToFront) {
|
|
state.reading = false;
|
|
}
|
|
|
|
return needMoreData(state);
|
|
}
|
|
|
|
|
|
|
|
// if it's past the high water mark, we can push in some more.
|
|
// Also, if we have no data yet, we can stand some
|
|
// more bytes. This is to work around cases where hwm=0,
|
|
// such as the repl. Also, if the push() triggered a
|
|
// readable event, and the user called read(largeNumber) such that
|
|
// needReadable was set, then we ought to push more, so that another
|
|
// 'readable' event will be triggered.
|
|
function needMoreData(state) {
|
|
return !state.ended &&
|
|
(state.needReadable ||
|
|
state.length < state.highWaterMark ||
|
|
state.length === 0);
|
|
}
|
|
|
|
// backwards compatibility.
|
|
Readable.prototype.setEncoding = function(enc) {
|
|
if (!StringDecoder)
|
|
StringDecoder = require('string_decoder').StringDecoder;
|
|
this._readableState.decoder = new StringDecoder(enc);
|
|
this._readableState.encoding = enc;
|
|
};
|
|
|
|
// Don't raise the hwm > 128MB
|
|
var MAX_HWM = 0x800000;
|
|
function roundUpToNextPowerOf2(n) {
|
|
if (n >= MAX_HWM) {
|
|
n = MAX_HWM;
|
|
} else {
|
|
// Get the next highest power of 2
|
|
n--;
|
|
for (var p = 1; p < 32; p <<= 1) n |= n >> p;
|
|
n++;
|
|
}
|
|
return n;
|
|
}
|
|
|
|
function howMuchToRead(n, state) {
|
|
if (state.length === 0 && state.ended)
|
|
return 0;
|
|
|
|
if (state.objectMode)
|
|
return n === 0 ? 0 : 1;
|
|
|
|
if (isNaN(n) || n === null) {
|
|
// only flow one buffer at a time
|
|
if (state.flowing && state.buffer.length)
|
|
return state.buffer[0].length;
|
|
else
|
|
return state.length;
|
|
}
|
|
|
|
if (n <= 0)
|
|
return 0;
|
|
|
|
// If we're asking for more than the target buffer level,
|
|
// then raise the water mark. Bump up to the next highest
|
|
// power of 2, to prevent increasing it excessively in tiny
|
|
// amounts.
|
|
if (n > state.highWaterMark)
|
|
state.highWaterMark = roundUpToNextPowerOf2(n);
|
|
|
|
// don't have that much. return null, unless we've ended.
|
|
if (n > state.length) {
|
|
if (!state.ended) {
|
|
state.needReadable = true;
|
|
return 0;
|
|
} else
|
|
return state.length;
|
|
}
|
|
|
|
return n;
|
|
}
|
|
|
|
// you can override either this method, or the async _read(n) below.
|
|
Readable.prototype.read = function(n) {
|
|
var state = this._readableState;
|
|
state.calledRead = true;
|
|
var nOrig = n;
|
|
|
|
if (typeof n !== 'number' || n > 0)
|
|
state.emittedReadable = false;
|
|
|
|
// if we're doing read(0) to trigger a readable event, but we
|
|
// already have a bunch of data in the buffer, then just trigger
|
|
// the 'readable' event and move on.
|
|
if (n === 0 &&
|
|
state.needReadable &&
|
|
(state.length >= state.highWaterMark || state.ended)) {
|
|
emitReadable(this);
|
|
return null;
|
|
}
|
|
|
|
n = howMuchToRead(n, state);
|
|
|
|
// if we've ended, and we're now clear, then finish it up.
|
|
if (n === 0 && state.ended) {
|
|
if (state.length === 0)
|
|
endReadable(this);
|
|
return null;
|
|
}
|
|
|
|
// All the actual chunk generation logic needs to be
|
|
// *below* the call to _read. The reason is that in certain
|
|
// synthetic stream cases, such as passthrough streams, _read
|
|
// may be a completely synchronous operation which may change
|
|
// the state of the read buffer, providing enough data when
|
|
// before there was *not* enough.
|
|
//
|
|
// So, the steps are:
|
|
// 1. Figure out what the state of things will be after we do
|
|
// a read from the buffer.
|
|
//
|
|
// 2. If that resulting state will trigger a _read, then call _read.
|
|
// Note that this may be asynchronous, or synchronous. Yes, it is
|
|
// deeply ugly to write APIs this way, but that still doesn't mean
|
|
// that the Readable class should behave improperly, as streams are
|
|
// designed to be sync/async agnostic.
|
|
// Take note if the _read call is sync or async (ie, if the read call
|
|
// has returned yet), so that we know whether or not it's safe to emit
|
|
// 'readable' etc.
|
|
//
|
|
// 3. Actually pull the requested chunks out of the buffer and return.
|
|
|
|
// if we need a readable event, then we need to do some reading.
|
|
var doRead = state.needReadable;
|
|
|
|
// if we currently have less than the highWaterMark, then also read some
|
|
if (state.length - n <= state.highWaterMark)
|
|
doRead = true;
|
|
|
|
// however, if we've ended, then there's no point, and if we're already
|
|
// reading, then it's unnecessary.
|
|
if (state.ended || state.reading)
|
|
doRead = false;
|
|
|
|
if (doRead) {
|
|
state.reading = true;
|
|
state.sync = true;
|
|
// if the length is currently zero, then we *need* a readable event.
|
|
if (state.length === 0)
|
|
state.needReadable = true;
|
|
// call internal read method
|
|
this._read(state.highWaterMark);
|
|
state.sync = false;
|
|
}
|
|
|
|
// If _read called its callback synchronously, then `reading`
|
|
// will be false, and we need to re-evaluate how much data we
|
|
// can return to the user.
|
|
if (doRead && !state.reading)
|
|
n = howMuchToRead(nOrig, state);
|
|
|
|
var ret;
|
|
if (n > 0)
|
|
ret = fromList(n, state);
|
|
else
|
|
ret = null;
|
|
|
|
if (ret === null) {
|
|
state.needReadable = true;
|
|
n = 0;
|
|
}
|
|
|
|
state.length -= n;
|
|
|
|
// If we have nothing in the buffer, then we want to know
|
|
// as soon as we *do* get something into the buffer.
|
|
if (state.length === 0 && !state.ended)
|
|
state.needReadable = true;
|
|
|
|
// If we happened to read() exactly the remaining amount in the
|
|
// buffer, and the EOF has been seen at this point, then make sure
|
|
// that we emit 'end' on the very next tick.
|
|
if (state.ended && !state.endEmitted && state.length === 0)
|
|
endReadable(this);
|
|
|
|
return ret;
|
|
};
|
|
|
|
function chunkInvalid(state, chunk) {
|
|
var er = null;
|
|
if (!Buffer.isBuffer(chunk) &&
|
|
'string' !== typeof chunk &&
|
|
chunk !== null &&
|
|
chunk !== undefined &&
|
|
!state.objectMode &&
|
|
!er) {
|
|
er = new TypeError('Invalid non-string/buffer chunk');
|
|
}
|
|
return er;
|
|
}
|
|
|
|
|
|
function onEofChunk(stream, state) {
|
|
if (state.decoder && !state.ended) {
|
|
var chunk = state.decoder.end();
|
|
if (chunk && chunk.length) {
|
|
state.buffer.push(chunk);
|
|
state.length += state.objectMode ? 1 : chunk.length;
|
|
}
|
|
}
|
|
state.ended = true;
|
|
|
|
// if we've ended and we have some data left, then emit
|
|
// 'readable' now to make sure it gets picked up.
|
|
if (state.length > 0)
|
|
emitReadable(stream);
|
|
else
|
|
endReadable(stream);
|
|
}
|
|
|
|
// Don't emit readable right away in sync mode, because this can trigger
|
|
// another read() call => stack overflow. This way, it might trigger
|
|
// a nextTick recursion warning, but that's not so bad.
|
|
function emitReadable(stream) {
|
|
var state = stream._readableState;
|
|
state.needReadable = false;
|
|
if (state.emittedReadable)
|
|
return;
|
|
|
|
state.emittedReadable = true;
|
|
if (state.sync)
|
|
timers.setImmediate(function() {
|
|
emitReadable_(stream);
|
|
});
|
|
else
|
|
emitReadable_(stream);
|
|
}
|
|
|
|
function emitReadable_(stream) {
|
|
stream.emit('readable');
|
|
}
|
|
|
|
|
|
// at this point, the user has presumably seen the 'readable' event,
|
|
// and called read() to consume some data. that may have triggered
|
|
// in turn another _read(n) call, in which case reading = true if
|
|
// it's in progress.
|
|
// However, if we're not ended, or reading, and the length < hwm,
|
|
// then go ahead and try to read some more preemptively.
|
|
function maybeReadMore(stream, state) {
|
|
if (!state.readingMore) {
|
|
state.readingMore = true;
|
|
timers.setImmediate(function() {
|
|
maybeReadMore_(stream, state);
|
|
});
|
|
}
|
|
}
|
|
|
|
function maybeReadMore_(stream, state) {
|
|
var len = state.length;
|
|
while (!state.reading && !state.flowing && !state.ended &&
|
|
state.length < state.highWaterMark) {
|
|
stream.read(0);
|
|
if (len === state.length)
|
|
// didn't get any data, stop spinning.
|
|
break;
|
|
else
|
|
len = state.length;
|
|
}
|
|
state.readingMore = false;
|
|
}
|
|
|
|
// abstract method. to be overridden in specific implementation classes.
|
|
// call cb(er, data) where data is <= n in length.
|
|
// for virtual (non-string, non-buffer) streams, "length" is somewhat
|
|
// arbitrary, and perhaps not very meaningful.
|
|
Readable.prototype._read = function(n) {
|
|
this.emit('error', new Error('not implemented'));
|
|
};
|
|
|
|
Readable.prototype.pipe = function(dest, pipeOpts) {
|
|
var src = this;
|
|
var state = this._readableState;
|
|
|
|
switch (state.pipesCount) {
|
|
case 0:
|
|
state.pipes = dest;
|
|
break;
|
|
case 1:
|
|
state.pipes = [state.pipes, dest];
|
|
break;
|
|
default:
|
|
state.pipes.push(dest);
|
|
break;
|
|
}
|
|
state.pipesCount += 1;
|
|
|
|
var doEnd = (!pipeOpts || pipeOpts.end !== false) &&
|
|
dest !== process.stdout &&
|
|
dest !== process.stderr;
|
|
|
|
var endFn = doEnd ? onend : cleanup;
|
|
if (state.endEmitted)
|
|
timers.setImmediate(endFn);
|
|
else
|
|
src.once('end', endFn);
|
|
|
|
dest.on('unpipe', onunpipe);
|
|
function onunpipe(readable) {
|
|
if (readable !== src) return;
|
|
cleanup();
|
|
}
|
|
|
|
function onend() {
|
|
dest.end();
|
|
}
|
|
|
|
// when the dest drains, it reduces the awaitDrain counter
|
|
// on the source. This would be more elegant with a .once()
|
|
// handler in flow(), but adding and removing repeatedly is
|
|
// too slow.
|
|
var ondrain = pipeOnDrain(src);
|
|
dest.on('drain', ondrain);
|
|
|
|
function cleanup() {
|
|
// cleanup event handlers once the pipe is broken
|
|
dest.removeListener('close', onclose);
|
|
dest.removeListener('finish', onfinish);
|
|
dest.removeListener('drain', ondrain);
|
|
dest.removeListener('error', onerror);
|
|
dest.removeListener('unpipe', onunpipe);
|
|
src.removeListener('end', onend);
|
|
src.removeListener('end', cleanup);
|
|
|
|
// if the reader is waiting for a drain event from this
|
|
// specific writer, then it would cause it to never start
|
|
// flowing again.
|
|
// So, if this is awaiting a drain, then we just call it now.
|
|
// If we don't know, then assume that we are waiting for one.
|
|
if (!dest._writableState || dest._writableState.needDrain)
|
|
ondrain();
|
|
}
|
|
|
|
// if the dest has an error, then stop piping into it.
|
|
// however, don't suppress the throwing behavior for this.
|
|
// check for listeners before emit removes one-time listeners.
|
|
var errListeners = EE.listenerCount(dest, 'error');
|
|
function onerror(er) {
|
|
unpipe();
|
|
if (errListeners === 0 && EE.listenerCount(dest, 'error') === 0)
|
|
dest.emit('error', er);
|
|
}
|
|
dest.once('error', onerror);
|
|
|
|
// Both close and finish should trigger unpipe, but only once.
|
|
function onclose() {
|
|
dest.removeListener('finish', onfinish);
|
|
unpipe();
|
|
}
|
|
dest.once('close', onclose);
|
|
function onfinish() {
|
|
dest.removeListener('close', onclose);
|
|
unpipe();
|
|
}
|
|
dest.once('finish', onfinish);
|
|
|
|
function unpipe() {
|
|
src.unpipe(dest);
|
|
}
|
|
|
|
// tell the dest that it's being piped to
|
|
dest.emit('pipe', src);
|
|
|
|
// start the flow if it hasn't been started already.
|
|
if (!state.flowing) {
|
|
// the handler that waits for readable events after all
|
|
// the data gets sucked out in flow.
|
|
// This would be easier to follow with a .once() handler
|
|
// in flow(), but that is too slow.
|
|
this.on('readable', pipeOnReadable);
|
|
|
|
state.flowing = true;
|
|
timers.setImmediate(function() {
|
|
flow(src);
|
|
});
|
|
}
|
|
|
|
return dest;
|
|
};
|
|
|
|
function pipeOnDrain(src) {
|
|
return function() {
|
|
var dest = this;
|
|
var state = src._readableState;
|
|
state.awaitDrain--;
|
|
if (state.awaitDrain === 0)
|
|
flow(src);
|
|
};
|
|
}
|
|
|
|
function flow(src) {
|
|
var state = src._readableState;
|
|
var chunk;
|
|
state.awaitDrain = 0;
|
|
|
|
function write(dest, i, list) {
|
|
var written = dest.write(chunk);
|
|
if (false === written) {
|
|
state.awaitDrain++;
|
|
}
|
|
}
|
|
|
|
while (state.pipesCount && null !== (chunk = src.read())) {
|
|
|
|
if (state.pipesCount === 1)
|
|
write(state.pipes, 0, null);
|
|
else
|
|
shims.forEach(state.pipes, write);
|
|
|
|
src.emit('data', chunk);
|
|
|
|
// if anyone needs a drain, then we have to wait for that.
|
|
if (state.awaitDrain > 0)
|
|
return;
|
|
}
|
|
|
|
// if every destination was unpiped, either before entering this
|
|
// function, or in the while loop, then stop flowing.
|
|
//
|
|
// NB: This is a pretty rare edge case.
|
|
if (state.pipesCount === 0) {
|
|
state.flowing = false;
|
|
|
|
// if there were data event listeners added, then switch to old mode.
|
|
if (EE.listenerCount(src, 'data') > 0)
|
|
emitDataEvents(src);
|
|
return;
|
|
}
|
|
|
|
// at this point, no one needed a drain, so we just ran out of data
|
|
// on the next readable event, start it over again.
|
|
state.ranOut = true;
|
|
}
|
|
|
|
function pipeOnReadable() {
|
|
if (this._readableState.ranOut) {
|
|
this._readableState.ranOut = false;
|
|
flow(this);
|
|
}
|
|
}
|
|
|
|
|
|
Readable.prototype.unpipe = function(dest) {
|
|
var state = this._readableState;
|
|
|
|
// if we're not piping anywhere, then do nothing.
|
|
if (state.pipesCount === 0)
|
|
return this;
|
|
|
|
// just one destination. most common case.
|
|
if (state.pipesCount === 1) {
|
|
// passed in one, but it's not the right one.
|
|
if (dest && dest !== state.pipes)
|
|
return this;
|
|
|
|
if (!dest)
|
|
dest = state.pipes;
|
|
|
|
// got a match.
|
|
state.pipes = null;
|
|
state.pipesCount = 0;
|
|
this.removeListener('readable', pipeOnReadable);
|
|
state.flowing = false;
|
|
if (dest)
|
|
dest.emit('unpipe', this);
|
|
return this;
|
|
}
|
|
|
|
// slow case. multiple pipe destinations.
|
|
|
|
if (!dest) {
|
|
// remove all.
|
|
var dests = state.pipes;
|
|
var len = state.pipesCount;
|
|
state.pipes = null;
|
|
state.pipesCount = 0;
|
|
this.removeListener('readable', pipeOnReadable);
|
|
state.flowing = false;
|
|
|
|
for (var i = 0; i < len; i++)
|
|
dests[i].emit('unpipe', this);
|
|
return this;
|
|
}
|
|
|
|
// try to find the right one.
|
|
var i = shims.indexOf(state.pipes, dest);
|
|
if (i === -1)
|
|
return this;
|
|
|
|
state.pipes.splice(i, 1);
|
|
state.pipesCount -= 1;
|
|
if (state.pipesCount === 1)
|
|
state.pipes = state.pipes[0];
|
|
|
|
dest.emit('unpipe', this);
|
|
|
|
return this;
|
|
};
|
|
|
|
// set up data events if they are asked for
|
|
// Ensure readable listeners eventually get something
|
|
Readable.prototype.on = function(ev, fn) {
|
|
var res = Stream.prototype.on.call(this, ev, fn);
|
|
|
|
if (ev === 'data' && !this._readableState.flowing)
|
|
emitDataEvents(this);
|
|
|
|
if (ev === 'readable' && this.readable) {
|
|
var state = this._readableState;
|
|
if (!state.readableListening) {
|
|
state.readableListening = true;
|
|
state.emittedReadable = false;
|
|
state.needReadable = true;
|
|
if (!state.reading) {
|
|
this.read(0);
|
|
} else if (state.length) {
|
|
emitReadable(this, state);
|
|
}
|
|
}
|
|
}
|
|
|
|
return res;
|
|
};
|
|
Readable.prototype.addListener = Readable.prototype.on;
|
|
|
|
// pause() and resume() are remnants of the legacy readable stream API
|
|
// If the user uses them, then switch into old mode.
|
|
Readable.prototype.resume = function() {
|
|
emitDataEvents(this);
|
|
this.read(0);
|
|
this.emit('resume');
|
|
};
|
|
|
|
Readable.prototype.pause = function() {
|
|
emitDataEvents(this, true);
|
|
this.emit('pause');
|
|
};
|
|
|
|
function emitDataEvents(stream, startPaused) {
|
|
var state = stream._readableState;
|
|
|
|
if (state.flowing) {
|
|
// https://github.com/isaacs/readable-stream/issues/16
|
|
throw new Error('Cannot switch to old mode now.');
|
|
}
|
|
|
|
var paused = startPaused || false;
|
|
var readable = false;
|
|
|
|
// convert to an old-style stream.
|
|
stream.readable = true;
|
|
stream.pipe = Stream.prototype.pipe;
|
|
stream.on = stream.addListener = Stream.prototype.on;
|
|
|
|
stream.on('readable', function() {
|
|
readable = true;
|
|
|
|
var c;
|
|
while (!paused && (null !== (c = stream.read())))
|
|
stream.emit('data', c);
|
|
|
|
if (c === null) {
|
|
readable = false;
|
|
stream._readableState.needReadable = true;
|
|
}
|
|
});
|
|
|
|
stream.pause = function() {
|
|
paused = true;
|
|
this.emit('pause');
|
|
};
|
|
|
|
stream.resume = function() {
|
|
paused = false;
|
|
if (readable)
|
|
timers.setImmediate(function() {
|
|
stream.emit('readable');
|
|
});
|
|
else
|
|
this.read(0);
|
|
this.emit('resume');
|
|
};
|
|
|
|
// now make it start, just in case it hadn't already.
|
|
stream.emit('readable');
|
|
}
|
|
|
|
// wrap an old-style stream as the async data source.
|
|
// This is *not* part of the readable stream interface.
|
|
// It is an ugly unfortunate mess of history.
|
|
Readable.prototype.wrap = function(stream) {
|
|
var state = this._readableState;
|
|
var paused = false;
|
|
|
|
var self = this;
|
|
stream.on('end', function() {
|
|
if (state.decoder && !state.ended) {
|
|
var chunk = state.decoder.end();
|
|
if (chunk && chunk.length)
|
|
self.push(chunk);
|
|
}
|
|
|
|
self.push(null);
|
|
});
|
|
|
|
stream.on('data', function(chunk) {
|
|
if (state.decoder)
|
|
chunk = state.decoder.write(chunk);
|
|
if (!chunk || !state.objectMode && !chunk.length)
|
|
return;
|
|
|
|
var ret = self.push(chunk);
|
|
if (!ret) {
|
|
paused = true;
|
|
stream.pause();
|
|
}
|
|
});
|
|
|
|
// proxy all the other methods.
|
|
// important when wrapping filters and duplexes.
|
|
for (var i in stream) {
|
|
if (typeof stream[i] === 'function' &&
|
|
typeof this[i] === 'undefined') {
|
|
this[i] = function(method) { return function() {
|
|
return stream[method].apply(stream, arguments);
|
|
}}(i);
|
|
}
|
|
}
|
|
|
|
// proxy certain important events.
|
|
var events = ['error', 'close', 'destroy', 'pause', 'resume'];
|
|
shims.forEach(events, function(ev) {
|
|
stream.on(ev, shims.bind(self.emit, self, ev));
|
|
});
|
|
|
|
// when we try to consume some more bytes, simply unpause the
|
|
// underlying stream.
|
|
self._read = function(n) {
|
|
if (paused) {
|
|
paused = false;
|
|
stream.resume();
|
|
}
|
|
};
|
|
|
|
return self;
|
|
};
|
|
|
|
|
|
|
|
// exposed for testing purposes only.
|
|
Readable._fromList = fromList;
|
|
|
|
// Pluck off n bytes from an array of buffers.
|
|
// Length is the combined lengths of all the buffers in the list.
|
|
function fromList(n, state) {
|
|
var list = state.buffer;
|
|
var length = state.length;
|
|
var stringMode = !!state.decoder;
|
|
var objectMode = !!state.objectMode;
|
|
var ret;
|
|
|
|
// nothing in the list, definitely empty.
|
|
if (list.length === 0)
|
|
return null;
|
|
|
|
if (length === 0)
|
|
ret = null;
|
|
else if (objectMode)
|
|
ret = list.shift();
|
|
else if (!n || n >= length) {
|
|
// read it all, truncate the array.
|
|
if (stringMode)
|
|
ret = list.join('');
|
|
else
|
|
ret = Buffer.concat(list, length);
|
|
list.length = 0;
|
|
} else {
|
|
// read just some of it.
|
|
if (n < list[0].length) {
|
|
// just take a part of the first list item.
|
|
// slice is the same for buffers and strings.
|
|
var buf = list[0];
|
|
ret = buf.slice(0, n);
|
|
list[0] = buf.slice(n);
|
|
} else if (n === list[0].length) {
|
|
// first list is a perfect match
|
|
ret = list.shift();
|
|
} else {
|
|
// complex case.
|
|
// we have enough to cover it, but it spans past the first buffer.
|
|
if (stringMode)
|
|
ret = '';
|
|
else
|
|
ret = new Buffer(n);
|
|
|
|
var c = 0;
|
|
for (var i = 0, l = list.length; i < l && c < n; i++) {
|
|
var buf = list[0];
|
|
var cpy = Math.min(n - c, buf.length);
|
|
|
|
if (stringMode)
|
|
ret += buf.slice(0, cpy);
|
|
else
|
|
buf.copy(ret, c, 0, cpy);
|
|
|
|
if (cpy < buf.length)
|
|
list[0] = buf.slice(cpy);
|
|
else
|
|
list.shift();
|
|
|
|
c += cpy;
|
|
}
|
|
}
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
function endReadable(stream) {
|
|
var state = stream._readableState;
|
|
|
|
// If we get here before consuming all the bytes, then that is a
|
|
// bug in node. Should never happen.
|
|
if (state.length > 0)
|
|
throw new Error('endReadable called on non-empty stream');
|
|
|
|
if (!state.endEmitted && state.calledRead) {
|
|
state.ended = true;
|
|
timers.setImmediate(function() {
|
|
// Check that we didn't get one last unshift.
|
|
if (!state.endEmitted && state.length === 0) {
|
|
state.endEmitted = true;
|
|
stream.readable = false;
|
|
stream.emit('end');
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
},{"__browserify_process":91,"_shims":50,"buffer":70,"events":58,"stream":63,"string_decoder":64,"timers":65,"util":68}],54:[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.
|
|
|
|
// a transform stream is a readable/writable stream where you do
|
|
// something with the data. Sometimes it's called a "filter",
|
|
// but that's not a great name for it, since that implies a thing where
|
|
// some bits pass through, and others are simply ignored. (That would
|
|
// be a valid example of a transform, of course.)
|
|
//
|
|
// While the output is causally related to the input, it's not a
|
|
// necessarily symmetric or synchronous transformation. For example,
|
|
// a zlib stream might take multiple plain-text writes(), and then
|
|
// emit a single compressed chunk some time in the future.
|
|
//
|
|
// Here's how this works:
|
|
//
|
|
// The Transform stream has all the aspects of the readable and writable
|
|
// stream classes. When you write(chunk), that calls _write(chunk,cb)
|
|
// internally, and returns false if there's a lot of pending writes
|
|
// buffered up. When you call read(), that calls _read(n) until
|
|
// there's enough pending readable data buffered up.
|
|
//
|
|
// In a transform stream, the written data is placed in a buffer. When
|
|
// _read(n) is called, it transforms the queued up data, calling the
|
|
// buffered _write cb's as it consumes chunks. If consuming a single
|
|
// written chunk would result in multiple output chunks, then the first
|
|
// outputted bit calls the readcb, and subsequent chunks just go into
|
|
// the read buffer, and will cause it to emit 'readable' if necessary.
|
|
//
|
|
// This way, back-pressure is actually determined by the reading side,
|
|
// since _read has to be called to start processing a new chunk. However,
|
|
// a pathological inflate type of transform can cause excessive buffering
|
|
// here. For example, imagine a stream where every byte of input is
|
|
// interpreted as an integer from 0-255, and then results in that many
|
|
// bytes of output. Writing the 4 bytes {ff,ff,ff,ff} would result in
|
|
// 1kb of data being output. In this case, you could write a very small
|
|
// amount of input, and end up with a very large amount of output. In
|
|
// such a pathological inflating mechanism, there'd be no way to tell
|
|
// the system to stop doing the transform. A single 4MB write could
|
|
// cause the system to run out of memory.
|
|
//
|
|
// However, even in such a pathological case, only a single written chunk
|
|
// would be consumed, and then the rest would wait (un-transformed) until
|
|
// the results of the previous transformed chunk were consumed.
|
|
|
|
module.exports = Transform;
|
|
|
|
var Duplex = require('_stream_duplex');
|
|
var util = require('util');
|
|
util.inherits(Transform, Duplex);
|
|
|
|
|
|
function TransformState(options, stream) {
|
|
this.afterTransform = function(er, data) {
|
|
return afterTransform(stream, er, data);
|
|
};
|
|
|
|
this.needTransform = false;
|
|
this.transforming = false;
|
|
this.writecb = null;
|
|
this.writechunk = null;
|
|
}
|
|
|
|
function afterTransform(stream, er, data) {
|
|
var ts = stream._transformState;
|
|
ts.transforming = false;
|
|
|
|
var cb = ts.writecb;
|
|
|
|
if (!cb)
|
|
return stream.emit('error', new Error('no writecb in Transform class'));
|
|
|
|
ts.writechunk = null;
|
|
ts.writecb = null;
|
|
|
|
if (data !== null && data !== undefined)
|
|
stream.push(data);
|
|
|
|
if (cb)
|
|
cb(er);
|
|
|
|
var rs = stream._readableState;
|
|
rs.reading = false;
|
|
if (rs.needReadable || rs.length < rs.highWaterMark) {
|
|
stream._read(rs.highWaterMark);
|
|
}
|
|
}
|
|
|
|
|
|
function Transform(options) {
|
|
if (!(this instanceof Transform))
|
|
return new Transform(options);
|
|
|
|
Duplex.call(this, options);
|
|
|
|
var ts = this._transformState = new TransformState(options, this);
|
|
|
|
// when the writable side finishes, then flush out anything remaining.
|
|
var stream = this;
|
|
|
|
// start out asking for a readable event once data is transformed.
|
|
this._readableState.needReadable = true;
|
|
|
|
// we have implemented the _read method, and done the other things
|
|
// that Readable wants before the first _read call, so unset the
|
|
// sync guard flag.
|
|
this._readableState.sync = false;
|
|
|
|
this.once('finish', function() {
|
|
if ('function' === typeof this._flush)
|
|
this._flush(function(er) {
|
|
done(stream, er);
|
|
});
|
|
else
|
|
done(stream);
|
|
});
|
|
}
|
|
|
|
Transform.prototype.push = function(chunk, encoding) {
|
|
this._transformState.needTransform = false;
|
|
return Duplex.prototype.push.call(this, chunk, encoding);
|
|
};
|
|
|
|
// This is the part where you do stuff!
|
|
// override this function in implementation classes.
|
|
// 'chunk' is an input chunk.
|
|
//
|
|
// Call `push(newChunk)` to pass along transformed output
|
|
// to the readable side. You may call 'push' zero or more times.
|
|
//
|
|
// Call `cb(err)` when you are done with this chunk. If you pass
|
|
// an error, then that'll put the hurt on the whole operation. If you
|
|
// never call cb(), then you'll never get another chunk.
|
|
Transform.prototype._transform = function(chunk, encoding, cb) {
|
|
throw new Error('not implemented');
|
|
};
|
|
|
|
Transform.prototype._write = function(chunk, encoding, cb) {
|
|
var ts = this._transformState;
|
|
ts.writecb = cb;
|
|
ts.writechunk = chunk;
|
|
ts.writeencoding = encoding;
|
|
if (!ts.transforming) {
|
|
var rs = this._readableState;
|
|
if (ts.needTransform ||
|
|
rs.needReadable ||
|
|
rs.length < rs.highWaterMark)
|
|
this._read(rs.highWaterMark);
|
|
}
|
|
};
|
|
|
|
// Doesn't matter what the args are here.
|
|
// _transform does all the work.
|
|
// That we got here means that the readable side wants more data.
|
|
Transform.prototype._read = function(n) {
|
|
var ts = this._transformState;
|
|
|
|
if (ts.writechunk && ts.writecb && !ts.transforming) {
|
|
ts.transforming = true;
|
|
this._transform(ts.writechunk, ts.writeencoding, ts.afterTransform);
|
|
} else {
|
|
// mark that we need a transform, so that any data that comes in
|
|
// will get processed, now that we've asked for it.
|
|
ts.needTransform = true;
|
|
}
|
|
};
|
|
|
|
|
|
function done(stream, er) {
|
|
if (er)
|
|
return stream.emit('error', er);
|
|
|
|
// if there's nothing in the write buffer, then that means
|
|
// that nothing more will ever be provided
|
|
var ws = stream._writableState;
|
|
var rs = stream._readableState;
|
|
var ts = stream._transformState;
|
|
|
|
if (ws.length)
|
|
throw new Error('calling transform done when ws.length != 0');
|
|
|
|
if (ts.transforming)
|
|
throw new Error('calling transform done when still transforming');
|
|
|
|
return stream.push(null);
|
|
}
|
|
|
|
},{"_stream_duplex":51,"util":68}],55:[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.
|
|
|
|
// A bit simpler than readable streams.
|
|
// Implement an async ._write(chunk, cb), and it'll handle all
|
|
// the drain event emission and buffering.
|
|
|
|
module.exports = Writable;
|
|
Writable.WritableState = WritableState;
|
|
|
|
var util = require('util');
|
|
var Stream = require('stream');
|
|
var timers = require('timers');
|
|
var Buffer = require('buffer').Buffer;
|
|
|
|
util.inherits(Writable, Stream);
|
|
|
|
function WriteReq(chunk, encoding, cb) {
|
|
this.chunk = chunk;
|
|
this.encoding = encoding;
|
|
this.callback = cb;
|
|
}
|
|
|
|
function WritableState(options, stream) {
|
|
options = options || {};
|
|
|
|
// the point at which write() starts returning false
|
|
// Note: 0 is a valid value, means that we always return false if
|
|
// the entire buffer is not flushed immediately on write()
|
|
var hwm = options.highWaterMark;
|
|
this.highWaterMark = (hwm || hwm === 0) ? hwm : 16 * 1024;
|
|
|
|
// object stream flag to indicate whether or not this stream
|
|
// contains buffers or objects.
|
|
this.objectMode = !!options.objectMode;
|
|
|
|
// cast to ints.
|
|
this.highWaterMark = ~~this.highWaterMark;
|
|
|
|
this.needDrain = false;
|
|
// at the start of calling end()
|
|
this.ending = false;
|
|
// when end() has been called, and returned
|
|
this.ended = false;
|
|
// when 'finish' is emitted
|
|
this.finished = false;
|
|
|
|
// should we decode strings into buffers before passing to _write?
|
|
// this is here so that some node-core streams can optimize string
|
|
// handling at a lower level.
|
|
var noDecode = options.decodeStrings === false;
|
|
this.decodeStrings = !noDecode;
|
|
|
|
// Crypto is kind of old and crusty. Historically, its default string
|
|
// encoding is 'binary' so we have to make this configurable.
|
|
// Everything else in the universe uses 'utf8', though.
|
|
this.defaultEncoding = options.defaultEncoding || 'utf8';
|
|
|
|
// not an actual buffer we keep track of, but a measurement
|
|
// of how much we're waiting to get pushed to some underlying
|
|
// socket or file.
|
|
this.length = 0;
|
|
|
|
// a flag to see when we're in the middle of a write.
|
|
this.writing = false;
|
|
|
|
// a flag to be able to tell if the onwrite cb is called immediately,
|
|
// or on a later tick. We set this to true at first, becuase any
|
|
// actions that shouldn't happen until "later" should generally also
|
|
// not happen before the first write call.
|
|
this.sync = true;
|
|
|
|
// a flag to know if we're processing previously buffered items, which
|
|
// may call the _write() callback in the same tick, so that we don't
|
|
// end up in an overlapped onwrite situation.
|
|
this.bufferProcessing = false;
|
|
|
|
// the callback that's passed to _write(chunk,cb)
|
|
this.onwrite = function(er) {
|
|
onwrite(stream, er);
|
|
};
|
|
|
|
// the callback that the user supplies to write(chunk,encoding,cb)
|
|
this.writecb = null;
|
|
|
|
// the amount that is being written when _write is called.
|
|
this.writelen = 0;
|
|
|
|
this.buffer = [];
|
|
}
|
|
|
|
function Writable(options) {
|
|
// Writable ctor is applied to Duplexes, though they're not
|
|
// instanceof Writable, they're instanceof Readable.
|
|
if (!(this instanceof Writable) && !(this instanceof Stream.Duplex))
|
|
return new Writable(options);
|
|
|
|
this._writableState = new WritableState(options, this);
|
|
|
|
// legacy.
|
|
this.writable = true;
|
|
|
|
Stream.call(this);
|
|
}
|
|
|
|
// Otherwise people can pipe Writable streams, which is just wrong.
|
|
Writable.prototype.pipe = function() {
|
|
this.emit('error', new Error('Cannot pipe. Not readable.'));
|
|
};
|
|
|
|
|
|
function writeAfterEnd(stream, state, cb) {
|
|
var er = new Error('write after end');
|
|
// TODO: defer error events consistently everywhere, not just the cb
|
|
stream.emit('error', er);
|
|
timers.setImmediate(function() {
|
|
cb(er);
|
|
});
|
|
}
|
|
|
|
// If we get something that is not a buffer, string, null, or undefined,
|
|
// and we're not in objectMode, then that's an error.
|
|
// Otherwise stream chunks are all considered to be of length=1, and the
|
|
// watermarks determine how many objects to keep in the buffer, rather than
|
|
// how many bytes or characters.
|
|
function validChunk(stream, state, chunk, cb) {
|
|
var valid = true;
|
|
if (!Buffer.isBuffer(chunk) &&
|
|
'string' !== typeof chunk &&
|
|
chunk !== null &&
|
|
chunk !== undefined &&
|
|
!state.objectMode) {
|
|
var er = new TypeError('Invalid non-string/buffer chunk');
|
|
stream.emit('error', er);
|
|
timers.setImmediate(function() {
|
|
cb(er);
|
|
});
|
|
valid = false;
|
|
}
|
|
return valid;
|
|
}
|
|
|
|
Writable.prototype.write = function(chunk, encoding, cb) {
|
|
var state = this._writableState;
|
|
var ret = false;
|
|
|
|
if (typeof encoding === 'function') {
|
|
cb = encoding;
|
|
encoding = null;
|
|
}
|
|
|
|
if (Buffer.isBuffer(chunk))
|
|
encoding = 'buffer';
|
|
else if (!encoding)
|
|
encoding = state.defaultEncoding;
|
|
|
|
if (typeof cb !== 'function')
|
|
cb = function() {};
|
|
|
|
if (state.ended)
|
|
writeAfterEnd(this, state, cb);
|
|
else if (validChunk(this, state, chunk, cb))
|
|
ret = writeOrBuffer(this, state, chunk, encoding, cb);
|
|
|
|
return ret;
|
|
};
|
|
|
|
function decodeChunk(state, chunk, encoding) {
|
|
if (!state.objectMode &&
|
|
state.decodeStrings !== false &&
|
|
typeof chunk === 'string') {
|
|
chunk = new Buffer(chunk, encoding);
|
|
}
|
|
return chunk;
|
|
}
|
|
|
|
// if we're already writing something, then just put this
|
|
// in the queue, and wait our turn. Otherwise, call _write
|
|
// If we return false, then we need a drain event, so set that flag.
|
|
function writeOrBuffer(stream, state, chunk, encoding, cb) {
|
|
chunk = decodeChunk(state, chunk, encoding);
|
|
var len = state.objectMode ? 1 : chunk.length;
|
|
|
|
state.length += len;
|
|
|
|
var ret = state.length < state.highWaterMark;
|
|
state.needDrain = !ret;
|
|
|
|
if (state.writing)
|
|
state.buffer.push(new WriteReq(chunk, encoding, cb));
|
|
else
|
|
doWrite(stream, state, len, chunk, encoding, cb);
|
|
|
|
return ret;
|
|
}
|
|
|
|
function doWrite(stream, state, len, chunk, encoding, cb) {
|
|
state.writelen = len;
|
|
state.writecb = cb;
|
|
state.writing = true;
|
|
state.sync = true;
|
|
stream._write(chunk, encoding, state.onwrite);
|
|
state.sync = false;
|
|
}
|
|
|
|
function onwriteError(stream, state, sync, er, cb) {
|
|
if (sync)
|
|
timers.setImmediate(function() {
|
|
cb(er);
|
|
});
|
|
else
|
|
cb(er);
|
|
|
|
stream.emit('error', er);
|
|
}
|
|
|
|
function onwriteStateUpdate(state) {
|
|
state.writing = false;
|
|
state.writecb = null;
|
|
state.length -= state.writelen;
|
|
state.writelen = 0;
|
|
}
|
|
|
|
function onwrite(stream, er) {
|
|
var state = stream._writableState;
|
|
var sync = state.sync;
|
|
var cb = state.writecb;
|
|
|
|
onwriteStateUpdate(state);
|
|
|
|
if (er)
|
|
onwriteError(stream, state, sync, er, cb);
|
|
else {
|
|
// Check if we're actually ready to finish, but don't emit yet
|
|
var finished = needFinish(stream, state);
|
|
|
|
if (!finished && !state.bufferProcessing && state.buffer.length)
|
|
clearBuffer(stream, state);
|
|
|
|
if (sync) {
|
|
timers.setImmediate(function() {
|
|
afterWrite(stream, state, finished, cb);
|
|
});
|
|
} else {
|
|
afterWrite(stream, state, finished, cb);
|
|
}
|
|
}
|
|
}
|
|
|
|
function afterWrite(stream, state, finished, cb) {
|
|
if (!finished)
|
|
onwriteDrain(stream, state);
|
|
cb();
|
|
if (finished)
|
|
finishMaybe(stream, state);
|
|
}
|
|
|
|
// Must force callback to be called on nextTick, so that we don't
|
|
// emit 'drain' before the write() consumer gets the 'false' return
|
|
// value, and has a chance to attach a 'drain' listener.
|
|
function onwriteDrain(stream, state) {
|
|
if (state.length === 0 && state.needDrain) {
|
|
state.needDrain = false;
|
|
stream.emit('drain');
|
|
}
|
|
}
|
|
|
|
|
|
// if there's something in the buffer waiting, then process it
|
|
function clearBuffer(stream, state) {
|
|
state.bufferProcessing = true;
|
|
|
|
for (var c = 0; c < state.buffer.length; c++) {
|
|
var entry = state.buffer[c];
|
|
var chunk = entry.chunk;
|
|
var encoding = entry.encoding;
|
|
var cb = entry.callback;
|
|
var len = state.objectMode ? 1 : chunk.length;
|
|
|
|
doWrite(stream, state, len, chunk, encoding, cb);
|
|
|
|
// if we didn't call the onwrite immediately, then
|
|
// it means that we need to wait until it does.
|
|
// also, that means that the chunk and cb are currently
|
|
// being processed, so move the buffer counter past them.
|
|
if (state.writing) {
|
|
c++;
|
|
break;
|
|
}
|
|
}
|
|
|
|
state.bufferProcessing = false;
|
|
if (c < state.buffer.length)
|
|
state.buffer = state.buffer.slice(c);
|
|
else
|
|
state.buffer.length = 0;
|
|
}
|
|
|
|
Writable.prototype._write = function(chunk, encoding, cb) {
|
|
cb(new Error('not implemented'));
|
|
};
|
|
|
|
Writable.prototype.end = function(chunk, encoding, cb) {
|
|
var state = this._writableState;
|
|
|
|
if (typeof chunk === 'function') {
|
|
cb = chunk;
|
|
chunk = null;
|
|
encoding = null;
|
|
} else if (typeof encoding === 'function') {
|
|
cb = encoding;
|
|
encoding = null;
|
|
}
|
|
|
|
if (typeof chunk !== 'undefined' && chunk !== null)
|
|
this.write(chunk, encoding);
|
|
|
|
// ignore unnecessary end() calls.
|
|
if (!state.ending && !state.finished)
|
|
endWritable(this, state, cb);
|
|
};
|
|
|
|
|
|
function needFinish(stream, state) {
|
|
return (state.ending &&
|
|
state.length === 0 &&
|
|
!state.finished &&
|
|
!state.writing);
|
|
}
|
|
|
|
function finishMaybe(stream, state) {
|
|
var need = needFinish(stream, state);
|
|
if (need) {
|
|
state.finished = true;
|
|
stream.emit('finish');
|
|
}
|
|
return need;
|
|
}
|
|
|
|
function endWritable(stream, state, cb) {
|
|
state.ending = true;
|
|
finishMaybe(stream, state);
|
|
if (cb) {
|
|
if (state.finished)
|
|
timers.setImmediate(cb);
|
|
else
|
|
stream.once('finish', cb);
|
|
}
|
|
state.ended = true;
|
|
}
|
|
|
|
},{"buffer":70,"stream":63,"timers":65,"util":68}],56:[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":50,"util":68}],57:[function(require,module,exports){
|
|
|
|
// not implemented
|
|
// The reason for having an empty file and not throwing is to allow
|
|
// untraditional implementation of this module.
|
|
|
|
},{}],58:[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":68}],59:[function(require,module,exports){
|
|
module.exports=require(57)
|
|
},{}],60:[function(require,module,exports){
|
|
var http = require('http');
|
|
|
|
var https = module.exports;
|
|
|
|
for (var key in http) {
|
|
if (http.hasOwnProperty(key)) https[key] = http[key];
|
|
};
|
|
|
|
https.request = function (params, cb) {
|
|
if (!params) params = {};
|
|
params.scheme = 'https';
|
|
return http.request.call(this, params, cb);
|
|
}
|
|
},{"http":72}],61:[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":91,"_shims":50,"util":68}],62:[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":50,"buffer":70,"util":68}],63:[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.
|
|
|
|
module.exports = Stream;
|
|
|
|
var EE = require('events').EventEmitter;
|
|
var util = require('util');
|
|
|
|
util.inherits(Stream, EE);
|
|
Stream.Readable = require('_stream_readable');
|
|
Stream.Writable = require('_stream_writable');
|
|
Stream.Duplex = require('_stream_duplex');
|
|
Stream.Transform = require('_stream_transform');
|
|
Stream.PassThrough = require('_stream_passthrough');
|
|
|
|
// Backwards-compat with node 0.4.x
|
|
Stream.Stream = Stream;
|
|
|
|
|
|
|
|
// old-style streams. Note that the pipe method (the only relevant
|
|
// part of this class) is overridden in the Readable class.
|
|
|
|
function Stream() {
|
|
EE.call(this);
|
|
}
|
|
|
|
Stream.prototype.pipe = function(dest, options) {
|
|
var source = this;
|
|
|
|
function ondata(chunk) {
|
|
if (dest.writable) {
|
|
if (false === dest.write(chunk) && source.pause) {
|
|
source.pause();
|
|
}
|
|
}
|
|
}
|
|
|
|
source.on('data', ondata);
|
|
|
|
function ondrain() {
|
|
if (source.readable && source.resume) {
|
|
source.resume();
|
|
}
|
|
}
|
|
|
|
dest.on('drain', ondrain);
|
|
|
|
// If the 'end' option is not supplied, dest.end() will be called when
|
|
// source gets the 'end' or 'close' events. Only dest.end() once.
|
|
if (!dest._isStdio && (!options || options.end !== false)) {
|
|
source.on('end', onend);
|
|
source.on('close', onclose);
|
|
}
|
|
|
|
var didOnEnd = false;
|
|
function onend() {
|
|
if (didOnEnd) return;
|
|
didOnEnd = true;
|
|
|
|
dest.end();
|
|
}
|
|
|
|
|
|
function onclose() {
|
|
if (didOnEnd) return;
|
|
didOnEnd = true;
|
|
|
|
if (typeof dest.destroy === 'function') dest.destroy();
|
|
}
|
|
|
|
// don't leave dangling pipes when there are errors.
|
|
function onerror(er) {
|
|
cleanup();
|
|
if (EE.listenerCount(this, 'error') === 0) {
|
|
throw er; // Unhandled stream error in pipe.
|
|
}
|
|
}
|
|
|
|
source.on('error', onerror);
|
|
dest.on('error', onerror);
|
|
|
|
// remove all the event listeners that were added.
|
|
function cleanup() {
|
|
source.removeListener('data', ondata);
|
|
dest.removeListener('drain', ondrain);
|
|
|
|
source.removeListener('end', onend);
|
|
source.removeListener('close', onclose);
|
|
|
|
source.removeListener('error', onerror);
|
|
dest.removeListener('error', onerror);
|
|
|
|
source.removeListener('end', cleanup);
|
|
source.removeListener('close', cleanup);
|
|
|
|
dest.removeListener('close', cleanup);
|
|
}
|
|
|
|
source.on('end', cleanup);
|
|
source.on('close', cleanup);
|
|
|
|
dest.on('close', cleanup);
|
|
|
|
dest.emit('pipe', source);
|
|
|
|
// Allow for unix-like usage: A.pipe(B).pipe(C)
|
|
return dest;
|
|
};
|
|
|
|
},{"_stream_duplex":51,"_stream_passthrough":52,"_stream_readable":53,"_stream_transform":54,"_stream_writable":55,"events":58,"util":68}],64:[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 Buffer = require('buffer').Buffer;
|
|
|
|
function assertEncoding(encoding) {
|
|
if (encoding && !Buffer.isEncoding(encoding)) {
|
|
throw new Error('Unknown encoding: ' + encoding);
|
|
}
|
|
}
|
|
|
|
var StringDecoder = exports.StringDecoder = function(encoding) {
|
|
this.encoding = (encoding || 'utf8').toLowerCase().replace(/[-_]/, '');
|
|
assertEncoding(encoding);
|
|
switch (this.encoding) {
|
|
case 'utf8':
|
|
// CESU-8 represents each of Surrogate Pair by 3-bytes
|
|
this.surrogateSize = 3;
|
|
break;
|
|
case 'ucs2':
|
|
case 'utf16le':
|
|
// UTF-16 represents each of Surrogate Pair by 2-bytes
|
|
this.surrogateSize = 2;
|
|
this.detectIncompleteChar = utf16DetectIncompleteChar;
|
|
break;
|
|
case 'base64':
|
|
// Base-64 stores 3 bytes in 4 chars, and pads the remainder.
|
|
this.surrogateSize = 3;
|
|
this.detectIncompleteChar = base64DetectIncompleteChar;
|
|
break;
|
|
default:
|
|
this.write = passThroughWrite;
|
|
return;
|
|
}
|
|
|
|
this.charBuffer = new Buffer(6);
|
|
this.charReceived = 0;
|
|
this.charLength = 0;
|
|
};
|
|
|
|
|
|
StringDecoder.prototype.write = function(buffer) {
|
|
var charStr = '';
|
|
var offset = 0;
|
|
|
|
// if our last write ended with an incomplete multibyte character
|
|
while (this.charLength) {
|
|
// determine how many remaining bytes this buffer has to offer for this char
|
|
var i = (buffer.length >= this.charLength - this.charReceived) ?
|
|
this.charLength - this.charReceived :
|
|
buffer.length;
|
|
|
|
// add the new bytes to the char buffer
|
|
buffer.copy(this.charBuffer, this.charReceived, offset, i);
|
|
this.charReceived += (i - offset);
|
|
offset = i;
|
|
|
|
if (this.charReceived < this.charLength) {
|
|
// still not enough chars in this buffer? wait for more ...
|
|
return '';
|
|
}
|
|
|
|
// get the character that was split
|
|
charStr = this.charBuffer.slice(0, this.charLength).toString(this.encoding);
|
|
|
|
// lead surrogate (D800-DBFF) is also the incomplete character
|
|
var charCode = charStr.charCodeAt(charStr.length - 1);
|
|
if (charCode >= 0xD800 && charCode <= 0xDBFF) {
|
|
this.charLength += this.surrogateSize;
|
|
charStr = '';
|
|
continue;
|
|
}
|
|
this.charReceived = this.charLength = 0;
|
|
|
|
// if there are no more bytes in this buffer, just emit our char
|
|
if (i == buffer.length) return charStr;
|
|
|
|
// otherwise cut off the characters end from the beginning of this buffer
|
|
buffer = buffer.slice(i, buffer.length);
|
|
break;
|
|
}
|
|
|
|
var lenIncomplete = this.detectIncompleteChar(buffer);
|
|
|
|
var end = buffer.length;
|
|
if (this.charLength) {
|
|
// buffer the incomplete character bytes we got
|
|
buffer.copy(this.charBuffer, 0, buffer.length - lenIncomplete, end);
|
|
this.charReceived = lenIncomplete;
|
|
end -= lenIncomplete;
|
|
}
|
|
|
|
charStr += buffer.toString(this.encoding, 0, end);
|
|
|
|
var end = charStr.length - 1;
|
|
var charCode = charStr.charCodeAt(end);
|
|
// lead surrogate (D800-DBFF) is also the incomplete character
|
|
if (charCode >= 0xD800 && charCode <= 0xDBFF) {
|
|
var size = this.surrogateSize;
|
|
this.charLength += size;
|
|
this.charReceived += size;
|
|
this.charBuffer.copy(this.charBuffer, size, 0, size);
|
|
this.charBuffer.write(charStr.charAt(charStr.length - 1), this.encoding);
|
|
return charStr.substring(0, end);
|
|
}
|
|
|
|
// or just emit the charStr
|
|
return charStr;
|
|
};
|
|
|
|
StringDecoder.prototype.detectIncompleteChar = function(buffer) {
|
|
// determine how many bytes we have to check at the end of this buffer
|
|
var i = (buffer.length >= 3) ? 3 : buffer.length;
|
|
|
|
// Figure out if one of the last i bytes of our buffer announces an
|
|
// incomplete char.
|
|
for (; i > 0; i--) {
|
|
var c = buffer[buffer.length - i];
|
|
|
|
// See http://en.wikipedia.org/wiki/UTF-8#Description
|
|
|
|
// 110XXXXX
|
|
if (i == 1 && c >> 5 == 0x06) {
|
|
this.charLength = 2;
|
|
break;
|
|
}
|
|
|
|
// 1110XXXX
|
|
if (i <= 2 && c >> 4 == 0x0E) {
|
|
this.charLength = 3;
|
|
break;
|
|
}
|
|
|
|
// 11110XXX
|
|
if (i <= 3 && c >> 3 == 0x1E) {
|
|
this.charLength = 4;
|
|
break;
|
|
}
|
|
}
|
|
|
|
return i;
|
|
};
|
|
|
|
StringDecoder.prototype.end = function(buffer) {
|
|
var res = '';
|
|
if (buffer && buffer.length)
|
|
res = this.write(buffer);
|
|
|
|
if (this.charReceived) {
|
|
var cr = this.charReceived;
|
|
var buf = this.charBuffer;
|
|
var enc = this.encoding;
|
|
res += buf.slice(0, cr).toString(enc);
|
|
}
|
|
|
|
return res;
|
|
};
|
|
|
|
function passThroughWrite(buffer) {
|
|
return buffer.toString(this.encoding);
|
|
}
|
|
|
|
function utf16DetectIncompleteChar(buffer) {
|
|
var incomplete = this.charReceived = buffer.length % 2;
|
|
this.charLength = incomplete ? 2 : 0;
|
|
return incomplete;
|
|
}
|
|
|
|
function base64DetectIncompleteChar(buffer) {
|
|
var incomplete = this.charReceived = buffer.length % 3;
|
|
this.charLength = incomplete ? 3 : 0;
|
|
return incomplete;
|
|
}
|
|
|
|
},{"buffer":70}],65:[function(require,module,exports){
|
|
try {
|
|
// Old IE browsers that do not curry arguments
|
|
if (!setTimeout.call) {
|
|
var slicer = Array.prototype.slice;
|
|
exports.setTimeout = function(fn) {
|
|
var args = slicer.call(arguments, 1);
|
|
return setTimeout(function() {
|
|
return fn.apply(this, args);
|
|
})
|
|
};
|
|
|
|
exports.setInterval = function(fn) {
|
|
var args = slicer.call(arguments, 1);
|
|
return setInterval(function() {
|
|
return fn.apply(this, args);
|
|
});
|
|
};
|
|
} else {
|
|
exports.setTimeout = setTimeout;
|
|
exports.setInterval = setInterval;
|
|
}
|
|
exports.clearTimeout = clearTimeout;
|
|
exports.clearInterval = clearInterval;
|
|
|
|
if (window.setImmediate) {
|
|
exports.setImmediate = window.setImmediate;
|
|
exports.clearImmediate = window.clearImmediate;
|
|
}
|
|
|
|
// Chrome and PhantomJS seems to depend on `this` pseudo variable being a
|
|
// `window` and throws invalid invocation exception otherwise. If this code
|
|
// runs in such JS runtime next line will throw and `catch` clause will
|
|
// exported timers functions bound to a window.
|
|
exports.setTimeout(function() {});
|
|
} catch (_) {
|
|
function bind(f, context) {
|
|
return function () { return f.apply(context, arguments) };
|
|
}
|
|
|
|
if (typeof window !== 'undefined') {
|
|
exports.setTimeout = bind(setTimeout, window);
|
|
exports.setInterval = bind(setInterval, window);
|
|
exports.clearTimeout = bind(clearTimeout, window);
|
|
exports.clearInterval = bind(clearInterval, window);
|
|
if (window.setImmediate) {
|
|
exports.setImmediate = bind(window.setImmediate, window);
|
|
exports.clearImmediate = bind(window.clearImmediate, window);
|
|
}
|
|
} else {
|
|
if (typeof setTimeout !== 'undefined') {
|
|
exports.setTimeout = setTimeout;
|
|
}
|
|
if (typeof setInterval !== 'undefined') {
|
|
exports.setInterval = setInterval;
|
|
}
|
|
if (typeof clearTimeout !== 'undefined') {
|
|
exports.clearTimeout = clearTimeout;
|
|
}
|
|
if (typeof clearInterval === 'function') {
|
|
exports.clearInterval = clearInterval;
|
|
}
|
|
}
|
|
}
|
|
|
|
exports.unref = function unref() {};
|
|
exports.ref = function ref() {};
|
|
|
|
if (!exports.setImmediate) {
|
|
var currentKey = 0, queue = {}, active = false;
|
|
|
|
exports.setImmediate = (function () {
|
|
function drain() {
|
|
active = false;
|
|
for (var key in queue) {
|
|
if (queue.hasOwnProperty(currentKey, key)) {
|
|
var fn = queue[key];
|
|
delete queue[key];
|
|
fn();
|
|
}
|
|
}
|
|
}
|
|
|
|
if (typeof window !== 'undefined' &&
|
|
window.postMessage && window.addEventListener) {
|
|
window.addEventListener('message', function (ev) {
|
|
if (ev.source === window && ev.data === 'browserify-tick') {
|
|
ev.stopPropagation();
|
|
drain();
|
|
}
|
|
}, true);
|
|
|
|
return function setImmediate(fn) {
|
|
var id = ++currentKey;
|
|
queue[id] = fn;
|
|
if (!active) {
|
|
active = true;
|
|
window.postMessage('browserify-tick', '*');
|
|
}
|
|
return id;
|
|
};
|
|
} else {
|
|
return function setImmediate(fn) {
|
|
var id = ++currentKey;
|
|
queue[id] = fn;
|
|
if (!active) {
|
|
active = true;
|
|
setTimeout(drain, 0);
|
|
}
|
|
return id;
|
|
};
|
|
}
|
|
})();
|
|
|
|
exports.clearImmediate = function clearImmediate(id) {
|
|
delete queue[id];
|
|
};
|
|
}
|
|
|
|
},{}],66:[function(require,module,exports){
|
|
|
|
exports.isatty = function () { return false; };
|
|
|
|
function ReadStream() {
|
|
throw new Error('tty.ReadStream is not implemented');
|
|
}
|
|
exports.ReadStream = ReadStream;
|
|
|
|
function WriteStream() {
|
|
throw new Error('tty.ReadStream is not implemented');
|
|
}
|
|
exports.WriteStream = WriteStream;
|
|
|
|
},{}],67:[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":50,"querystring":62,"util":68}],68:[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":50}],69:[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;
|
|
};
|
|
|
|
},{}],70:[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":69,"assert":56,"base64-js":71}],71:[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;
|
|
}());
|
|
|
|
},{}],72:[function(require,module,exports){
|
|
var http = module.exports;
|
|
var EventEmitter = require('events').EventEmitter;
|
|
var Request = require('./lib/request');
|
|
|
|
http.request = function (params, cb) {
|
|
if (!params) params = {};
|
|
if (!params.host) params.host = window.location.host.split(':')[0];
|
|
if (!params.port) params.port = window.location.port;
|
|
if (!params.scheme) params.scheme = window.location.protocol.split(':')[0];
|
|
|
|
var req = new Request(new xhrHttp, params);
|
|
if (cb) req.on('response', cb);
|
|
return req;
|
|
};
|
|
|
|
http.get = function (params, cb) {
|
|
params.method = 'GET';
|
|
var req = http.request(params, cb);
|
|
req.end();
|
|
return req;
|
|
};
|
|
|
|
http.Agent = function () {};
|
|
http.Agent.defaultMaxSockets = 4;
|
|
|
|
var xhrHttp = (function () {
|
|
if (typeof window === 'undefined') {
|
|
throw new Error('no window object present');
|
|
}
|
|
else if (window.XMLHttpRequest) {
|
|
return window.XMLHttpRequest;
|
|
}
|
|
else if (window.ActiveXObject) {
|
|
var axs = [
|
|
'Msxml2.XMLHTTP.6.0',
|
|
'Msxml2.XMLHTTP.3.0',
|
|
'Microsoft.XMLHTTP'
|
|
];
|
|
for (var i = 0; i < axs.length; i++) {
|
|
try {
|
|
var ax = new(window.ActiveXObject)(axs[i]);
|
|
return function () {
|
|
if (ax) {
|
|
var ax_ = ax;
|
|
ax = null;
|
|
return ax_;
|
|
}
|
|
else {
|
|
return new(window.ActiveXObject)(axs[i]);
|
|
}
|
|
};
|
|
}
|
|
catch (e) {}
|
|
}
|
|
throw new Error('ajax not supported in this browser')
|
|
}
|
|
else {
|
|
throw new Error('ajax not supported in this browser');
|
|
}
|
|
})();
|
|
|
|
},{"./lib/request":73,"events":58}],73:[function(require,module,exports){
|
|
var Stream = require('stream');
|
|
var Response = require('./response');
|
|
var concatStream = require('concat-stream');
|
|
var Base64 = require('Base64');
|
|
var util = require('util');
|
|
|
|
var Request = module.exports = function (xhr, params) {
|
|
var self = this;
|
|
self.writable = true;
|
|
self.xhr = xhr;
|
|
self.body = concatStream()
|
|
|
|
var uri = params.host
|
|
+ (params.port ? ':' + params.port : '')
|
|
+ (params.path || '/')
|
|
;
|
|
|
|
xhr.open(
|
|
params.method || 'GET',
|
|
(params.scheme || 'http') + '://' + uri,
|
|
true
|
|
);
|
|
|
|
if (params.headers) {
|
|
var keys = objectKeys(params.headers);
|
|
for (var i = 0; i < keys.length; i++) {
|
|
var key = keys[i];
|
|
if (!self.isSafeRequestHeader(key)) continue;
|
|
var value = params.headers[key];
|
|
if (isArray(value)) {
|
|
for (var j = 0; j < value.length; j++) {
|
|
xhr.setRequestHeader(key, value[j]);
|
|
}
|
|
}
|
|
else xhr.setRequestHeader(key, value)
|
|
}
|
|
}
|
|
|
|
if (params.auth) {
|
|
//basic auth
|
|
this.setHeader('Authorization', 'Basic ' + Base64.btoa(params.auth));
|
|
}
|
|
|
|
var res = new Response;
|
|
res.on('close', function () {
|
|
self.emit('close');
|
|
});
|
|
|
|
res.on('ready', function () {
|
|
self.emit('response', res);
|
|
});
|
|
|
|
xhr.onreadystatechange = function () {
|
|
res.handle(xhr);
|
|
};
|
|
};
|
|
|
|
util.inherits(Request, Stream);
|
|
|
|
Request.prototype.setHeader = function (key, value) {
|
|
if (isArray(value)) {
|
|
for (var i = 0; i < value.length; i++) {
|
|
this.xhr.setRequestHeader(key, value[i]);
|
|
}
|
|
}
|
|
else {
|
|
this.xhr.setRequestHeader(key, value);
|
|
}
|
|
};
|
|
|
|
Request.prototype.write = function (s) {
|
|
this.body.write(s);
|
|
};
|
|
|
|
Request.prototype.destroy = function (s) {
|
|
this.xhr.abort();
|
|
this.emit('close');
|
|
};
|
|
|
|
Request.prototype.end = function (s) {
|
|
if (s !== undefined) this.body.write(s);
|
|
this.body.end()
|
|
this.xhr.send(this.body.getBody());
|
|
};
|
|
|
|
// Taken from http://dxr.mozilla.org/mozilla/mozilla-central/content/base/src/nsXMLHttpRequest.cpp.html
|
|
Request.unsafeHeaders = [
|
|
"accept-charset",
|
|
"accept-encoding",
|
|
"access-control-request-headers",
|
|
"access-control-request-method",
|
|
"connection",
|
|
"content-length",
|
|
"cookie",
|
|
"cookie2",
|
|
"content-transfer-encoding",
|
|
"date",
|
|
"expect",
|
|
"host",
|
|
"keep-alive",
|
|
"origin",
|
|
"referer",
|
|
"te",
|
|
"trailer",
|
|
"transfer-encoding",
|
|
"upgrade",
|
|
"user-agent",
|
|
"via"
|
|
];
|
|
|
|
Request.prototype.isSafeRequestHeader = function (headerName) {
|
|
if (!headerName) return false;
|
|
return indexOf(Request.unsafeHeaders, headerName.toLowerCase()) === -1;
|
|
};
|
|
|
|
var objectKeys = Object.keys || function (obj) {
|
|
var keys = [];
|
|
for (var key in obj) keys.push(key);
|
|
return keys;
|
|
};
|
|
|
|
var isArray = Array.isArray || function (xs) {
|
|
return Object.prototype.toString.call(xs) === '[object Array]';
|
|
};
|
|
|
|
var indexOf = function (xs, x) {
|
|
if (xs.indexOf) return xs.indexOf(x);
|
|
for (var i = 0; i < xs.length; i++) {
|
|
if (xs[i] === x) return i;
|
|
}
|
|
return -1;
|
|
};
|
|
|
|
},{"./response":74,"Base64":75,"concat-stream":76,"stream":63,"util":68}],74:[function(require,module,exports){
|
|
var Stream = require('stream');
|
|
var util = require('util');
|
|
|
|
var Response = module.exports = function (res) {
|
|
this.offset = 0;
|
|
this.readable = true;
|
|
};
|
|
|
|
util.inherits(Response, Stream);
|
|
|
|
var capable = {
|
|
streaming : true,
|
|
status2 : true
|
|
};
|
|
|
|
function parseHeaders (res) {
|
|
var lines = res.getAllResponseHeaders().split(/\r?\n/);
|
|
var headers = {};
|
|
for (var i = 0; i < lines.length; i++) {
|
|
var line = lines[i];
|
|
if (line === '') continue;
|
|
|
|
var m = line.match(/^([^:]+):\s*(.*)/);
|
|
if (m) {
|
|
var key = m[1].toLowerCase(), value = m[2];
|
|
|
|
if (headers[key] !== undefined) {
|
|
|
|
if (isArray(headers[key])) {
|
|
headers[key].push(value);
|
|
}
|
|
else {
|
|
headers[key] = [ headers[key], value ];
|
|
}
|
|
}
|
|
else {
|
|
headers[key] = value;
|
|
}
|
|
}
|
|
else {
|
|
headers[line] = true;
|
|
}
|
|
}
|
|
return headers;
|
|
}
|
|
|
|
Response.prototype.getResponse = function (xhr) {
|
|
var respType = String(xhr.responseType).toLowerCase();
|
|
if (respType === 'blob') return xhr.responseBlob || xhr.response;
|
|
if (respType === 'arraybuffer') return xhr.response;
|
|
return xhr.responseText;
|
|
}
|
|
|
|
Response.prototype.getHeader = function (key) {
|
|
return this.headers[key.toLowerCase()];
|
|
};
|
|
|
|
Response.prototype.handle = function (res) {
|
|
if (res.readyState === 2 && capable.status2) {
|
|
try {
|
|
this.statusCode = res.status;
|
|
this.headers = parseHeaders(res);
|
|
}
|
|
catch (err) {
|
|
capable.status2 = false;
|
|
}
|
|
|
|
if (capable.status2) {
|
|
this.emit('ready');
|
|
}
|
|
}
|
|
else if (capable.streaming && res.readyState === 3) {
|
|
try {
|
|
if (!this.statusCode) {
|
|
this.statusCode = res.status;
|
|
this.headers = parseHeaders(res);
|
|
this.emit('ready');
|
|
}
|
|
}
|
|
catch (err) {}
|
|
|
|
try {
|
|
this._emitData(res);
|
|
}
|
|
catch (err) {
|
|
capable.streaming = false;
|
|
}
|
|
}
|
|
else if (res.readyState === 4) {
|
|
if (!this.statusCode) {
|
|
this.statusCode = res.status;
|
|
this.emit('ready');
|
|
}
|
|
this._emitData(res);
|
|
|
|
if (res.error) {
|
|
this.emit('error', this.getResponse(res));
|
|
}
|
|
else this.emit('end');
|
|
|
|
this.emit('close');
|
|
}
|
|
};
|
|
|
|
Response.prototype._emitData = function (res) {
|
|
var respBody = this.getResponse(res);
|
|
if (respBody.toString().match(/ArrayBuffer/)) {
|
|
this.emit('data', new Uint8Array(respBody, this.offset));
|
|
this.offset = respBody.byteLength;
|
|
return;
|
|
}
|
|
if (respBody.length > this.offset) {
|
|
this.emit('data', respBody.slice(this.offset));
|
|
this.offset = respBody.length;
|
|
}
|
|
};
|
|
|
|
var isArray = Array.isArray || function (xs) {
|
|
return Object.prototype.toString.call(xs) === '[object Array]';
|
|
};
|
|
|
|
},{"stream":63,"util":68}],75:[function(require,module,exports){
|
|
;(function () {
|
|
|
|
var
|
|
object = typeof exports != 'undefined' ? exports : this, // #8: web workers
|
|
chars = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=',
|
|
INVALID_CHARACTER_ERR = (function () {
|
|
// fabricate a suitable error object
|
|
try { document.createElement('$'); }
|
|
catch (error) { return error; }}());
|
|
|
|
// encoder
|
|
// [https://gist.github.com/999166] by [https://github.com/nignag]
|
|
object.btoa || (
|
|
object.btoa = function (input) {
|
|
for (
|
|
// initialize result and counter
|
|
var block, charCode, idx = 0, map = chars, output = '';
|
|
// if the next input index does not exist:
|
|
// change the mapping table to "="
|
|
// check if d has no fractional digits
|
|
input.charAt(idx | 0) || (map = '=', idx % 1);
|
|
// "8 - idx % 1 * 8" generates the sequence 2, 4, 6, 8
|
|
output += map.charAt(63 & block >> 8 - idx % 1 * 8)
|
|
) {
|
|
charCode = input.charCodeAt(idx += 3/4);
|
|
if (charCode > 0xFF) throw INVALID_CHARACTER_ERR;
|
|
block = block << 8 | charCode;
|
|
}
|
|
return output;
|
|
});
|
|
|
|
// decoder
|
|
// [https://gist.github.com/1020396] by [https://github.com/atk]
|
|
object.atob || (
|
|
object.atob = function (input) {
|
|
input = input.replace(/=+$/, '')
|
|
if (input.length % 4 == 1) throw INVALID_CHARACTER_ERR;
|
|
for (
|
|
// initialize result and counters
|
|
var bc = 0, bs, buffer, idx = 0, output = '';
|
|
// get next character
|
|
buffer = input.charAt(idx++);
|
|
// character found in table? initialize bit storage and add its ascii value;
|
|
~buffer && (bs = bc % 4 ? bs * 64 + buffer : buffer,
|
|
// and if not first of each 4 characters,
|
|
// convert the first 8 bits to one ascii character
|
|
bc++ % 4) ? output += String.fromCharCode(255 & bs >> (-2 * bc & 6)) : 0
|
|
) {
|
|
// try to find character in table (0-63, not found => -1)
|
|
buffer = chars.indexOf(buffer);
|
|
}
|
|
return output;
|
|
});
|
|
|
|
}());
|
|
|
|
},{}],76:[function(require,module,exports){
|
|
var stream = require('stream')
|
|
var bops = require('bops')
|
|
var util = require('util')
|
|
|
|
function ConcatStream(cb) {
|
|
stream.Stream.call(this)
|
|
this.writable = true
|
|
if (cb) this.cb = cb
|
|
this.body = []
|
|
this.on('error', function(err) {
|
|
// no-op
|
|
})
|
|
}
|
|
|
|
util.inherits(ConcatStream, stream.Stream)
|
|
|
|
ConcatStream.prototype.write = function(chunk) {
|
|
this.body.push(chunk)
|
|
}
|
|
|
|
ConcatStream.prototype.destroy = function() {}
|
|
|
|
ConcatStream.prototype.arrayConcat = function(arrs) {
|
|
if (arrs.length === 0) return []
|
|
if (arrs.length === 1) return arrs[0]
|
|
return arrs.reduce(function (a, b) { return a.concat(b) })
|
|
}
|
|
|
|
ConcatStream.prototype.isArray = function(arr) {
|
|
return Array.isArray(arr)
|
|
}
|
|
|
|
ConcatStream.prototype.getBody = function () {
|
|
if (this.body.length === 0) return
|
|
if (typeof(this.body[0]) === "string") return this.body.join('')
|
|
if (this.isArray(this.body[0])) return this.arrayConcat(this.body)
|
|
if (bops.is(this.body[0])) return bops.join(this.body)
|
|
return this.body
|
|
}
|
|
|
|
ConcatStream.prototype.end = function() {
|
|
if (this.cb) this.cb(this.getBody())
|
|
}
|
|
|
|
module.exports = function(cb) {
|
|
return new ConcatStream(cb)
|
|
}
|
|
|
|
module.exports.ConcatStream = ConcatStream
|
|
|
|
},{"bops":77,"stream":63,"util":68}],77:[function(require,module,exports){
|
|
var proto = {}
|
|
module.exports = proto
|
|
|
|
proto.from = require('./from.js')
|
|
proto.to = require('./to.js')
|
|
proto.is = require('./is.js')
|
|
proto.subarray = require('./subarray.js')
|
|
proto.join = require('./join.js')
|
|
proto.copy = require('./copy.js')
|
|
proto.create = require('./create.js')
|
|
|
|
mix(require('./read.js'), proto)
|
|
mix(require('./write.js'), proto)
|
|
|
|
function mix(from, into) {
|
|
for(var key in from) {
|
|
into[key] = from[key]
|
|
}
|
|
}
|
|
|
|
},{"./copy.js":80,"./create.js":81,"./from.js":82,"./is.js":83,"./join.js":84,"./read.js":86,"./subarray.js":87,"./to.js":88,"./write.js":89}],78:[function(require,module,exports){
|
|
module.exports=require(71)
|
|
},{}],79:[function(require,module,exports){
|
|
module.exports = to_utf8
|
|
|
|
var out = []
|
|
, col = []
|
|
, fcc = String.fromCharCode
|
|
, mask = [0x40, 0x20, 0x10, 0x08, 0x04, 0x02, 0x01]
|
|
, unmask = [
|
|
0x00
|
|
, 0x01
|
|
, 0x02 | 0x01
|
|
, 0x04 | 0x02 | 0x01
|
|
, 0x08 | 0x04 | 0x02 | 0x01
|
|
, 0x10 | 0x08 | 0x04 | 0x02 | 0x01
|
|
, 0x20 | 0x10 | 0x08 | 0x04 | 0x02 | 0x01
|
|
, 0x40 | 0x20 | 0x10 | 0x08 | 0x04 | 0x02 | 0x01
|
|
]
|
|
|
|
function to_utf8(bytes, start, end) {
|
|
start = start === undefined ? 0 : start
|
|
end = end === undefined ? bytes.length : end
|
|
|
|
var idx = 0
|
|
, hi = 0x80
|
|
, collecting = 0
|
|
, pos
|
|
, by
|
|
|
|
col.length =
|
|
out.length = 0
|
|
|
|
while(idx < bytes.length) {
|
|
by = bytes[idx]
|
|
if(!collecting && by & hi) {
|
|
pos = find_pad_position(by)
|
|
collecting += pos
|
|
if(pos < 8) {
|
|
col[col.length] = by & unmask[6 - pos]
|
|
}
|
|
} else if(collecting) {
|
|
col[col.length] = by & unmask[6]
|
|
--collecting
|
|
if(!collecting && col.length) {
|
|
out[out.length] = fcc(reduced(col, pos))
|
|
col.length = 0
|
|
}
|
|
} else {
|
|
out[out.length] = fcc(by)
|
|
}
|
|
++idx
|
|
}
|
|
if(col.length && !collecting) {
|
|
out[out.length] = fcc(reduced(col, pos))
|
|
col.length = 0
|
|
}
|
|
return out.join('')
|
|
}
|
|
|
|
function find_pad_position(byt) {
|
|
for(var i = 0; i < 7; ++i) {
|
|
if(!(byt & mask[i])) {
|
|
break
|
|
}
|
|
}
|
|
return i
|
|
}
|
|
|
|
function reduced(list) {
|
|
var out = 0
|
|
for(var i = 0, len = list.length; i < len; ++i) {
|
|
out |= list[i] << ((len - i - 1) * 6)
|
|
}
|
|
return out
|
|
}
|
|
|
|
},{}],80:[function(require,module,exports){
|
|
module.exports = copy
|
|
|
|
var slice = [].slice
|
|
|
|
function copy(source, target, target_start, source_start, source_end) {
|
|
target_start = arguments.length < 3 ? 0 : target_start
|
|
source_start = arguments.length < 4 ? 0 : source_start
|
|
source_end = arguments.length < 5 ? source.length : source_end
|
|
|
|
if(source_end === source_start) {
|
|
return
|
|
}
|
|
|
|
if(target.length === 0 || source.length === 0) {
|
|
return
|
|
}
|
|
|
|
if(source_end > source.length) {
|
|
source_end = source.length
|
|
}
|
|
|
|
if(target.length - target_start < source_end - source_start) {
|
|
source_end = target.length - target_start + start
|
|
}
|
|
|
|
if(source.buffer !== target.buffer) {
|
|
return fast_copy(source, target, target_start, source_start, source_end)
|
|
}
|
|
return slow_copy(source, target, target_start, source_start, source_end)
|
|
}
|
|
|
|
function fast_copy(source, target, target_start, source_start, source_end) {
|
|
var len = (source_end - source_start) + target_start
|
|
|
|
for(var i = target_start, j = source_start;
|
|
i < len;
|
|
++i,
|
|
++j) {
|
|
target[i] = source[j]
|
|
}
|
|
}
|
|
|
|
function slow_copy(from, to, j, i, jend) {
|
|
// the buffers could overlap.
|
|
var iend = jend + i
|
|
, tmp = new Uint8Array(slice.call(from, i, iend))
|
|
, x = 0
|
|
|
|
for(; i < iend; ++i, ++x) {
|
|
to[j++] = tmp[x]
|
|
}
|
|
}
|
|
|
|
},{}],81:[function(require,module,exports){
|
|
module.exports = function(size) {
|
|
return new Uint8Array(size)
|
|
}
|
|
|
|
},{}],82:[function(require,module,exports){
|
|
module.exports = from
|
|
|
|
var base64 = require('base64-js')
|
|
|
|
var decoders = {
|
|
hex: from_hex
|
|
, utf8: from_utf
|
|
, base64: from_base64
|
|
}
|
|
|
|
function from(source, encoding) {
|
|
if(Array.isArray(source)) {
|
|
return new Uint8Array(source)
|
|
}
|
|
|
|
return decoders[encoding || 'utf8'](source)
|
|
}
|
|
|
|
function from_hex(str) {
|
|
var size = str.length / 2
|
|
, buf = new Uint8Array(size)
|
|
, character = ''
|
|
|
|
for(var i = 0, len = str.length; i < len; ++i) {
|
|
character += str.charAt(i)
|
|
|
|
if(i > 0 && (i % 2) === 1) {
|
|
buf[i>>>1] = parseInt(character, 16)
|
|
character = ''
|
|
}
|
|
}
|
|
|
|
return buf
|
|
}
|
|
|
|
function from_utf(str) {
|
|
var bytes = []
|
|
, tmp
|
|
, ch
|
|
|
|
for(var i = 0, len = str.length; i < len; ++i) {
|
|
ch = str.charCodeAt(i)
|
|
if(ch & 0x80) {
|
|
tmp = encodeURIComponent(str.charAt(i)).substr(1).split('%')
|
|
for(var j = 0, jlen = tmp.length; j < jlen; ++j) {
|
|
bytes[bytes.length] = parseInt(tmp[j], 16)
|
|
}
|
|
} else {
|
|
bytes[bytes.length] = ch
|
|
}
|
|
}
|
|
|
|
return new Uint8Array(bytes)
|
|
}
|
|
|
|
function from_base64(str) {
|
|
return new Uint8Array(base64.toByteArray(str))
|
|
}
|
|
|
|
},{"base64-js":78}],83:[function(require,module,exports){
|
|
|
|
module.exports = function(buffer) {
|
|
return buffer instanceof Uint8Array;
|
|
}
|
|
|
|
},{}],84:[function(require,module,exports){
|
|
module.exports = join
|
|
|
|
function join(targets, hint) {
|
|
if(!targets.length) {
|
|
return new Uint8Array(0)
|
|
}
|
|
|
|
var len = hint !== undefined ? hint : get_length(targets)
|
|
, out = new Uint8Array(len)
|
|
, cur = targets[0]
|
|
, curlen = cur.length
|
|
, curidx = 0
|
|
, curoff = 0
|
|
, i = 0
|
|
|
|
while(i < len) {
|
|
if(curoff === curlen) {
|
|
curoff = 0
|
|
++curidx
|
|
cur = targets[curidx]
|
|
curlen = cur && cur.length
|
|
continue
|
|
}
|
|
out[i++] = cur[curoff++]
|
|
}
|
|
|
|
return out
|
|
}
|
|
|
|
function get_length(targets) {
|
|
var size = 0
|
|
for(var i = 0, len = targets.length; i < len; ++i) {
|
|
size += targets[i].byteLength
|
|
}
|
|
return size
|
|
}
|
|
|
|
},{}],85:[function(require,module,exports){
|
|
var proto
|
|
, map
|
|
|
|
module.exports = proto = {}
|
|
|
|
map = typeof WeakMap === 'undefined' ? null : new WeakMap
|
|
|
|
proto.get = !map ? no_weakmap_get : get
|
|
|
|
function no_weakmap_get(target) {
|
|
return new DataView(target.buffer, 0)
|
|
}
|
|
|
|
function get(target) {
|
|
var out = map.get(target.buffer)
|
|
if(!out) {
|
|
map.set(target.buffer, out = new DataView(target.buffer, 0))
|
|
}
|
|
return out
|
|
}
|
|
|
|
},{}],86:[function(require,module,exports){
|
|
module.exports = {
|
|
readUInt8: read_uint8
|
|
, readInt8: read_int8
|
|
, readUInt16LE: read_uint16_le
|
|
, readUInt32LE: read_uint32_le
|
|
, readInt16LE: read_int16_le
|
|
, readInt32LE: read_int32_le
|
|
, readFloatLE: read_float_le
|
|
, readDoubleLE: read_double_le
|
|
, readUInt16BE: read_uint16_be
|
|
, readUInt32BE: read_uint32_be
|
|
, readInt16BE: read_int16_be
|
|
, readInt32BE: read_int32_be
|
|
, readFloatBE: read_float_be
|
|
, readDoubleBE: read_double_be
|
|
}
|
|
|
|
var map = require('./mapped.js')
|
|
|
|
function read_uint8(target, at) {
|
|
return target[at]
|
|
}
|
|
|
|
function read_int8(target, at) {
|
|
var v = target[at];
|
|
return v < 0x80 ? v : v - 0x100
|
|
}
|
|
|
|
function read_uint16_le(target, at) {
|
|
var dv = map.get(target);
|
|
return dv.getUint16(at + target.byteOffset, true)
|
|
}
|
|
|
|
function read_uint32_le(target, at) {
|
|
var dv = map.get(target);
|
|
return dv.getUint32(at + target.byteOffset, true)
|
|
}
|
|
|
|
function read_int16_le(target, at) {
|
|
var dv = map.get(target);
|
|
return dv.getInt16(at + target.byteOffset, true)
|
|
}
|
|
|
|
function read_int32_le(target, at) {
|
|
var dv = map.get(target);
|
|
return dv.getInt32(at + target.byteOffset, true)
|
|
}
|
|
|
|
function read_float_le(target, at) {
|
|
var dv = map.get(target);
|
|
return dv.getFloat32(at + target.byteOffset, true)
|
|
}
|
|
|
|
function read_double_le(target, at) {
|
|
var dv = map.get(target);
|
|
return dv.getFloat64(at + target.byteOffset, true)
|
|
}
|
|
|
|
function read_uint16_be(target, at) {
|
|
var dv = map.get(target);
|
|
return dv.getUint16(at + target.byteOffset, false)
|
|
}
|
|
|
|
function read_uint32_be(target, at) {
|
|
var dv = map.get(target);
|
|
return dv.getUint32(at + target.byteOffset, false)
|
|
}
|
|
|
|
function read_int16_be(target, at) {
|
|
var dv = map.get(target);
|
|
return dv.getInt16(at + target.byteOffset, false)
|
|
}
|
|
|
|
function read_int32_be(target, at) {
|
|
var dv = map.get(target);
|
|
return dv.getInt32(at + target.byteOffset, false)
|
|
}
|
|
|
|
function read_float_be(target, at) {
|
|
var dv = map.get(target);
|
|
return dv.getFloat32(at + target.byteOffset, false)
|
|
}
|
|
|
|
function read_double_be(target, at) {
|
|
var dv = map.get(target);
|
|
return dv.getFloat64(at + target.byteOffset, false)
|
|
}
|
|
|
|
},{"./mapped.js":85}],87:[function(require,module,exports){
|
|
module.exports = subarray
|
|
|
|
function subarray(buf, from, to) {
|
|
return buf.subarray(from || 0, to || buf.length)
|
|
}
|
|
|
|
},{}],88:[function(require,module,exports){
|
|
module.exports = to
|
|
|
|
var base64 = require('base64-js')
|
|
, toutf8 = require('to-utf8')
|
|
|
|
var encoders = {
|
|
hex: to_hex
|
|
, utf8: to_utf
|
|
, base64: to_base64
|
|
}
|
|
|
|
function to(buf, encoding) {
|
|
return encoders[encoding || 'utf8'](buf)
|
|
}
|
|
|
|
function to_hex(buf) {
|
|
var str = ''
|
|
, byt
|
|
|
|
for(var i = 0, len = buf.length; i < len; ++i) {
|
|
byt = buf[i]
|
|
str += ((byt & 0xF0) >>> 4).toString(16)
|
|
str += (byt & 0x0F).toString(16)
|
|
}
|
|
|
|
return str
|
|
}
|
|
|
|
function to_utf(buf) {
|
|
return toutf8(buf)
|
|
}
|
|
|
|
function to_base64(buf) {
|
|
return base64.fromByteArray(buf)
|
|
}
|
|
|
|
|
|
},{"base64-js":78,"to-utf8":79}],89:[function(require,module,exports){
|
|
module.exports = {
|
|
writeUInt8: write_uint8
|
|
, writeInt8: write_int8
|
|
, writeUInt16LE: write_uint16_le
|
|
, writeUInt32LE: write_uint32_le
|
|
, writeInt16LE: write_int16_le
|
|
, writeInt32LE: write_int32_le
|
|
, writeFloatLE: write_float_le
|
|
, writeDoubleLE: write_double_le
|
|
, writeUInt16BE: write_uint16_be
|
|
, writeUInt32BE: write_uint32_be
|
|
, writeInt16BE: write_int16_be
|
|
, writeInt32BE: write_int32_be
|
|
, writeFloatBE: write_float_be
|
|
, writeDoubleBE: write_double_be
|
|
}
|
|
|
|
var map = require('./mapped.js')
|
|
|
|
function write_uint8(target, value, at) {
|
|
return target[at] = value
|
|
}
|
|
|
|
function write_int8(target, value, at) {
|
|
return target[at] = value < 0 ? value + 0x100 : value
|
|
}
|
|
|
|
function write_uint16_le(target, value, at) {
|
|
var dv = map.get(target);
|
|
return dv.setUint16(at + target.byteOffset, value, true)
|
|
}
|
|
|
|
function write_uint32_le(target, value, at) {
|
|
var dv = map.get(target);
|
|
return dv.setUint32(at + target.byteOffset, value, true)
|
|
}
|
|
|
|
function write_int16_le(target, value, at) {
|
|
var dv = map.get(target);
|
|
return dv.setInt16(at + target.byteOffset, value, true)
|
|
}
|
|
|
|
function write_int32_le(target, value, at) {
|
|
var dv = map.get(target);
|
|
return dv.setInt32(at + target.byteOffset, value, true)
|
|
}
|
|
|
|
function write_float_le(target, value, at) {
|
|
var dv = map.get(target);
|
|
return dv.setFloat32(at + target.byteOffset, value, true)
|
|
}
|
|
|
|
function write_double_le(target, value, at) {
|
|
var dv = map.get(target);
|
|
return dv.setFloat64(at + target.byteOffset, value, true)
|
|
}
|
|
|
|
function write_uint16_be(target, value, at) {
|
|
var dv = map.get(target);
|
|
return dv.setUint16(at + target.byteOffset, value, false)
|
|
}
|
|
|
|
function write_uint32_be(target, value, at) {
|
|
var dv = map.get(target);
|
|
return dv.setUint32(at + target.byteOffset, value, false)
|
|
}
|
|
|
|
function write_int16_be(target, value, at) {
|
|
var dv = map.get(target);
|
|
return dv.setInt16(at + target.byteOffset, value, false)
|
|
}
|
|
|
|
function write_int32_be(target, value, at) {
|
|
var dv = map.get(target);
|
|
return dv.setInt32(at + target.byteOffset, value, false)
|
|
}
|
|
|
|
function write_float_be(target, value, at) {
|
|
var dv = map.get(target);
|
|
return dv.setFloat32(at + target.byteOffset, value, false)
|
|
}
|
|
|
|
function write_double_be(target, value, at) {
|
|
var dv = map.get(target);
|
|
return dv.setFloat64(at + target.byteOffset, value, false)
|
|
}
|
|
|
|
},{"./mapped.js":85}],90:[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")
|
|
|
|
},{}],91:[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');
|
|
};
|
|
|
|
},{}],92:[function(require,module,exports){
|
|
module.exports = require('./lib/js-yaml.js');
|
|
|
|
},{"./lib/js-yaml.js":93}],93:[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":94,"./js-yaml/dumper":95,"./js-yaml/exception":96,"./js-yaml/loader":97,"./js-yaml/require":99,"./js-yaml/schema":100,"./js-yaml/schema/core":101,"./js-yaml/schema/default_full":102,"./js-yaml/schema/default_safe":103,"./js-yaml/schema/failsafe":104,"./js-yaml/schema/json":105,"./js-yaml/type":106}],94:[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;
|
|
|
|
},{}],95:[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":94,"./exception":96,"./schema/default_full":102,"./schema/default_safe":103}],96:[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;
|
|
|
|
},{}],97:[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":94,"./exception":96,"./mark":98,"./schema/default_full":102,"./schema/default_safe":103}],98:[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":94}],99:[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":97,"fs":59}],100:[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":94,"./exception":96,"./type":106}],101:[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":100,"./json":105}],102:[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":100,"../type/js/function":111,"../type/js/regexp":112,"../type/js/undefined":113,"./default_safe":103}],103:[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":100,"../type/binary":107,"../type/merge":115,"../type/omap":117,"../type/pairs":118,"../type/set":120,"../type/timestamp":122,"./core":101}],104:[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":100,"../type/map":114,"../type/seq":119,"../type/str":121}],105:[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":100,"../type/bool":108,"../type/float":109,"../type/int":110,"../type/null":116,"./failsafe":104}],106:[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":96}],107:[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":94,"../type":106,"buffer":70}],108:[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":94,"../type":106}],109:[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":94,"../type":106}],110:[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":94,"../type":106}],111:[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":94,"../../type":106,"esprima":123}],112:[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":94,"../../type":106}],113:[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":106}],114:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
|
|
var Type = require('../type');
|
|
|
|
|
|
module.exports = new Type('tag:yaml.org,2002:map', {
|
|
loader: {
|
|
kind: 'object'
|
|
}
|
|
});
|
|
|
|
},{"../type":106}],115:[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":94,"../type":106}],116:[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":94,"../type":106}],117:[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":94,"../type":106}],118:[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":94,"../type":106}],119:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
|
|
var Type = require('../type');
|
|
|
|
|
|
module.exports = new Type('tag:yaml.org,2002:seq', {
|
|
loader: {
|
|
kind: 'array'
|
|
}
|
|
});
|
|
|
|
},{"../type":106}],120:[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":94,"../type":106}],121:[function(require,module,exports){
|
|
'use strict';
|
|
|
|
|
|
var Type = require('../type');
|
|
|
|
|
|
module.exports = new Type('tag:yaml.org,2002:str', {
|
|
loader: {
|
|
kind: 'string'
|
|
}
|
|
});
|
|
|
|
},{"../type":106}],122:[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":94,"../type":106}],123:[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 : */
|
|
|
|
},{}],124:[function(require,module,exports){
|
|
var global=typeof self !== "undefined" ? self : typeof window !== "undefined" ? window : {};/**
|
|
* @license
|
|
* Lo-Dash 2.2.1 (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 = /^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
|
|
};
|
|
}
|
|
|
|
/**
|
|
* A no-operation function.
|
|
*
|
|
* @private
|
|
*/
|
|
function noop() {
|
|
// no operation performed
|
|
}
|
|
|
|
/**
|
|
* 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 detect if a method is native */
|
|
var reNative = RegExp('^' +
|
|
String(objectProto.valueOf)
|
|
.replace(/[.*+?^${}()|[\]\\]/g, '\\$&')
|
|
.replace(/valueOf|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,
|
|
setImmediate = context.setImmediate,
|
|
setTimeout = context.setTimeout,
|
|
splice = arrayRef.splice,
|
|
toString = objectProto.toString,
|
|
unshift = arrayRef.unshift;
|
|
|
|
var defineProperty = (function() {
|
|
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 nativeBind = reNative.test(nativeBind = toString.bind) && nativeBind,
|
|
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,
|
|
nativeSlice = arrayRef.slice;
|
|
|
|
/** Detect various environments */
|
|
var isIeOpera = reNative.test(context.attachEvent),
|
|
isV8 = nativeBind && !/\n|true/.test(nativeBind + isIeOpera);
|
|
|
|
/** 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`, `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 `Function#bind` exists and is inferred to be fast (all but V8).
|
|
*
|
|
* @memberOf _.support
|
|
* @type boolean
|
|
*/
|
|
support.fastBind = nativeBind && !isV8;
|
|
|
|
/**
|
|
* 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 `_.clone` without argument juggling or support
|
|
* for `thisArg` binding.
|
|
*
|
|
* @private
|
|
* @param {*} value The value to clone.
|
|
* @param {boolean} [deep=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, deep, 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 (deep) {
|
|
// 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 (!deep) {
|
|
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, deep, callback, stackA, stackB);
|
|
});
|
|
|
|
if (initedStack) {
|
|
releaseArray(stackA);
|
|
releaseArray(stackB);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* 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 if there is no `thisArg`
|
|
if (typeof thisArg == 'undefined') {
|
|
return func;
|
|
}
|
|
var bindData = func.__bindData__ || (support.funcNames && !func.name);
|
|
if (typeof bindData == 'undefined') {
|
|
var source = reThis && fnToString.call(func);
|
|
if (!support.funcNames && source && !reFuncName.test(source)) {
|
|
bindData = true;
|
|
}
|
|
if (support.funcNames || !bindData) {
|
|
// checks if `func` references the `this` keyword and stores the result
|
|
bindData = !support.funcDecomp || reThis.test(source);
|
|
setBindData(func, bindData);
|
|
}
|
|
}
|
|
// exit early if there are no `this` references or `func` is bound
|
|
if (bindData !== true && (bindData && 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 `_.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} [isArgArrays=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, isArgArrays, 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, isArgArrays);
|
|
}
|
|
var valIndex = -1,
|
|
valLength = value.length,
|
|
resIndex = result.length;
|
|
|
|
result.length += valLength;
|
|
while (++valIndex < valLength) {
|
|
result[resIndex++] = value[valIndex];
|
|
}
|
|
} else if (!isArgArrays) {
|
|
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
|
|
if (hasOwnProperty.call(a, '__wrapped__ ') || hasOwnProperty.call(b, '__wrapped__')) {
|
|
return baseIsEqual(a.__wrapped__ || a, 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
|
|
)) {
|
|
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 `_.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 bound function.
|
|
*/
|
|
function createBound(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,
|
|
key = func;
|
|
|
|
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) {
|
|
if (isBind && !(bindData[1] & 1)) {
|
|
bindData[4] = thisArg;
|
|
}
|
|
if (!isBind && bindData[1] & 1) {
|
|
bitmask |= 8;
|
|
}
|
|
if (isCurry && !(bindData[1] & 4)) {
|
|
bindData[5] = arity;
|
|
}
|
|
if (isPartial) {
|
|
push.apply(bindData[2] || (bindData[2] = []), partialArgs);
|
|
}
|
|
if (isPartialRight) {
|
|
push.apply(bindData[3] || (bindData[3] = []), partialRightArgs);
|
|
}
|
|
bindData[1] |= bitmask;
|
|
return createBound.apply(null, bindData);
|
|
}
|
|
// use `Function#bind` if it exists and is fast
|
|
// (in V8 `Function#bind` is slower except when partially applied)
|
|
if (isBind && !(isBindKey || isCurry || isPartialRight) &&
|
|
(support.fastBind || (nativeBind && isPartial))) {
|
|
if (isPartial) {
|
|
var args = [thisArg];
|
|
push.apply(args, partialArgs);
|
|
}
|
|
var bound = isPartial
|
|
? nativeBind.apply(func, args)
|
|
: nativeBind.call(func, thisArg);
|
|
}
|
|
else {
|
|
bound = function() {
|
|
// `Function#bind` spec
|
|
// http://es5.github.io/#x15.3.4.5
|
|
var args = arguments,
|
|
thisBinding = isBind ? thisArg : this;
|
|
|
|
if (isCurry || isPartial || isPartialRight) {
|
|
args = nativeSlice.call(args);
|
|
if (isPartial) {
|
|
unshift.apply(args, partialArgs);
|
|
}
|
|
if (isPartialRight) {
|
|
push.apply(args, partialRightArgs);
|
|
}
|
|
if (isCurry && args.length < arity) {
|
|
bitmask |= 16 & ~32;
|
|
return createBound(func, (isCurryBound ? bitmask : bitmask & ~3), args, null, thisArg, arity);
|
|
}
|
|
}
|
|
if (isBindKey) {
|
|
func = thisBinding[key];
|
|
}
|
|
if (this instanceof bound) {
|
|
// ensure `new bound` is an instance of `func`
|
|
thisBinding = createObject(func.prototype);
|
|
|
|
// mimic the constructor's `return` behavior
|
|
// http://es5.github.io/#x13.2.2
|
|
var result = func.apply(thisBinding, args);
|
|
return isObject(result) ? result : thisBinding;
|
|
}
|
|
return func.apply(thisBinding, args);
|
|
};
|
|
}
|
|
setBindData(bound, nativeSlice.call(arguments));
|
|
return bound;
|
|
}
|
|
|
|
/**
|
|
* Creates a new object with the specified `prototype`.
|
|
*
|
|
* @private
|
|
* @param {Object} prototype The prototype object.
|
|
* @returns {Object} Returns the new object.
|
|
*/
|
|
function createObject(prototype) {
|
|
return isObject(prototype) ? nativeCreate(prototype) : {};
|
|
}
|
|
// fallback for browsers without `Object.create`
|
|
if (!nativeCreate) {
|
|
createObject = function(prototype) {
|
|
if (isObject(prototype)) {
|
|
noop.prototype = prototype;
|
|
var result = new noop;
|
|
noop.prototype = null;
|
|
}
|
|
return result || {};
|
|
};
|
|
}
|
|
|
|
/**
|
|
* 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 {*} value The value 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': 'moe' }, { 'age': 40 });
|
|
* // => { 'name': 'moe', 'age': 40 }
|
|
*
|
|
* var defaults = _.partialRight(_.assign, function(a, b) {
|
|
* return typeof a == 'undefined' ? b : a;
|
|
* });
|
|
*
|
|
* var food = { 'name': 'apple' };
|
|
* defaults(food, { 'name': 'banana', 'type': 'fruit' });
|
|
* // => { 'name': 'apple', 'type': 'fruit' }
|
|
*/
|
|
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 `deep` 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} [deep=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 stooges = [
|
|
* { 'name': 'moe', 'age': 40 },
|
|
* { 'name': 'larry', 'age': 50 }
|
|
* ];
|
|
*
|
|
* var shallow = _.clone(stooges);
|
|
* shallow[0] === stooges[0];
|
|
* // => true
|
|
*
|
|
* var deep = _.clone(stooges, true);
|
|
* deep[0] === stooges[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, deep, callback, thisArg) {
|
|
// allows working with "Collections" methods without using their `index`
|
|
// and `collection` arguments for `deep` and `callback`
|
|
if (typeof deep != 'boolean' && deep != null) {
|
|
thisArg = callback;
|
|
callback = deep;
|
|
deep = false;
|
|
}
|
|
return baseClone(value, deep, 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 stooges = [
|
|
* { 'name': 'moe', 'age': 40 },
|
|
* { 'name': 'larry', 'age': 50 }
|
|
* ];
|
|
*
|
|
* var deep = _.cloneDeep(stooges);
|
|
* deep[0] === stooges[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));
|
|
}
|
|
|
|
/**
|
|
* 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 food = { 'name': 'apple' };
|
|
* _.defaults(food, { 'name': 'banana', 'type': 'fruit' });
|
|
* // => { 'name': 'apple', 'type': 'fruit' }
|
|
*/
|
|
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.
|
|
*
|
|
* @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
|
|
*
|
|
* _.findKey({ 'a': 1, 'b': 2, 'c': 3, 'd': 4 }, function(num) {
|
|
* return num % 2 == 0;
|
|
* });
|
|
* // => 'b' (property order is not guaranteed across environments)
|
|
*/
|
|
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.
|
|
*
|
|
* @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
|
|
*
|
|
* _.findLastKey({ 'a': 1, 'b': 2, 'c': 3, 'd': 4 }, function(num) {
|
|
* return num % 2 == 1;
|
|
* });
|
|
* // => returns `c`, assuming `_.findKey` returns `a`
|
|
*/
|
|
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 Dog(name) {
|
|
* this.name = name;
|
|
* }
|
|
*
|
|
* Dog.prototype.bark = function() {
|
|
* console.log('Woof, woof!');
|
|
* };
|
|
*
|
|
* _.forIn(new Dog('Dagny'), function(value, key) {
|
|
* console.log(key);
|
|
* });
|
|
* // => logs 'bark' and 'name' (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 Dog(name) {
|
|
* this.name = name;
|
|
* }
|
|
*
|
|
* Dog.prototype.bark = function() {
|
|
* console.log('Woof, woof!');
|
|
* };
|
|
*
|
|
* _.forInRight(new Dog('Dagny'), function(value, key) {
|
|
* console.log(key);
|
|
* });
|
|
* // => logs 'name' and 'bark' assuming `_.forIn ` logs 'bark' and 'name'
|
|
*/
|
|
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': 'moe', 'second': 'larry' });
|
|
* // => { 'moe': 'first', 'larry': '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 || toString.call(value) == boolClass;
|
|
}
|
|
|
|
/**
|
|
* 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 moe = { 'name': 'moe', 'age': 40 };
|
|
* var copy = { 'name': 'moe', 'age': 40 };
|
|
*
|
|
* moe == copy;
|
|
* // => false
|
|
*
|
|
* _.isEqual(moe, 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' || toString.call(value) == numberClass;
|
|
}
|
|
|
|
/**
|
|
* 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 Stooge(name, age) {
|
|
* this.name = name;
|
|
* this.age = age;
|
|
* }
|
|
*
|
|
* _.isPlainObject(new Stooge('moe', 40));
|
|
* // => false
|
|
*
|
|
* _.isPlainObject([1, 2, 3]);
|
|
* // => false
|
|
*
|
|
* _.isPlainObject({ 'name': 'moe', 'age': 40 });
|
|
* // => 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(/moe/);
|
|
* // => 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('moe');
|
|
* // => true
|
|
*/
|
|
function isString(value) {
|
|
return typeof value == 'string' || toString.call(value) == stringClass;
|
|
}
|
|
|
|
/**
|
|
* 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 = {
|
|
* 'stooges': [
|
|
* { 'name': 'moe' },
|
|
* { 'name': 'larry' }
|
|
* ]
|
|
* };
|
|
*
|
|
* var ages = {
|
|
* 'stooges': [
|
|
* { 'age': 40 },
|
|
* { 'age': 50 }
|
|
* ]
|
|
* };
|
|
*
|
|
* _.merge(names, ages);
|
|
* // => { 'stooges': [{ 'name': 'moe', 'age': 40 }, { 'name': 'larry', 'age': 50 }] }
|
|
*
|
|
* 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 = nativeSlice.call(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': 'moe', 'age': 40 }, 'age');
|
|
* // => { 'name': 'moe' }
|
|
*
|
|
* _.omit({ 'name': 'moe', 'age': 40 }, function(value) {
|
|
* return typeof value == 'number';
|
|
* });
|
|
* // => { 'name': 'moe' }
|
|
*/
|
|
function omit(object, callback, thisArg) {
|
|
var indexOf = getIndexOf(),
|
|
isFunc = typeof callback == 'function',
|
|
result = {};
|
|
|
|
if (isFunc) {
|
|
callback = lodash.createCallback(callback, thisArg, 3);
|
|
} else {
|
|
var props = baseFlatten(arguments, true, false, 1);
|
|
}
|
|
forIn(object, function(value, key, object) {
|
|
if (isFunc
|
|
? !callback(value, key, object)
|
|
: indexOf(props, key) < 0
|
|
) {
|
|
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({ 'moe': 30, 'larry': 40 });
|
|
* // => [['moe', 30], ['larry', 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': 'moe', '_userid': 'moe1' }, 'name');
|
|
* // => { 'name': 'moe' }
|
|
*
|
|
* _.pick({ 'name': 'moe', '_userid': 'moe1' }, function(value, key) {
|
|
* return key.charAt(0) != '_';
|
|
* });
|
|
* // => { 'name': 'moe' }
|
|
*/
|
|
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} collection The collection 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);
|
|
callback = baseCreateCallback(callback, thisArg, 4);
|
|
|
|
if (accumulator == null) {
|
|
if (isArr) {
|
|
accumulator = [];
|
|
} else {
|
|
var ctor = object && object.constructor,
|
|
proto = ctor && ctor.prototype;
|
|
|
|
accumulator = createObject(proto);
|
|
}
|
|
}
|
|
(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(['moe', 'larry', 'curly'], 0, 2);
|
|
* // => ['moe', 'curly']
|
|
*/
|
|
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': 'moe', 'age': 40 }, 'moe');
|
|
* // => true
|
|
*
|
|
* _.contains('curly', 'ur');
|
|
* // => 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'], Boolean);
|
|
* // => false
|
|
*
|
|
* var stooges = [
|
|
* { 'name': 'moe', 'age': 40 },
|
|
* { 'name': 'larry', 'age': 50 }
|
|
* ];
|
|
*
|
|
* // using "_.pluck" callback shorthand
|
|
* _.every(stooges, 'age');
|
|
* // => true
|
|
*
|
|
* // using "_.where" callback shorthand
|
|
* _.every(stooges, { 'age': 50 });
|
|
* // => 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 food = [
|
|
* { 'name': 'apple', 'organic': false, 'type': 'fruit' },
|
|
* { 'name': 'carrot', 'organic': true, 'type': 'vegetable' }
|
|
* ];
|
|
*
|
|
* // using "_.pluck" callback shorthand
|
|
* _.filter(food, 'organic');
|
|
* // => [{ 'name': 'carrot', 'organic': true, 'type': 'vegetable' }]
|
|
*
|
|
* // using "_.where" callback shorthand
|
|
* _.filter(food, { 'type': 'fruit' });
|
|
* // => [{ 'name': 'apple', 'organic': false, 'type': 'fruit' }]
|
|
*/
|
|
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
|
|
*
|
|
* _.find([1, 2, 3, 4], function(num) {
|
|
* return num % 2 == 0;
|
|
* });
|
|
* // => 2
|
|
*
|
|
* var food = [
|
|
* { 'name': 'apple', 'organic': false, 'type': 'fruit' },
|
|
* { 'name': 'banana', 'organic': true, 'type': 'fruit' },
|
|
* { 'name': 'beet', 'organic': false, 'type': 'vegetable' }
|
|
* ];
|
|
*
|
|
* // using "_.where" callback shorthand
|
|
* _.find(food, { 'type': 'vegetable' });
|
|
* // => { 'name': 'beet', 'organic': false, 'type': 'vegetable' }
|
|
*
|
|
* // using "_.pluck" callback shorthand
|
|
* _.find(food, 'organic');
|
|
* // => { 'name': 'banana', 'organic': true, 'type': 'fruit' }
|
|
*/
|
|
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`.
|
|
*
|
|
* @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(stooges, 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 = nativeSlice.call(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 stooges = [
|
|
* { 'name': 'moe', 'age': 40 },
|
|
* { 'name': 'larry', 'age': 50 }
|
|
* ];
|
|
*
|
|
* // using "_.pluck" callback shorthand
|
|
* _.map(stooges, 'name');
|
|
* // => ['moe', 'larry']
|
|
*/
|
|
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 stooges = [
|
|
* { 'name': 'moe', 'age': 40 },
|
|
* { 'name': 'larry', 'age': 50 }
|
|
* ];
|
|
*
|
|
* _.max(stooges, function(stooge) { return stooge.age; });
|
|
* // => { 'name': 'larry', 'age': 50 };
|
|
*
|
|
* // using "_.pluck" callback shorthand
|
|
* _.max(stooges, 'age');
|
|
* // => { 'name': 'larry', 'age': 50 };
|
|
*/
|
|
function max(collection, callback, thisArg) {
|
|
var computed = -Infinity,
|
|
result = computed;
|
|
|
|
if (!callback && isArray(collection)) {
|
|
var index = -1,
|
|
length = collection.length;
|
|
|
|
while (++index < length) {
|
|
var value = collection[index];
|
|
if (value > result) {
|
|
result = value;
|
|
}
|
|
}
|
|
} else {
|
|
callback = (!callback && 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 stooges = [
|
|
* { 'name': 'moe', 'age': 40 },
|
|
* { 'name': 'larry', 'age': 50 }
|
|
* ];
|
|
*
|
|
* _.min(stooges, function(stooge) { return stooge.age; });
|
|
* // => { 'name': 'moe', 'age': 40 };
|
|
*
|
|
* // using "_.pluck" callback shorthand
|
|
* _.min(stooges, 'age');
|
|
* // => { 'name': 'moe', 'age': 40 };
|
|
*/
|
|
function min(collection, callback, thisArg) {
|
|
var computed = Infinity,
|
|
result = computed;
|
|
|
|
if (!callback && isArray(collection)) {
|
|
var index = -1,
|
|
length = collection.length;
|
|
|
|
while (++index < length) {
|
|
var value = collection[index];
|
|
if (value < result) {
|
|
result = value;
|
|
}
|
|
}
|
|
} else {
|
|
callback = (!callback && 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 stooges = [
|
|
* { 'name': 'moe', 'age': 40 },
|
|
* { 'name': 'larry', 'age': 50 }
|
|
* ];
|
|
*
|
|
* _.pluck(stooges, 'name');
|
|
* // => ['moe', 'larry']
|
|
*/
|
|
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 = baseCreateCallback(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 = baseCreateCallback(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 food = [
|
|
* { 'name': 'apple', 'organic': false, 'type': 'fruit' },
|
|
* { 'name': 'carrot', 'organic': true, 'type': 'vegetable' }
|
|
* ];
|
|
*
|
|
* // using "_.pluck" callback shorthand
|
|
* _.reject(food, 'organic');
|
|
* // => [{ 'name': 'apple', 'organic': false, 'type': 'fruit' }]
|
|
*
|
|
* // using "_.where" callback shorthand
|
|
* _.reject(food, { 'type': 'fruit' });
|
|
* // => [{ 'name': 'carrot', 'organic': true, 'type': 'vegetable' }]
|
|
*/
|
|
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 `key` and `object` arguments as sources.
|
|
* @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) {
|
|
var length = collection ? collection.length : 0;
|
|
if (typeof length != 'number') {
|
|
collection = values(collection);
|
|
}
|
|
if (n == null || guard) {
|
|
return collection ? collection[random(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 = random(++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('curly');
|
|
* // => 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 food = [
|
|
* { 'name': 'apple', 'organic': false, 'type': 'fruit' },
|
|
* { 'name': 'carrot', 'organic': true, 'type': 'vegetable' }
|
|
* ];
|
|
*
|
|
* // using "_.pluck" callback shorthand
|
|
* _.some(food, 'organic');
|
|
* // => true
|
|
*
|
|
* // using "_.where" callback shorthand
|
|
* _.some(food, { 'type': 'meat' });
|
|
* // => 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 stooges = [
|
|
* { 'name': 'curly', 'age': 30, 'quotes': ['Oh, a wise guy, eh?', 'Poifect!'] },
|
|
* { 'name': 'moe', 'age': 40, 'quotes': ['Spread out!', 'You knucklehead!'] }
|
|
* ];
|
|
*
|
|
* _.where(stooges, { 'age': 40 });
|
|
* // => [{ 'name': 'moe', 'age': 40, 'quotes': ['Spread out!', 'You knucklehead!'] }]
|
|
*
|
|
* _.where(stooges, { 'quotes': ['Poifect!'] });
|
|
* // => [{ 'name': 'curly', 'age': 30, 'quotes': ['Oh, a wise guy, eh?', 'Poifect!'] }]
|
|
*/
|
|
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} [array] 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) {
|
|
var index = -1,
|
|
indexOf = getIndexOf(),
|
|
length = array ? array.length : 0,
|
|
seen = baseFlatten(arguments, true, true, 1),
|
|
result = [];
|
|
|
|
var isLarge = length >= largeArraySize && indexOf === baseIndexOf;
|
|
|
|
if (isLarge) {
|
|
var cache = createCache(seen);
|
|
if (cache) {
|
|
indexOf = cacheIndexOf;
|
|
seen = cache;
|
|
} else {
|
|
isLarge = false;
|
|
}
|
|
}
|
|
while (++index < length) {
|
|
var value = array[index];
|
|
if (indexOf(seen, value) < 0) {
|
|
result.push(value);
|
|
}
|
|
}
|
|
if (isLarge) {
|
|
releaseObject(seen);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* 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.
|
|
*
|
|
* @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
|
|
*
|
|
* _.findIndex(['apple', 'banana', 'beet'], function(food) {
|
|
* return /^b/.test(food);
|
|
* });
|
|
* // => 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.
|
|
*
|
|
* @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
|
|
*
|
|
* _.findLastIndex(['apple', 'banana', 'beet'], function(food) {
|
|
* return /^b/.test(food);
|
|
* });
|
|
* // => 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 food = [
|
|
* { 'name': 'banana', 'organic': true },
|
|
* { 'name': 'beet', 'organic': false },
|
|
* ];
|
|
*
|
|
* // using "_.pluck" callback shorthand
|
|
* _.first(food, 'organic');
|
|
* // => [{ 'name': 'banana', 'organic': true }]
|
|
*
|
|
* var food = [
|
|
* { 'name': 'apple', 'type': 'fruit' },
|
|
* { 'name': 'banana', 'type': 'fruit' },
|
|
* { 'name': 'beet', 'type': 'vegetable' }
|
|
* ];
|
|
*
|
|
* // using "_.where" callback shorthand
|
|
* _.first(food, { 'type': 'fruit' });
|
|
* // => [{ 'name': 'apple', 'type': 'fruit' }, { 'name': 'banana', 'type': 'fruit' }]
|
|
*/
|
|
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 stooges = [
|
|
* { 'name': 'curly', 'quotes': ['Oh, a wise guy, eh?', 'Poifect!'] },
|
|
* { 'name': 'moe', 'quotes': ['Spread out!', 'You knucklehead!'] }
|
|
* ];
|
|
*
|
|
* // using "_.pluck" callback shorthand
|
|
* _.flatten(stooges, 'quotes');
|
|
* // => ['Oh, a wise guy, eh?', 'Poifect!', 'Spread out!', 'You knucklehead!']
|
|
*/
|
|
function flatten(array, isShallow, callback, thisArg) {
|
|
// juggle arguments
|
|
if (typeof isShallow != 'boolean' && isShallow != null) {
|
|
thisArg = callback;
|
|
callback = !(thisArg && thisArg[isShallow] === array) ? isShallow : null;
|
|
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 food = [
|
|
* { 'name': 'beet', 'organic': false },
|
|
* { 'name': 'carrot', 'organic': true }
|
|
* ];
|
|
*
|
|
* // using "_.pluck" callback shorthand
|
|
* _.initial(food, 'organic');
|
|
* // => [{ 'name': 'beet', 'organic': false }]
|
|
*
|
|
* var food = [
|
|
* { 'name': 'banana', 'type': 'fruit' },
|
|
* { 'name': 'beet', 'type': 'vegetable' },
|
|
* { 'name': 'carrot', 'type': 'vegetable' }
|
|
* ];
|
|
*
|
|
* // using "_.where" callback shorthand
|
|
* _.initial(food, { 'type': 'vegetable' });
|
|
* // => [{ 'name': 'banana', 'type': 'fruit' }]
|
|
*/
|
|
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 food = [
|
|
* { 'name': 'beet', 'organic': false },
|
|
* { 'name': 'carrot', 'organic': true }
|
|
* ];
|
|
*
|
|
* // using "_.pluck" callback shorthand
|
|
* _.last(food, 'organic');
|
|
* // => [{ 'name': 'carrot', 'organic': true }]
|
|
*
|
|
* var food = [
|
|
* { 'name': 'banana', 'type': 'fruit' },
|
|
* { 'name': 'beet', 'type': 'vegetable' },
|
|
* { 'name': 'carrot', 'type': 'vegetable' }
|
|
* ];
|
|
*
|
|
* // using "_.where" callback shorthand
|
|
* _.last(food, { 'type': 'vegetable' });
|
|
* // => [{ 'name': 'beet', 'type': 'vegetable' }, { 'name': 'carrot', 'type': 'vegetable' }]
|
|
*/
|
|
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.
|
|
*
|
|
* @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(10);
|
|
* // => [0, 1, 2, 3, 4, 5, 6, 7, 8, 9]
|
|
*
|
|
* _.range(1, 11);
|
|
* // => [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]
|
|
*
|
|
* _.range(0, 30, 5);
|
|
* // => [0, 5, 10, 15, 20, 25]
|
|
*
|
|
* _.range(0, -10, -1);
|
|
* // => [0, -1, -2, -3, -4, -5, -6, -7, -8, -9]
|
|
*
|
|
* _.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 food = [
|
|
* { 'name': 'banana', 'organic': true },
|
|
* { 'name': 'beet', 'organic': false },
|
|
* ];
|
|
*
|
|
* // using "_.pluck" callback shorthand
|
|
* _.rest(food, 'organic');
|
|
* // => [{ 'name': 'beet', 'organic': false }]
|
|
*
|
|
* var food = [
|
|
* { 'name': 'apple', 'type': 'fruit' },
|
|
* { 'name': 'banana', 'type': 'fruit' },
|
|
* { 'name': 'beet', 'type': 'vegetable' }
|
|
* ];
|
|
*
|
|
* // using "_.where" callback shorthand
|
|
* _.rest(food, { 'type': 'fruit' });
|
|
* // => [{ 'name': 'beet', 'type': 'vegetable' }]
|
|
*/
|
|
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 = !(thisArg && thisArg[isSorted] === array) ? isSorted : null;
|
|
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 difference(array, nativeSlice.call(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(['moe', 'larry'], [30, 40], [true, false]);
|
|
* // => [['moe', 30, true], ['larry', 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(['moe', 'larry'], [30, 40]);
|
|
* // => { 'moe': 30, 'larry': 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': 'moe' }, 'hi');
|
|
* func();
|
|
* // => 'hi moe'
|
|
*/
|
|
function bind(func, thisArg) {
|
|
return arguments.length > 2
|
|
? createBound(func, 17, nativeSlice.call(arguments, 2), null, thisArg)
|
|
: createBound(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] = createBound(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': 'moe',
|
|
* 'greet': function(greeting) {
|
|
* return greeting + ' ' + this.name;
|
|
* }
|
|
* };
|
|
*
|
|
* var func = _.bindKey(object, 'greet', 'hi');
|
|
* func();
|
|
* // => 'hi moe'
|
|
*
|
|
* object.greet = function(greeting) {
|
|
* return greeting + ', ' + this.name + '!';
|
|
* };
|
|
*
|
|
* func();
|
|
* // => 'hi, moe!'
|
|
*/
|
|
function bindKey(object, key) {
|
|
return arguments.length > 2
|
|
? createBound(key, 19, nativeSlice.call(arguments, 2), null, object)
|
|
: createBound(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 = {
|
|
* 'curly': 'jerome'
|
|
* };
|
|
*
|
|
* 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('curly');
|
|
* // => 'Hiya Jerome!'
|
|
*/
|
|
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 stooges = [
|
|
* { 'name': 'moe', 'age': 40 },
|
|
* { 'name': 'larry', 'age': 50 }
|
|
* ];
|
|
*
|
|
* // 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(stooges, 'age__gt45');
|
|
* // => [{ 'name': 'larry', 'age': 50 }]
|
|
*/
|
|
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 createBound(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);
|
|
}
|
|
} 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);
|
|
}
|
|
};
|
|
|
|
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);
|
|
if (remaining <= 0) {
|
|
if (maxTimeoutId) {
|
|
maxTimeoutId = clearTimeout(maxTimeoutId);
|
|
}
|
|
lastCalled = stamp;
|
|
result = func.apply(thisArg, args);
|
|
}
|
|
else if (!maxTimeoutId) {
|
|
maxTimeoutId = setTimeout(maxDelayed, remaining);
|
|
}
|
|
}
|
|
if (!timeoutId && wait !== maxWait) {
|
|
timeoutId = setTimeout(delayed, wait);
|
|
}
|
|
if (leadingCall) {
|
|
result = func.apply(thisArg, args);
|
|
}
|
|
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 = nativeSlice.call(arguments, 1);
|
|
return setTimeout(function() { func.apply(undefined, args); }, 1);
|
|
}
|
|
// use `setImmediate` if available in Node.js
|
|
if (isV8 && moduleExports && typeof setImmediate == 'function') {
|
|
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 = nativeSlice.call(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);
|
|
* });
|
|
*
|
|
* var data = {
|
|
* 'moe': { 'name': 'moe', 'age': 40 },
|
|
* 'curly': { 'name': 'curly', 'age': 60 }
|
|
* };
|
|
*
|
|
* // modifying the result cache
|
|
* var stooge = _.memoize(function(name) { return data[name]; }, _.identity);
|
|
* stooge('curly');
|
|
* // => { 'name': 'curly', 'age': 60 }
|
|
*
|
|
* stooge.cache.curly.name = 'jerome';
|
|
* stooge('curly');
|
|
* // => { 'name': 'jerome', 'age': 60 }
|
|
*/
|
|
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('moe');
|
|
* // => 'hi moe'
|
|
*/
|
|
function partial(func) {
|
|
return createBound(func, 16, nativeSlice.call(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 createBound(func, 32, null, nativeSlice.call(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;
|
|
|
|
var result = debounce(func, wait, debounceOptions);
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* 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 hello = function(name) { return 'hello ' + name; };
|
|
* hello = _.wrap(hello, function(func) {
|
|
* return 'before, ' + func('moe') + ', after';
|
|
* });
|
|
* hello();
|
|
* // => 'before, hello moe, after'
|
|
*/
|
|
function wrap(value, wrapper) {
|
|
if (!isFunction(wrapper)) {
|
|
throw new TypeError;
|
|
}
|
|
return function() {
|
|
var args = [value];
|
|
push.apply(args, arguments);
|
|
return wrapper.apply(this, args);
|
|
};
|
|
}
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
|
|
/**
|
|
* 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('Moe, Larry & Curly');
|
|
* // => 'Moe, Larry & Curly'
|
|
*/
|
|
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 moe = { 'name': 'moe' };
|
|
* moe === _.identity(moe);
|
|
* // => 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('moe');
|
|
* // => 'Moe'
|
|
*
|
|
* _('moe').capitalize();
|
|
* // => 'Moe'
|
|
*/
|
|
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;
|
|
}
|
|
|
|
/**
|
|
* 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 and Opera still 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;
|
|
}
|
|
var rand = nativeRandom();
|
|
return (floating || min % 1 || max % 1)
|
|
? nativeMin(min + (rand * (max - min + parseFloat('1e-' + ((rand +'').length - 1)))), max)
|
|
: min + floor(rand * (max - min + 1));
|
|
}
|
|
|
|
/**
|
|
* 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': 'moe' });
|
|
* // => 'hello moe'
|
|
*
|
|
* // 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': ['moe', 'larry'] });
|
|
* // => '<li>moe</li><li>larry</li>'
|
|
*
|
|
* // using the ES6 delimiter as an alternative to the default "interpolate" delimiter
|
|
* _.template('hello ${ name }', { 'name': 'curly' });
|
|
* // => 'hello curly'
|
|
*
|
|
* // using the internal `print` function in "evaluate" delimiters
|
|
* _.template('<% print("hello " + name); %>!', { 'name': 'larry' });
|
|
* // => 'hello larry!'
|
|
*
|
|
* // 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': ['moe', 'larry'] }, { 'imports': { '$': jQuery } });
|
|
* // => '<li>moe</li><li>larry</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 || (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('Moe, Larry & Curly');
|
|
* // => 'Moe, Larry & Curly'
|
|
*/
|
|
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 stooges = [
|
|
* { 'name': 'moe', 'age': 40 },
|
|
* { 'name': 'larry', 'age': 50 },
|
|
* { 'name': 'curly', 'age': 60 }
|
|
* ];
|
|
*
|
|
* var youngest = _.chain(stooges)
|
|
* .sortBy('age')
|
|
* .map(function(stooge) { return stooge.name + ' is ' + stooge.age; })
|
|
* .first()
|
|
* .value();
|
|
* // => 'moe is 40'
|
|
*/
|
|
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])
|
|
* .filter(function(num) { return num % 2 == 0; })
|
|
* .tap(function(array) { console.log(array); })
|
|
* .map(function(num) { return num * num; })
|
|
* .value();
|
|
* // => // [2, 4] (logged)
|
|
* // => [4, 16]
|
|
*/
|
|
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 stooges = [
|
|
* { 'name': 'moe', 'age': 40 },
|
|
* { 'name': 'larry', 'age': 50 }
|
|
* ];
|
|
*
|
|
* // without explicit chaining
|
|
* _(stooges).first();
|
|
* // => { 'name': 'moe', 'age': 40 }
|
|
*
|
|
* // with explicit chaining
|
|
* _(stooges).chain()
|
|
* .first()
|
|
* .pick('age')
|
|
* .value()
|
|
* // => { 'age': 40 }
|
|
*/
|
|
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.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.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.2.1';
|
|
|
|
// 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));
|
|
|
|
},{}],125:[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":91,"lru-cache":126,"path":61,"sigmund":127}],126:[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)
|
|
}
|
|
|
|
var max
|
|
if (typeof options === 'number') {
|
|
max = options
|
|
options = { max: max }
|
|
}
|
|
|
|
if (!options) options = {}
|
|
|
|
max = options.max
|
|
|
|
var lengthCalculator = options.length || naiveLength
|
|
|
|
if (typeof lengthCalculator !== "function") {
|
|
lengthCalculator = naiveLength
|
|
}
|
|
|
|
if (!max || !(typeof max === "number") || max <= 0 ) {
|
|
// a little bit silly. maybe this should throw?
|
|
max = Infinity
|
|
}
|
|
|
|
var allowStale = options.stale || false
|
|
|
|
var maxAge = options.maxAge || null
|
|
|
|
var dispose = options.dispose
|
|
|
|
var cache = Object.create(null) // hash of items by key
|
|
, lruList = Object.create(null) // list of items in order of use recency
|
|
, mru = 0 // most recently used
|
|
, lru = 0 // least recently used
|
|
, length = 0 // number of items in the list
|
|
, itemCount = 0
|
|
|
|
|
|
// resize the cache when the max changes.
|
|
Object.defineProperty(this, "max",
|
|
{ set : function (mL) {
|
|
if (!mL || !(typeof mL === "number") || mL <= 0 ) mL = Infinity
|
|
max = mL
|
|
// if it gets above double max, trim right away.
|
|
// otherwise, do it whenever it's convenient.
|
|
if (length > max) trim()
|
|
}
|
|
, get : function () { return max }
|
|
, enumerable : true
|
|
})
|
|
|
|
// resize the cache when the lengthCalculator changes.
|
|
Object.defineProperty(this, "lengthCalculator",
|
|
{ set : function (lC) {
|
|
if (typeof lC !== "function") {
|
|
lengthCalculator = naiveLength
|
|
length = itemCount
|
|
for (var key in cache) {
|
|
cache[key].length = 1
|
|
}
|
|
} else {
|
|
lengthCalculator = lC
|
|
length = 0
|
|
for (var key in cache) {
|
|
cache[key].length = lengthCalculator(cache[key].value)
|
|
length += cache[key].length
|
|
}
|
|
}
|
|
|
|
if (length > max) trim()
|
|
}
|
|
, get : function () { return lengthCalculator }
|
|
, enumerable : true
|
|
})
|
|
|
|
Object.defineProperty(this, "length",
|
|
{ get : function () { return length }
|
|
, enumerable : true
|
|
})
|
|
|
|
|
|
Object.defineProperty(this, "itemCount",
|
|
{ get : function () { return itemCount }
|
|
, enumerable : true
|
|
})
|
|
|
|
this.forEach = function (fn, thisp) {
|
|
thisp = thisp || this
|
|
var i = 0;
|
|
for (var k = mru - 1; k >= 0 && i < itemCount; k--) if (lruList[k]) {
|
|
i++
|
|
var hit = lruList[k]
|
|
if (maxAge && (Date.now() - hit.now > maxAge)) {
|
|
del(hit)
|
|
if (!allowStale) hit = undefined
|
|
}
|
|
if (hit) {
|
|
fn.call(thisp, hit.value, hit.key, this)
|
|
}
|
|
}
|
|
}
|
|
|
|
this.keys = function () {
|
|
var keys = new Array(itemCount)
|
|
var i = 0
|
|
for (var k = mru - 1; k >= 0 && i < itemCount; k--) if (lruList[k]) {
|
|
var hit = lruList[k]
|
|
keys[i++] = hit.key
|
|
}
|
|
return keys
|
|
}
|
|
|
|
this.values = function () {
|
|
var values = new Array(itemCount)
|
|
var i = 0
|
|
for (var k = mru - 1; k >= 0 && i < itemCount; k--) if (lruList[k]) {
|
|
var hit = lruList[k]
|
|
values[i++] = hit.value
|
|
}
|
|
return values
|
|
}
|
|
|
|
this.reset = function () {
|
|
if (dispose) {
|
|
for (var k in cache) {
|
|
dispose(k, cache[k].value)
|
|
}
|
|
}
|
|
cache = {}
|
|
lruList = {}
|
|
lru = 0
|
|
mru = 0
|
|
length = 0
|
|
itemCount = 0
|
|
}
|
|
|
|
// Provided for debugging/dev purposes only. No promises whatsoever that
|
|
// this API stays stable.
|
|
this.dump = function () {
|
|
return cache
|
|
}
|
|
|
|
this.dumpLru = function () {
|
|
return lruList
|
|
}
|
|
|
|
this.set = function (key, value) {
|
|
if (hOP(cache, key)) {
|
|
// dispose of the old one before overwriting
|
|
if (dispose) dispose(key, cache[key].value)
|
|
if (maxAge) cache[key].now = Date.now()
|
|
cache[key].value = value
|
|
this.get(key)
|
|
return true
|
|
}
|
|
|
|
var len = lengthCalculator(value)
|
|
var age = maxAge ? Date.now() : 0
|
|
var hit = new Entry(key, value, mru++, len, age)
|
|
|
|
// oversized objects fall out of cache automatically.
|
|
if (hit.length > max) {
|
|
if (dispose) dispose(key, value)
|
|
return false
|
|
}
|
|
|
|
length += hit.length
|
|
lruList[hit.lu] = cache[key] = hit
|
|
itemCount ++
|
|
|
|
if (length > max) trim()
|
|
return true
|
|
}
|
|
|
|
this.has = function (key) {
|
|
if (!hOP(cache, key)) return false
|
|
var hit = cache[key]
|
|
if (maxAge && (Date.now() - hit.now > maxAge)) {
|
|
return false
|
|
}
|
|
return true
|
|
}
|
|
|
|
this.get = function (key) {
|
|
return get(key, true)
|
|
}
|
|
|
|
this.peek = function (key) {
|
|
return get(key, false)
|
|
}
|
|
|
|
function get (key, doUse) {
|
|
var hit = cache[key]
|
|
if (hit) {
|
|
if (maxAge && (Date.now() - hit.now > maxAge)) {
|
|
del(hit)
|
|
if (!allowStale) hit = undefined
|
|
} else {
|
|
if (doUse) use(hit)
|
|
}
|
|
if (hit) hit = hit.value
|
|
}
|
|
return hit
|
|
}
|
|
|
|
function use (hit) {
|
|
shiftLU(hit)
|
|
hit.lu = mru ++
|
|
lruList[hit.lu] = hit
|
|
}
|
|
|
|
this.del = function (key) {
|
|
del(cache[key])
|
|
}
|
|
|
|
function trim () {
|
|
while (lru < mru && length > max)
|
|
del(lruList[lru])
|
|
}
|
|
|
|
function shiftLU(hit) {
|
|
delete lruList[ hit.lu ]
|
|
while (lru < mru && !lruList[lru]) lru ++
|
|
}
|
|
|
|
function del(hit) {
|
|
if (hit) {
|
|
if (dispose) dispose(hit.key, hit.value)
|
|
length -= hit.length
|
|
itemCount --
|
|
delete cache[ hit.key ]
|
|
shiftLU(hit)
|
|
}
|
|
}
|
|
}
|
|
|
|
// classy, since V8 prefers predictable objects.
|
|
function Entry (key, value, mru, len, age) {
|
|
this.key = key
|
|
this.value = value
|
|
this.lu = mru
|
|
this.length = len
|
|
this.now = age
|
|
}
|
|
|
|
})()
|
|
|
|
},{}],127:[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:
|
|
|
|
},{}],128:[function(require,module,exports){
|
|
var es = {
|
|
Client: require('./lib/client')
|
|
};
|
|
|
|
module.exports = es;
|
|
},{"./lib/client":130}],129:[function(require,module,exports){
|
|
/* jshint maxlen: false */
|
|
|
|
var ca = require('./client_action');
|
|
var errors = require('./errors');
|
|
|
|
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({
|
|
methods: [
|
|
'POST',
|
|
'PUT'
|
|
],
|
|
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
|
|
});
|
|
|
|
|
|
/**
|
|
* 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|ArrayOfStrings|Boolean} params.scrollId - A comma-separated list of scroll IDs to clear
|
|
*/
|
|
api.clearScroll = ca({
|
|
methods: [
|
|
'DELETE'
|
|
],
|
|
params: {},
|
|
urls: [
|
|
{
|
|
fmt: '/_search/scroll/<%=scrollId%>',
|
|
req: {
|
|
scrollId: {
|
|
type: 'list'
|
|
}
|
|
}
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
api.cluster = function ClusterNS(client) {
|
|
if (this instanceof ClusterNS) {
|
|
this.client = client;
|
|
} else {
|
|
return new ClusterNS(client);
|
|
}
|
|
};
|
|
|
|
/**
|
|
* 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({
|
|
methods: [
|
|
'GET'
|
|
],
|
|
params: {},
|
|
urls: [
|
|
{
|
|
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|Number} params.masterTimeout - Explicit operation timeout for connection to master node
|
|
* @param {Date|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({
|
|
methods: [
|
|
'GET'
|
|
],
|
|
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|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|ArrayOfStrings|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({
|
|
methods: [
|
|
'GET'
|
|
],
|
|
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|Number} params.timeout - Explicit operation timeout
|
|
* @param {Boolean} params.transport - Return information about transport
|
|
* @param {String|ArrayOfStrings|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({
|
|
methods: [
|
|
'GET'
|
|
],
|
|
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|Number} params.delay - Set the delay for the operation (default: 1s)
|
|
* @param {Boolean} params.exit - Exit the JVM as well (default: true)
|
|
* @param {String|ArrayOfStrings|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({
|
|
methods: [
|
|
'POST'
|
|
],
|
|
params: {
|
|
delay: {
|
|
type: 'time'
|
|
},
|
|
exit: {
|
|
type: 'boolean'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/_cluster/nodes/<%=nodeId%>/_shutdown',
|
|
req: {
|
|
nodeId: {
|
|
type: 'list'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/_shutdown'
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* 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|ArrayOfStrings|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|ArrayOfStrings|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({
|
|
methods: [
|
|
'GET'
|
|
],
|
|
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({
|
|
methods: [
|
|
'PUT'
|
|
],
|
|
params: {},
|
|
urls: [
|
|
{
|
|
fmt: '/_cluster/settings'
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* 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({
|
|
methods: [
|
|
'POST'
|
|
],
|
|
params: {
|
|
dryRun: {
|
|
type: 'boolean',
|
|
name: 'dry_run'
|
|
},
|
|
filterMetadata: {
|
|
type: 'boolean',
|
|
name: 'filter_metadata'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/_cluster/reroute'
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* 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|ArrayOfStrings|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|Number} params.masterTimeout - Specify timeout for connection to master
|
|
*/
|
|
api.cluster.prototype.state = ca({
|
|
methods: [
|
|
'GET'
|
|
],
|
|
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'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
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|ArrayOfStrings|Boolean} params.index - A comma-separated list of indices to restrict the results
|
|
* @param {String|ArrayOfStrings|Boolean} params.type - A comma-separated list of types to restrict the results
|
|
*/
|
|
api.count = ca({
|
|
methods: [
|
|
'POST',
|
|
'GET'
|
|
],
|
|
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'
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* 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|Number} params.timeout - Explicit operation timeout
|
|
* @param {Date|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({
|
|
methods: [
|
|
'POST',
|
|
'PUT'
|
|
],
|
|
params: {
|
|
consistency: {
|
|
type: 'enum',
|
|
options: [
|
|
'one',
|
|
'quorum',
|
|
'all'
|
|
]
|
|
},
|
|
id: {
|
|
type: 'string'
|
|
},
|
|
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%>/_create',
|
|
req: {
|
|
index: {
|
|
type: 'string'
|
|
},
|
|
type: {
|
|
type: 'string'
|
|
},
|
|
id: {
|
|
type: 'string'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/<%=index%>/<%=type%>',
|
|
req: {
|
|
index: {
|
|
type: 'string'
|
|
},
|
|
type: {
|
|
type: 'string'
|
|
}
|
|
}
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* 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|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({
|
|
methods: [
|
|
'DELETE'
|
|
],
|
|
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'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>/<%=type%>/<%=id%>',
|
|
req: {
|
|
index: {
|
|
type: 'string'
|
|
},
|
|
type: {
|
|
type: 'string'
|
|
},
|
|
id: {
|
|
type: 'string'
|
|
}
|
|
}
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* 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|Number} params.timeout - Explicit operation timeout
|
|
* @param {String|ArrayOfStrings|Boolean} params.index - A comma-separated list of indices to restrict the operation; use `_all` to perform the operation on all indices
|
|
* @param {String|ArrayOfStrings|Boolean} params.type - A comma-separated list of types to restrict the operation
|
|
*/
|
|
api.deleteByQuery = ca({
|
|
methods: [
|
|
'DELETE'
|
|
],
|
|
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'
|
|
}
|
|
}
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* 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({
|
|
methods: [
|
|
'HEAD'
|
|
],
|
|
params: {
|
|
parent: {
|
|
type: 'string'
|
|
},
|
|
preference: {
|
|
type: 'string'
|
|
},
|
|
realtime: {
|
|
type: 'boolean'
|
|
},
|
|
refresh: {
|
|
type: 'boolean'
|
|
},
|
|
routing: {
|
|
type: 'string'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>/<%=type%>/<%=id%>',
|
|
opt: {
|
|
type: {
|
|
type: 'string',
|
|
'default': '_all'
|
|
}
|
|
},
|
|
req: {
|
|
index: {
|
|
type: 'string'
|
|
},
|
|
id: {
|
|
type: 'string'
|
|
}
|
|
}
|
|
}
|
|
],
|
|
castExists: true
|
|
});
|
|
|
|
|
|
/**
|
|
* 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|ArrayOfStrings|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|ArrayOfStrings|Boolean} params.source - True or false to return the _source field or not, or a list of fields to return
|
|
* @param {String|ArrayOfStrings|Boolean} params.sourceExclude - A list of fields to exclude from the returned _source field
|
|
* @param {String|ArrayOfStrings|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({
|
|
methods: [
|
|
'GET',
|
|
'POST'
|
|
],
|
|
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'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>/<%=type%>/<%=id%>/_explain',
|
|
req: {
|
|
index: {
|
|
type: 'string'
|
|
},
|
|
type: {
|
|
type: 'string'
|
|
},
|
|
id: {
|
|
type: 'string'
|
|
}
|
|
}
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* 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|ArrayOfStrings|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|ArrayOfStrings|Boolean} params.source - True or false to return the _source field or not, or a list of fields to return
|
|
* @param {String|ArrayOfStrings|Boolean} params.sourceExclude - A list of fields to exclude from the returned _source field
|
|
* @param {String|ArrayOfStrings|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({
|
|
methods: [
|
|
'GET'
|
|
],
|
|
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'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>/<%=type%>/<%=id%>',
|
|
opt: {
|
|
type: {
|
|
type: 'string',
|
|
'default': '_all'
|
|
}
|
|
},
|
|
req: {
|
|
index: {
|
|
type: 'string'
|
|
},
|
|
id: {
|
|
type: 'string'
|
|
}
|
|
}
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* 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|ArrayOfStrings|Boolean} params.exclude - A list of fields to exclude from the returned _source field
|
|
* @param {String|ArrayOfStrings|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({
|
|
methods: [
|
|
'GET'
|
|
],
|
|
params: {
|
|
exclude: {
|
|
type: 'list'
|
|
},
|
|
include: {
|
|
type: 'list'
|
|
},
|
|
parent: {
|
|
type: 'string'
|
|
},
|
|
preference: {
|
|
type: 'string'
|
|
},
|
|
realtime: {
|
|
type: 'boolean'
|
|
},
|
|
refresh: {
|
|
type: 'boolean'
|
|
},
|
|
routing: {
|
|
type: 'string'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>/<%=type%>/<%=id%>/_source',
|
|
opt: {
|
|
type: {
|
|
type: 'string',
|
|
'default': '_all'
|
|
}
|
|
},
|
|
req: {
|
|
index: {
|
|
type: 'string'
|
|
},
|
|
id: {
|
|
type: 'string'
|
|
}
|
|
}
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* 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|Number} params.timeout - Explicit operation timeout
|
|
* @param {Date|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({
|
|
methods: [
|
|
'POST',
|
|
'PUT'
|
|
],
|
|
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'
|
|
}
|
|
}
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
api.indices = function IndicesNS(client) {
|
|
if (this instanceof IndicesNS) {
|
|
this.client = client;
|
|
} else {
|
|
return new IndicesNS(client);
|
|
}
|
|
};
|
|
|
|
/**
|
|
* 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|ArrayOfStrings|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({
|
|
methods: [
|
|
'GET',
|
|
'POST'
|
|
],
|
|
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'
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* 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|ArrayOfStrings|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|ArrayOfStrings|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({
|
|
methods: [
|
|
'POST',
|
|
'GET'
|
|
],
|
|
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'
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* 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|Number} params.timeout - Explicit operation timeout
|
|
* @param {Date|Number} params.masterTimeout - Specify timeout for connection to master
|
|
* @param {String} params.index - The name of the index
|
|
*/
|
|
api.indices.prototype.close = ca({
|
|
methods: [
|
|
'POST'
|
|
],
|
|
params: {
|
|
timeout: {
|
|
type: 'time'
|
|
},
|
|
masterTimeout: {
|
|
type: 'time',
|
|
name: 'master_timeout'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>/_close',
|
|
req: {
|
|
index: {
|
|
type: 'string'
|
|
}
|
|
}
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* 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|Number} params.timeout - Explicit operation timeout
|
|
* @param {Date|Number} params.masterTimeout - Specify timeout for connection to master
|
|
* @param {String} params.index - The name of the index
|
|
*/
|
|
api.indices.prototype.create = ca({
|
|
methods: [
|
|
'PUT',
|
|
'POST'
|
|
],
|
|
params: {
|
|
timeout: {
|
|
type: 'time'
|
|
},
|
|
masterTimeout: {
|
|
type: 'time',
|
|
name: 'master_timeout'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>',
|
|
req: {
|
|
index: {
|
|
type: 'string'
|
|
}
|
|
}
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* 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|Number} params.timeout - Explicit operation timeout
|
|
* @param {Date|Number} params.masterTimeout - Specify timeout for connection to master
|
|
* @param {String|ArrayOfStrings|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({
|
|
methods: [
|
|
'DELETE'
|
|
],
|
|
params: {
|
|
timeout: {
|
|
type: 'time'
|
|
},
|
|
masterTimeout: {
|
|
type: 'time',
|
|
name: 'master_timeout'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/'
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* 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|Number} params.timeout - Explicit timestamp for the document
|
|
* @param {Date|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({
|
|
methods: [
|
|
'DELETE'
|
|
],
|
|
params: {
|
|
timeout: {
|
|
type: 'time'
|
|
},
|
|
masterTimeout: {
|
|
type: 'time',
|
|
name: 'master_timeout'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>/_alias/<%=name%>',
|
|
req: {
|
|
index: {
|
|
type: 'string'
|
|
},
|
|
name: {
|
|
type: 'string'
|
|
}
|
|
}
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* 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|Number} params.masterTimeout - Specify timeout for connection to master
|
|
* @param {String|ArrayOfStrings|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({
|
|
methods: [
|
|
'DELETE'
|
|
],
|
|
params: {
|
|
masterTimeout: {
|
|
type: 'time',
|
|
name: 'master_timeout'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>/<%=type%>',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
},
|
|
type: {
|
|
type: 'string'
|
|
}
|
|
}
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* 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|Number} params.timeout - Explicit operation timeout
|
|
* @param {Date|Number} params.masterTimeout - Specify timeout for connection to master
|
|
* @param {String} params.name - The name of the template
|
|
*/
|
|
api.indices.prototype.deleteTemplate = ca({
|
|
methods: [
|
|
'DELETE'
|
|
],
|
|
params: {
|
|
timeout: {
|
|
type: 'time'
|
|
},
|
|
masterTimeout: {
|
|
type: 'time',
|
|
name: 'master_timeout'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/_template/<%=name%>',
|
|
req: {
|
|
name: {
|
|
type: 'string'
|
|
}
|
|
}
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* 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|Number} params.masterTimeout - Specify timeout for connection to master
|
|
* @param {String|ArrayOfStrings|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|ArrayOfStrings|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({
|
|
methods: [
|
|
'DELETE'
|
|
],
|
|
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'
|
|
}
|
|
}
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* 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|ArrayOfStrings|Boolean} params.index - A comma-separated list of indices to check
|
|
*/
|
|
api.indices.prototype.exists = ca({
|
|
methods: [
|
|
'HEAD'
|
|
],
|
|
params: {},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
}
|
|
}
|
|
}
|
|
],
|
|
castExists: true
|
|
});
|
|
|
|
|
|
/**
|
|
* 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|ArrayOfStrings|Boolean} params.index - A comma-separated list of index names to filter aliases
|
|
* @param {String|ArrayOfStrings|Boolean} params.name - A comma-separated list of alias names to return
|
|
*/
|
|
api.indices.prototype.existsAlias = ca({
|
|
methods: [
|
|
'HEAD'
|
|
],
|
|
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
|
|
});
|
|
|
|
|
|
/**
|
|
* 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|ArrayOfStrings|Boolean} params.index - A comma-separated list of index names; use `_all` to check the types across all indices
|
|
* @param {String|ArrayOfStrings|Boolean} params.type - A comma-separated list of document types to check
|
|
*/
|
|
api.indices.prototype.existsType = ca({
|
|
methods: [
|
|
'HEAD'
|
|
],
|
|
params: {
|
|
ignoreIndices: {
|
|
type: 'enum',
|
|
'default': 'none',
|
|
options: [
|
|
'none',
|
|
'missing'
|
|
],
|
|
name: 'ignore_indices'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>/<%=type%>',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
},
|
|
type: {
|
|
type: 'list'
|
|
}
|
|
}
|
|
}
|
|
],
|
|
castExists: true
|
|
});
|
|
|
|
|
|
/**
|
|
* 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|ArrayOfStrings|Boolean} params.index - A comma-separated list of index names; use `_all` or empty string for all indices
|
|
*/
|
|
api.indices.prototype.flush = ca({
|
|
methods: [
|
|
'POST',
|
|
'GET'
|
|
],
|
|
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'
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* 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|ArrayOfStrings|Boolean} params.index - A comma-separated list of index names to filter aliases
|
|
* @param {String|ArrayOfStrings|Boolean} params.name - A comma-separated list of alias names to return
|
|
*/
|
|
api.indices.prototype.getAlias = ca({
|
|
methods: [
|
|
'GET'
|
|
],
|
|
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|Number} params.timeout - Explicit operation timeout
|
|
* @param {String|ArrayOfStrings|Boolean} params.index - A comma-separated list of index names to filter aliases
|
|
*/
|
|
api.indices.prototype.getAliases = ca({
|
|
methods: [
|
|
'GET'
|
|
],
|
|
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|ArrayOfStrings|Boolean} params.index - A comma-separated list of index names
|
|
* @param {String|ArrayOfStrings|Boolean} params.type - A comma-separated list of document types
|
|
* @param {String|ArrayOfStrings|Boolean} params.field - A comma-separated list of fields
|
|
*/
|
|
api.indices.prototype.getFieldMapping = ca({
|
|
methods: [
|
|
'GET'
|
|
],
|
|
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|ArrayOfStrings|Boolean} params.index - A comma-separated list of index names
|
|
* @param {String|ArrayOfStrings|Boolean} params.type - A comma-separated list of document types
|
|
*/
|
|
api.indices.prototype.getMapping = ca({
|
|
methods: [
|
|
'GET'
|
|
],
|
|
params: {},
|
|
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|ArrayOfStrings|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({
|
|
methods: [
|
|
'GET'
|
|
],
|
|
params: {},
|
|
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({
|
|
methods: [
|
|
'GET'
|
|
],
|
|
params: {},
|
|
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|ArrayOfStrings|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|ArrayOfStrings|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({
|
|
methods: [
|
|
'GET'
|
|
],
|
|
params: {},
|
|
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|Number} params.timeout - Explicit operation timeout
|
|
* @param {Date|Number} params.masterTimeout - Specify timeout for connection to master
|
|
* @param {String} params.index - The name of the index
|
|
*/
|
|
api.indices.prototype.open = ca({
|
|
methods: [
|
|
'POST'
|
|
],
|
|
params: {
|
|
timeout: {
|
|
type: 'time'
|
|
},
|
|
masterTimeout: {
|
|
type: 'time',
|
|
name: 'master_timeout'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>/_open',
|
|
req: {
|
|
index: {
|
|
type: 'string'
|
|
}
|
|
}
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* 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|ArrayOfStrings|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({
|
|
methods: [
|
|
'POST',
|
|
'GET'
|
|
],
|
|
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'
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* 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|Number} params.timeout - Explicit timestamp for the document
|
|
* @param {Date|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({
|
|
methods: [
|
|
'PUT'
|
|
],
|
|
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'
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* 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|Number} params.timeout - Explicit operation timeout
|
|
* @param {Date|Number} params.masterTimeout - Specify timeout for connection to master
|
|
* @param {String|ArrayOfStrings|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({
|
|
methods: [
|
|
'PUT',
|
|
'POST'
|
|
],
|
|
params: {
|
|
ignoreConflicts: {
|
|
type: 'boolean',
|
|
name: 'ignore_conflicts'
|
|
},
|
|
timeout: {
|
|
type: 'time'
|
|
},
|
|
masterTimeout: {
|
|
type: 'time',
|
|
name: 'master_timeout'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>/<%=type%>/_mapping',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
},
|
|
type: {
|
|
type: 'string'
|
|
}
|
|
}
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* 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|Number} params.masterTimeout - Specify timeout for connection to master
|
|
* @param {String|ArrayOfStrings|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({
|
|
methods: [
|
|
'PUT'
|
|
],
|
|
params: {
|
|
masterTimeout: {
|
|
type: 'time',
|
|
name: 'master_timeout'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>/_settings',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/_settings'
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* 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|Number} params.timeout - Explicit operation timeout
|
|
* @param {Date|Number} params.masterTimeout - Specify timeout for connection to master
|
|
* @param {String} params.name - The name of the template
|
|
*/
|
|
api.indices.prototype.putTemplate = ca({
|
|
methods: [
|
|
'PUT',
|
|
'POST'
|
|
],
|
|
params: {
|
|
order: {
|
|
type: 'number'
|
|
},
|
|
timeout: {
|
|
type: 'time'
|
|
},
|
|
masterTimeout: {
|
|
type: 'time',
|
|
name: 'master_timeout'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/_template/<%=name%>',
|
|
req: {
|
|
name: {
|
|
type: 'string'
|
|
}
|
|
}
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* 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|Number} params.masterTimeout - Specify timeout for connection to master
|
|
* @param {String|ArrayOfStrings|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|ArrayOfStrings|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({
|
|
methods: [
|
|
'PUT'
|
|
],
|
|
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'
|
|
}
|
|
}
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* 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|ArrayOfStrings|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({
|
|
methods: [
|
|
'POST',
|
|
'GET'
|
|
],
|
|
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'
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* 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|ArrayOfStrings|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({
|
|
methods: [
|
|
'GET'
|
|
],
|
|
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|ArrayOfStrings|Boolean} params.index - A comma-separated list of index names; use `_all` or empty string for all indices
|
|
*/
|
|
api.indices.prototype.snapshotIndex = ca({
|
|
methods: [
|
|
'POST'
|
|
],
|
|
params: {
|
|
ignoreIndices: {
|
|
type: 'enum',
|
|
'default': 'none',
|
|
options: [
|
|
'none',
|
|
'missing'
|
|
],
|
|
name: 'ignore_indices'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>/_gateway/snapshot',
|
|
req: {
|
|
index: {
|
|
type: 'list'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/_gateway/snapshot'
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* 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|ArrayOfStrings|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|ArrayOfStrings|Boolean} params.fielddataFields - A comma-separated list of fields for `fielddata` metric (supports wildcards)
|
|
* @param {String|ArrayOfStrings|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|ArrayOfStrings|Boolean} params.index - A comma-separated list of index names; use `_all` or empty string to perform the operation on all indices
|
|
* @param {String|ArrayOfStrings|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|ArrayOfStrings|Boolean} params.searchGroups - A comma-separated list of search groups to include in the `search` statistics
|
|
*/
|
|
api.indices.prototype.stats = ca({
|
|
methods: [
|
|
'GET'
|
|
],
|
|
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|ArrayOfStrings|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({
|
|
methods: [
|
|
'GET'
|
|
],
|
|
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|Number} params.timeout - Request timeout
|
|
* @param {Date|Number} params.masterTimeout - Specify timeout for connection to master
|
|
* @param {String|ArrayOfStrings|Boolean} params.index - A comma-separated list of index names to filter aliases
|
|
*/
|
|
api.indices.prototype.updateAliases = ca({
|
|
methods: [
|
|
'POST'
|
|
],
|
|
params: {
|
|
timeout: {
|
|
type: 'time'
|
|
},
|
|
masterTimeout: {
|
|
type: 'time',
|
|
name: 'master_timeout'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/_aliases'
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* 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|ArrayOfStrings|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|ArrayOfStrings|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({
|
|
methods: [
|
|
'GET',
|
|
'POST'
|
|
],
|
|
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'
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* Perform a [info](http://elasticsearch.org/guide/) request
|
|
*
|
|
* @param {Object} params - An object with parameters used to carry out this action
|
|
*/
|
|
api.info = ca({
|
|
methods: [
|
|
'GET',
|
|
'HEAD'
|
|
],
|
|
params: {},
|
|
urls: [
|
|
{
|
|
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|ArrayOfStrings|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|ArrayOfStrings|Boolean} params.source - True or false to return the _source field or not, or a list of fields to return
|
|
* @param {String|ArrayOfStrings|Boolean} params.sourceExclude - A list of fields to exclude from the returned _source field
|
|
* @param {String|ArrayOfStrings|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({
|
|
methods: [
|
|
'GET',
|
|
'POST'
|
|
],
|
|
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'
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* 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|ArrayOfStrings|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|ArrayOfStrings|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|ArrayOfStrings|Boolean} params.searchTypes - A comma-separated list of types to perform the query against (default: the same type as the document)
|
|
* @param {String|ArrayOfStrings|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({
|
|
methods: [
|
|
'GET',
|
|
'POST'
|
|
],
|
|
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'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>/<%=type%>/<%=id%>/_mlt',
|
|
req: {
|
|
index: {
|
|
type: 'string'
|
|
},
|
|
type: {
|
|
type: 'string'
|
|
},
|
|
id: {
|
|
type: 'string'
|
|
}
|
|
}
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* 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|ArrayOfStrings|Boolean} params.index - A comma-separated list of index names to use as default
|
|
* @param {String|ArrayOfStrings|Boolean} params.type - A comma-separated list of document types to use as default
|
|
*/
|
|
api.msearch = ca({
|
|
methods: [
|
|
'GET',
|
|
'POST'
|
|
],
|
|
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
|
|
});
|
|
|
|
|
|
/**
|
|
* 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({
|
|
methods: [
|
|
'GET',
|
|
'POST'
|
|
],
|
|
params: {
|
|
preferLocal: {
|
|
type: 'boolean',
|
|
name: 'prefer_local'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>/<%=type%>/_percolate',
|
|
req: {
|
|
index: {
|
|
type: 'string'
|
|
},
|
|
type: {
|
|
type: 'string'
|
|
}
|
|
}
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* 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({
|
|
methods: [
|
|
'GET',
|
|
'POST'
|
|
],
|
|
params: {
|
|
scroll: {
|
|
type: 'duration'
|
|
},
|
|
scrollId: {
|
|
type: 'string',
|
|
name: 'scroll_id'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/_search/scroll/<%=scrollId%>',
|
|
req: {
|
|
scrollId: {
|
|
type: 'string'
|
|
}
|
|
}
|
|
},
|
|
{
|
|
fmt: '/_search/scroll'
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* 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|ArrayOfStrings|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|ArrayOfStrings|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|ArrayOfStrings|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|ArrayOfStrings|Boolean} params.sort - A comma-separated list of <field>:<direction> pairs
|
|
* @param {String|ArrayOfStrings|Boolean} params.source - True or false to return the _source field or not, or a list of fields to return
|
|
* @param {String|ArrayOfStrings|Boolean} params.sourceExclude - A list of fields to exclude from the returned _source field
|
|
* @param {String|ArrayOfStrings|Boolean} params.sourceInclude - A list of fields to extract and return from the _source field
|
|
* @param {String|ArrayOfStrings|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|Number} params.timeout - Explicit operation timeout
|
|
* @param {Boolean} params.version - Specify whether to return document version as part of a hit
|
|
* @param {String|ArrayOfStrings|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|ArrayOfStrings|Boolean} params.type - A comma-separated list of document types to search; leave empty to perform the operation on all types
|
|
*/
|
|
api.search = ca({
|
|
methods: [
|
|
'GET',
|
|
'POST'
|
|
],
|
|
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'
|
|
}
|
|
}
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* 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|ArrayOfStrings|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({
|
|
methods: [
|
|
'POST',
|
|
'GET'
|
|
],
|
|
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'
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
/**
|
|
* 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|ArrayOfStrings|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|Number} params.timeout - Explicit operation timeout
|
|
* @param {Date|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({
|
|
methods: [
|
|
'POST'
|
|
],
|
|
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'
|
|
}
|
|
},
|
|
urls: [
|
|
{
|
|
fmt: '/<%=index%>/<%=type%>/<%=id%>/_update',
|
|
req: {
|
|
index: {
|
|
type: 'string'
|
|
},
|
|
type: {
|
|
type: 'string'
|
|
},
|
|
id: {
|
|
type: 'string'
|
|
}
|
|
}
|
|
}
|
|
]
|
|
});
|
|
|
|
|
|
},{"./client_action":131,"./errors":139}],130:[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
|
|
* @param {Object} [config={}] - Configuration for the transport
|
|
* @param {Object} [config.transport] - Transport settings passed to {{#crossLink "Transport"}}Transport Constructor{{/crossLink}}
|
|
* @param {String|Array<String>} [config.log] - Log output settings {{#crossLink "Log"}}Log Constructor{{/crossLink}}
|
|
* @param {Object} [config.trace=false] - Create a log output to stdio that only tracks trace logs
|
|
*/
|
|
|
|
module.exports = Client;
|
|
|
|
var _ = require('./utils');
|
|
var ClientConfig = require('./client_config');
|
|
var api = require('./api.js');
|
|
|
|
function Client(config) {
|
|
this.client = this;
|
|
|
|
// setup the config.. this config will be passed EVERYWHERE so for good measure it is locked down
|
|
Object.defineProperty(this, 'config', {
|
|
configurable: false,
|
|
enumerable: false,
|
|
writable: false,
|
|
value: !config || _.isPlainObject(config) ? new ClientConfig(config) : config,
|
|
});
|
|
this.config.client = this;
|
|
|
|
// instansiate the api's namespaces
|
|
for (var i = 0; i < this._namespaces.length; i++) {
|
|
this[this._namespaces[i]] = new this[this._namespaces[i]](this);
|
|
}
|
|
}
|
|
|
|
Client.prototype = _.clone(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 = function (params, cb) {
|
|
if (typeof params === 'function') {
|
|
cb = params;
|
|
params = {};
|
|
}
|
|
|
|
this.config.transport.request({
|
|
method: 'HEAD',
|
|
path: '/'
|
|
}, cb);
|
|
};
|
|
|
|
Client.prototype.close = function () {
|
|
this.config.close();
|
|
};
|
|
|
|
},{"./api.js":129,"./client_config":132,"./utils":153}],131:[function(require,module,exports){
|
|
/**
|
|
* Constructs a function that can be called to make a request to ES
|
|
* @type {[type]}
|
|
*/
|
|
module.exports = function ClientAction(spec, client) {
|
|
return function (params, cb) {
|
|
return exec((client || this.client).config.transport, spec, params, cb);
|
|
};
|
|
};
|
|
|
|
var errors = require('./errors');
|
|
var _ = require('./utils');
|
|
var urlParamRE = /\{(\w+)\}/g;
|
|
|
|
var castType = {
|
|
enum: function (param, val, name) {
|
|
if (_.contains(param.options, val)) {
|
|
return val;
|
|
} else {
|
|
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 'string':
|
|
return val;
|
|
case 'object':
|
|
if (_.isArray(val)) {
|
|
return val.join(',');
|
|
} else {
|
|
throw new TypeError('Invalid ' + name + ': expected be a comma seperated list, array, or boolean.');
|
|
}
|
|
break;
|
|
default:
|
|
return !!val;
|
|
}
|
|
},
|
|
boolean: function (param, val, name) {
|
|
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) {
|
|
if (typeof val !== 'object' && val) {
|
|
return '' + val;
|
|
} else {
|
|
throw new TypeError('Invalid ' + name + ': expected a string.');
|
|
}
|
|
},
|
|
time: function (param, val, name) {
|
|
if (typeof val === 'string' || _.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 {
|
|
// copy param vals into vars
|
|
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];
|
|
}, {}));
|
|
}
|
|
|
|
function exec(transport, spec, params, cb) {
|
|
if (typeof params === 'function') {
|
|
cb = params;
|
|
params = {};
|
|
} else {
|
|
params = params || {};
|
|
cb = typeof cb === 'function' ? cb : _.noop;
|
|
}
|
|
|
|
var request = {};
|
|
var parts = {};
|
|
var query = {};
|
|
var i;
|
|
|
|
if (spec.needsBody && !params.body) {
|
|
return _.nextTick(cb, new TypeError('A request body is required.'));
|
|
}
|
|
|
|
params.body && (request.body = params.body);
|
|
params.ignore && (request.ignore = _.isArray(params.ignore) ? params.ignore : [params.ignore]);
|
|
if (params.timeout === void 0) {
|
|
request.timeout = 10000;
|
|
} else {
|
|
request.timeout = params.timeout;
|
|
}
|
|
|
|
// copy over some properties from the spec
|
|
spec.bulkBody && (request.bulkBody = true);
|
|
spec.castExists && (request.castExists = true);
|
|
|
|
if (spec.methods.length === 1) {
|
|
request.method = spec.methods[0];
|
|
} else {
|
|
// if set, uppercase the user's choice, other wise returns ""
|
|
request.method = _.toUpperString(params.method);
|
|
|
|
if (request.method) {
|
|
// use the one specified as long as it's a valid option
|
|
if (!_.contains(spec.methods, request.method)) {
|
|
return _.nextTick(cb, new TypeError('Invalid method: should be one of ' + spec.methods.join(', ')));
|
|
}
|
|
} else {
|
|
// pick a method
|
|
if (request.body) {
|
|
// first method that isn't "GET"
|
|
request.method = spec.methodWithBody || (
|
|
spec.methodWithBody = _.find(spec.methods, function (m) { return m !== 'GET'; })
|
|
);
|
|
} else {
|
|
// just use the first option
|
|
request.method = spec.methods[0];
|
|
}
|
|
}
|
|
}
|
|
|
|
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
|
|
return _.nextTick(
|
|
cb,
|
|
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);
|
|
}
|
|
var key, param, name;
|
|
for (i = 0; i < spec.paramKeys.length; i++) {
|
|
key = spec.paramKeys[i];
|
|
param = spec.params[key];
|
|
// param keys don't always match the param name, in those cases it's stored in the param def as "name"
|
|
name = param.name || key;
|
|
try {
|
|
if (params[key] != null) {
|
|
query[name] = castType[param.type] ? castType[param.type](param, params[key], key) : params[key];
|
|
if (param['default'] && query[name] === param['default']) {
|
|
delete query[name];
|
|
}
|
|
} else if (param.required) {
|
|
throw new TypeError('Missing required parameter ' + key);
|
|
}
|
|
} catch (e) {
|
|
return _.nextTick(cb, e);
|
|
}
|
|
}
|
|
|
|
request.query = query;
|
|
|
|
return transport.request(request, cb);
|
|
}
|
|
|
|
},{"./errors":139,"./utils":153}],132:[function(require,module,exports){
|
|
var process=require("__browserify_process");/**
|
|
* Manages the configuration of the client.
|
|
*
|
|
* @class ClientConfig
|
|
* @type {Function}
|
|
*/
|
|
module.exports = ClientConfig;
|
|
|
|
var url = require('url');
|
|
var _ = require('./utils');
|
|
var Host = require('./host');
|
|
var selectors = require('./selectors');
|
|
|
|
var connectors = {};
|
|
if (process.browser) {
|
|
connectors.Xhr = require('./connectors/xhr');
|
|
connectors.Angular = require('./connectors/angular');
|
|
connectors.jQuery = require('./connectors/jquery');
|
|
} else {
|
|
connectors.Http = require('./connectors/http');
|
|
}
|
|
|
|
_.each(connectors, function (conn, name) {
|
|
if (typeof conn !== 'function') {
|
|
delete connectors[name];
|
|
}
|
|
});
|
|
|
|
var serializers = {
|
|
Json: require('./serializers/json')
|
|
};
|
|
|
|
var extractHostPartsRE = /\[([^:]+):(\d+)]/;
|
|
var hostProtocolRE = /^([a-z]+:)?\/\//;
|
|
|
|
var defaultClasses = {
|
|
log: require('./log'),
|
|
serializer: serializers.Json,
|
|
connectionPool: require('./connection_pool'),
|
|
transport: require('./transport'),
|
|
};
|
|
|
|
var defaultConfig = {
|
|
loggers: [
|
|
{
|
|
level: 'warning'
|
|
}
|
|
],
|
|
hosts: [
|
|
{
|
|
host: 'localhost',
|
|
port: 9200,
|
|
protocol: 'http'
|
|
}
|
|
],
|
|
connectionClass: process.browser ? connectors.Xhr : connectors.Http,
|
|
selector: selectors.roundRobin,
|
|
sniffOnStart: false,
|
|
sniffAfterRequests: null,
|
|
sniffOnConnectionFail: false,
|
|
maxRetries: 3,
|
|
timeout: 10000,
|
|
deadTimeout: 60000,
|
|
maxSockets: 10,
|
|
nodesToHostCallback: function (nodes) {
|
|
var hosts = [];
|
|
_.each(nodes, function (node, id) {
|
|
var hostnameMatches = extractHostPartsRE.exec(node.host);
|
|
hosts.push({
|
|
host: hostnameMatches[1],
|
|
port: hostnameMatches[2],
|
|
_meta: {
|
|
id: id,
|
|
name: node.name,
|
|
servername: node.host,
|
|
version: node.version
|
|
}
|
|
});
|
|
});
|
|
return hosts;
|
|
}
|
|
};
|
|
|
|
// remove connector classes that were not included in the build
|
|
connectors = _.transform(connectors, function (note, connector, name) {
|
|
if (connector) {
|
|
note[name] = connector;
|
|
}
|
|
}, {});
|
|
|
|
function ClientConfig(config) {
|
|
_.extend(this, defaultConfig, config);
|
|
|
|
if (this.log) {
|
|
// treat log as an alias for loggers in the config.
|
|
this.loggers = this.log;
|
|
delete this.log;
|
|
}
|
|
|
|
// validate connectionClass
|
|
if (typeof this.connectionClass === 'string') {
|
|
this.connectionClass = connectors[_.studlyCase(this.connectionClass)];
|
|
}
|
|
if (typeof this.connectionClass !== 'function') {
|
|
throw new TypeError('Invalid connectionClass "' + this.connectionClass + '". ' +
|
|
'Expected a constructor or one of ' + _.keys(connectors).join(', '));
|
|
}
|
|
|
|
// validate selector
|
|
if (typeof this.selector === 'string') {
|
|
this.selector = selectors[_.camelCase(this.selector)];
|
|
}
|
|
if (typeof this.selector !== 'function') {
|
|
throw new TypeError('Invalid Selector "' + this.selector + '". ' +
|
|
'Expected a function or one of ' + _.keys(selectors).join(', '));
|
|
}
|
|
|
|
_.each(defaultClasses, function (DefaultClass, prop) {
|
|
this[prop] = typeof this[prop] === 'function' ? new this[prop](this) : new DefaultClass(this);
|
|
}, this);
|
|
|
|
// populate the connection pool
|
|
this.connectionPool.setNodes(this.prepareHosts(this.hosts));
|
|
|
|
// nodes are completely managed by the connection pool, remove traces of the config
|
|
// value to prevent confusion
|
|
delete this.hosts;
|
|
}
|
|
|
|
ClientConfig.prototype.prepareHosts = function (hosts) {
|
|
var host;
|
|
var i;
|
|
|
|
if (!_.isArray(hosts)) {
|
|
hosts = [hosts];
|
|
}
|
|
|
|
return _.map(hosts, function (host) {
|
|
return new Host(host);
|
|
});
|
|
};
|
|
|
|
/**
|
|
* Shutdown the connectionPool, log outputs, and clear timers
|
|
*/
|
|
ClientConfig.prototype.close = function () {
|
|
this.log.close();
|
|
this.connectionPool.close();
|
|
};
|
|
|
|
},{"./connection_pool":134,"./connectors/angular":135,"./connectors/http":136,"./connectors/jquery":137,"./connectors/xhr":138,"./host":140,"./log":141,"./selectors":147,"./serializers/json":150,"./transport":151,"./utils":153,"__browserify_process":91,"url":67}],133:[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) {
|
|
EventEmitter.call(this);
|
|
this.config = config;
|
|
this.host = host;
|
|
this.requestCount = 0;
|
|
|
|
if (!this.host) {
|
|
throw new Error('Missing host config');
|
|
}
|
|
|
|
_.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 (params, cb) {
|
|
if (typeof params === 'function') {
|
|
cb = params;
|
|
} else 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 (status === 'dead' || status === 'closed') {
|
|
if (this.__deadTimeout) {
|
|
clearTimeout(this.__deadTimeout);
|
|
}
|
|
if (status === 'dead') {
|
|
this.__deadTimeout = setTimeout(this.bound.resuscitate, this.config.deadTimeout);
|
|
}
|
|
}
|
|
|
|
this.emit('status changed', status, origStatus, this);
|
|
};
|
|
|
|
ConnectionAbstract.prototype.resuscitate = _.scheduled(function () {
|
|
var self = this;
|
|
|
|
if (self.status === 'dead') {
|
|
self.ping(function (err) {
|
|
if (!err) {
|
|
self.setStatus('alive');
|
|
} else {
|
|
self.emit('dead');
|
|
}
|
|
});
|
|
}
|
|
});
|
|
|
|
},{"./utils":153,"events":58}],134:[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
|
|
* @param {Client} client - The client this pool belongs to
|
|
*/
|
|
|
|
module.exports = ConnectionPool;
|
|
|
|
var _ = require('./utils');
|
|
var selectors = require('./selectors');
|
|
var EventEmitter = require('events').EventEmitter;
|
|
var errors = require('./errors');
|
|
var Host = require('./host');
|
|
|
|
function ConnectionPool(config) {
|
|
_.makeBoundMethods(this);
|
|
this.config = config;
|
|
this.index = {};
|
|
this.connections = {
|
|
alive: [],
|
|
dead: []
|
|
};
|
|
}
|
|
|
|
ConnectionPool.prototype.select = function (cb) {
|
|
if (this.connections.alive.length) {
|
|
if (this.config.selector.length > 1) {
|
|
this.config.selector(this.connections.alive, cb);
|
|
} else {
|
|
try {
|
|
_.nextTick(cb, null, this.config.selector(this.connections.alive));
|
|
} catch (e) {
|
|
this.config.log.error(e);
|
|
cb(e);
|
|
}
|
|
}
|
|
} else {
|
|
cb();
|
|
}
|
|
};
|
|
|
|
ConnectionPool.prototype.onStatusChanged = _.handler(function (status, oldStatus, connection) {
|
|
var from, to, index;
|
|
|
|
if (oldStatus === status) {
|
|
return true;
|
|
} else {
|
|
this.config.log.info('connection id:', connection.__id, 'is', status);
|
|
}
|
|
|
|
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 '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._add = function (connection) {
|
|
if (!this.index[connection.__id]) {
|
|
this.index[connection.__id] = connection;
|
|
connection.on('status changed', this.bound.onStatusChanged);
|
|
connection.setStatus('alive');
|
|
}
|
|
};
|
|
|
|
ConnectionPool.prototype._remove = function (connection) {
|
|
if (this.index[connection.__id]) {
|
|
delete this.index[connection.__id];
|
|
connection.setStatus('closed');
|
|
connection.removeListener('status changed', this.bound.onStatusChanged);
|
|
}
|
|
};
|
|
|
|
ConnectionPool.prototype.setNodes = function (nodeConfigs) {
|
|
var connection;
|
|
var i;
|
|
var id;
|
|
var node;
|
|
var toRemove = _.clone(this.index);
|
|
|
|
for (i = 0; i < nodeConfigs.length; i++) {
|
|
node = nodeConfigs[i];
|
|
if (node instanceof Host) {
|
|
id = node.toString();
|
|
if (this.index[id]) {
|
|
delete toRemove[id];
|
|
} else {
|
|
connection = new this.config.connectionClass(node, this.config);
|
|
connection.__id = id;
|
|
this._add(connection);
|
|
}
|
|
}
|
|
}
|
|
|
|
_.each(toRemove, this._remove, this);
|
|
};
|
|
|
|
ConnectionPool.prototype.close = function () {
|
|
this.setNodes([]);
|
|
};
|
|
ConnectionPool.prototype.empty = ConnectionPool.prototype.close;
|
|
|
|
},{"./errors":139,"./host":140,"./selectors":147,"./utils":153,"events":58}],135:[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;
|
|
|
|
/* global angular */
|
|
|
|
function AngularConnector(host, config) {
|
|
ConnectionAbstract.call(this, host, config);
|
|
}
|
|
_.inherits(AngularConnector, ConnectionAbstract);
|
|
|
|
AngularConnector.prototype.request = function (params, cb) {
|
|
var timeoutId;
|
|
|
|
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":133,"../errors":139,"../utils":153}],136:[function(require,module,exports){
|
|
/**
|
|
* A Connection that operates using Node's http module
|
|
*
|
|
* @param client {Client} - The Client that this class belongs to
|
|
* @param config {Object} - Configuration options
|
|
* @param [config.protocol=http:] {String} - The HTTP protocol that this connection will use, can be set to https:
|
|
* @class HttpConnector
|
|
*/
|
|
module.exports = HttpConnector;
|
|
|
|
var http = require('http');
|
|
var https = require('https');
|
|
var _ = require('../utils');
|
|
var errors = require('../errors');
|
|
var qs = require('querystring');
|
|
var KeepAliveAgent = require('agentkeepalive/lib/agent');
|
|
var ConnectionAbstract = require('../connection');
|
|
var defaultHeaders = {
|
|
'connection': 'keep-alive'
|
|
};
|
|
|
|
|
|
function HttpConnector(host, config) {
|
|
ConnectionAbstract.call(this, host, config);
|
|
|
|
this.hand = require(this.host.protocol);
|
|
this.agent = new KeepAliveAgent({
|
|
maxSockets: 1,
|
|
maxKeepAliveRequests: 0, // max requests per keepalive socket, default is 0, no limit.
|
|
maxKeepAliveTime: 30000 // keepalive for 30 seconds
|
|
});
|
|
|
|
this.on('closed', this.bound.onClosed);
|
|
this.on('alive', this.bound.onAlive);
|
|
}
|
|
_.inherits(HttpConnector, ConnectionAbstract);
|
|
|
|
HttpConnector.prototype.onClosed = _.handler(function () {
|
|
this.agent.destroy();
|
|
this.removeAllListeners();
|
|
});
|
|
|
|
HttpConnector.prototype.onAlive = _.handler(function () {
|
|
// only set the agents max agents config once the connection is verified to be alive
|
|
this.agent.maxSockets = this.config.maxSockets;
|
|
});
|
|
|
|
HttpConnector.prototype.makeReqParams = function (params) {
|
|
var reqParams = {
|
|
method: params.method,
|
|
protocol: this.host.protocol + ':',
|
|
auth: this.host.auth,
|
|
hostname: this.host.host,
|
|
port: this.host.port,
|
|
path: this.host.path + params.path,
|
|
headers: this.host.headers,
|
|
agent: this.agent
|
|
};
|
|
var query = this.host.query ? this.host.query : null;
|
|
var queryStr;
|
|
if (typeof query === 'string') {
|
|
query = qs.parse(query);
|
|
}
|
|
|
|
if (params.query) {
|
|
query = _.defaults({},
|
|
typeof params.query === 'string' ? qs.parse(params.query) : params.query,
|
|
query || {}
|
|
);
|
|
}
|
|
|
|
if (query) {
|
|
queryStr = qs.stringify(query);
|
|
}
|
|
|
|
if (queryStr) {
|
|
reqParams.path = reqParams.path + '?' + queryStr;
|
|
}
|
|
|
|
return reqParams;
|
|
};
|
|
|
|
HttpConnector.prototype.request = function (params, cb) {
|
|
var incoming;
|
|
var timeoutId;
|
|
var request;
|
|
var requestId = this.requestCount;
|
|
var response;
|
|
var responseStarted = false;
|
|
var status = 0;
|
|
var timeout = params.timeout || this.config.timeout;
|
|
var log = this.config.log;
|
|
|
|
var reqParams = this.makeReqParams(params);
|
|
|
|
// general clean-up procedure to run after the request
|
|
// completes, has an error, or is aborted.
|
|
var cleanUp = _.bind(function (err) {
|
|
clearTimeout(timeoutId);
|
|
|
|
request && request.removeAllListeners();
|
|
incoming && incoming.removeAllListeners();
|
|
|
|
if ((err instanceof Error) === false) {
|
|
err = void 0;
|
|
} else {
|
|
log.error(err);
|
|
this.setStatus('dead');
|
|
}
|
|
|
|
log.trace(params.method, reqParams, params.body, response, status);
|
|
cb(err, response, status);
|
|
}, this);
|
|
|
|
request = this.hand.request(reqParams, function (_incoming) {
|
|
incoming = _incoming;
|
|
status = incoming.statusCode;
|
|
incoming.setEncoding('utf8');
|
|
response = '';
|
|
|
|
incoming.on('data', function (d) {
|
|
response += d;
|
|
});
|
|
|
|
incoming.on('error', cleanUp);
|
|
incoming.on('end', cleanUp);
|
|
});
|
|
|
|
request.on('error', cleanUp);
|
|
|
|
if (timeout !== Infinity) {
|
|
// timeout for the entire request.
|
|
timeoutId = setTimeout(function () {
|
|
request.abort();
|
|
request.emit('error', new errors.RequestTimeout('Request timed out at ' + timeout + 'ms'));
|
|
}, timeout);
|
|
}
|
|
|
|
request.setNoDelay(true);
|
|
request.setSocketKeepAlive(true);
|
|
|
|
request.end(params.body);
|
|
this.requestCount++;
|
|
};
|
|
|
|
},{"../connection":133,"../errors":139,"../utils":153,"agentkeepalive/lib/agent":1,"http":72,"https":60,"querystring":62}],137:[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) {
|
|
var $xhr = jQuery.ajax(params).done(cb);
|
|
};
|
|
|
|
|
|
|
|
},{}],138:[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) {
|
|
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 (e) {
|
|
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":133,"../errors":139,"../utils":153}],139:[function(require,module,exports){
|
|
var process=require("__browserify_process");var _ = require('./utils'),
|
|
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);
|
|
}
|
|
}
|
|
_.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);
|
|
|
|
/**
|
|
* 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":153,"__browserify_process":91}],140:[function(require,module,exports){
|
|
/**
|
|
* Class to wrap URLS, formatting them and maintaining their seperate 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
|
|
};
|
|
|
|
function Host(config) {
|
|
if (this instanceof Host) {
|
|
if (typeof config === 'string') {
|
|
return Host.fromString(config);
|
|
} else {
|
|
_.extend(this, config || {});
|
|
}
|
|
} else {
|
|
return new Host(config);
|
|
}
|
|
}
|
|
|
|
Host.fromString = function (urlString) {
|
|
if (!startsWithProtocolRE.test(urlString)) {
|
|
urlString = 'http://' + urlString;
|
|
}
|
|
var u = url.parse(urlString, true, true);
|
|
return new Host({
|
|
protocol: u.protocol ? u.protocol.substring(0, u.protocol.length - 1) : 'http',
|
|
host: u.hostname || 'localhost',
|
|
port: u.port || 9200,
|
|
auth: u.auth || '',
|
|
path: u.pathname,
|
|
query: u.query,
|
|
});
|
|
};
|
|
|
|
Host.prototype = {
|
|
protocol: 'http',
|
|
host: 'localhost',
|
|
port: 9200,
|
|
auth: '',
|
|
path: '',
|
|
query: false
|
|
};
|
|
|
|
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 = '';
|
|
// add the path prefix if set
|
|
if (this.path) {
|
|
path += this.path;
|
|
}
|
|
// then the path from the params
|
|
if (params.path) {
|
|
path += params.path;
|
|
}
|
|
// if we still have a path, and it doesn't start with '/' add it.
|
|
if (path && 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 : '');
|
|
};
|
|
|
|
},{"./utils":153,"querystring":62,"url":67}],141:[function(require,module,exports){
|
|
var process=require("__browserify_process");var _ = require('./utils');
|
|
var url = require('url');
|
|
var EventEmitter = require('events').EventEmitter;
|
|
if (process.browser) {
|
|
var loggers = {
|
|
Console: require('./loggers/console')
|
|
};
|
|
} else {
|
|
var loggers = {
|
|
File: require('./loggers/file'),
|
|
Stream: require('./loggers/file'),
|
|
Stdio: require('./loggers/stdio')
|
|
};
|
|
}
|
|
|
|
/**
|
|
* 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) {
|
|
this.config = config || {};
|
|
|
|
var i;
|
|
var output = config.loggers ? config.loggers : 'warning';
|
|
|
|
if (_.isString(output) || _.isFinite(output)) {
|
|
output = [
|
|
{
|
|
level: output
|
|
}
|
|
];
|
|
} else if (_.isPlainObject(output)) {
|
|
output = [output];
|
|
} else if (_.isArray(output)) {
|
|
for (i = 0; i < output.length; i++) {
|
|
if (_.isString(output[i])) {
|
|
output[i] = {
|
|
level: output[i]
|
|
};
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!_.isArrayOfPlainObjects(output)) {
|
|
throw new TypeError('Invalid Logging output config');
|
|
}
|
|
|
|
for (i = 0; i < output.length; i++) {
|
|
this.addOutput(output[i]);
|
|
}
|
|
|
|
}
|
|
_.inherits(Log, EventEmitter);
|
|
|
|
|
|
Log.prototype.close = function () {
|
|
this.emit('closing');
|
|
if (EventEmitter.listenerCount(this)) {
|
|
console.error('Something is still listening for log events, but the logger is closing.');
|
|
this.clearAllListeners();
|
|
}
|
|
};
|
|
|
|
/**
|
|
* 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) {
|
|
if (_.isString(input)) {
|
|
return Log.levels.slice(0, _.indexOf(Log.levels, input) + 1);
|
|
}
|
|
else if (_.isArray(input)) {
|
|
return _.intersection(input, Log.levels);
|
|
}
|
|
};
|
|
|
|
/**
|
|
* 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) {
|
|
var levels = Log.parseLevels(config.levels || config.level || 'warning');
|
|
|
|
_.defaults(config || {}, {
|
|
type: process.browser ? 'Console' : 'Stdio',
|
|
});
|
|
|
|
// force the levels config
|
|
delete config.level;
|
|
config.levels = levels;
|
|
|
|
var Logger = loggers[_.studlyCase(config.type)];
|
|
if (Logger) {
|
|
return new Logger(config, this);
|
|
} else {
|
|
throw new Error('Invalid logger type "' + config.type + '". Expected one of ' + _.keys(loggers).join(', '));
|
|
}
|
|
};
|
|
|
|
/**
|
|
* 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 (EventEmitter.listenerCount(this, '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 (EventEmitter.listenerCount(this, '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 (EventEmitter.listenerCount(this, '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 (EventEmitter.listenerCount(this, '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 (EventEmitter.listenerCount(this, '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;
|
|
return this.emit('trace', method, url.format(requestUrl), body, responseBody, responseStatus);
|
|
}
|
|
};
|
|
|
|
module.exports = Log;
|
|
|
|
},{"./loggers/console":143,"./loggers/file":144,"./loggers/stdio":145,"./utils":153,"__browserify_process":91,"events":58,"url":67}],142:[function(require,module,exports){
|
|
var Log = require('./log'),
|
|
_ = require('./utils');
|
|
|
|
/**
|
|
* Abstract class providing common functionality to loggers
|
|
* @param {[type]} config [description]
|
|
* @param {[type]} bridge [description]
|
|
*/
|
|
function LoggerAbstract(config, bridge) {
|
|
|
|
this.bridge = bridge;
|
|
this.listeningLevels = [];
|
|
|
|
_.makeBoundMethods(this);
|
|
|
|
// when the bridge closes, remove our event listeners
|
|
this.bridge.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.bridge.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.bridge.removeListener(level, this.bound['on' + _.ucfirst(level)]);
|
|
}, this);
|
|
});
|
|
|
|
/**
|
|
* Handler for the bridges "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 bridges "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 bridges "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 bridges "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 bridges "trace" event
|
|
*
|
|
* @method onTrace
|
|
* @private
|
|
* @param {String} msg - The message to be logged
|
|
* @return {undefined}
|
|
*/
|
|
LoggerAbstract.prototype.onTrace = _.handler(function (method, url, body, responseBody, responseStatus) {
|
|
var message = 'curl "' + url.replace(/"/g, '\\"') + '" -X' + method.toUpperCase();
|
|
if (body) {
|
|
message += ' -d "' + body.replace(/"/g, '\\"') + '"';
|
|
}
|
|
message += '\n<- ' + responseStatus + '\n' + responseBody;
|
|
this.write('TRACE', message);
|
|
});
|
|
|
|
|
|
module.exports = LoggerAbstract;
|
|
|
|
},{"./log":141,"./utils":153}],143:[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(config, bridge) {
|
|
LoggerAbstract.call(this, config, bridge);
|
|
|
|
// 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.onWarning = this.onWarning;
|
|
this.bound.onInfo = this.onInfo;
|
|
this.bound.onDebug = this.onDebug;
|
|
|
|
// 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 = _.handler(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 = _.bindKey(console, console.warn ? 'warn' : 'log', 'WARNING');
|
|
|
|
/**
|
|
* Handler for the bridges "info" event
|
|
*
|
|
* @method onInfo
|
|
* @private
|
|
* @param {String} msg - The message to be logged
|
|
* @return {undefined}
|
|
*/
|
|
Console.prototype.onInfo = _.bindKey(console, console.info ? 'info' : 'log', 'INFO');
|
|
|
|
/**
|
|
* Handler for the bridges "debug" event
|
|
*
|
|
* @method onDebug
|
|
* @private
|
|
* @param {String} msg - The message to be logged
|
|
* @return {undefined}
|
|
*/
|
|
Console.prototype.onDebug = _.bindKey(console, console.debug ? 'debug' : 'log', 'DEBUG');
|
|
|
|
/**
|
|
* Handler for the bridges "trace" event
|
|
*
|
|
* @method onTrace
|
|
* @private
|
|
* @return {undefined}
|
|
*/
|
|
Console.prototype.onTrace = _.handler(function (method, url, body, responseBody, responseStatus) {
|
|
var message = 'curl "' + url.replace(/"/g, '\\"') + '" -X' + method.toUpperCase();
|
|
if (body) {
|
|
message += ' -d "' + body.replace(/"/g, '\\"') + '"';
|
|
}
|
|
message += '\n<- ' + responseStatus + '\n' + responseBody;
|
|
console.log('TRACE:\n' + message + '\n\n');
|
|
});
|
|
|
|
},{"../logger":142,"../utils":153}],144:[function(require,module,exports){
|
|
/**
|
|
* Logger that writes to a file
|
|
*
|
|
* @class Loggers.File
|
|
* @extends StreamLogger
|
|
* @constructor
|
|
* @param {Object} config - The configuration for the Logger (See LoggerAbstract for generic options)
|
|
* @param {String} config.path - The location to write
|
|
* @param {Log} bridge - The object that triggers logging events, which we will record
|
|
*/
|
|
|
|
module.exports = File;
|
|
|
|
var StreamLogger = require('./stream'),
|
|
_ = require('../utils'),
|
|
fs = require('fs');
|
|
|
|
function File(config, bridge) {
|
|
this.path = config.path;
|
|
|
|
config.stream = fs.createWriteStream(config.path, {
|
|
flags: 'a',
|
|
encoding: 'utf8'
|
|
});
|
|
|
|
File.callSuper(this, arguments);
|
|
}
|
|
_.inherits(File, StreamLogger);
|
|
|
|
File.prototype.onProcessExit = _.handler(function () {
|
|
// flush the write buffer to disk
|
|
var writeBuffer = this.stream._writableState.buffer;
|
|
var out = '';
|
|
if (writeBuffer) {
|
|
writeBuffer.forEach(function (buffered) {
|
|
out += buffered.chunk.toString();
|
|
});
|
|
fs.appendFileSync(this.path, out);
|
|
}
|
|
});
|
|
|
|
},{"../utils":153,"./stream":146,"fs":59}],145:[function(require,module,exports){
|
|
var process=require("__browserify_process");/**
|
|
* Special version of the Stream logger, which logs errors and warnings to stderr and all other
|
|
* levels to stdout.
|
|
*
|
|
* @class Loggers.Stdio
|
|
* @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 = Stdio;
|
|
|
|
var clc = require('cli-color'),
|
|
LoggerAbstract = require('../logger'),
|
|
_ = require('../utils');
|
|
|
|
function Stdio(config, bridge) {
|
|
Stdio.callSuper(this, arguments);
|
|
|
|
// config/state
|
|
this.color = _.has(config, 'color') ? !!config.color : true;
|
|
}
|
|
|
|
_.inherits(Stdio, LoggerAbstract);
|
|
|
|
/**
|
|
* Sends output to a stream, does some formatting first
|
|
*
|
|
* @method write
|
|
* @private
|
|
* @param {WritableStream} to - The stream that should receive this message
|
|
* @param {String} label - The text that should be used at the beginning the message
|
|
* @param {function} colorize - A function that receives a string and returned a colored version of it
|
|
* @param {*} what - The message to log
|
|
* @return {undefined}
|
|
*/
|
|
Stdio.prototype.write = function (to, label, colorize, message) {
|
|
if (this.color) {
|
|
label = colorize(label);
|
|
}
|
|
to.write(this.format(label, message));
|
|
};
|
|
|
|
/**
|
|
* Handler for the bridges "error" event
|
|
*
|
|
* @method onError
|
|
* @private
|
|
* @param {Error} e - The Error object to log
|
|
* @return {undefined}
|
|
*/
|
|
Stdio.prototype.onError = _.handler(function (e) {
|
|
this.write(process.stderr, e.name === 'Error' ? 'ERROR' : e.name, clc.red.bold, e.stack);
|
|
});
|
|
|
|
/**
|
|
* Handler for the bridges "warning" event
|
|
*
|
|
* @method onWarning
|
|
* @private
|
|
* @param {String} msg - The message to be logged
|
|
* @return {undefined}
|
|
*/
|
|
Stdio.prototype.onWarning = _.handler(function (msg) {
|
|
this.write(process.stderr, 'WARNING', clc.yellow.bold, msg);
|
|
});
|
|
|
|
/**
|
|
* Handler for the bridges "info" event
|
|
*
|
|
* @method onInfo
|
|
* @private
|
|
* @param {String} msg - The message to be logged
|
|
* @return {undefined}
|
|
*/
|
|
Stdio.prototype.onInfo = _.handler(function (msg) {
|
|
this.write(process.stdout, 'INFO', clc.cyan.bold, msg);
|
|
});
|
|
|
|
/**
|
|
* Handler for the bridges "debug" event
|
|
*
|
|
* @method onDebug
|
|
* @private
|
|
* @param {String} msg - The message to be logged
|
|
* @return {undefined}
|
|
*/
|
|
Stdio.prototype.onDebug = _.handler(function (msg) {
|
|
this.write(process.stdout, 'DEBUG', clc.magentaBright.bold, msg);
|
|
});
|
|
|
|
/**
|
|
* Handler for the bridges "trace" event
|
|
*
|
|
* @method onTrace
|
|
* @private
|
|
* @return {undefined}
|
|
*/
|
|
Stdio.prototype.onTrace = _.handler(function (method, url, body, responseBody, responseStatus) {
|
|
var message = 'curl "' + url.replace(/"/g, '\\"') + '" -X' + method.toUpperCase();
|
|
if (body) {
|
|
message += ' -d "' + body.replace(/"/g, '\\"') + '"';
|
|
}
|
|
message += '\n<- ';
|
|
if (this.color) {
|
|
if (responseStatus >= 200 && responseStatus < 300) {
|
|
message += clc.green.bold(responseStatus);
|
|
} else {
|
|
message += clc.red.bold(responseStatus);
|
|
}
|
|
} else {
|
|
message += responseStatus;
|
|
}
|
|
message += '\n' + responseBody;
|
|
|
|
this.write(process.stdout, 'TRACE', clc.cyanBright.bold, message);
|
|
});
|
|
|
|
},{"../logger":142,"../utils":153,"__browserify_process":91,"cli-color":5}],146:[function(require,module,exports){
|
|
var process=require("__browserify_process");/**
|
|
* Logger that writes to a file
|
|
*
|
|
* @class Loggers.File
|
|
* @extends LoggerAbstract
|
|
* @constructor
|
|
* @see LoggerAbstract
|
|
* @param {Object} config - The configuration for the Logger (See LoggerAbstract for generic options)
|
|
* @param {String} config.path - The location to write
|
|
* @param {Log} bridge - The object that triggers logging events, which we will record
|
|
*/
|
|
|
|
module.exports = Stream;
|
|
|
|
var LoggerAbstract = require('../logger'),
|
|
nodeStreams = require('stream'),
|
|
_ = require('../utils'),
|
|
fs = require('fs');
|
|
|
|
function Stream(config, bridge) {
|
|
Stream.callSuper(this, arguments);
|
|
_.makeBoundMethods(this);
|
|
|
|
if (config.stream instanceof nodeStreams.Writable) {
|
|
this.stream = config.stream;
|
|
} else {
|
|
throw new TypeError('Invalid stream, use an instance of stream.Writeable');
|
|
}
|
|
|
|
process.on('exit', this.bound.onProcessExit);
|
|
}
|
|
_.inherits(Stream, LoggerAbstract);
|
|
|
|
// flush the write buffer to stderr synchronously
|
|
Stream.prototype.onProcessExit = _.handler(function () {
|
|
var writeBuffer = this.stream._writableState.buffer;
|
|
if (writeBuffer && writeBuffer.length) {
|
|
console.error('Log stream did not get to finish writing. Flushing to stderr');
|
|
writeBuffer.forEach(function (buffered) {
|
|
console.error(buffered.chunk.toString());
|
|
});
|
|
}
|
|
});
|
|
|
|
Stream.prototype.write = function (label, message) {
|
|
this.stream.write(this.format(label, message), 'utf8');
|
|
};
|
|
|
|
Stream.prototype.close = function () {
|
|
this.stream.end();
|
|
};
|
|
|
|
},{"../logger":142,"../utils":153,"__browserify_process":91,"fs":59,"stream":63}],147:[function(require,module,exports){
|
|
module.exports = {
|
|
random: require('./random'),
|
|
roundRobin: require('./round_robin')
|
|
};
|
|
|
|
},{"./random":148,"./round_robin":149}],148:[function(require,module,exports){
|
|
module.exports = RandomSelect;
|
|
|
|
function RandomSelect(connections) {
|
|
return connections[Math.floor(Math.random() * connections.length)];
|
|
}
|
|
|
|
},{}],149:[function(require,module,exports){
|
|
/**
|
|
* Selects a connection the simplest way possible, Round Robin
|
|
*
|
|
* @class selector.roundRobin
|
|
* @constructor
|
|
* @type {Function}
|
|
*/
|
|
module.exports = RoundRobinSelect;
|
|
|
|
function RoundRobinSelect(connections) {
|
|
connections.unshift(connections.pop());
|
|
return connections[0];
|
|
}
|
|
|
|
},{}],150:[function(require,module,exports){
|
|
/**
|
|
* Simple JSON serializer
|
|
* @type {[type]}
|
|
*/
|
|
module.exports = Json;
|
|
|
|
var _ = require('../utils');
|
|
|
|
function Json(client) {
|
|
this.client = client;
|
|
}
|
|
|
|
Json.prototype.serialize = function (val, replacer, spaces) {
|
|
if (val == null) {
|
|
return null;
|
|
}
|
|
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) {
|
|
this.client.log.error(new Error('unable to parse', str));
|
|
return null;
|
|
}
|
|
} 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":153}],151:[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 TransportRequest = require('./transport_request');
|
|
var errors = require('./errors');
|
|
|
|
function Transport(config) {
|
|
this.config = config;
|
|
}
|
|
|
|
/**
|
|
* 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) {
|
|
return new TransportRequest(this.config, params, cb);
|
|
};
|
|
|
|
/**
|
|
* 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 config = this.config;
|
|
|
|
// 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 nodes = config.nodesToHostCallback(resp.nodes);
|
|
config.connectionPool.setNodes(nodes);
|
|
}
|
|
cb(err, resp);
|
|
});
|
|
};
|
|
|
|
},{"./errors":139,"./transport_request":152,"./utils":153}],152:[function(require,module,exports){
|
|
var process=require("__browserify_process");/**
|
|
* Constructs a function that can be called to make a request to ES
|
|
* @type {[type]}
|
|
*/
|
|
module.exports = TransportRequest;
|
|
|
|
var _ = require('./utils');
|
|
var EventEmitter = require('events').EventEmitter;
|
|
var errors = require('./errors');
|
|
|
|
function TransportRequest(config, params, cb) {
|
|
// setup event emitter
|
|
EventEmitter.call(this);
|
|
// copy cetain methods into the bound object
|
|
_.makeBoundMethods(this);
|
|
|
|
this._params = params;
|
|
this._log = config.log;
|
|
this._serializer = config.serializer;
|
|
this._connectionPool = config.connectionPool;
|
|
this._remainingRetries = config.maxRetries;
|
|
|
|
// in cb isn't a function make it one
|
|
if (typeof cb === 'function') {
|
|
this.once('done', cb);
|
|
}
|
|
|
|
this._startRequest();
|
|
}
|
|
_.inherits(TransportRequest, EventEmitter);
|
|
|
|
TransportRequest.prototype._startRequest = function () {
|
|
var params = this._params;
|
|
|
|
this._log.debug('starting request', params);
|
|
|
|
if (params.body && params.method === 'GET') {
|
|
process.nextTick(_.bindKey(this, 'respond', new TypeError('Body can not be sent with method "GET"')));
|
|
return;
|
|
}
|
|
|
|
// serialize the body
|
|
if (params.body) {
|
|
params.body = this._serializer[params.bulkBody ? 'bulkBody' : 'serialize'](params.body);
|
|
}
|
|
|
|
params.req = {
|
|
timeout: params.timeout,
|
|
path: params.path,
|
|
query: params.query,
|
|
method: params.method,
|
|
body: params.body,
|
|
};
|
|
|
|
this._connectionPool.select(this.bound._sendReqWithCon);
|
|
};
|
|
|
|
TransportRequest.prototype._sendReqWithCon = _.handler(function (err, con) {
|
|
if (err) {
|
|
this._respond(err);
|
|
} else if (con) {
|
|
this._connection = con;
|
|
this._log.info('Selected', con.status, 'Connection, making request');
|
|
this._request = con.request(this._params.req, this.bound._checkRespForFail);
|
|
} else {
|
|
this._log.warning('No living connections');
|
|
this._respond(new errors.ConnectionFault('No living connections.'));
|
|
}
|
|
});
|
|
|
|
TransportRequest.prototype._checkRespForFail = _.handler(function (err, body, status) {
|
|
if (err && this._remainingRetries) {
|
|
this._remainingRetries--;
|
|
this._log.error(err.message, '-- retrying');
|
|
this._connectionPool.select(this.bound._sendReqWithCon);
|
|
} else {
|
|
this._log.info('Request complete');
|
|
this._respond(err, body, status);
|
|
}
|
|
});
|
|
|
|
TransportRequest.prototype._respond = _.handler(function (err, body, status) {
|
|
if (this._response) {
|
|
throw new Error('Request responded twice');
|
|
}
|
|
|
|
var parsedBody;
|
|
var serializer = this._serializer;
|
|
|
|
// get ignore and ensure that it's an array
|
|
var ignore = this._params.ignore;
|
|
if (ignore && !_.isArray(ignore)) {
|
|
ignore = [ignore];
|
|
}
|
|
|
|
if (!err && body) {
|
|
parsedBody = serializer.unserialize(body);
|
|
if (parsedBody == null) {
|
|
err = new errors.Serialization();
|
|
}
|
|
}
|
|
|
|
if (!err) {
|
|
if ((status < 200 || status >= 300) && !_.contains(ignore, status)) {
|
|
if (errors[status]) {
|
|
err = new errors[status](parsedBody && parsedBody.error);
|
|
} else {
|
|
err = new errors.Generic('unknown error');
|
|
}
|
|
}
|
|
}
|
|
|
|
if (this._params.castExists) {
|
|
if (err && err instanceof errors.NotFound) {
|
|
parsedBody = false;
|
|
err = void 0;
|
|
} else {
|
|
parsedBody = !err;
|
|
}
|
|
}
|
|
|
|
this._error = err;
|
|
this._response = {
|
|
body: parsedBody,
|
|
status: status
|
|
};
|
|
|
|
this.emit('done', this._error, this._response.body, this._response.status);
|
|
});
|
|
|
|
TransportRequest.prototype.abort = function () {
|
|
this.aborted = true;
|
|
if (this.__request) {
|
|
this.__request.abort();
|
|
return true;
|
|
}
|
|
return false;
|
|
};
|
|
|
|
TransportRequest.prototype.then = function (callback, errback) {
|
|
if (this._error) {
|
|
errback(this._error);
|
|
} else if (this._response) {
|
|
callback(this._response);
|
|
} else {
|
|
this.once('done', _.bindKey(this, 'then', callback, errback));
|
|
}
|
|
};
|
|
|
|
},{"./errors":139,"./utils":153,"__browserify_process":91,"events":58}],153:[function(require,module,exports){
|
|
var process=require("__browserify_process");var path = require('path'),
|
|
_ = require('lodash'),
|
|
nodeUtils = require('util');
|
|
|
|
/**
|
|
* 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) {
|
|
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 && typeof prop === 'object') {
|
|
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
|
|
*
|
|
* @todo Tests
|
|
* @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 inWord = false;
|
|
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
|
|
*
|
|
* @todo Tests
|
|
* @method studlyCase
|
|
* @param {String} string
|
|
* @return {String}
|
|
*/
|
|
utils.studlyCase = adjustWordCase(true, true, '');
|
|
|
|
/**
|
|
* Transform a string into camelCase
|
|
*
|
|
* @todo Tests
|
|
* @method camelCase
|
|
* @param {String} string
|
|
* @return {String}
|
|
*/
|
|
utils.camelCase = adjustWordCase(false, true, '');
|
|
|
|
/**
|
|
* Transform a string into snakeCase
|
|
*
|
|
* @todo Tests
|
|
* @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
|
|
*
|
|
* @todo Tests
|
|
* @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
|
|
*
|
|
* @todo Tests
|
|
* @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
|
|
*
|
|
* @todo Tests
|
|
* @method isNumeric
|
|
* @param {*} val
|
|
* @return {Boolean}
|
|
*/
|
|
utils.isNumeric = function (val) {
|
|
return !isNaN(val === null ? NaN : val * 1);
|
|
};
|
|
|
|
// regexp to test for intervals
|
|
var intervalRE = /^(\d+(?:\.\d+)?)([Mwdhmsy])$/;
|
|
|
|
/**
|
|
* Test if a string represents an interval (eg. 1m, 2Y)
|
|
*
|
|
* @todo Test
|
|
* @method isInterval
|
|
* @param {String} val
|
|
* @return {Boolean}
|
|
*/
|
|
utils.isInterval = function (val) {
|
|
return !!(val.match && val.match(intervalRE));
|
|
};
|
|
|
|
/**
|
|
* Repeat a string n times
|
|
*
|
|
* @todo Test
|
|
* @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 to also supply a callSuper function on the child class that can be called
|
|
* with the instance and the arguments passed to the child's constructor. This should only be called from within the
|
|
* constructor of the child class and should be removed from the code once the constructor is "done".
|
|
*
|
|
* @param constructor {Function} - the constructor that should subClass superConstructor
|
|
* @param superConstructor {Function} - The parent constructor
|
|
*/
|
|
utils.inherits = function (constructor, superConstructor) {
|
|
nodeUtils.inherits(constructor, superConstructor);
|
|
constructor.callSuper = function (inst, args) {
|
|
if (args) {
|
|
if (_.isArguments(args)) {
|
|
utils.applyArgs(superConstructor, inst, args);
|
|
} else {
|
|
utils.applyArgs(superConstructor, inst, arguments, 1);
|
|
}
|
|
} else {
|
|
superConstructor.call(inst);
|
|
}
|
|
};
|
|
};
|
|
|
|
/**
|
|
* 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 () {};
|
|
|
|
// _.getStackTrace = function (callee) {
|
|
// var e = {};
|
|
// if (typeof Error.captureStackTrace === 'function') {
|
|
// Error.captureStackTrace(e, callee || _.getStackTrace);
|
|
// } else {
|
|
// e.stack = (new Error()).stack;
|
|
// console.log(e.stack);
|
|
// }
|
|
// return '\n' + e.stack.split('\n').slice(1).join('\n');
|
|
// };
|
|
|
|
module.exports = utils;
|
|
|
|
},{"__browserify_process":91,"lodash":124,"path":61,"util":68}],154:[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) {
|
|
module.exports = defaults;
|
|
} else {
|
|
module.exports = require('optimist')
|
|
.default(defaults)
|
|
.boolean('createServer')
|
|
.argv;
|
|
}
|
|
|
|
},{"../../../src/lib/utils":153,"__browserify_process":91,"optimist":false,"path":61}],155:[function(require,module,exports){
|
|
var process=require("__browserify_process"),__dirname="/";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');
|
|
|
|
// location that the logger will write to
|
|
var logFile = path.resolve(__dirname, './log');
|
|
|
|
// 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) {
|
|
externalExists = !err;
|
|
create(cb);
|
|
});
|
|
});
|
|
} else {
|
|
doCreateClient(done);
|
|
}
|
|
|
|
function done(err) {
|
|
cb(err, client);
|
|
}
|
|
|
|
function doCreateClient(cb) {
|
|
// close existing client
|
|
if (client) {
|
|
client.close();
|
|
}
|
|
|
|
if (!process.browser) {
|
|
// delete existing log file
|
|
try {
|
|
fs.unlinkSync(logFile);
|
|
} catch (e) {
|
|
if (!~e.message.indexOf('ENOENT')) {
|
|
return _.nextTick(cb, e);
|
|
}
|
|
}
|
|
}
|
|
|
|
client = new es.Client({
|
|
hosts: [
|
|
{
|
|
host: esServer ? esServer.__hostname : argv.host,
|
|
port: esServer ? esServer.__port : argv.port
|
|
}
|
|
],
|
|
log: {
|
|
type: process.browser ? 'console' : 'file',
|
|
level: 'trace',
|
|
path: logFile
|
|
}
|
|
});
|
|
|
|
_.nextTick(cb);
|
|
}
|
|
},
|
|
get: function () {
|
|
return client;
|
|
}
|
|
};
|
|
|
|
},{"../../../src/elasticsearch":128,"../../../src/lib/utils":153,"./argv":154,"./server":157,"__browserify_process":91,"fs":59,"path":61}],156:[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":128,"../../../src/lib/utils":153,"./argv":154,"./client_manager":155,"./yaml_file":159,"./yaml_tests.json":160,"async":2,"expect.js":49,"js-yaml":92,"minimatch":125,"path":61}],157:[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":153,"./argv":154,"__browserify_process":91,"child_process":57,"events":58,"fs":59,"path":61}],158:[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 (typeof action.args === 'object') {
|
|
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":153,"./client_manager":155,"__browserify_process":91,"expect.js":49}],159:[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":153,"./client_manager":155,"./yaml_doc":158,"async":2}],160:[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": [
|
|
{
|
|
"Basic tests for suggest API": [
|
|
{
|
|
"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": {}
|
|
}
|
|
},
|
|
{
|
|
"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"
|
|
}
|
|
}
|
|
]
|
|
}
|
|
],
|
|
"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"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
]
|
|
}
|
|
},{}]},{},[156])
|
|
; |