appcd-agent
Advanced tools
Comparing version 1.0.1 to 1.1.0-0
@@ -1,18 +0,14 @@ | ||
'use strict'; | ||
"use strict"; | ||
Object.defineProperty(exports, "__esModule", { | ||
value: true | ||
value: true | ||
}); | ||
exports.default = undefined; | ||
exports.default = void 0; | ||
var _collection = require('./collection'); | ||
var _collection = _interopRequireDefault(require("./collection")); | ||
var _collection2 = _interopRequireDefault(_collection); | ||
var _os = _interopRequireDefault(require("os")); | ||
var _os = require('os'); | ||
var _events = require("events"); | ||
var _os2 = _interopRequireDefault(_os); | ||
var _events = require('events'); | ||
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } | ||
@@ -24,200 +20,203 @@ | ||
let Agent = class Agent extends _events.EventEmitter { | ||
/** | ||
* Creates the Agent instance. | ||
* | ||
* @param {Object} [opts] - Various options. | ||
* @param {Number} [opts.pollInterval=1000] - The number of milliseconds to poll the metrics. | ||
* @access public | ||
*/ | ||
constructor(opts = {}) { | ||
if (!opts || typeof opts !== 'object') { | ||
throw new TypeError('Expected options to be an object'); | ||
} | ||
/** | ||
* Creates the Agent instance. | ||
* | ||
* @param {Object} [opts] - Various options. | ||
* @param {Number} [opts.pollInterval=1000] - The number of milliseconds to poll the metrics. | ||
* @access public | ||
*/ | ||
constructor(opts = {}) { | ||
if (!opts || typeof opts !== 'object') { | ||
throw new TypeError('Expected options to be an object'); | ||
} | ||
super(); | ||
super(); | ||
/** | ||
* The number of milliseconds to wait before polling. | ||
* @type {Number} | ||
*/ | ||
/** | ||
* The number of milliseconds to wait before polling. | ||
* @type {Number} | ||
this.pollInterval = Math.max(~~opts.pollInterval || 1000, 1); | ||
/** | ||
* A map of names to collections. | ||
* @type {Object} | ||
*/ | ||
this.buckets = {}; | ||
/** | ||
* A list of functions to call when polling for stats. | ||
* @type {Array} | ||
*/ | ||
this.collectors = []; | ||
} | ||
/** | ||
* Fetches the stats for the given bucket name. | ||
* | ||
* @param {String} name - The bucket name. | ||
* @returns {Object} | ||
* @access public | ||
*/ | ||
this.pollInterval = Math.max(~~opts.pollInterval || 1000, 1); | ||
/** | ||
* A map of names to collections. | ||
* @type {Object} | ||
getStats(name) { | ||
return this.buckets[name] ? this.buckets[name].collection.stats : null; | ||
} | ||
/** | ||
* Starts monitoring the system. | ||
* | ||
* @returns {Agent} | ||
* @access public | ||
*/ | ||
this.buckets = {}; | ||
/** | ||
* A list of functions to call when polling for stats. | ||
* @type {Array} | ||
start() { | ||
this.initCpu = this.currentCpu = process.cpuUsage(); | ||
this.initHrtime = this.currentHrTime = process.hrtime(); | ||
clearTimeout(this.pollTimer); | ||
this.poll(); | ||
return this; | ||
} | ||
/** | ||
* Stops monitoring the system. | ||
* | ||
* @returns {Agent} | ||
* @access public | ||
*/ | ||
this.collectors = []; | ||
} | ||
/** | ||
* Fetches the stats for the given bucket name. | ||
* | ||
* @param {String} name - The bucket name. | ||
* @returns {Object} | ||
* @access public | ||
*/ | ||
getStats(name) { | ||
return this.buckets[name] ? this.buckets[name].collection.stats : null; | ||
} | ||
/** | ||
* Starts monitoring the system. | ||
* | ||
* @returns {Agent} | ||
* @access public | ||
*/ | ||
start() { | ||
this.initCpu = this.currentCpu = process.cpuUsage(); | ||
this.initHrtime = this.currentHrTime = process.hrtime(); | ||
stop() { | ||
clearTimeout(this.pollTimer); | ||
return this; | ||
} | ||
/** | ||
* Adds a new collector. | ||
* | ||
* @param {Function} fn - The collector callback to add. | ||
* @returns {Agent} | ||
* @access public | ||
*/ | ||
clearTimeout(this.pollTimer); | ||
this.poll(); | ||
return this; | ||
} | ||
addCollector(fn) { | ||
if (!fn || typeof fn !== 'function') { | ||
throw new TypeError('Expected collector to be a function'); | ||
} | ||
/** | ||
* Stops monitoring the system. | ||
* | ||
* @returns {Agent} | ||
* @access public | ||
*/ | ||
stop() { | ||
clearTimeout(this.pollTimer); | ||
return this; | ||
} | ||
this.collectors.push(fn); | ||
return this; | ||
} | ||
/** | ||
* Removes a collector. | ||
* | ||
* @param {Function} fn - The collector callback to remove. | ||
* @returns {Agent} | ||
* @access public | ||
*/ | ||
/** | ||
* Adds a new collector. | ||
* | ||
* @param {Function} fn - The collector callback to add. | ||
* @returns {Agent} | ||
* @access public | ||
*/ | ||
addCollector(fn) { | ||
if (!fn || typeof fn !== 'function') { | ||
throw new TypeError('Expected collector to be a function'); | ||
} | ||
this.collectors.push(fn); | ||
return this; | ||
} | ||
/** | ||
* Removes a collector. | ||
* | ||
* @param {Function} fn - The collector callback to remove. | ||
* @returns {Agent} | ||
* @access public | ||
*/ | ||
removeCollector(fn) { | ||
if (!fn || typeof fn !== 'function') { | ||
throw new TypeError('Expected collector to be a function'); | ||
} | ||
removeCollector(fn) { | ||
if (!fn || typeof fn !== 'function') { | ||
throw new TypeError('Expected collector to be a function'); | ||
} | ||
for (let i = 0; i < this.collectors.length; i++) { | ||
if (this.collectors[i] === fn) { | ||
this.collectors.splice(i--, 1); | ||
} | ||
} | ||
for (let i = 0; i < this.collectors.length; i++) { | ||
if (this.collectors[i] === fn) { | ||
this.collectors.splice(i--, 1); | ||
} | ||
} | ||
return this; | ||
} | ||
return this; | ||
} | ||
/** | ||
* Polls the CPU and memory. | ||
* | ||
* @access private | ||
*/ | ||
/** | ||
* Polls the CPU and memory. | ||
* | ||
* @access private | ||
*/ | ||
poll() { | ||
const cpu = process.cpuUsage(this.initCpu); | ||
const hrtime = process.hrtime(this.initHrtime); | ||
const mem = process.memoryUsage(); | ||
const stats = { | ||
cpu: (cpu.user + cpu.system) / (hrtime[0] * 1000000 + hrtime[1] / 1000) * 100, | ||
freemem: _os2.default.freemem(), | ||
heapTotal: mem.heapTotal, | ||
heapUsed: mem.heapUsed, | ||
rss: mem.rss | ||
}; | ||
Promise.all(this.collectors.map(fn => Promise.resolve().then(() => fn()).then(result => { | ||
if (result && typeof result === 'object') { | ||
Object.assign(stats, result); | ||
} | ||
}))).then(() => { | ||
// figure out how long it's been since we started the last poll | ||
const now = Date.now(); | ||
const delta = this.lastTimestamp ? now - this.lastTimestamp : 0; | ||
this.lastTimestamp = now; | ||
poll() { | ||
const cpu = process.cpuUsage(this.initCpu); | ||
const hrtime = process.hrtime(this.initHrtime); | ||
const mem = process.memoryUsage(); | ||
const stats = { | ||
cpu: (cpu.user + cpu.system) / (hrtime[0] * 1000000 + hrtime[1] / 1000) * 100, | ||
freemem: _os.default.freemem(), | ||
heapTotal: mem.heapTotal, | ||
heapUsed: mem.heapUsed, | ||
rss: mem.rss | ||
}; | ||
Promise.all(this.collectors.map(fn => Promise.resolve().then(() => fn()).then(result => { | ||
if (result && typeof result === 'object') { | ||
Object.assign(stats, result); | ||
} | ||
}))).then(() => { | ||
// figure out how long it's been since we started the last poll | ||
const now = Date.now(); | ||
const delta = this.lastTimestamp ? now - this.lastTimestamp : 0; | ||
this.lastTimestamp = now; // schedule the next poll | ||
// schedule the next poll | ||
const next = this.pollInterval - delta % this.pollInterval; | ||
this.pollTimer = setTimeout(this.poll.bind(this), next); | ||
const next = this.pollInterval - delta % this.pollInterval; | ||
this.pollTimer = setTimeout(this.poll.bind(this), next); // figure out how many intervals we missed, we'll interpolate the values later | ||
// figure out how many intervals we missed, we'll interpolate the values later | ||
const missed = Math.floor((delta - this.pollInterval) / this.pollInterval) + 1; | ||
const missed = Math.floor((delta - this.pollInterval) / this.pollInterval) + 1; // make sure we have collections for all the data we want to store | ||
// make sure we have collections for all the data we want to store | ||
for (const name of Object.keys(stats)) { | ||
if (!this.buckets[name]) { | ||
this.buckets[name] = { | ||
collection: new _collection2.default(60 * 15), // 15 minutes worth of data | ||
last: null | ||
}; | ||
} | ||
} | ||
for (const name of Object.keys(stats)) { | ||
if (!this.buckets[name]) { | ||
this.buckets[name] = { | ||
collection: new _collection.default(60 * 15), | ||
// 15 minutes worth of data | ||
last: null | ||
}; | ||
} | ||
} // add the values for each stat to its bucket, or set zero if we don't have a value | ||
// for this poll | ||
// add the values for each stat to its bucket, or set zero if we don't have a value | ||
// for this poll | ||
for (const name of Object.keys(this.buckets)) { | ||
if (stats.hasOwnProperty(name)) { | ||
const value = stats[name]; | ||
const bucket = this.buckets[name]; | ||
let last = bucket.last || 0; | ||
// if we missed any intervals, interpolate the values | ||
// note that we don't emit stats for missed intervals | ||
for (let i = 1; i < missed; i++) { | ||
bucket.collection.add((value - last) * i / missed + last); | ||
} | ||
for (const name of Object.keys(this.buckets)) { | ||
if (stats.hasOwnProperty(name)) { | ||
const value = stats[name]; | ||
const bucket = this.buckets[name]; | ||
let last = bucket.last || 0; // if we missed any intervals, interpolate the values | ||
// note that we don't emit stats for missed intervals | ||
// add the value and set it as the last value | ||
bucket.collection.add(value); | ||
bucket.last = value; | ||
} else { | ||
// the bucket didn't receive a value so that probably means the collector | ||
// was removed or the bucket is no longer monitored, so remove the bucket | ||
delete this.buckets[name]; | ||
} | ||
} | ||
for (let i = 1; i < missed; i++) { | ||
bucket.collection.add((value - last) * i / missed + last); | ||
} // add the value and set it as the last value | ||
// add in the timestamp | ||
stats.ts = process.uptime(); | ||
// emit the stats! | ||
this.emit('stats', stats); | ||
}).catch(err => this.emit('error', err)); | ||
} | ||
bucket.collection.add(value); | ||
bucket.last = value; | ||
} else { | ||
// the bucket didn't receive a value so that probably means the collector | ||
// was removed or the bucket is no longer monitored, so remove the bucket | ||
delete this.buckets[name]; | ||
} | ||
} // add in the timestamp | ||
/** | ||
* Generates a snapshot of the collected data. | ||
* | ||
* @returns {Object} | ||
* @access public | ||
*/ | ||
health() { | ||
const result = {}; | ||
for (const [name, bucket] of Object.entries(this.buckets)) { | ||
result[name] = bucket.collection.stats; | ||
} | ||
return result; | ||
} | ||
stats.ts = process.uptime(); // emit the stats! | ||
this.emit('stats', stats); | ||
}).catch(err => this.emit('error', err)); | ||
} | ||
/** | ||
* Generates a snapshot of the collected data. | ||
* | ||
* @returns {Object} | ||
* @access public | ||
*/ | ||
health() { | ||
const result = {}; | ||
for (const [name, bucket] of Object.entries(this.buckets)) { | ||
result[name] = bucket.collection.stats; | ||
} | ||
return result; | ||
} | ||
}; | ||
exports.default = Agent; | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["agent.js"],"names":["Agent","constructor","opts","TypeError","pollInterval","Math","max","buckets","collectors","getStats","name","collection","stats","start","initCpu","currentCpu","process","cpuUsage","initHrtime","currentHrTime","hrtime","clearTimeout","pollTimer","poll","stop","addCollector","fn","push","removeCollector","i","length","splice","cpu","mem","memoryUsage","user","system","freemem","heapTotal","heapUsed","rss","Promise","all","map","resolve","then","result","Object","assign","now","Date","delta","lastTimestamp","next","setTimeout","bind","missed","floor","keys","last","hasOwnProperty","value","bucket","add","ts","uptime","emit","catch","err","health","entries"],"mappings":"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","file":"agent.js","sourceRoot":"src","sourcesContent":["import Collection from './collection';\nimport os from 'os';\n\nimport { EventEmitter } from 'events';\n\n/**\n * Monitors the current process for CPU and memory usage.\n */\nexport default class Agent extends EventEmitter {\n\t/**\n\t * Creates the Agent instance.\n\t *\n\t * @param {Object} [opts] - Various options.\n\t * @param {Number} [opts.pollInterval=1000] - The number of milliseconds to poll the metrics.\n\t * @access public\n\t */\n\tconstructor(opts = {}) {\n\t\tif (!opts || typeof opts !== 'object') {\n\t\t\tthrow new TypeError('Expected options to be an object');\n\t\t}\n\n\t\tsuper();\n\n\t\t/**\n\t\t * The number of milliseconds to wait before polling.\n\t\t * @type {Number}\n\t\t */\n\t\tthis.pollInterval = Math.max(~~opts.pollInterval || 1000, 1);\n\n\t\t/**\n\t\t * A map of names to collections.\n\t\t * @type {Object}\n\t\t */\n\t\tthis.buckets = {};\n\n\t\t/**\n\t\t * A list of functions to call when polling for stats.\n\t\t * @type {Array}\n\t\t */\n\t\tthis.collectors = [];\n\t}\n\n\t/**\n\t * Fetches the stats for the given bucket name.\n\t *\n\t * @param {String} name - The bucket name.\n\t * @returns {Object}\n\t * @access public\n\t */\n\tgetStats(name) {\n\t\treturn this.buckets[name] ? this.buckets[name].collection.stats : null;\n\t}\n\n\t/**\n\t * Starts monitoring the system.\n\t *\n\t * @returns {Agent}\n\t * @access public\n\t */\n\tstart() {\n\t\tthis.initCpu = this.currentCpu = process.cpuUsage();\n\t\tthis.initHrtime = this.currentHrTime = process.hrtime();\n\n\t\tclearTimeout(this.pollTimer);\n\t\tthis.poll();\n\n\t\treturn this;\n\t}\n\n\t/**\n\t * Stops monitoring the system.\n\t *\n\t * @returns {Agent}\n\t * @access public\n\t */\n\tstop() {\n\t\tclearTimeout(this.pollTimer);\n\t\treturn this;\n\t}\n\n\t/**\n\t * Adds a new collector.\n\t *\n\t * @param {Function} fn - The collector callback to add.\n\t * @returns {Agent}\n\t * @access public\n\t */\n\taddCollector(fn) {\n\t\tif (!fn || typeof fn !== 'function') {\n\t\t\tthrow new TypeError('Expected collector to be a function');\n\t\t}\n\t\tthis.collectors.push(fn);\n\t\treturn this;\n\t}\n\n\t/**\n\t * Removes a collector.\n\t *\n\t * @param {Function} fn - The collector callback to remove.\n\t * @returns {Agent}\n\t * @access public\n\t */\n\tremoveCollector(fn) {\n\t\tif (!fn || typeof fn !== 'function') {\n\t\t\tthrow new TypeError('Expected collector to be a function');\n\t\t}\n\n\t\tfor (let i = 0; i < this.collectors.length; i++) {\n\t\t\tif (this.collectors[i] === fn) {\n\t\t\t\tthis.collectors.splice(i--, 1);\n\t\t\t}\n\t\t}\n\n\t\treturn this;\n\t}\n\n\t/**\n\t * Polls the CPU and memory.\n\t *\n\t * @access private\n\t */\n\tpoll() {\n\t\tconst cpu    = process.cpuUsage(this.initCpu);\n\t\tconst hrtime = process.hrtime(this.initHrtime);\n\t\tconst mem    = process.memoryUsage();\n\t\tconst stats = {\n\t\t\tcpu:       (cpu.user + cpu.system) / (hrtime[0] * 1000000 + hrtime[1] / 1000) * 100,\n\t\t\tfreemem:   os.freemem(),\n\t\t\theapTotal: mem.heapTotal,\n\t\t\theapUsed:  mem.heapUsed,\n\t\t\trss:       mem.rss\n\t\t};\n\n\t\tPromise\n\t\t\t.all(this.collectors.map(fn => Promise.resolve()\n\t\t\t\t.then(() => fn())\n\t\t\t\t.then(result => {\n\t\t\t\t\tif (result && typeof result === 'object') {\n\t\t\t\t\t\tObject.assign(stats, result);\n\t\t\t\t\t}\n\t\t\t\t})\n\t\t\t))\n\t\t\t.then(() => {\n\t\t\t\t// figure out how long it's been since we started the last poll\n\t\t\t\tconst now = Date.now();\n\t\t\t\tconst delta = this.lastTimestamp ? (now - this.lastTimestamp) : 0;\n\t\t\t\tthis.lastTimestamp = now;\n\n\t\t\t\t// schedule the next poll\n\t\t\t\tconst next = this.pollInterval - (delta % this.pollInterval);\n\t\t\t\tthis.pollTimer = setTimeout(this.poll.bind(this), next);\n\n\t\t\t\t// figure out how many intervals we missed, we'll interpolate the values later\n\t\t\t\tconst missed = Math.floor((delta - this.pollInterval) / this.pollInterval) + 1;\n\n\t\t\t\t// make sure we have collections for all the data we want to store\n\t\t\t\tfor (const name of Object.keys(stats)) {\n\t\t\t\t\tif (!this.buckets[name]) {\n\t\t\t\t\t\tthis.buckets[name] = {\n\t\t\t\t\t\t\tcollection: new Collection(60 * 15), // 15 minutes worth of data\n\t\t\t\t\t\t\tlast:       null\n\t\t\t\t\t\t};\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\t// add the values for each stat to its bucket, or set zero if we don't have a value\n\t\t\t\t// for this poll\n\t\t\t\tfor (const name of Object.keys(this.buckets)) {\n\t\t\t\t\tif (stats.hasOwnProperty(name)) {\n\t\t\t\t\t\tconst value = stats[name];\n\t\t\t\t\t\tconst bucket = this.buckets[name];\n\t\t\t\t\t\tlet last = bucket.last || 0;\n\n\t\t\t\t\t\t// if we missed any intervals, interpolate the values\n\t\t\t\t\t\t// note that we don't emit stats for missed intervals\n\t\t\t\t\t\tfor (let i = 1; i < missed; i++) {\n\t\t\t\t\t\t\tbucket.collection.add((value - last) * i / missed + last);\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\t// add the value and set it as the last value\n\t\t\t\t\t\tbucket.collection.add(value);\n\t\t\t\t\t\tbucket.last = value;\n\t\t\t\t\t} else {\n\t\t\t\t\t\t// the bucket didn't receive a value so that probably means the collector\n\t\t\t\t\t\t// was removed or the bucket is no longer monitored, so remove the bucket\n\t\t\t\t\t\tdelete this.buckets[name];\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\t// add in the timestamp\n\t\t\t\tstats.ts = process.uptime();\n\n\t\t\t\t// emit the stats!\n\t\t\t\tthis.emit('stats', stats);\n\t\t\t})\n\t\t\t.catch(err => this.emit('error', err));\n\t}\n\n\t/**\n\t * Generates a snapshot of the collected data.\n\t *\n\t * @returns {Object}\n\t * @access public\n\t */\n\thealth() {\n\t\tconst result = {};\n\t\tfor (const [ name, bucket ] of Object.entries(this.buckets)) {\n\t\t\tresult[name] = bucket.collection.stats;\n\t\t}\n\t\treturn result;\n\t}\n}\n"]} | ||
exports.default = Agent; |
@@ -1,12 +0,10 @@ | ||
'use strict'; | ||
"use strict"; | ||
Object.defineProperty(exports, "__esModule", { | ||
value: true | ||
value: true | ||
}); | ||
exports.default = undefined; | ||
exports.default = void 0; | ||
var _nanobuffer = require('nanobuffer'); | ||
var _nanobuffer = _interopRequireDefault(require("nanobuffer")); | ||
var _nanobuffer2 = _interopRequireDefault(_nanobuffer); | ||
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } | ||
@@ -18,68 +16,70 @@ | ||
let Collection = class Collection { | ||
/** | ||
* Creates the collection instance and initializes the value buffer. | ||
* | ||
* @param {Number} size - The number of values to buffer. | ||
* @access public | ||
*/ | ||
constructor(size) { | ||
this.values = new _nanobuffer2.default(size); | ||
this.min = null; | ||
this.max = null; | ||
this.avg = null; | ||
} | ||
/** | ||
* Creates the collection instance and initializes the value buffer. | ||
* | ||
* @param {Number} size - The number of values to buffer. | ||
* @access public | ||
*/ | ||
constructor(size) { | ||
this.values = new _nanobuffer.default(size); | ||
this.min = null; | ||
this.max = null; | ||
this.avg = null; | ||
} | ||
/** | ||
* Adds a value to the collection and recomputes the minimum, maximum, and average. | ||
* | ||
* @param {Number} value - The value to add to the collection. | ||
* @returns {Collection} | ||
* @access public | ||
*/ | ||
/** | ||
* Adds a value to the collection and recomputes the minimum, maximum, and average. | ||
* | ||
* @param {Number} value - The value to add to the collection. | ||
* @returns {Collection} | ||
* @access public | ||
*/ | ||
add(value) { | ||
if (typeof value !== 'number' || isNaN(value)) { | ||
throw TypeError('Expected value to be a number'); | ||
} | ||
this.values.push(value); | ||
add(value) { | ||
if (typeof value !== 'number' || isNaN(value)) { | ||
throw TypeError('Expected value to be a number'); | ||
} | ||
if (this.values.size === 1) { | ||
this.min = this.max = this.avg = value; | ||
} else { | ||
let total = 0; | ||
this.values.push(value); | ||
for (const counter of this.values) { | ||
total += counter; | ||
if (this.values.size === 1) { | ||
this.min = this.max = this.avg = value; | ||
} else { | ||
let total = 0; | ||
if (counter < this.min) { | ||
this.min = counter; | ||
} | ||
for (const counter of this.values) { | ||
total += counter; | ||
if (counter > this.max) { | ||
this.max = counter; | ||
} | ||
} | ||
if (counter < this.min) { | ||
this.min = counter; | ||
} | ||
this.avg = total / this.values.size; | ||
} | ||
if (counter > this.max) { | ||
this.max = counter; | ||
} | ||
} | ||
return this; | ||
} | ||
this.avg = total / this.values.size; | ||
} | ||
/** | ||
* Returns a snapshot of the contents of this collection. | ||
* | ||
* @returns {Object} | ||
* @access public | ||
*/ | ||
get stats() { | ||
return { | ||
values: Array.from(this.values), | ||
min: this.min, | ||
max: this.max, | ||
avg: this.avg | ||
}; | ||
} | ||
return this; | ||
} | ||
/** | ||
* Returns a snapshot of the contents of this collection. | ||
* | ||
* @returns {Object} | ||
* @access public | ||
*/ | ||
get stats() { | ||
return { | ||
values: Array.from(this.values), | ||
min: this.min, | ||
max: this.max, | ||
avg: this.avg | ||
}; | ||
} | ||
}; | ||
exports.default = Collection; | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbImNvbGxlY3Rpb24uanMiXSwibmFtZXMiOlsiQ29sbGVjdGlvbiIsImNvbnN0cnVjdG9yIiwic2l6ZSIsInZhbHVlcyIsIm1pbiIsIm1heCIsImF2ZyIsImFkZCIsInZhbHVlIiwiaXNOYU4iLCJUeXBlRXJyb3IiLCJwdXNoIiwidG90YWwiLCJjb3VudGVyIiwic3RhdHMiLCJBcnJheSIsImZyb20iXSwibWFwcGluZ3MiOiI7Ozs7Ozs7QUFBQTs7Ozs7O0FBRUE7OztJQUdxQkEsVSxHQUFOLE1BQU1BLFVBQU4sQ0FBaUI7QUFDL0I7Ozs7OztBQU1BQyxhQUFZQyxJQUFaLEVBQWtCO0FBQ2pCLE9BQUtDLE1BQUwsR0FBYyx5QkFBZUQsSUFBZixDQUFkO0FBQ0EsT0FBS0UsR0FBTCxHQUFXLElBQVg7QUFDQSxPQUFLQyxHQUFMLEdBQVcsSUFBWDtBQUNBLE9BQUtDLEdBQUwsR0FBVyxJQUFYO0FBQ0E7O0FBRUQ7Ozs7Ozs7QUFPQUMsS0FBSUMsS0FBSixFQUFXO0FBQ1YsTUFBSSxPQUFPQSxLQUFQLEtBQWlCLFFBQWpCLElBQTZCQyxNQUFNRCxLQUFOLENBQWpDLEVBQStDO0FBQzlDLFNBQU1FLFVBQVUsK0JBQVYsQ0FBTjtBQUNBOztBQUVELE9BQUtQLE1BQUwsQ0FBWVEsSUFBWixDQUFpQkgsS0FBakI7O0FBRUEsTUFBSSxLQUFLTCxNQUFMLENBQVlELElBQVosS0FBcUIsQ0FBekIsRUFBNEI7QUFDM0IsUUFBS0UsR0FBTCxHQUFXLEtBQUtDLEdBQUwsR0FBVyxLQUFLQyxHQUFMLEdBQVdFLEtBQWpDO0FBQ0EsR0FGRCxNQUVPO0FBQ04sT0FBSUksUUFBUSxDQUFaOztBQUVBLFFBQUssTUFBTUMsT0FBWCxJQUFzQixLQUFLVixNQUEzQixFQUFtQztBQUNsQ1MsYUFBU0MsT0FBVDs7QUFFQSxRQUFJQSxVQUFVLEtBQUtULEdBQW5CLEVBQXdCO0FBQ3ZCLFVBQUtBLEdBQUwsR0FBV1MsT0FBWDtBQUNBOztBQUVELFFBQUlBLFVBQVUsS0FBS1IsR0FBbkIsRUFBd0I7QUFDdkIsVUFBS0EsR0FBTCxHQUFXUSxPQUFYO0FBQ0E7QUFDRDs7QUFFRCxRQUFLUCxHQUFMLEdBQVdNLFFBQVEsS0FBS1QsTUFBTCxDQUFZRCxJQUEvQjtBQUNBOztBQUVELFNBQU8sSUFBUDtBQUNBOztBQUVEOzs7Ozs7QUFNQSxLQUFJWSxLQUFKLEdBQVk7QUFDWCxTQUFPO0FBQ05YLFdBQVFZLE1BQU1DLElBQU4sQ0FBVyxLQUFLYixNQUFoQixDQURGO0FBRU5DLFFBQVEsS0FBS0EsR0FGUDtBQUdOQyxRQUFRLEtBQUtBLEdBSFA7QUFJTkMsUUFBUSxLQUFLQTtBQUpQLEdBQVA7QUFNQTtBQWhFOEIsQztrQkFBWE4sVSIsImZpbGUiOiJjb2xsZWN0aW9uLmpzIiwic291cmNlUm9vdCI6InNyYyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBOYW5vQnVmZmVyIGZyb20gJ25hbm9idWZmZXInO1xuXG4vKipcbiAqIFN0b3JlcyBhbmQgY29tcHV0ZXMgc3RhdHMgb24gYSBjb2xsZWN0aW9uIG9mIG51bWJlcnMuXG4gKi9cbmV4cG9ydCBkZWZhdWx0IGNsYXNzIENvbGxlY3Rpb24ge1xuXHQvKipcblx0ICogQ3JlYXRlcyB0aGUgY29sbGVjdGlvbiBpbnN0YW5jZSBhbmQgaW5pdGlhbGl6ZXMgdGhlIHZhbHVlIGJ1ZmZlci5cblx0ICpcblx0ICogQHBhcmFtIHtOdW1iZXJ9IHNpemUgLSBUaGUgbnVtYmVyIG9mIHZhbHVlcyB0byBidWZmZXIuXG5cdCAqIEBhY2Nlc3MgcHVibGljXG5cdCAqL1xuXHRjb25zdHJ1Y3RvcihzaXplKSB7XG5cdFx0dGhpcy52YWx1ZXMgPSBuZXcgTmFub0J1ZmZlcihzaXplKTtcblx0XHR0aGlzLm1pbiA9IG51bGw7XG5cdFx0dGhpcy5tYXggPSBudWxsO1xuXHRcdHRoaXMuYXZnID0gbnVsbDtcblx0fVxuXG5cdC8qKlxuXHQgKiBBZGRzIGEgdmFsdWUgdG8gdGhlIGNvbGxlY3Rpb24gYW5kIHJlY29tcHV0ZXMgdGhlIG1pbmltdW0sIG1heGltdW0sIGFuZCBhdmVyYWdlLlxuXHQgKlxuXHQgKiBAcGFyYW0ge051bWJlcn0gdmFsdWUgLSBUaGUgdmFsdWUgdG8gYWRkIHRvIHRoZSBjb2xsZWN0aW9uLlxuXHQgKiBAcmV0dXJucyB7Q29sbGVjdGlvbn1cblx0ICogQGFjY2VzcyBwdWJsaWNcblx0ICovXG5cdGFkZCh2YWx1ZSkge1xuXHRcdGlmICh0eXBlb2YgdmFsdWUgIT09ICdudW1iZXInIHx8IGlzTmFOKHZhbHVlKSkge1xuXHRcdFx0dGhyb3cgVHlwZUVycm9yKCdFeHBlY3RlZCB2YWx1ZSB0byBiZSBhIG51bWJlcicpO1xuXHRcdH1cblxuXHRcdHRoaXMudmFsdWVzLnB1c2godmFsdWUpO1xuXG5cdFx0aWYgKHRoaXMudmFsdWVzLnNpemUgPT09IDEpIHtcblx0XHRcdHRoaXMubWluID0gdGhpcy5tYXggPSB0aGlzLmF2ZyA9IHZhbHVlO1xuXHRcdH0gZWxzZSB7XG5cdFx0XHRsZXQgdG90YWwgPSAwO1xuXG5cdFx0XHRmb3IgKGNvbnN0IGNvdW50ZXIgb2YgdGhpcy52YWx1ZXMpIHtcblx0XHRcdFx0dG90YWwgKz0gY291bnRlcjtcblxuXHRcdFx0XHRpZiAoY291bnRlciA8IHRoaXMubWluKSB7XG5cdFx0XHRcdFx0dGhpcy5taW4gPSBjb3VudGVyO1xuXHRcdFx0XHR9XG5cblx0XHRcdFx0aWYgKGNvdW50ZXIgPiB0aGlzLm1heCkge1xuXHRcdFx0XHRcdHRoaXMubWF4ID0gY291bnRlcjtcblx0XHRcdFx0fVxuXHRcdFx0fVxuXG5cdFx0XHR0aGlzLmF2ZyA9IHRvdGFsIC8gdGhpcy52YWx1ZXMuc2l6ZTtcblx0XHR9XG5cblx0XHRyZXR1cm4gdGhpcztcblx0fVxuXG5cdC8qKlxuXHQgKiBSZXR1cm5zIGEgc25hcHNob3Qgb2YgdGhlIGNvbnRlbnRzIG9mIHRoaXMgY29sbGVjdGlvbi5cblx0ICpcblx0ICogQHJldHVybnMge09iamVjdH1cblx0ICogQGFjY2VzcyBwdWJsaWNcblx0ICovXG5cdGdldCBzdGF0cygpIHtcblx0XHRyZXR1cm4ge1xuXHRcdFx0dmFsdWVzOiBBcnJheS5mcm9tKHRoaXMudmFsdWVzKSxcblx0XHRcdG1pbjogICAgdGhpcy5taW4sXG5cdFx0XHRtYXg6ICAgIHRoaXMubWF4LFxuXHRcdFx0YXZnOiAgICB0aGlzLmF2Z1xuXHRcdH07XG5cdH1cbn1cbiJdfQ== | ||
exports.default = Collection; |
@@ -1,16 +0,15 @@ | ||
'use strict'; | ||
"use strict"; | ||
Object.defineProperty(exports, "__esModule", { | ||
value: true | ||
value: true | ||
}); | ||
Object.defineProperty(exports, "default", { | ||
enumerable: true, | ||
get: function () { | ||
return _agent.default; | ||
} | ||
}); | ||
var _agent = require('./agent'); | ||
var _agent = _interopRequireDefault(require("./agent")); | ||
Object.defineProperty(exports, 'default', { | ||
enumerable: true, | ||
get: function () { | ||
return _interopRequireDefault(_agent).default; | ||
} | ||
}); | ||
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } | ||
@@ -20,4 +19,3 @@ | ||
if (!Error.prepareStackTrace) { | ||
require('source-map-support/register'); | ||
} | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbImluZGV4LmpzIl0sIm5hbWVzIjpbImRlZmF1bHQiLCJFcnJvciIsInByZXBhcmVTdGFja1RyYWNlIiwicmVxdWlyZSJdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7d0NBS1NBLE87Ozs7OztBQUxUO0FBQ0EsSUFBSSxDQUFDQyxNQUFNQyxpQkFBWCxFQUE4QjtBQUM3QkMsU0FBUSw2QkFBUjtBQUNBIiwiZmlsZSI6ImluZGV4LmpzIiwic291cmNlUm9vdCI6InNyYyIsInNvdXJjZXNDb250ZW50IjpbIi8qIGlzdGFuYnVsIGlnbm9yZSBpZiAqL1xuaWYgKCFFcnJvci5wcmVwYXJlU3RhY2tUcmFjZSkge1xuXHRyZXF1aXJlKCdzb3VyY2UtbWFwLXN1cHBvcnQvcmVnaXN0ZXInKTtcbn1cblxuZXhwb3J0IHsgZGVmYXVsdCB9IGZyb20gJy4vYWdlbnQnO1xuIl19 | ||
require('source-map-support/register'); | ||
} |
{ | ||
"name": "appcd-agent", | ||
"version": "1.0.1", | ||
"version": "1.1.0-0", | ||
"description": "Monitors a Node.js process and tracks CPU and memory usage.", | ||
@@ -20,6 +20,6 @@ "main": "./dist/index", | ||
"nanobuffer": "^1.0.0", | ||
"source-map-support": "^0.5.0" | ||
"source-map-support": "^0.5.4" | ||
}, | ||
"devDependencies": { | ||
"appcd-gulp": "^1.0.1", | ||
"appcd-gulp": "^1.1.0-0", | ||
"gulp": "^3.9.1" | ||
@@ -26,0 +26,0 @@ }, |
Sorry, the diff of this file is not supported yet
License Policy Violation
LicenseThis package is not allowed per your license policy. Review the package's license to ensure compliance.
Found 1 instance in 1 package
No v1
QualityPackage is not semver >=1. This means it is not stable and does not support ^ ranges.
Found 1 instance in 1 package
License Policy Violation
LicenseThis package is not allowed per your license policy. Review the package's license to ensure compliance.
Found 1 instance in 1 package
9470
6
261
1
Updatedsource-map-support@^0.5.4