appcd-client
Advanced tools
Comparing version 1.0.1 to 1.1.0-0
@@ -1,34 +0,26 @@ | ||
'use strict'; | ||
"use strict"; | ||
Object.defineProperty(exports, "__esModule", { | ||
value: true | ||
value: true | ||
}); | ||
exports.default = undefined; | ||
exports.default = void 0; | ||
var _fs = require('fs'); | ||
var _appcdLogger = _interopRequireDefault(require("appcd-logger")); | ||
var _fs2 = _interopRequireDefault(_fs); | ||
var _fs = _interopRequireDefault(require("fs")); | ||
var _msgpackLite = require('msgpack-lite'); | ||
var _msgpackLite = _interopRequireDefault(require("msgpack-lite")); | ||
var _msgpackLite2 = _interopRequireDefault(_msgpackLite); | ||
var _path = _interopRequireDefault(require("path")); | ||
var _path = require('path'); | ||
var _uuid = _interopRequireDefault(require("uuid")); | ||
var _path2 = _interopRequireDefault(_path); | ||
var _ws = _interopRequireDefault(require("ws")); | ||
var _uuid = require('uuid'); | ||
var _appcdUtil = require("appcd-util"); | ||
var _uuid2 = _interopRequireDefault(_uuid); | ||
var _events = require("events"); | ||
var _ws = require('ws'); | ||
var _appcdResponse = require("appcd-response"); | ||
var _ws2 = _interopRequireDefault(_ws); | ||
var _appcdUtil = require('appcd-util'); | ||
var _events = require('events'); | ||
var _appcdResponse = require('appcd-response'); | ||
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } | ||
@@ -38,194 +30,229 @@ | ||
if (!Error.prepareStackTrace) { | ||
require('source-map-support/register'); | ||
require('source-map-support/register'); | ||
} | ||
const { | ||
log | ||
} = (0, _appcdLogger.default)('appcd:client'); | ||
const { | ||
alert, | ||
highlight, | ||
note, | ||
ok | ||
} = _appcdLogger.default.styles; | ||
/** | ||
* The client for connecting to the appcd server. | ||
*/ | ||
let Client = class Client { | ||
/** | ||
* Initializes the client. | ||
* | ||
* @param {Object} [opts] - Various options. | ||
* @param {String} [opts.host='127.0.0.1'] - The host to connect to. | ||
* @param {Number} [opts.port=1732] - The port to connect to. | ||
* @param {String} [opts.userAgent] - The user agent containing the name and | ||
* version of the client. If not specified, one will be generated. | ||
* @access public | ||
*/ | ||
constructor(opts = {}) { | ||
/** | ||
* The websocket to the server. | ||
* @type {WebSocket} | ||
* @access private | ||
/** | ||
* Initializes the client. | ||
* | ||
* @param {Object} [opts] - Various options. | ||
* @param {String} [opts.host='127.0.0.1'] - The host to connect to. | ||
* @param {Number} [opts.port=1732] - The port to connect to. | ||
* @param {String} [opts.userAgent] - The user agent containing the name and | ||
* version of the client. If not specified, one will be generated. | ||
* @access public | ||
*/ | ||
this.socket = null; | ||
constructor(opts = {}) { | ||
/** | ||
* The websocket to the server. | ||
* @type {WebSocket} | ||
* @access private | ||
*/ | ||
this.socket = null; | ||
/** | ||
* An internal map used to dispatch responses to requesters. | ||
* @type {Object} | ||
* @access private | ||
*/ | ||
/** | ||
* An internal map used to dispatch responses to requesters. | ||
* @type {Object} | ||
* @access private | ||
*/ | ||
this.requests = {}; | ||
this.requests = {}; | ||
/** | ||
* The host to connect to. | ||
* @type {String} | ||
* @access private | ||
*/ | ||
/** | ||
* The host to connect to. | ||
* @type {String} | ||
* @access private | ||
*/ | ||
this.host = opts.host || '127.0.0.1'; | ||
this.host = opts.host || '127.0.0.1'; | ||
/** | ||
* The port to connect to. | ||
* @type {Number} | ||
* @access private | ||
*/ | ||
/** | ||
* The port to connect to. | ||
* @type {Number} | ||
* @access private | ||
*/ | ||
if (opts.port && (typeof opts.port !== 'number' || opts.port < 1 || opts.port > 65535)) { | ||
throw new TypeError('Invalid port, expected a number between 1 and 65535'); | ||
} | ||
this.port = opts.port || 1732; | ||
if (opts.port && (typeof opts.port !== 'number' || opts.port < 1 || opts.port > 65535)) { | ||
throw new TypeError('Invalid port, expected a number between 1 and 65535'); | ||
} | ||
/** | ||
* The user agent containing the name and version of the client. If not | ||
* specified, one will be generated. | ||
* @type {String} | ||
* @access private | ||
this.port = opts.port || 1732; | ||
/** | ||
* The user agent containing the name and version of the client. If not | ||
* specified, one will be generated. | ||
* @type {String} | ||
* @access private | ||
*/ | ||
this.userAgent = constructUserAgent(opts.userAgent); | ||
} | ||
/** | ||
* Connects to the server via a websocket. You do not need to call this. | ||
* `request()` will automatically call this function. | ||
* | ||
* @returns {EventEmitter} Emits events `connected`, `close`, and `error`. | ||
* @access public | ||
*/ | ||
this.userAgent = constructUserAgent(opts.userAgent); | ||
} | ||
/** | ||
* Connects to the server via a websocket. You do not need to call this. | ||
* `request()` will automatically call this function. | ||
* | ||
* @returns {EventEmitter} Emits events `connected`, `close`, and `error`. | ||
* @access public | ||
*/ | ||
connect() { | ||
const emitter = new _events.EventEmitter(); | ||
// need to delay request so event emitter can be returned and events can | ||
// be wired up | ||
setImmediate(async () => { | ||
try { | ||
if (this.socket) { | ||
emitter.emit('connected', this); | ||
return; | ||
} | ||
connect() { | ||
const emitter = new _events.EventEmitter(); // need to delay request so event emitter can be returned and events can | ||
// be wired up | ||
const headers = { | ||
'User-Agent': this.userAgent | ||
}; | ||
setImmediate(async () => { | ||
try { | ||
if (this.socket) { | ||
emitter.emit('connected', this); | ||
return; | ||
} | ||
const localeValue = process.env.APPCD_LOCALE || (await (0, _appcdResponse.locale)()); | ||
if (localeValue) { | ||
headers['Accept-Language'] = localeValue; | ||
} | ||
const headers = { | ||
'User-Agent': this.userAgent | ||
}; | ||
const localeValue = process.env.APPCD_LOCALE || (await (0, _appcdResponse.locale)()); | ||
const socket = this.socket = new _ws2.default(`ws://${this.host}:${this.port}`, { | ||
headers | ||
}); | ||
if (localeValue) { | ||
headers['Accept-Language'] = localeValue; | ||
} | ||
socket.on('message', data => { | ||
let json = null; | ||
if (typeof data === 'string') { | ||
try { | ||
json = JSON.parse(data); | ||
} catch (e) { | ||
// bad response, shouldn't ever happen | ||
emitter.emit('warning', `Server returned invalid JSON: ${e.message}`); | ||
return; | ||
} | ||
} else { | ||
json = _msgpackLite2.default.decode(data); | ||
} | ||
const url = `ws://${this.host}:${this.port}`; | ||
log(`Connecting to ${highlight(url)}`); | ||
const socket = this.socket = new _ws.default(url, { | ||
headers | ||
}); | ||
socket.on('message', data => { | ||
let json = null; | ||
if (json && typeof json === 'object' && this.requests[json.id]) { | ||
this.requests[json.id](json); | ||
} else { | ||
emitter.emit('warning', 'Server response is not an object or has an invalid id'); | ||
} | ||
}); | ||
if (typeof data === 'string') { | ||
try { | ||
json = JSON.parse(data); | ||
} catch (e) { | ||
// bad response, shouldn't ever happen | ||
emitter.emit('warning', `Server returned invalid JSON: ${e.message}`); | ||
return; | ||
} | ||
} else { | ||
json = _msgpackLite.default.decode(data); | ||
} | ||
socket.on('open', () => emitter.emit('connected', this)).once('close', () => emitter.emit('close')).once('error', err => { | ||
socket.close(); | ||
this.socket = null; | ||
emitter.emit('error', err); | ||
}); | ||
} catch (e) { | ||
emitter.emit('error', e); | ||
} | ||
}); | ||
if (json && typeof json === 'object' && this.requests[json.id]) { | ||
this.requests[json.id](json); | ||
} else { | ||
emitter.emit('warning', 'Server response is not an object or has an invalid id'); | ||
} | ||
}).on('open', () => emitter.emit('connected', this)).once('close', () => emitter.emit('close')).once('error', err => { | ||
socket.close(); | ||
this.socket = null; | ||
emitter.emit('error', err); | ||
}); | ||
} catch (e) { | ||
emitter.emit('error', e); | ||
} | ||
}); | ||
return emitter; | ||
} | ||
/** | ||
* Issues a request to the server over a websocket. | ||
* | ||
* @param {String} path - The path to send. | ||
* @param {Object} [data] - An object to send. | ||
* @param {String} [type] - The request type. Valid types include `call`, `subscribe`, and | ||
* `unsubscribe`. | ||
* @returns {EventEmitter} Emits events `response` and `error`. | ||
* @access public | ||
*/ | ||
return emitter; | ||
} | ||
/** | ||
* Issues a request to the server over a websocket. | ||
* | ||
* @param {String} path - The path to send. | ||
* @param {Object} [data] - An object to send. | ||
* @param {String} [type] - The request type. Valid types include `call`, `subscribe`, and | ||
* `unsubscribe`. | ||
* @returns {EventEmitter} Emits events `response` and `error`. | ||
* @access public | ||
*/ | ||
request({ path, data, type } = {}) { | ||
const emitter = new _events.EventEmitter(); | ||
request({ | ||
path, | ||
data, | ||
type | ||
} = {}) { | ||
const emitter = new _events.EventEmitter(); | ||
const startTime = new Date(); // need to delay request so event emitter can be returned and events can | ||
// be wired up | ||
// need to delay request so event emitter can be returned and events can | ||
// be wired up | ||
setImmediate(() => { | ||
const id = _uuid2.default.v4(); | ||
setImmediate(() => { | ||
const id = _uuid.default.v4(); | ||
return this.connect().on('connected', client => { | ||
this.requests[id] = response => { | ||
const status = ~~response.status || 500; | ||
const statusClass = Math.floor(status / 100); | ||
return this.connect().on('connected', client => { | ||
this.requests[id] = response => { | ||
const status = response.status = ~~response.status || 500; | ||
const statusClass = Math.floor(status / 100); | ||
const style = status < 400 ? ok : alert; // no need for the id anymore | ||
switch (statusClass) { | ||
case 2: | ||
emitter.emit('response', response.message, response); | ||
break; | ||
delete response.id; | ||
log(`${style(status)} ${highlight(path)} ${note(`${new Date() - startTime}ms`)}`); | ||
case 4: | ||
case 5: | ||
const err = new Error(response.message || 'Server Error'); | ||
err.errorCode = status; | ||
err.code = String(response.statusCode ? response.statusCode : status); | ||
emitter.emit('error', err, response); | ||
} | ||
}; | ||
switch (statusClass) { | ||
case 2: | ||
emitter.emit('response', response.message, response); | ||
break; | ||
client.socket.send(JSON.stringify({ | ||
version: '1.0', | ||
path: path, | ||
id: id, | ||
data, | ||
type | ||
})); | ||
}).on('warning', (...args) => emitter.emit('warning', ...args)).once('close', () => { | ||
delete this.requests[id]; | ||
}).once('error', err => { | ||
delete this.requests[id]; | ||
emitter.emit('error', err); | ||
}); | ||
}); | ||
case 4: | ||
case 5: | ||
const err = new Error(response.message || 'Server Error'); | ||
return emitter; | ||
} | ||
if (!response.statusCode) { | ||
response.statusCode = String(status); | ||
} | ||
/** | ||
* Disconnects from the server. | ||
* | ||
* @access public | ||
*/ | ||
disconnect() { | ||
if (this.socket) { | ||
this.socket.close(); | ||
} | ||
this.socket = null; | ||
} | ||
for (const prop of Object.keys(response)) { | ||
// we need to use defineProperty() to force properties to be created | ||
Object.defineProperty(err, prop, { | ||
configurable: true, | ||
enumerable: true, | ||
value: response[prop], | ||
writable: true | ||
}); | ||
} | ||
emitter.emit('error', err, response); | ||
} | ||
}; | ||
const req = { | ||
version: '1.0', | ||
path, | ||
id, | ||
data, | ||
type | ||
}; | ||
log('Sending request:', req); | ||
client.socket.send(JSON.stringify(req)); | ||
}).on('warning', (...args) => emitter.emit('warning', ...args)).once('close', () => { | ||
delete this.requests[id]; | ||
}).once('error', err => { | ||
delete this.requests[id]; | ||
emitter.emit('error', err); | ||
}); | ||
}); | ||
return emitter; | ||
} | ||
/** | ||
* Disconnects from the server. | ||
* | ||
* @access public | ||
*/ | ||
disconnect() { | ||
if (this.socket) { | ||
this.socket.close(); | ||
} | ||
this.socket = null; | ||
} | ||
}; | ||
/** | ||
@@ -241,56 +268,60 @@ * Generates a user agent string containing the name of the parent-most script | ||
exports.default = Client; | ||
function constructUserAgent(userAgent) { | ||
if (userAgent && typeof userAgent !== 'string') { | ||
throw new TypeError('Expected user agent to be a string'); | ||
} | ||
if (userAgent && typeof userAgent !== 'string') { | ||
throw new TypeError('Expected user agent to be a string'); | ||
} | ||
const parts = userAgent ? userAgent.split(' ') : []; | ||
const parts = userAgent ? userAgent.split(' ') : []; | ||
if (!parts.length) { | ||
let entry = module; | ||
while (entry.parent) { | ||
entry = entry.parent; | ||
} | ||
if (!parts.length) { | ||
let entry = module; | ||
const name = _path2.default.basename(entry.filename); | ||
const root = _path2.default.resolve('/'); | ||
let dir = _path2.default.dirname(entry.filename); | ||
while (entry.parent) { | ||
entry = entry.parent; | ||
} | ||
do { | ||
const pkgJsonFile = _path2.default.join(dir, 'package.json'); | ||
const name = _path.default.basename(entry.filename); | ||
try { | ||
if (_fs2.default.statSync(pkgJsonFile)) { | ||
parts.push(`${name}/${JSON.parse(_fs2.default.readFileSync(pkgJsonFile)).version || ''}`); | ||
break; | ||
} | ||
} catch (e) { | ||
// either the package.json doesn't exist or the JSON was malformed | ||
if (e.code !== 'ENOENT') { | ||
// must be malformed JSON, we can stop | ||
break; | ||
} | ||
} | ||
const root = _path.default.resolve('/'); | ||
dir = _path2.default.dirname(dir); | ||
} while (dir !== root); | ||
} | ||
let dir = _path.default.dirname(entry.filename); | ||
parts.push(`appcd-client/${JSON.parse(_fs2.default.readFileSync(_path2.default.resolve(__dirname, '..', 'package.json'))).version}`); | ||
do { | ||
const pkgJsonFile = _path.default.join(dir, 'package.json'); | ||
if (!parts.some(p => p.indexOf('node/') === 0)) { | ||
parts.push(`node/${process.version.replace(/^v/, '')}`); | ||
} | ||
try { | ||
if (_fs.default.statSync(pkgJsonFile)) { | ||
parts.push(`${name}/${JSON.parse(_fs.default.readFileSync(pkgJsonFile)).version || ''}`); | ||
break; | ||
} | ||
} catch (e) { | ||
// either the package.json doesn't exist or the JSON was malformed | ||
if (e.code !== 'ENOENT') { | ||
// must be malformed JSON, we can stop | ||
break; | ||
} | ||
} | ||
if (!parts.some(p => p === process.platform)) { | ||
parts.push(process.platform); | ||
} | ||
dir = _path.default.dirname(dir); | ||
} while (dir !== root); | ||
} | ||
const architecture = (0, _appcdUtil.arch)(); | ||
if (!parts.some(p => p === architecture)) { | ||
parts.push(architecture); | ||
} | ||
parts.push(`appcd-client/${JSON.parse(_fs.default.readFileSync(_path.default.resolve(__dirname, '..', 'package.json'))).version}`); | ||
return parts.join(' '); | ||
} | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["client.js"],"names":["Error","prepareStackTrace","require","Client","constructor","opts","socket","requests","host","port","TypeError","userAgent","constructUserAgent","connect","emitter","setImmediate","emit","headers","localeValue","process","env","APPCD_LOCALE","on","data","json","JSON","parse","e","message","decode","id","once","err","close","request","path","type","v4","client","response","status","statusClass","Math","floor","errorCode","code","String","statusCode","send","stringify","version","args","disconnect","parts","split","length","entry","module","parent","name","basename","filename","root","resolve","dir","dirname","pkgJsonFile","join","statSync","push","readFileSync","__dirname","some","p","indexOf","replace","platform","architecture"],"mappings":"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","file":"client.js","sourceRoot":"src","sourcesContent":["/* istanbul ignore if */\nif (!Error.prepareStackTrace) {\n\trequire('source-map-support/register');\n}\n\nimport fs from 'fs';\nimport msgpack from 'msgpack-lite';\nimport path from 'path';\nimport uuid from 'uuid';\nimport WebSocket from 'ws';\n\nimport { arch } from 'appcd-util';\nimport { EventEmitter } from 'events';\nimport { locale } from 'appcd-response';\n\n/**\n * The client for connecting to the appcd server.\n */\nexport default class Client {\n\t/**\n\t * Initializes the client.\n\t *\n\t * @param {Object} [opts] - Various options.\n\t * @param {String} [opts.host='127.0.0.1'] - The host to connect to.\n\t * @param {Number} [opts.port=1732] - The port to connect to.\n\t * @param {String} [opts.userAgent] - The user agent containing the name and\n\t * version of the client. If not specified, one will be generated.\n\t * @access public\n\t */\n\tconstructor(opts = {}) {\n\t\t/**\n\t\t * The websocket to the server.\n\t\t * @type {WebSocket}\n\t\t * @access private\n\t\t */\n\t\tthis.socket = null;\n\n\t\t/**\n\t\t * An internal map used to dispatch responses to requesters.\n\t\t * @type {Object}\n\t\t * @access private\n\t\t */\n\t\tthis.requests = {};\n\n\t\t/**\n\t\t * The host to connect to.\n\t\t * @type {String}\n\t\t * @access private\n\t\t */\n\t\tthis.host = opts.host || '127.0.0.1';\n\n\t\t/**\n\t\t * The port to connect to.\n\t\t * @type {Number}\n\t\t * @access private\n\t\t */\n\t\tif (opts.port && (typeof opts.port !== 'number' || opts.port < 1 || opts.port > 65535)) {\n\t\t\tthrow new TypeError('Invalid port, expected a number between 1 and 65535');\n\t\t}\n\t\tthis.port = opts.port || 1732;\n\n\t\t/**\n\t\t * The user agent containing the name and version of the client. If not\n\t\t * specified, one will be generated.\n\t\t * @type {String}\n\t\t * @access private\n\t\t */\n\t\tthis.userAgent = constructUserAgent(opts.userAgent);\n\t}\n\n\t/**\n\t * Connects to the server via a websocket. You do not need to call this.\n\t * `request()` will automatically call this function.\n\t *\n\t * @returns {EventEmitter} Emits events `connected`, `close`, and `error`.\n\t * @access public\n\t */\n\tconnect() {\n\t\tconst emitter = new EventEmitter();\n\n\t\t// need to delay request so event emitter can be returned and events can\n\t\t// be wired up\n\t\tsetImmediate(async () => {\n\t\t\ttry {\n\t\t\t\tif (this.socket) {\n\t\t\t\t\temitter.emit('connected', this);\n\t\t\t\t\treturn;\n\t\t\t\t}\n\n\t\t\t\tconst headers = {\n\t\t\t\t\t'User-Agent': this.userAgent\n\t\t\t\t};\n\n\t\t\t\tconst localeValue = process.env.APPCD_LOCALE || await locale();\n\t\t\t\tif (localeValue) {\n\t\t\t\t\theaders['Accept-Language'] = localeValue;\n\t\t\t\t}\n\n\t\t\t\tconst socket = this.socket = new WebSocket(`ws://${this.host}:${this.port}`, {\n\t\t\t\t\theaders\n\t\t\t\t});\n\n\t\t\t\tsocket.on('message', data => {\n\t\t\t\t\tlet json = null;\n\t\t\t\t\tif (typeof data === 'string') {\n\t\t\t\t\t\ttry {\n\t\t\t\t\t\t\tjson = JSON.parse(data);\n\t\t\t\t\t\t} catch (e) {\n\t\t\t\t\t\t\t// bad response, shouldn't ever happen\n\t\t\t\t\t\t\temitter.emit('warning', `Server returned invalid JSON: ${e.message}`);\n\t\t\t\t\t\t\treturn;\n\t\t\t\t\t\t}\n\t\t\t\t\t} else {\n\t\t\t\t\t\tjson = msgpack.decode(data);\n\t\t\t\t\t}\n\n\t\t\t\t\tif (json && typeof json === 'object' && this.requests[json.id]) {\n\t\t\t\t\t\tthis.requests[json.id](json);\n\t\t\t\t\t} else {\n\t\t\t\t\t\temitter.emit('warning', 'Server response is not an object or has an invalid id');\n\t\t\t\t\t}\n\t\t\t\t});\n\n\t\t\t\tsocket\n\t\t\t\t\t.on('open', () => emitter.emit('connected', this))\n\t\t\t\t\t.once('close', () => emitter.emit('close'))\n\t\t\t\t\t.once('error', err => {\n\t\t\t\t\t\tsocket.close();\n\t\t\t\t\t\tthis.socket = null;\n\t\t\t\t\t\temitter.emit('error', err);\n\t\t\t\t\t});\n\t\t\t} catch (e) {\n\t\t\t\temitter.emit('error', e);\n\t\t\t}\n\t\t});\n\n\t\treturn emitter;\n\t}\n\n\t/**\n\t * Issues a request to the server over a websocket.\n\t *\n\t * @param {String} path - The path to send.\n\t * @param {Object} [data] - An object to send.\n\t * @param {String} [type] - The request type. Valid types include `call`, `subscribe`, and\n\t * `unsubscribe`.\n\t * @returns {EventEmitter} Emits events `response` and `error`.\n\t * @access public\n\t */\n\trequest({ path, data, type } = {}) {\n\t\tconst emitter = new EventEmitter();\n\n\t\t// need to delay request so event emitter can be returned and events can\n\t\t// be wired up\n\t\tsetImmediate(() => {\n\t\t\tconst id = uuid.v4();\n\n\t\t\treturn this.connect()\n\t\t\t\t.on('connected', client => {\n\t\t\t\t\tthis.requests[id] = response => {\n\t\t\t\t\t\tconst status = ~~response.status || 500;\n\t\t\t\t\t\tconst statusClass = Math.floor(status / 100);\n\n\t\t\t\t\t\tswitch (statusClass) {\n\t\t\t\t\t\t\tcase 2:\n\t\t\t\t\t\t\t\temitter.emit('response', response.message, response);\n\t\t\t\t\t\t\t\tbreak;\n\n\t\t\t\t\t\t\tcase 4:\n\t\t\t\t\t\t\tcase 5:\n\t\t\t\t\t\t\t\tconst err = new Error(response.message || 'Server Error');\n\t\t\t\t\t\t\t\terr.errorCode = status;\n\t\t\t\t\t\t\t\terr.code = String(response.statusCode ? response.statusCode : status);\n\t\t\t\t\t\t\t\temitter.emit('error', err, response);\n\t\t\t\t\t\t}\n\t\t\t\t\t};\n\n\t\t\t\t\tclient.socket.send(JSON.stringify({\n\t\t\t\t\t\tversion: '1.0',\n\t\t\t\t\t\tpath:    path,\n\t\t\t\t\t\tid:      id,\n\t\t\t\t\t\tdata,\n\t\t\t\t\t\ttype\n\t\t\t\t\t}));\n\t\t\t\t})\n\t\t\t\t.on('warning', (...args) => emitter.emit('warning', ...args))\n\t\t\t\t.once('close', () => {\n\t\t\t\t\tdelete this.requests[id];\n\t\t\t\t})\n\t\t\t\t.once('error', err => {\n\t\t\t\t\tdelete this.requests[id];\n\t\t\t\t\temitter.emit('error', err);\n\t\t\t\t});\n\t\t});\n\n\t\treturn emitter;\n\t}\n\n\t/**\n\t * Disconnects from the server.\n\t *\n\t * @access public\n\t */\n\tdisconnect() {\n\t\tif (this.socket) {\n\t\t\tthis.socket.close();\n\t\t}\n\t\tthis.socket = null;\n\t}\n}\n\n/**\n * Generates a user agent string containing the name of the parent-most script\n * name, Node.js version, platform name, and architecture.\n *\n * @param {String} [userAgent] - The invoking client's user agent. This simply needs to be the\n * `name/version`.\n * @returns {String}\n */\nfunction constructUserAgent(userAgent) {\n\tif (userAgent && typeof userAgent !== 'string') {\n\t\tthrow new TypeError('Expected user agent to be a string');\n\t}\n\n\tconst parts = userAgent ? userAgent.split(' ') : [];\n\n\tif (!parts.length) {\n\t\tlet entry = module;\n\t\twhile (entry.parent) {\n\t\t\tentry = entry.parent;\n\t\t}\n\n\t\tconst name = path.basename(entry.filename);\n\t\tconst root = path.resolve('/');\n\t\tlet dir = path.dirname(entry.filename);\n\n\t\tdo {\n\t\t\tconst pkgJsonFile = path.join(dir, 'package.json');\n\n\t\t\ttry {\n\t\t\t\tif (fs.statSync(pkgJsonFile)) {\n\t\t\t\t\tparts.push(`${name}/${JSON.parse(fs.readFileSync(pkgJsonFile)).version || ''}`);\n\t\t\t\t\tbreak;\n\t\t\t\t}\n\t\t\t} catch (e) {\n\t\t\t\t// either the package.json doesn't exist or the JSON was malformed\n\t\t\t\tif (e.code !== 'ENOENT') {\n\t\t\t\t\t// must be malformed JSON, we can stop\n\t\t\t\t\tbreak;\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tdir = path.dirname(dir);\n\t\t} while (dir !== root);\n\t}\n\n\tparts.push(`appcd-client/${JSON.parse(fs.readFileSync(path.resolve(__dirname, '..', 'package.json'))).version}`);\n\n\tif (!parts.some(p => p.indexOf('node/') === 0)) {\n\t\tparts.push(`node/${process.version.replace(/^v/, '')}`);\n\t}\n\n\tif (!parts.some(p => p === process.platform)) {\n\t\tparts.push(process.platform);\n\t}\n\n\tconst architecture = arch();\n\tif (!parts.some(p => p === architecture)) {\n\t\tparts.push(architecture);\n\t}\n\n\treturn parts.join(' ');\n}\n"]} | ||
if (!parts.some(p => p.indexOf('node/') === 0)) { | ||
parts.push(`node/${process.version.replace(/^v/, '')}`); | ||
} | ||
if (!parts.some(p => p === process.platform)) { | ||
parts.push(process.platform); | ||
} | ||
const architecture = (0, _appcdUtil.arch)(); | ||
if (!parts.some(p => p === architecture)) { | ||
parts.push(architecture); | ||
} | ||
return parts.join(' '); | ||
} |
{ | ||
"name": "appcd-client", | ||
"version": "1.0.1", | ||
"version": "1.1.0-0", | ||
"description": "The Node.js client for connecting to the Appc Daemon.", | ||
@@ -25,11 +25,12 @@ "main": "./dist/client", | ||
"dependencies": { | ||
"appcd-response": "^1.0.1", | ||
"appcd-util": "^1.0.1", | ||
"appcd-logger": "^1.1.0-0", | ||
"appcd-response": "^1.1.0-0", | ||
"appcd-util": "^1.1.0-0", | ||
"msgpack-lite": "^0.1.26", | ||
"source-map-support": "^0.5.0", | ||
"uuid": "^3.1.0", | ||
"ws": "^3.3.2" | ||
"source-map-support": "^0.5.4", | ||
"uuid": "^3.2.1", | ||
"ws": "^5.1.0" | ||
}, | ||
"devDependencies": { | ||
"appcd-gulp": "^1.0.1", | ||
"appcd-gulp": "^1.1.0-0", | ||
"gulp": "^3.9.1" | ||
@@ -36,0 +37,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
Filesystem access
Supply chain riskAccesses the file system, and could potentially read sensitive data.
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
Filesystem access
Supply chain riskAccesses the file system, and could potentially read sensitive data.
Found 1 instance in 1 package
265
10967
7
4
1
+ Addedappcd-logger@^1.1.0-0
+ Addedansi-styles@3.2.1(transitive)
+ Addedappcd-logger@1.1.5(transitive)
+ Addedbase64-js@1.5.1(transitive)
+ Addedbrotli@1.3.3(transitive)
+ Addedbryt@1.0.2(transitive)
+ Addedchalk@2.4.2(transitive)
+ Addedcolor-convert@1.9.3(transitive)
+ Addedcolor-name@1.1.3(transitive)
+ Addedescape-string-regexp@1.0.5(transitive)
+ Addedfigures@2.0.0(transitive)
+ Addedhas-flag@3.0.0(transitive)
+ Addedhumanize@0.0.9(transitive)
+ Addedmoment@2.30.1(transitive)
+ Addednanobuffer@1.1.7(transitive)
+ Addedpluralize@7.0.0(transitive)
+ Addedsnooplogg@1.13.2(transitive)
+ Addedsupports-color@5.5.0(transitive)
+ Addedws@5.2.4(transitive)
- Removedsafe-buffer@5.1.2(transitive)
- Removedultron@1.1.1(transitive)
- Removedws@3.3.3(transitive)
Updatedappcd-response@^1.1.0-0
Updatedappcd-util@^1.1.0-0
Updatedsource-map-support@^0.5.4
Updateduuid@^3.2.1
Updatedws@^5.1.0