2017-08-27 15:52:07 +08:00
|
|
|
/* eslint no-param-reassign: 0 */
|
|
|
|
const assert = require('assert');
|
|
|
|
const crypto = require('crypto');
|
|
|
|
const validate = require('@segment/loosely-validate-event');
|
|
|
|
const debug = require('debug')('analytics-node');
|
2017-09-27 02:33:08 +08:00
|
|
|
const version = `3.0.0`;
|
2017-08-27 15:52:07 +08:00
|
|
|
|
|
|
|
// BG: Dependencies of analytics-node I lifted in
|
|
|
|
|
|
|
|
// https://github.com/segmentio/crypto-token/blob/master/lib/index.js
|
|
|
|
const chars = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789';
|
|
|
|
|
|
|
|
function uid(length, fn) {
|
2017-09-27 02:33:08 +08:00
|
|
|
const str = bytes => {
|
2017-08-27 15:52:07 +08:00
|
|
|
const res = [];
|
|
|
|
for (let i = 0; i < bytes.length; i++) {
|
|
|
|
res.push(chars[bytes[i] % chars.length]);
|
|
|
|
}
|
|
|
|
return res.join('');
|
2017-09-27 02:33:08 +08:00
|
|
|
};
|
2017-08-27 15:52:07 +08:00
|
|
|
|
|
|
|
if (typeof length === 'function') {
|
|
|
|
fn = length;
|
|
|
|
length = null;
|
|
|
|
}
|
|
|
|
if (!length) {
|
|
|
|
length = 10;
|
|
|
|
}
|
|
|
|
if (!fn) {
|
|
|
|
return str(crypto.randomBytes(length));
|
|
|
|
}
|
|
|
|
|
|
|
|
crypto.randomBytes(length, (err, bytes) => {
|
|
|
|
if (err) {
|
|
|
|
fn(err);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
fn(null, str(bytes));
|
|
|
|
});
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
// https://github.com/stephenmathieson/remove-trailing-slash/blob/master/index.js
|
|
|
|
function removeSlash(str) {
|
|
|
|
return String(str).replace(/\/+$/, '');
|
|
|
|
}
|
|
|
|
|
2017-09-27 02:33:08 +08:00
|
|
|
const setImmediate = global.setImmediate || process.nextTick.bind(process);
|
|
|
|
const noop = () => {};
|
2017-08-27 15:52:07 +08:00
|
|
|
|
|
|
|
export default class Analytics {
|
|
|
|
/**
|
|
|
|
* Initialize a new `Analytics` with your Segment project's `writeKey` and an
|
|
|
|
* optional dictionary of `options`.
|
|
|
|
*
|
|
|
|
* @param {String} writeKey
|
|
|
|
* @param {Object} [options] (optional)
|
|
|
|
* @property {Number} flushAt (default: 20)
|
|
|
|
* @property {Number} flushInterval (default: 10000)
|
|
|
|
* @property {String} host (default: 'https://api.segment.io')
|
|
|
|
*/
|
|
|
|
|
|
|
|
constructor(writeKey, options) {
|
2017-09-27 02:33:08 +08:00
|
|
|
options = options || {};
|
2017-08-27 15:52:07 +08:00
|
|
|
|
2017-09-27 02:33:08 +08:00
|
|
|
assert(writeKey, "You must pass your Segment project's write key.");
|
2017-08-27 15:52:07 +08:00
|
|
|
|
2017-09-27 02:33:08 +08:00
|
|
|
this.queue = [];
|
|
|
|
this.writeKey = writeKey;
|
|
|
|
this.host = removeSlash(options.host || 'https://api.segment.io');
|
|
|
|
this.flushAt = Math.max(options.flushAt, 1) || 20;
|
|
|
|
this.flushInterval = options.flushInterval || 10000;
|
|
|
|
this.flushed = false;
|
2017-08-27 15:52:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Send an identify `message`.
|
|
|
|
*
|
|
|
|
* @param {Object} message
|
|
|
|
* @param {Function} [callback] (optional)
|
|
|
|
* @return {Analytics}
|
|
|
|
*/
|
|
|
|
|
|
|
|
identify(message, callback) {
|
2017-09-27 02:33:08 +08:00
|
|
|
validate(message, 'identify');
|
|
|
|
this.enqueue('identify', message, callback);
|
|
|
|
return this;
|
2017-08-27 15:52:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Send a group `message`.
|
|
|
|
*
|
|
|
|
* @param {Object} message
|
|
|
|
* @param {Function} [callback] (optional)
|
|
|
|
* @return {Analytics}
|
|
|
|
*/
|
|
|
|
|
|
|
|
group(message, callback) {
|
2017-09-27 02:33:08 +08:00
|
|
|
validate(message, 'group');
|
|
|
|
this.enqueue('group', message, callback);
|
|
|
|
return this;
|
2017-08-27 15:52:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Send a track `message`.
|
|
|
|
*
|
|
|
|
* @param {Object} message
|
|
|
|
* @param {Function} [callback] (optional)
|
|
|
|
* @return {Analytics}
|
|
|
|
*/
|
|
|
|
|
|
|
|
track(message, callback) {
|
2017-09-27 02:33:08 +08:00
|
|
|
validate(message, 'track');
|
|
|
|
this.enqueue('track', message, callback);
|
|
|
|
return this;
|
2017-08-27 15:52:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Send a page `message`.
|
|
|
|
*
|
|
|
|
* @param {Object} message
|
|
|
|
* @param {Function} [callback] (optional)
|
|
|
|
* @return {Analytics}
|
|
|
|
*/
|
|
|
|
|
|
|
|
page(message, callback) {
|
2017-09-27 02:33:08 +08:00
|
|
|
validate(message, 'page');
|
|
|
|
this.enqueue('page', message, callback);
|
|
|
|
return this;
|
2017-08-27 15:52:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Send a screen `message`.
|
|
|
|
*
|
|
|
|
* @param {Object} message
|
|
|
|
* @param {Function} fn (optional)
|
|
|
|
* @return {Analytics}
|
|
|
|
*/
|
|
|
|
|
|
|
|
screen(message, callback) {
|
2017-09-27 02:33:08 +08:00
|
|
|
validate(message, 'screen');
|
|
|
|
this.enqueue('screen', message, callback);
|
|
|
|
return this;
|
2017-08-27 15:52:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Send an alias `message`.
|
|
|
|
*
|
|
|
|
* @param {Object} message
|
|
|
|
* @param {Function} [callback] (optional)
|
|
|
|
* @return {Analytics}
|
|
|
|
*/
|
|
|
|
|
|
|
|
alias(message, callback) {
|
2017-09-27 02:33:08 +08:00
|
|
|
validate(message, 'alias');
|
|
|
|
this.enqueue('alias', message, callback);
|
|
|
|
return this;
|
2017-08-27 15:52:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Add a `message` of type `type` to the queue and
|
|
|
|
* check whether it should be flushed.
|
|
|
|
*
|
|
|
|
* @param {String} type
|
|
|
|
* @param {Object} message
|
|
|
|
* @param {Functino} [callback] (optional)
|
|
|
|
* @api private
|
|
|
|
*/
|
|
|
|
|
|
|
|
enqueue(type, message, callback) {
|
2017-09-27 02:33:08 +08:00
|
|
|
callback = callback || noop;
|
|
|
|
|
|
|
|
message = Object.assign({}, message);
|
|
|
|
message.type = type;
|
|
|
|
message.context = Object.assign(
|
|
|
|
{
|
|
|
|
library: {
|
|
|
|
name: 'analytics-node',
|
|
|
|
version,
|
|
|
|
},
|
2017-08-27 15:52:07 +08:00
|
|
|
},
|
2017-09-27 02:33:08 +08:00
|
|
|
message.context
|
|
|
|
);
|
2017-08-27 15:52:07 +08:00
|
|
|
|
2017-09-27 02:33:08 +08:00
|
|
|
message._metadata = Object.assign(
|
|
|
|
{
|
|
|
|
nodeVersion: process.versions.node,
|
|
|
|
},
|
|
|
|
message._metadata
|
|
|
|
);
|
2017-08-27 15:52:07 +08:00
|
|
|
|
|
|
|
if (!message.timestamp) {
|
2017-09-27 02:33:08 +08:00
|
|
|
message.timestamp = new Date();
|
2017-08-27 15:52:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!message.messageId) {
|
2017-09-27 02:33:08 +08:00
|
|
|
message.messageId = `node-${uid(32)}`;
|
2017-08-27 15:52:07 +08:00
|
|
|
}
|
|
|
|
|
2017-09-27 02:33:08 +08:00
|
|
|
debug('%s: %o', type, message);
|
2017-08-27 15:52:07 +08:00
|
|
|
|
2017-09-27 02:33:08 +08:00
|
|
|
this.queue.push({ message, callback });
|
2017-08-27 15:52:07 +08:00
|
|
|
|
|
|
|
if (!this.flushed) {
|
2017-09-27 02:33:08 +08:00
|
|
|
this.flushed = true;
|
|
|
|
this.flush();
|
|
|
|
return;
|
2017-08-27 15:52:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (this.queue.length >= this.flushAt) {
|
2017-09-27 02:33:08 +08:00
|
|
|
this.flush();
|
2017-08-27 15:52:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (this.flushInterval && !this.timer) {
|
2017-09-27 02:33:08 +08:00
|
|
|
this.timer = setTimeout(this.flush.bind(this), this.flushInterval);
|
2017-08-27 15:52:07 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Flush the current queue
|
|
|
|
*
|
|
|
|
* @param {Function} [callback] (optional)
|
|
|
|
* @return {Analytics}
|
|
|
|
*/
|
|
|
|
|
|
|
|
async flush(callback) {
|
2017-09-27 02:33:08 +08:00
|
|
|
callback = callback || noop;
|
2017-08-27 15:52:07 +08:00
|
|
|
|
|
|
|
if (this.timer) {
|
2017-09-27 02:33:08 +08:00
|
|
|
clearTimeout(this.timer);
|
|
|
|
this.timer = null;
|
2017-08-27 15:52:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!this.queue.length) {
|
2017-09-27 02:33:08 +08:00
|
|
|
setImmediate(callback);
|
2017-08-27 15:52:07 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-09-27 02:33:08 +08:00
|
|
|
const items = this.queue.splice(0, this.flushAt);
|
|
|
|
const callbacks = items.map(item => item.callback);
|
|
|
|
const messages = items.map(item => item.message);
|
2017-08-27 15:52:07 +08:00
|
|
|
|
|
|
|
const data = {
|
|
|
|
batch: messages,
|
|
|
|
timestamp: new Date(),
|
|
|
|
sentAt: new Date(),
|
2017-09-27 02:33:08 +08:00
|
|
|
};
|
2017-08-27 15:52:07 +08:00
|
|
|
|
2017-09-27 02:33:08 +08:00
|
|
|
debug('flush: %o', data);
|
2017-08-27 15:52:07 +08:00
|
|
|
|
|
|
|
const options = {
|
|
|
|
body: JSON.stringify(data),
|
|
|
|
credentials: 'include',
|
|
|
|
headers: new Headers(),
|
|
|
|
method: 'POST',
|
|
|
|
};
|
|
|
|
options.headers.set('Accept', 'application/json');
|
2017-09-27 02:33:08 +08:00
|
|
|
options.headers.set('Authorization', `Basic ${btoa(`${this.writeKey}:`)}`);
|
2017-08-27 15:52:07 +08:00
|
|
|
options.headers.set('Content-Type', 'application/json');
|
|
|
|
|
2017-09-27 02:33:08 +08:00
|
|
|
const runCallbacks = err => {
|
|
|
|
callbacks.forEach(cb => cb(err));
|
2017-08-27 15:52:07 +08:00
|
|
|
callback(err, data);
|
|
|
|
debug('flushed: %o', data);
|
|
|
|
};
|
|
|
|
|
|
|
|
let resp = null;
|
|
|
|
try {
|
|
|
|
resp = await fetch(`${this.host}/v1/batch`, options);
|
|
|
|
} catch (err) {
|
|
|
|
runCallbacks(err, null);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!resp.ok) {
|
|
|
|
runCallbacks(new Error(`${resp.statusCode}: ${resp.statusText}`), null);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const body = await resp.json();
|
|
|
|
if (body.error) {
|
|
|
|
runCallbacks(new Error(body.error.message), null);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
runCallbacks(null, {
|
|
|
|
status: resp.statusCode,
|
|
|
|
body: body,
|
|
|
|
ok: true,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|