This repository has been archived on 2020-01-20. You can view files and clone it, but cannot push or open issues or pull requests.
node-spearhead/lib/common.js

373 lines
9.6 KiB
JavaScript
Raw Normal View History

2014-02-07 23:21:24 +02:00
/**
* Copyright (c) 2015 Joyent Inc. All rights reserved.
2014-02-07 23:21:24 +02:00
*/
var assert = require('assert-plus');
var util = require('util'),
format = util.format;
var errors = require('./errors'),
InternalError = errors.InternalError;
// ---- globals
var p = console.log;
2015-08-26 06:53:48 +03:00
var UUID_RE = /^[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}$/;
var TABULA_OPTIONS = [
{
group: 'Output options'
},
{
names: ['H'],
type: 'bool',
help: 'Omit table header row.'
},
{
names: ['o'],
type: 'string',
help: 'Specify fields (columns) to output.',
helpArg: 'field1,...'
},
{
names: ['s'],
type: 'string',
help: 'Sort on the given fields.',
helpArg: 'field1,...'
},
{
names: ['json', 'j'],
type: 'bool',
help: 'JSON output.'
}
];
// ---- support stuff
2014-02-07 23:21:24 +02:00
function objCopy(obj, target) {
assert.object(obj, 'obj');
assert.optionalObject(obj, 'target');
2014-02-07 23:21:24 +02:00
if (target === undefined) {
target = {};
}
Object.keys(obj).forEach(function (k) {
target[k] = obj[k];
});
return target;
}
function deepObjCopy(obj) {
return JSON.parse(JSON.stringify(obj));
}
function zeroPad(n, width) {
var s = String(n);
assert.number(width, 'width');
assert.string(s, 'string');
2014-02-07 23:21:24 +02:00
while (s.length < width) {
s = '0' + s;
}
return s;
}
/**
* Convert a boolean or string representation into a boolean, or
* raise TypeError trying.
*
* @param value {Boolean|String} The input value to convert.
* @param default_ {Boolean} The default value is `value` is undefined.
2015-08-31 22:23:20 +03:00
* @param errName {String} The context to quote in the possibly
2014-02-07 23:21:24 +02:00
* raised TypeError.
*/
function boolFromString(value, default_, errName) {
if (value === undefined) {
return default_;
} else if (value === 'false' || value === '0') {
return false;
} else if (value === 'true' || value === '1') {
return true;
} else if (typeof (value) === 'boolean') {
return value;
} else {
2015-08-31 22:23:20 +03:00
var errmsg = format('invalid boolean value: %j', value);
if (errName) {
errmsg = format('invalid boolean value for %s: %j', errName, value);
}
throw new TypeError(errmsg);
2014-02-07 23:21:24 +02:00
}
}
/**
* given an array return a string with each element
* JSON-stringifed separated by newlines
*/
2015-08-26 23:53:23 +03:00
function jsonStream(arr, stream) {
stream = stream || process.stdout;
2015-09-02 22:38:27 +03:00
2015-08-26 23:53:23 +03:00
arr.forEach(function (elem) {
stream.write(JSON.stringify(elem) + '\n');
});
}
2014-02-07 23:21:24 +02:00
/**
* given an array of key=value pairs, break them into an object
2014-02-07 23:21:24 +02:00
*
* @param {Array} kvs - an array of key=value pairs
* @param {Array} valid (optional) - an array to validate pairs
2014-02-07 23:21:24 +02:00
*/
function kvToObj(kvs, valid) {
assert.arrayOfString(kvs, 'kvs');
assert.optionalArrayOfString(valid, 'valid');
2015-09-02 22:38:27 +03:00
var o = {};
for (var i = 0; i < kvs.length; i++) {
var kv = kvs[i];
var idx = kv.indexOf('=');
if (idx === -1)
throw new errors.UsageError(format(
2015-09-02 20:47:06 +03:00
'invalid filter: "%s" (must be of the form "field=value")',
kv));
var k = kv.slice(0, idx);
var v = kv.slice(idx + 1);
2015-09-02 22:38:27 +03:00
if (valid && valid.indexOf(k) === -1)
throw new errors.UsageError(format(
2015-09-02 20:47:06 +03:00
'invalid filter name: "%s" (must be one of "%s")',
k, valid.join('", "')));
o[k] = v;
2014-02-07 23:21:24 +02:00
}
return o;
2014-02-07 23:21:24 +02:00
}
2015-08-26 03:00:50 +03:00
/**
* return how long ago something happened
*
* @param {Date} when - a date object in the past
* @param {Date} now (optional) - a date object to compare to
* @return {String} - printable string
*/
function longAgo(when, now) {
now = now || new Date();
assert.date(now, 'now');
2015-08-26 03:00:50 +03:00
var seconds = Math.round((now - when) / 1000);
var times = [
seconds / 60 / 60 / 24 / 365, // years
2015-08-26 20:00:01 +03:00
seconds / 60 / 60 / 24 / 7, // weeks
2015-08-26 03:00:50 +03:00
seconds / 60 / 60 / 24, // days
seconds / 60 / 60, // hours
seconds / 60, // minutes
seconds // seconds
];
2015-08-26 20:00:01 +03:00
var names = ['y', 'w', 'd', 'h', 'm', 's'];
2015-08-26 03:00:50 +03:00
for (var i = 0; i < names.length; i++) {
var time = Math.floor(times[i]);
if (time > 0)
return util.format('%d%s', time, names[i]);
}
return '0s';
}
2015-08-26 03:27:46 +03:00
/**
* checks a string and returns a boolean based on if it
* is a UUID or not
*/
function isUUID(s) {
assert.string(s, 's');
2015-08-26 03:27:46 +03:00
return /^([a-f\d]{8}(-[a-f\d]{4}){3}-[a-f\d]{12}?)$/i.test(s);
}
2015-08-26 06:53:48 +03:00
function humanDurationFromMs(ms) {
assert.number(ms, 'ms');
var sizes = [
['ms', 1000, 's'],
['s', 60, 'm'],
['m', 60, 'h'],
2015-09-01 10:31:00 +03:00
['h', 24, 'd'],
['d', 7, 'w']
2015-08-26 06:53:48 +03:00
];
if (ms === 0) {
return '0ms';
}
var bits = [];
var n = ms;
for (var i = 0; i < sizes.length; i++) {
var size = sizes[i];
var remainder = n % size[1];
if (remainder === 0) {
bits.unshift('');
} else {
bits.unshift(format('%d%s', remainder, size[0]));
}
n = Math.floor(n / size[1]);
if (n === 0) {
break;
2015-09-01 10:31:00 +03:00
} else if (i === sizes.length - 1) {
2015-08-26 06:53:48 +03:00
bits.unshift(format('%d%s', n, size[2]));
break;
}
}
2015-08-26 19:36:22 +03:00
if (bits.length > 1 && bits[bits.length - 1].slice(-2) === 'ms') {
bits.pop();
}
2015-08-26 06:53:48 +03:00
return bits.slice(0, 2).join('');
}
2015-08-26 07:27:59 +03:00
/**
* Adapted from <http://stackoverflow.com/a/18650828>
*
* @param {Number} opts.precision The number of decimal places of precision to
* include. Note: This is just clipping (i.e. floor) instead of rounding.
* TODO: round
* @param {Boolean} opts.narrow Make it as narrow as possible: short units,
* no space between value and unit, drop precision if it is all zeros.
2015-08-26 07:27:59 +03:00
*/
function humanSizeFromBytes(opts, bytes) {
if (bytes === undefined) {
bytes = opts;
2015-09-01 20:44:10 +03:00
opts = {};
}
2015-08-26 07:27:59 +03:00
assert.number(bytes, 'bytes');
// The number of decimal places, default 1.
assert.optionalNumber(opts.precision, 'opts.precision');
var precision = opts.precision === undefined ? 1 : opts.precision;
assert.ok(precision >= 0);
assert.optionalBool(opts.narrow, 'opts.narrow');
var sizes = ['B', 'KiB', 'MiB', 'GiB', 'TiB', 'PiB'];
if (opts.narrow) {
sizes = ['B', 'K', 'M', 'G', 'T', 'P'];
}
2015-09-01 20:47:35 +03:00
var template = opts.narrow ? '%s%s%s' : '%s%s %s';
2015-08-26 07:27:59 +03:00
if (bytes === 0) {
return '0 B';
}
2015-09-01 20:47:35 +03:00
var sign = bytes < 0 ? '-' : '';
bytes = Math.abs(bytes);
2015-08-26 07:27:59 +03:00
var i = Number(Math.floor(Math.log(bytes) / Math.log(1024)));
var s = String(bytes / Math.pow(1024, i));
var hasDecimal = s.indexOf('.') !== -1;
if (precision === 0) {
if (hasDecimal) {
s = s.slice(0, s.indexOf('.'));
}
} else if (opts.narrow && !hasDecimal) {
/* skip all-zero precision */
2015-09-01 19:00:45 +03:00
/* jsl:pass */
} else {
if (!hasDecimal) {
s += '.';
}
var places = s.length - s.indexOf('.') - 1;
while (places < precision) {
s += '0';
places++;
}
if (places > precision) {
s = s.slice(0, s.length - places + precision);
}
}
//var precision1 = (s.indexOf('.') === -1
// ? s + '.0' : s.slice(0, s.indexOf('.') + 2));
2015-09-01 20:47:35 +03:00
return format(template, sign, s, sizes[i]);
2015-08-26 07:27:59 +03:00
}
2015-08-26 06:53:48 +03:00
2015-08-26 08:57:27 +03:00
function capitalize(s) {
assert.string(s, 's');
2015-08-26 08:57:27 +03:00
return s[0].toUpperCase() + s.substr(1);
}
/*
* Normalize a short ID. Returns undefined if the given string isn't a valid
* short id.
*
* Short IDs:
* - UUID prefix
* - allow '-' to be elided (to support using containers IDs from
* docker)
* - support docker ID *longer* than a UUID? The curr implementation does.
*/
function normShortId(s) {
assert.string(s, 's');
var shortIdCharsRe = /^[a-f0-9]+$/;
var shortId;
if (s.indexOf('-') === -1) {
if (!shortIdCharsRe.test(s)) {
return;
}
shortId = s.substr(0, 8) + '-'
+ s.substr(8, 4) + '-'
+ s.substr(12, 4) + '-'
+ s.substr(16, 4) + '-'
+ s.substr(20, 12);
shortId = shortId.replace(/-+$/, '');
} else {
// UUID prefix.
2015-09-01 19:00:45 +03:00
shortId = '';
var remaining = s;
var spans = [8, 4, 4, 4, 12];
for (var i = 0; i < spans.length; i++) {
var span = spans[i];
2015-09-01 19:00:45 +03:00
var head = remaining.slice(0, span);
remaining = remaining.slice(span + 1);
if (!shortIdCharsRe.test(head)) {
return;
}
shortId += head;
if (remaining) {
shortId += '-';
} else {
break;
}
}
}
return shortId;
}
/*
* take a "profile" object and return a slug based on the account name
* and DC URL. This is currently used to create a filesystem-safe name
* to use for caching
*/
function slug(o) {
var acct = o.account.replace(/[@]/g, '_');
var url = o.url.replace(/^https?:\/\//, '');
var s = format('%s@%s', acct, url).replace(/[!#$%\^&\*:'"\?\/\\\.]/g, '_');
return s;
}
2014-02-07 23:21:24 +02:00
//---- exports
module.exports = {
2015-08-26 06:53:48 +03:00
UUID_RE: UUID_RE,
2014-02-07 23:21:24 +02:00
objCopy: objCopy,
deepObjCopy: deepObjCopy,
zeroPad: zeroPad,
boolFromString: boolFromString,
jsonStream: jsonStream,
2015-08-26 03:00:50 +03:00
kvToObj: kvToObj,
2015-08-26 03:27:46 +03:00
longAgo: longAgo,
isUUID: isUUID,
2015-08-26 07:27:59 +03:00
humanDurationFromMs: humanDurationFromMs,
2015-08-26 08:57:27 +03:00
humanSizeFromBytes: humanSizeFromBytes,
capitalize: capitalize,
normShortId: normShortId,
slug: slug,
TABULA_OPTIONS: TABULA_OPTIONS
2014-02-07 23:21:24 +02:00
};
// vim: set softtabstop=4 shiftwidth=4: