2019-03-12 22:07:57 +00:00
|
|
|
/**
|
|
|
|
* @license
|
2020-03-04 16:31:54 +00:00
|
|
|
* Copyright 2019-2020 Balena Ltd.
|
2019-03-12 22:07:57 +00:00
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
2020-03-04 16:31:54 +00:00
|
|
|
import * as Sentry from '@sentry/node';
|
|
|
|
import * as Bluebird from 'bluebird';
|
|
|
|
import * as _ from 'lodash';
|
|
|
|
import * as Mixpanel from 'mixpanel';
|
2019-03-12 22:07:57 +00:00
|
|
|
|
2020-03-04 16:31:54 +00:00
|
|
|
import * as packageJSON from '../package.json';
|
2020-02-27 14:55:30 +00:00
|
|
|
import { getBalenaSdk } from './utils/lazy';
|
2017-12-20 21:46:01 +00:00
|
|
|
|
2019-02-22 07:59:36 +00:00
|
|
|
const getMixpanel = _.once<any>(() => {
|
2019-02-07 15:36:48 +00:00
|
|
|
const settings = require('balena-settings-client');
|
2019-09-18 15:00:22 +00:00
|
|
|
return Mixpanel.init('balena-main', {
|
2019-02-07 15:36:48 +00:00
|
|
|
host: `api.${settings.get('balenaUrl')}`,
|
|
|
|
path: '/mixpanel',
|
|
|
|
protocol: 'https',
|
|
|
|
});
|
|
|
|
});
|
2017-12-20 21:46:01 +00:00
|
|
|
|
2020-03-04 16:31:54 +00:00
|
|
|
/**
|
|
|
|
* Mixpanel.com analytics tracking (information on balena CLI usage).
|
|
|
|
*
|
|
|
|
* @param commandSignature A string like, for example:
|
|
|
|
* "push <applicationOrDevice>"
|
|
|
|
* That's literally so: "applicationOrDevice" is NOT replaced with the actual
|
|
|
|
* application ID or device ID. The purpose is to find out the most / least
|
|
|
|
* used command verbs, so we can focus our development effort where it is most
|
|
|
|
* beneficial to end users.
|
|
|
|
*
|
|
|
|
* The username and command signature are also added as extra context
|
|
|
|
* information in Sentry.io error reporting, for CLI debugging purposes
|
|
|
|
* (mainly unexpected/unhandled exceptions -- see also `lib/errors.ts`).
|
|
|
|
*/
|
2019-08-28 01:14:19 +00:00
|
|
|
export function trackCommand(commandSignature: string) {
|
2019-01-12 21:20:19 +00:00
|
|
|
const balena = getBalenaSdk();
|
2020-03-04 16:31:54 +00:00
|
|
|
return Bluebird.props({
|
2018-10-19 14:38:50 +00:00
|
|
|
balenaUrl: balena.settings.get('balenaUrl'),
|
|
|
|
username: balena.auth.whoami().catchReturn(undefined),
|
2018-01-04 14:07:55 +00:00
|
|
|
mixpanel: getMixpanel(),
|
2018-01-09 15:05:24 +00:00
|
|
|
})
|
2018-10-19 14:38:50 +00:00
|
|
|
.then(({ username, balenaUrl, mixpanel }) => {
|
2020-03-04 16:31:54 +00:00
|
|
|
Sentry.configureScope(scope => {
|
|
|
|
scope.setExtra('command', commandSignature);
|
|
|
|
scope.setUser({
|
|
|
|
id: username,
|
|
|
|
username,
|
2018-01-09 15:05:24 +00:00
|
|
|
});
|
2017-12-20 21:46:01 +00:00
|
|
|
});
|
2020-03-04 16:31:54 +00:00
|
|
|
return mixpanel.track(`[CLI] ${commandSignature}`, {
|
|
|
|
distinct_id: username,
|
|
|
|
version: packageJSON.version,
|
|
|
|
node: process.version,
|
|
|
|
arch: process.arch,
|
|
|
|
balenaUrl, // e.g. 'balena-cloud.com' or 'balena-staging.com'
|
|
|
|
platform: process.platform,
|
|
|
|
});
|
2018-01-09 15:05:24 +00:00
|
|
|
})
|
|
|
|
.timeout(100)
|
|
|
|
.catchReturn(undefined);
|
2018-01-04 14:07:55 +00:00
|
|
|
}
|