@local-labs-jpollock/local-cli 0.0.5 → 0.0.6
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/addon-dist/package.json +1 -1
- package/lib/analytics.d.ts +55 -0
- package/lib/analytics.js +447 -0
- package/lib/index.js +110 -10
- package/package.json +1 -1
package/addon-dist/package.json
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@local-labs-jpollock/local-addon-cli",
|
|
3
3
|
"productName": "Local CLI",
|
|
4
|
-
"version": "0.0.
|
|
4
|
+
"version": "0.0.5",
|
|
5
5
|
"description": "Command-line interface for Local WordPress development",
|
|
6
6
|
"main": "lib/main/index.js",
|
|
7
7
|
"renderer": "lib/renderer/index.js",
|
|
@@ -0,0 +1,55 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Anonymous Usage Analytics - Phase 3 (Signed Authentication)
|
|
3
|
+
*
|
|
4
|
+
* Collects anonymous usage data with user consent and transmits to server.
|
|
5
|
+
* All requests are signed with HMAC-SHA256 for authentication.
|
|
6
|
+
*
|
|
7
|
+
* Privacy: Only tracks command names, success/failure, duration, and system info.
|
|
8
|
+
* Never tracks: arguments, site names, paths, or any PII.
|
|
9
|
+
*
|
|
10
|
+
* Identifiers:
|
|
11
|
+
* - installationId: Random UUID, identifies this CLI installation (not the user)
|
|
12
|
+
* - secretKey: Random 32 bytes, used for HMAC signing (never transmitted after registration)
|
|
13
|
+
* - sessionId: Random UUID per CLI invocation, correlates commands in a session
|
|
14
|
+
*/
|
|
15
|
+
export type ErrorCategory = 'site_not_found' | 'site_not_running' | 'local_not_running' | 'timeout' | 'network_error' | 'validation_error' | 'unknown';
|
|
16
|
+
interface AnalyticsEvent {
|
|
17
|
+
command: string;
|
|
18
|
+
success: boolean;
|
|
19
|
+
duration_ms: number;
|
|
20
|
+
timestamp: string;
|
|
21
|
+
installation_id: string;
|
|
22
|
+
session_id: string;
|
|
23
|
+
cli_version: string;
|
|
24
|
+
os: string;
|
|
25
|
+
node_version: string;
|
|
26
|
+
error_category?: ErrorCategory;
|
|
27
|
+
}
|
|
28
|
+
export declare function getInstallationId(): string;
|
|
29
|
+
export declare function getSecretKey(): string;
|
|
30
|
+
export declare function getSessionId(): string;
|
|
31
|
+
export declare function isAnalyticsEnabled(): boolean;
|
|
32
|
+
export declare function setAnalyticsEnabled(enabled: boolean): void;
|
|
33
|
+
export declare function hasBeenPrompted(): boolean;
|
|
34
|
+
export declare function showOptInPrompt(): Promise<boolean>;
|
|
35
|
+
export declare function recordEvent(event: AnalyticsEvent): void;
|
|
36
|
+
export declare function readEvents(): AnalyticsEvent[];
|
|
37
|
+
export declare function clearEvents(): void;
|
|
38
|
+
/**
|
|
39
|
+
* Reset analytics: clear events and regenerate installationId + secretKey
|
|
40
|
+
*/
|
|
41
|
+
export declare function resetAnalytics(): void;
|
|
42
|
+
export declare function startTracking(commandName: string): void;
|
|
43
|
+
export declare function finishTracking(success: boolean, errorCategory?: ErrorCategory): void;
|
|
44
|
+
/**
|
|
45
|
+
* Generate a signed dashboard URL for viewing personal analytics
|
|
46
|
+
* URL expires in 1 hour
|
|
47
|
+
*/
|
|
48
|
+
export declare function getDashboardUrl(): string;
|
|
49
|
+
export declare function getStatus(): {
|
|
50
|
+
enabled: boolean;
|
|
51
|
+
eventCount: number;
|
|
52
|
+
installationId: string;
|
|
53
|
+
};
|
|
54
|
+
export declare function getSummary(): string;
|
|
55
|
+
export {};
|
package/lib/analytics.js
ADDED
|
@@ -0,0 +1,447 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* Anonymous Usage Analytics - Phase 3 (Signed Authentication)
|
|
4
|
+
*
|
|
5
|
+
* Collects anonymous usage data with user consent and transmits to server.
|
|
6
|
+
* All requests are signed with HMAC-SHA256 for authentication.
|
|
7
|
+
*
|
|
8
|
+
* Privacy: Only tracks command names, success/failure, duration, and system info.
|
|
9
|
+
* Never tracks: arguments, site names, paths, or any PII.
|
|
10
|
+
*
|
|
11
|
+
* Identifiers:
|
|
12
|
+
* - installationId: Random UUID, identifies this CLI installation (not the user)
|
|
13
|
+
* - secretKey: Random 32 bytes, used for HMAC signing (never transmitted after registration)
|
|
14
|
+
* - sessionId: Random UUID per CLI invocation, correlates commands in a session
|
|
15
|
+
*/
|
|
16
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
17
|
+
if (k2 === undefined) k2 = k;
|
|
18
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
19
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
20
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
21
|
+
}
|
|
22
|
+
Object.defineProperty(o, k2, desc);
|
|
23
|
+
}) : (function(o, m, k, k2) {
|
|
24
|
+
if (k2 === undefined) k2 = k;
|
|
25
|
+
o[k2] = m[k];
|
|
26
|
+
}));
|
|
27
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
28
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
29
|
+
}) : function(o, v) {
|
|
30
|
+
o["default"] = v;
|
|
31
|
+
});
|
|
32
|
+
var __importStar = (this && this.__importStar) || (function () {
|
|
33
|
+
var ownKeys = function(o) {
|
|
34
|
+
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
35
|
+
var ar = [];
|
|
36
|
+
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
37
|
+
return ar;
|
|
38
|
+
};
|
|
39
|
+
return ownKeys(o);
|
|
40
|
+
};
|
|
41
|
+
return function (mod) {
|
|
42
|
+
if (mod && mod.__esModule) return mod;
|
|
43
|
+
var result = {};
|
|
44
|
+
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
45
|
+
__setModuleDefault(result, mod);
|
|
46
|
+
return result;
|
|
47
|
+
};
|
|
48
|
+
})();
|
|
49
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
50
|
+
exports.getInstallationId = getInstallationId;
|
|
51
|
+
exports.getSecretKey = getSecretKey;
|
|
52
|
+
exports.getSessionId = getSessionId;
|
|
53
|
+
exports.isAnalyticsEnabled = isAnalyticsEnabled;
|
|
54
|
+
exports.setAnalyticsEnabled = setAnalyticsEnabled;
|
|
55
|
+
exports.hasBeenPrompted = hasBeenPrompted;
|
|
56
|
+
exports.showOptInPrompt = showOptInPrompt;
|
|
57
|
+
exports.recordEvent = recordEvent;
|
|
58
|
+
exports.readEvents = readEvents;
|
|
59
|
+
exports.clearEvents = clearEvents;
|
|
60
|
+
exports.resetAnalytics = resetAnalytics;
|
|
61
|
+
exports.startTracking = startTracking;
|
|
62
|
+
exports.finishTracking = finishTracking;
|
|
63
|
+
exports.getDashboardUrl = getDashboardUrl;
|
|
64
|
+
exports.getStatus = getStatus;
|
|
65
|
+
exports.getSummary = getSummary;
|
|
66
|
+
const fs = __importStar(require("fs"));
|
|
67
|
+
const path = __importStar(require("path"));
|
|
68
|
+
const os = __importStar(require("os"));
|
|
69
|
+
const crypto = __importStar(require("crypto"));
|
|
70
|
+
// ============================================================================
|
|
71
|
+
// Constants
|
|
72
|
+
// ============================================================================
|
|
73
|
+
const MAX_EVENTS = 10000;
|
|
74
|
+
const EXCLUDED_PREFIXES = ['wpe.', 'analytics.'];
|
|
75
|
+
const ANALYTICS_BASE_URL = process.env.LWP_ANALYTICS_ENDPOINT?.replace('/v1/events', '') ||
|
|
76
|
+
'https://lwp-analytics.jeremy7746.workers.dev';
|
|
77
|
+
const ANALYTICS_ENDPOINT = `${ANALYTICS_BASE_URL}/v1/events`;
|
|
78
|
+
const TRANSMISSION_TIMEOUT = 5000; // 5 seconds
|
|
79
|
+
// Session ID generated once per CLI invocation
|
|
80
|
+
const SESSION_ID = crypto.randomUUID();
|
|
81
|
+
// CLI version from package.json
|
|
82
|
+
const CLI_VERSION = require('../package.json').version;
|
|
83
|
+
// Lazy-initialized paths (for testability)
|
|
84
|
+
function getLwpDir() {
|
|
85
|
+
return path.join(os.homedir(), '.lwp');
|
|
86
|
+
}
|
|
87
|
+
function getConfigPath() {
|
|
88
|
+
return path.join(getLwpDir(), 'config.json');
|
|
89
|
+
}
|
|
90
|
+
function getEventsDir() {
|
|
91
|
+
return path.join(getLwpDir(), 'analytics');
|
|
92
|
+
}
|
|
93
|
+
function getEventsPath() {
|
|
94
|
+
return path.join(getEventsDir(), 'events.jsonl');
|
|
95
|
+
}
|
|
96
|
+
const CI_ENV_VARS = [
|
|
97
|
+
'CI',
|
|
98
|
+
'GITHUB_ACTIONS',
|
|
99
|
+
'GITLAB_CI',
|
|
100
|
+
'JENKINS_URL',
|
|
101
|
+
'TRAVIS',
|
|
102
|
+
'CIRCLECI',
|
|
103
|
+
'BUILDKITE',
|
|
104
|
+
];
|
|
105
|
+
// ============================================================================
|
|
106
|
+
// Config Management
|
|
107
|
+
// ============================================================================
|
|
108
|
+
function ensureDir(dirPath) {
|
|
109
|
+
if (!fs.existsSync(dirPath)) {
|
|
110
|
+
fs.mkdirSync(dirPath, { recursive: true, mode: 0o700 });
|
|
111
|
+
}
|
|
112
|
+
}
|
|
113
|
+
function generateInstallationId() {
|
|
114
|
+
return crypto.randomUUID();
|
|
115
|
+
}
|
|
116
|
+
function generateSecretKey() {
|
|
117
|
+
return crypto.randomBytes(32).toString('base64');
|
|
118
|
+
}
|
|
119
|
+
function readConfig() {
|
|
120
|
+
try {
|
|
121
|
+
const configPath = getConfigPath();
|
|
122
|
+
if (fs.existsSync(configPath)) {
|
|
123
|
+
const data = fs.readFileSync(configPath, 'utf-8');
|
|
124
|
+
const config = JSON.parse(data);
|
|
125
|
+
// Validate structure
|
|
126
|
+
if (typeof config.analytics?.enabled === 'boolean') {
|
|
127
|
+
let needsWrite = false;
|
|
128
|
+
// Ensure installationId exists (migrate from Phase 1/2)
|
|
129
|
+
if (!config.installationId) {
|
|
130
|
+
config.installationId = generateInstallationId();
|
|
131
|
+
needsWrite = true;
|
|
132
|
+
}
|
|
133
|
+
// Ensure secretKey exists (migrate from Phase 2)
|
|
134
|
+
if (!config.secretKey) {
|
|
135
|
+
config.secretKey = generateSecretKey();
|
|
136
|
+
needsWrite = true;
|
|
137
|
+
}
|
|
138
|
+
if (needsWrite) {
|
|
139
|
+
writeConfig(config);
|
|
140
|
+
}
|
|
141
|
+
return config;
|
|
142
|
+
}
|
|
143
|
+
}
|
|
144
|
+
}
|
|
145
|
+
catch {
|
|
146
|
+
// Corrupted config, will regenerate
|
|
147
|
+
}
|
|
148
|
+
// Default to enabled (opt-out model) with new credentials
|
|
149
|
+
return {
|
|
150
|
+
installationId: generateInstallationId(),
|
|
151
|
+
secretKey: generateSecretKey(),
|
|
152
|
+
analytics: { enabled: true, promptedAt: null },
|
|
153
|
+
};
|
|
154
|
+
}
|
|
155
|
+
function writeConfig(config) {
|
|
156
|
+
const configPath = getConfigPath();
|
|
157
|
+
ensureDir(getLwpDir());
|
|
158
|
+
const tempPath = `${configPath}.${process.pid}.tmp`;
|
|
159
|
+
fs.writeFileSync(tempPath, JSON.stringify(config, null, 2));
|
|
160
|
+
fs.chmodSync(tempPath, 0o600);
|
|
161
|
+
fs.renameSync(tempPath, configPath);
|
|
162
|
+
}
|
|
163
|
+
function getInstallationId() {
|
|
164
|
+
return readConfig().installationId || generateInstallationId();
|
|
165
|
+
}
|
|
166
|
+
function getSecretKey() {
|
|
167
|
+
return readConfig().secretKey || generateSecretKey();
|
|
168
|
+
}
|
|
169
|
+
function getSessionId() {
|
|
170
|
+
return SESSION_ID;
|
|
171
|
+
}
|
|
172
|
+
function isAnalyticsEnabled() {
|
|
173
|
+
const override = process.env.LWP_ANALYTICS;
|
|
174
|
+
if (override === '0')
|
|
175
|
+
return false;
|
|
176
|
+
if (override === '1')
|
|
177
|
+
return true;
|
|
178
|
+
if (CI_ENV_VARS.some((v) => process.env[v]))
|
|
179
|
+
return false;
|
|
180
|
+
return readConfig().analytics.enabled;
|
|
181
|
+
}
|
|
182
|
+
function setAnalyticsEnabled(enabled) {
|
|
183
|
+
const config = readConfig();
|
|
184
|
+
config.analytics.enabled = enabled;
|
|
185
|
+
config.analytics.promptedAt = config.analytics.promptedAt || new Date().toISOString();
|
|
186
|
+
writeConfig(config);
|
|
187
|
+
}
|
|
188
|
+
function hasBeenPrompted() {
|
|
189
|
+
return readConfig().analytics.promptedAt !== null;
|
|
190
|
+
}
|
|
191
|
+
// ============================================================================
|
|
192
|
+
// HMAC Signing
|
|
193
|
+
// ============================================================================
|
|
194
|
+
/**
|
|
195
|
+
* Sign data with HMAC-SHA256
|
|
196
|
+
*/
|
|
197
|
+
function signData(data, secretKey) {
|
|
198
|
+
const key = Buffer.from(secretKey, 'base64');
|
|
199
|
+
return crypto.createHmac('sha256', key).update(data).digest('hex');
|
|
200
|
+
}
|
|
201
|
+
/**
|
|
202
|
+
* Mark this installation as registered
|
|
203
|
+
*/
|
|
204
|
+
function markAsRegistered() {
|
|
205
|
+
const config = readConfig();
|
|
206
|
+
config.registeredAt = new Date().toISOString();
|
|
207
|
+
writeConfig(config);
|
|
208
|
+
}
|
|
209
|
+
// ============================================================================
|
|
210
|
+
// Opt-In Prompt
|
|
211
|
+
// ============================================================================
|
|
212
|
+
async function showOptInPrompt() {
|
|
213
|
+
// Mark as prompted and keep enabled (opt-out model)
|
|
214
|
+
const config = readConfig();
|
|
215
|
+
config.analytics.promptedAt = new Date().toISOString();
|
|
216
|
+
config.analytics.enabled = true;
|
|
217
|
+
writeConfig(config);
|
|
218
|
+
// In non-interactive mode, silently enable without message
|
|
219
|
+
if (!process.stdin.isTTY) {
|
|
220
|
+
return true;
|
|
221
|
+
}
|
|
222
|
+
// Show informational message about analytics
|
|
223
|
+
console.log('');
|
|
224
|
+
console.log('━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━');
|
|
225
|
+
console.log('Anonymous usage analytics enabled');
|
|
226
|
+
console.log('');
|
|
227
|
+
console.log('We collect anonymous data to improve the CLI.');
|
|
228
|
+
console.log('No personal information, site names, or command arguments are collected.');
|
|
229
|
+
console.log('');
|
|
230
|
+
console.log('To disable: lwp analytics off');
|
|
231
|
+
console.log('Learn more: https://github.com/jpollock/local-addon-cli#analytics');
|
|
232
|
+
console.log('━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━');
|
|
233
|
+
console.log('');
|
|
234
|
+
return true;
|
|
235
|
+
}
|
|
236
|
+
// ============================================================================
|
|
237
|
+
// Event Tracking
|
|
238
|
+
// ============================================================================
|
|
239
|
+
function isCommandExcluded(command) {
|
|
240
|
+
return EXCLUDED_PREFIXES.some((prefix) => command.startsWith(prefix));
|
|
241
|
+
}
|
|
242
|
+
/**
|
|
243
|
+
* Transmit event to analytics server with HMAC signature (fire-and-forget)
|
|
244
|
+
*/
|
|
245
|
+
async function transmitEvent(event) {
|
|
246
|
+
try {
|
|
247
|
+
const config = readConfig();
|
|
248
|
+
const installationId = config.installationId;
|
|
249
|
+
const secretKey = config.secretKey;
|
|
250
|
+
const isFirstRequest = !config.registeredAt;
|
|
251
|
+
const body = JSON.stringify(event);
|
|
252
|
+
const signature = signData(body, secretKey);
|
|
253
|
+
const headers = {
|
|
254
|
+
'Content-Type': 'application/json',
|
|
255
|
+
'X-Installation-Id': installationId,
|
|
256
|
+
'X-Signature': signature,
|
|
257
|
+
};
|
|
258
|
+
// On first request, send the secret key so server can store it
|
|
259
|
+
if (isFirstRequest) {
|
|
260
|
+
headers['X-Secret-Key'] = secretKey;
|
|
261
|
+
}
|
|
262
|
+
const controller = new AbortController();
|
|
263
|
+
const timeoutId = setTimeout(() => controller.abort(), TRANSMISSION_TIMEOUT);
|
|
264
|
+
const response = await fetch(ANALYTICS_ENDPOINT, {
|
|
265
|
+
method: 'POST',
|
|
266
|
+
headers,
|
|
267
|
+
body,
|
|
268
|
+
signal: controller.signal,
|
|
269
|
+
});
|
|
270
|
+
clearTimeout(timeoutId);
|
|
271
|
+
// If successful and this was first request, mark as registered
|
|
272
|
+
if (response.ok && isFirstRequest) {
|
|
273
|
+
markAsRegistered();
|
|
274
|
+
}
|
|
275
|
+
}
|
|
276
|
+
catch {
|
|
277
|
+
// Silently ignore transmission errors - never block CLI
|
|
278
|
+
}
|
|
279
|
+
}
|
|
280
|
+
function recordEvent(event) {
|
|
281
|
+
try {
|
|
282
|
+
if (!isAnalyticsEnabled())
|
|
283
|
+
return;
|
|
284
|
+
if (isCommandExcluded(event.command))
|
|
285
|
+
return;
|
|
286
|
+
const eventsDir = getEventsDir();
|
|
287
|
+
const eventsPath = getEventsPath();
|
|
288
|
+
ensureDir(eventsDir);
|
|
289
|
+
// Check event count and rotate if needed
|
|
290
|
+
if (fs.existsSync(eventsPath)) {
|
|
291
|
+
const content = fs.readFileSync(eventsPath, 'utf-8');
|
|
292
|
+
const lines = content.trim().split('\n').filter(Boolean);
|
|
293
|
+
if (lines.length >= MAX_EVENTS) {
|
|
294
|
+
// Keep newest 80%
|
|
295
|
+
const keepCount = Math.floor(MAX_EVENTS * 0.8);
|
|
296
|
+
const toKeep = lines.slice(-keepCount);
|
|
297
|
+
fs.writeFileSync(eventsPath, toKeep.join('\n') + '\n');
|
|
298
|
+
fs.chmodSync(eventsPath, 0o600);
|
|
299
|
+
}
|
|
300
|
+
}
|
|
301
|
+
// Append new event to local storage
|
|
302
|
+
const line = JSON.stringify(event) + '\n';
|
|
303
|
+
fs.appendFileSync(eventsPath, line);
|
|
304
|
+
// Ensure permissions on first write
|
|
305
|
+
fs.chmodSync(eventsPath, 0o600);
|
|
306
|
+
// Transmit to server (fire-and-forget, don't await)
|
|
307
|
+
transmitEvent(event).catch(() => {
|
|
308
|
+
// Ignore transmission errors
|
|
309
|
+
});
|
|
310
|
+
}
|
|
311
|
+
catch {
|
|
312
|
+
// Never let analytics errors affect command execution
|
|
313
|
+
}
|
|
314
|
+
}
|
|
315
|
+
function readEvents() {
|
|
316
|
+
try {
|
|
317
|
+
const eventsPath = getEventsPath();
|
|
318
|
+
if (!fs.existsSync(eventsPath))
|
|
319
|
+
return [];
|
|
320
|
+
const content = fs.readFileSync(eventsPath, 'utf-8');
|
|
321
|
+
return content
|
|
322
|
+
.trim()
|
|
323
|
+
.split('\n')
|
|
324
|
+
.filter(Boolean)
|
|
325
|
+
.map((line) => JSON.parse(line));
|
|
326
|
+
}
|
|
327
|
+
catch {
|
|
328
|
+
return [];
|
|
329
|
+
}
|
|
330
|
+
}
|
|
331
|
+
function clearEvents() {
|
|
332
|
+
try {
|
|
333
|
+
const eventsPath = getEventsPath();
|
|
334
|
+
if (fs.existsSync(eventsPath)) {
|
|
335
|
+
fs.unlinkSync(eventsPath);
|
|
336
|
+
}
|
|
337
|
+
}
|
|
338
|
+
catch {
|
|
339
|
+
// Ignore cleanup errors
|
|
340
|
+
}
|
|
341
|
+
}
|
|
342
|
+
/**
|
|
343
|
+
* Reset analytics: clear events and regenerate installationId + secretKey
|
|
344
|
+
*/
|
|
345
|
+
function resetAnalytics() {
|
|
346
|
+
clearEvents();
|
|
347
|
+
const config = readConfig();
|
|
348
|
+
config.installationId = generateInstallationId();
|
|
349
|
+
config.secretKey = generateSecretKey();
|
|
350
|
+
delete config.registeredAt; // Will need to re-register
|
|
351
|
+
config.analytics.enabled = false;
|
|
352
|
+
writeConfig(config);
|
|
353
|
+
}
|
|
354
|
+
// ============================================================================
|
|
355
|
+
// Command Tracking (for Commander hooks)
|
|
356
|
+
// ============================================================================
|
|
357
|
+
let commandStartTime = null;
|
|
358
|
+
let currentCommandName = null;
|
|
359
|
+
function startTracking(commandName) {
|
|
360
|
+
commandStartTime = Date.now();
|
|
361
|
+
currentCommandName = commandName;
|
|
362
|
+
}
|
|
363
|
+
function finishTracking(success, errorCategory) {
|
|
364
|
+
if (commandStartTime === null || currentCommandName === null)
|
|
365
|
+
return;
|
|
366
|
+
const duration = Date.now() - commandStartTime;
|
|
367
|
+
const event = {
|
|
368
|
+
command: currentCommandName,
|
|
369
|
+
success,
|
|
370
|
+
duration_ms: duration,
|
|
371
|
+
timestamp: new Date().toISOString(),
|
|
372
|
+
installation_id: getInstallationId(),
|
|
373
|
+
session_id: SESSION_ID,
|
|
374
|
+
cli_version: CLI_VERSION,
|
|
375
|
+
os: os.platform(),
|
|
376
|
+
node_version: process.version,
|
|
377
|
+
};
|
|
378
|
+
// Add error category for failures
|
|
379
|
+
if (!success && errorCategory) {
|
|
380
|
+
event.error_category = errorCategory;
|
|
381
|
+
}
|
|
382
|
+
recordEvent(event);
|
|
383
|
+
commandStartTime = null;
|
|
384
|
+
currentCommandName = null;
|
|
385
|
+
}
|
|
386
|
+
// ============================================================================
|
|
387
|
+
// Dashboard URL Generation
|
|
388
|
+
// ============================================================================
|
|
389
|
+
/**
|
|
390
|
+
* Generate a signed dashboard URL for viewing personal analytics
|
|
391
|
+
* URL expires in 1 hour
|
|
392
|
+
*/
|
|
393
|
+
function getDashboardUrl() {
|
|
394
|
+
const config = readConfig();
|
|
395
|
+
const installationId = config.installationId;
|
|
396
|
+
const secretKey = config.secretKey;
|
|
397
|
+
// Expire in 1 hour
|
|
398
|
+
const expiration = Math.floor(Date.now() / 1000) + 3600;
|
|
399
|
+
// Sign: installationId:expiration
|
|
400
|
+
const payload = `${installationId}:${expiration}`;
|
|
401
|
+
const signature = signData(payload, secretKey);
|
|
402
|
+
return `${ANALYTICS_BASE_URL}/dashboard/${installationId}?exp=${expiration}&sig=${signature}`;
|
|
403
|
+
}
|
|
404
|
+
// ============================================================================
|
|
405
|
+
// Analytics Summary
|
|
406
|
+
// ============================================================================
|
|
407
|
+
function getStatus() {
|
|
408
|
+
return {
|
|
409
|
+
enabled: isAnalyticsEnabled(),
|
|
410
|
+
eventCount: readEvents().length,
|
|
411
|
+
installationId: getInstallationId(),
|
|
412
|
+
};
|
|
413
|
+
}
|
|
414
|
+
function getSummary() {
|
|
415
|
+
const events = readEvents();
|
|
416
|
+
if (events.length === 0) {
|
|
417
|
+
return 'No analytics data collected yet.';
|
|
418
|
+
}
|
|
419
|
+
const total = events.length;
|
|
420
|
+
const successful = events.filter((e) => e.success).length;
|
|
421
|
+
const successRate = ((successful / total) * 100).toFixed(1);
|
|
422
|
+
// Count commands
|
|
423
|
+
const commandCounts = {};
|
|
424
|
+
events.forEach((e) => {
|
|
425
|
+
commandCounts[e.command] = (commandCounts[e.command] || 0) + 1;
|
|
426
|
+
});
|
|
427
|
+
// Sort by count
|
|
428
|
+
const topCommands = Object.entries(commandCounts)
|
|
429
|
+
.sort((a, b) => b[1] - a[1])
|
|
430
|
+
.slice(0, 5);
|
|
431
|
+
// Count recent failures (last 7 days)
|
|
432
|
+
const weekAgo = Date.now() - 7 * 24 * 60 * 60 * 1000;
|
|
433
|
+
const recentFailures = events.filter((e) => !e.success && new Date(e.timestamp).getTime() > weekAgo).length;
|
|
434
|
+
let output = `
|
|
435
|
+
Analytics Summary
|
|
436
|
+
─────────────────
|
|
437
|
+
Total commands: ${total}
|
|
438
|
+
Success rate: ${successRate}%
|
|
439
|
+
|
|
440
|
+
Top commands:`;
|
|
441
|
+
topCommands.forEach(([cmd, count]) => {
|
|
442
|
+
output += `\n ${cmd.padEnd(15)} ${count}`;
|
|
443
|
+
});
|
|
444
|
+
output += `\n\nRecent failures: ${recentFailures} in last 7 days`;
|
|
445
|
+
return output;
|
|
446
|
+
}
|
|
447
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"analytics.js","sourceRoot":"","sources":["../src/analytics.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Anonymous Usage Analytics - Phase 3 (Signed Authentication)\n *\n * Collects anonymous usage data with user consent and transmits to server.\n * All requests are signed with HMAC-SHA256 for authentication.\n *\n * Privacy: Only tracks command names, success/failure, duration, and system info.\n * Never tracks: arguments, site names, paths, or any PII.\n *\n * Identifiers:\n * - installationId: Random UUID, identifies this CLI installation (not the user)\n * - secretKey: Random 32 bytes, used for HMAC signing (never transmitted after registration)\n * - sessionId: Random UUID per CLI invocation, correlates commands in a session\n */\n\nimport * as fs from 'fs';\nimport * as path from 'path';\nimport * as os from 'os';\nimport * as crypto from 'crypto';\n\n// ============================================================================\n// Types\n// ============================================================================\n\ninterface AnalyticsConfig {\n  installationId?: string;\n  secretKey?: string;\n  registeredAt?: string; // When secretKey was first sent to server\n  analytics: {\n    enabled: boolean;\n    promptedAt: string | null;\n  };\n}\n\nexport type ErrorCategory =\n  | 'site_not_found'\n  | 'site_not_running'\n  | 'local_not_running'\n  | 'timeout'\n  | 'network_error'\n  | 'validation_error'\n  | 'unknown';\n\ninterface AnalyticsEvent {\n  // Phase 1 fields\n  command: string;\n  success: boolean;\n  duration_ms: number;\n  timestamp: string;\n\n  // Phase 2 fields\n  installation_id: string;\n  session_id: string;\n  cli_version: string;\n  os: string;\n  node_version: string;\n  error_category?: ErrorCategory;\n}\n\n// ============================================================================\n// Constants\n// ============================================================================\n\nconst MAX_EVENTS = 10000;\nconst EXCLUDED_PREFIXES = ['wpe.', 'analytics.'];\nconst ANALYTICS_BASE_URL =\n  process.env.LWP_ANALYTICS_ENDPOINT?.replace('/v1/events', '') ||\n  'https://lwp-analytics.jeremy7746.workers.dev';\nconst ANALYTICS_ENDPOINT = `${ANALYTICS_BASE_URL}/v1/events`;\nconst TRANSMISSION_TIMEOUT = 5000; // 5 seconds\n\n// Session ID generated once per CLI invocation\nconst SESSION_ID = crypto.randomUUID();\n\n// CLI version from package.json\nconst CLI_VERSION = require('../package.json').version;\n\n// Lazy-initialized paths (for testability)\nfunction getLwpDir(): string {\n  return path.join(os.homedir(), '.lwp');\n}\n\nfunction getConfigPath(): string {\n  return path.join(getLwpDir(), 'config.json');\n}\n\nfunction getEventsDir(): string {\n  return path.join(getLwpDir(), 'analytics');\n}\n\nfunction getEventsPath(): string {\n  return path.join(getEventsDir(), 'events.jsonl');\n}\n\nconst CI_ENV_VARS = [\n  'CI',\n  'GITHUB_ACTIONS',\n  'GITLAB_CI',\n  'JENKINS_URL',\n  'TRAVIS',\n  'CIRCLECI',\n  'BUILDKITE',\n];\n\n// ============================================================================\n// Config Management\n// ============================================================================\n\nfunction ensureDir(dirPath: string): void {\n  if (!fs.existsSync(dirPath)) {\n    fs.mkdirSync(dirPath, { recursive: true, mode: 0o700 });\n  }\n}\n\nfunction generateInstallationId(): string {\n  return crypto.randomUUID();\n}\n\nfunction generateSecretKey(): string {\n  return crypto.randomBytes(32).toString('base64');\n}\n\nfunction readConfig(): AnalyticsConfig {\n  try {\n    const configPath = getConfigPath();\n    if (fs.existsSync(configPath)) {\n      const data = fs.readFileSync(configPath, 'utf-8');\n      const config = JSON.parse(data);\n      // Validate structure\n      if (typeof config.analytics?.enabled === 'boolean') {\n        let needsWrite = false;\n\n        // Ensure installationId exists (migrate from Phase 1/2)\n        if (!config.installationId) {\n          config.installationId = generateInstallationId();\n          needsWrite = true;\n        }\n\n        // Ensure secretKey exists (migrate from Phase 2)\n        if (!config.secretKey) {\n          config.secretKey = generateSecretKey();\n          needsWrite = true;\n        }\n\n        if (needsWrite) {\n          writeConfig(config);\n        }\n        return config;\n      }\n    }\n  } catch {\n    // Corrupted config, will regenerate\n  }\n  // Default to enabled (opt-out model) with new credentials\n  return {\n    installationId: generateInstallationId(),\n    secretKey: generateSecretKey(),\n    analytics: { enabled: true, promptedAt: null },\n  };\n}\n\nfunction writeConfig(config: AnalyticsConfig): void {\n  const configPath = getConfigPath();\n  ensureDir(getLwpDir());\n  const tempPath = `${configPath}.${process.pid}.tmp`;\n  fs.writeFileSync(tempPath, JSON.stringify(config, null, 2));\n  fs.chmodSync(tempPath, 0o600);\n  fs.renameSync(tempPath, configPath);\n}\n\nexport function getInstallationId(): string {\n  return readConfig().installationId || generateInstallationId();\n}\n\nexport function getSecretKey(): string {\n  return readConfig().secretKey || generateSecretKey();\n}\n\nexport function getSessionId(): string {\n  return SESSION_ID;\n}\n\nexport function isAnalyticsEnabled(): boolean {\n  const override = process.env.LWP_ANALYTICS;\n  if (override === '0') return false;\n  if (override === '1') return true;\n  if (CI_ENV_VARS.some((v) => process.env[v])) return false;\n  return readConfig().analytics.enabled;\n}\n\nexport function setAnalyticsEnabled(enabled: boolean): void {\n  const config = readConfig();\n  config.analytics.enabled = enabled;\n  config.analytics.promptedAt = config.analytics.promptedAt || new Date().toISOString();\n  writeConfig(config);\n}\n\nexport function hasBeenPrompted(): boolean {\n  return readConfig().analytics.promptedAt !== null;\n}\n\n// ============================================================================\n// HMAC Signing\n// ============================================================================\n\n/**\n * Sign data with HMAC-SHA256\n */\nfunction signData(data: string, secretKey: string): string {\n  const key = Buffer.from(secretKey, 'base64');\n  return crypto.createHmac('sha256', key).update(data).digest('hex');\n}\n\n/**\n * Mark this installation as registered\n */\nfunction markAsRegistered(): void {\n  const config = readConfig();\n  config.registeredAt = new Date().toISOString();\n  writeConfig(config);\n}\n\n// ============================================================================\n// Opt-In Prompt\n// ============================================================================\n\nexport async function showOptInPrompt(): Promise<boolean> {\n  // Mark as prompted and keep enabled (opt-out model)\n  const config = readConfig();\n  config.analytics.promptedAt = new Date().toISOString();\n  config.analytics.enabled = true;\n  writeConfig(config);\n\n  // In non-interactive mode, silently enable without message\n  if (!process.stdin.isTTY) {\n    return true;\n  }\n\n  // Show informational message about analytics\n  console.log('');\n  console.log('━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━');\n  console.log('Anonymous usage analytics enabled');\n  console.log('');\n  console.log('We collect anonymous data to improve the CLI.');\n  console.log('No personal information, site names, or command arguments are collected.');\n  console.log('');\n  console.log('To disable: lwp analytics off');\n  console.log('Learn more: https://github.com/jpollock/local-addon-cli#analytics');\n  console.log('━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━');\n  console.log('');\n\n  return true;\n}\n\n// ============================================================================\n// Event Tracking\n// ============================================================================\n\nfunction isCommandExcluded(command: string): boolean {\n  return EXCLUDED_PREFIXES.some((prefix) => command.startsWith(prefix));\n}\n\n/**\n * Transmit event to analytics server with HMAC signature (fire-and-forget)\n */\nasync function transmitEvent(event: AnalyticsEvent): Promise<void> {\n  try {\n    const config = readConfig();\n    const installationId = config.installationId!;\n    const secretKey = config.secretKey!;\n    const isFirstRequest = !config.registeredAt;\n\n    const body = JSON.stringify(event);\n    const signature = signData(body, secretKey);\n\n    const headers: Record<string, string> = {\n      'Content-Type': 'application/json',\n      'X-Installation-Id': installationId,\n      'X-Signature': signature,\n    };\n\n    // On first request, send the secret key so server can store it\n    if (isFirstRequest) {\n      headers['X-Secret-Key'] = secretKey;\n    }\n\n    const controller = new AbortController();\n    const timeoutId = setTimeout(() => controller.abort(), TRANSMISSION_TIMEOUT);\n\n    const response = await fetch(ANALYTICS_ENDPOINT, {\n      method: 'POST',\n      headers,\n      body,\n      signal: controller.signal,\n    });\n\n    clearTimeout(timeoutId);\n\n    // If successful and this was first request, mark as registered\n    if (response.ok && isFirstRequest) {\n      markAsRegistered();\n    }\n  } catch {\n    // Silently ignore transmission errors - never block CLI\n  }\n}\n\nexport function recordEvent(event: AnalyticsEvent): void {\n  try {\n    if (!isAnalyticsEnabled()) return;\n    if (isCommandExcluded(event.command)) return;\n\n    const eventsDir = getEventsDir();\n    const eventsPath = getEventsPath();\n\n    ensureDir(eventsDir);\n\n    // Check event count and rotate if needed\n    if (fs.existsSync(eventsPath)) {\n      const content = fs.readFileSync(eventsPath, 'utf-8');\n      const lines = content.trim().split('\\n').filter(Boolean);\n      if (lines.length >= MAX_EVENTS) {\n        // Keep newest 80%\n        const keepCount = Math.floor(MAX_EVENTS * 0.8);\n        const toKeep = lines.slice(-keepCount);\n        fs.writeFileSync(eventsPath, toKeep.join('\\n') + '\\n');\n        fs.chmodSync(eventsPath, 0o600);\n      }\n    }\n\n    // Append new event to local storage\n    const line = JSON.stringify(event) + '\\n';\n    fs.appendFileSync(eventsPath, line);\n\n    // Ensure permissions on first write\n    fs.chmodSync(eventsPath, 0o600);\n\n    // Transmit to server (fire-and-forget, don't await)\n    transmitEvent(event).catch(() => {\n      // Ignore transmission errors\n    });\n  } catch {\n    // Never let analytics errors affect command execution\n  }\n}\n\nexport function readEvents(): AnalyticsEvent[] {\n  try {\n    const eventsPath = getEventsPath();\n    if (!fs.existsSync(eventsPath)) return [];\n    const content = fs.readFileSync(eventsPath, 'utf-8');\n    return content\n      .trim()\n      .split('\\n')\n      .filter(Boolean)\n      .map((line) => JSON.parse(line));\n  } catch {\n    return [];\n  }\n}\n\nexport function clearEvents(): void {\n  try {\n    const eventsPath = getEventsPath();\n    if (fs.existsSync(eventsPath)) {\n      fs.unlinkSync(eventsPath);\n    }\n  } catch {\n    // Ignore cleanup errors\n  }\n}\n\n/**\n * Reset analytics: clear events and regenerate installationId + secretKey\n */\nexport function resetAnalytics(): void {\n  clearEvents();\n  const config = readConfig();\n  config.installationId = generateInstallationId();\n  config.secretKey = generateSecretKey();\n  delete config.registeredAt; // Will need to re-register\n  config.analytics.enabled = false;\n  writeConfig(config);\n}\n\n// ============================================================================\n// Command Tracking (for Commander hooks)\n// ============================================================================\n\nlet commandStartTime: number | null = null;\nlet currentCommandName: string | null = null;\n\nexport function startTracking(commandName: string): void {\n  commandStartTime = Date.now();\n  currentCommandName = commandName;\n}\n\nexport function finishTracking(success: boolean, errorCategory?: ErrorCategory): void {\n  if (commandStartTime === null || currentCommandName === null) return;\n\n  const duration = Date.now() - commandStartTime;\n  const event: AnalyticsEvent = {\n    command: currentCommandName,\n    success,\n    duration_ms: duration,\n    timestamp: new Date().toISOString(),\n    installation_id: getInstallationId(),\n    session_id: SESSION_ID,\n    cli_version: CLI_VERSION,\n    os: os.platform(),\n    node_version: process.version,\n  };\n\n  // Add error category for failures\n  if (!success && errorCategory) {\n    event.error_category = errorCategory;\n  }\n\n  recordEvent(event);\n\n  commandStartTime = null;\n  currentCommandName = null;\n}\n\n// ============================================================================\n// Dashboard URL Generation\n// ============================================================================\n\n/**\n * Generate a signed dashboard URL for viewing personal analytics\n * URL expires in 1 hour\n */\nexport function getDashboardUrl(): string {\n  const config = readConfig();\n  const installationId = config.installationId!;\n  const secretKey = config.secretKey!;\n\n  // Expire in 1 hour\n  const expiration = Math.floor(Date.now() / 1000) + 3600;\n\n  // Sign: installationId:expiration\n  const payload = `${installationId}:${expiration}`;\n  const signature = signData(payload, secretKey);\n\n  return `${ANALYTICS_BASE_URL}/dashboard/${installationId}?exp=${expiration}&sig=${signature}`;\n}\n\n// ============================================================================\n// Analytics Summary\n// ============================================================================\n\nexport function getStatus(): { enabled: boolean; eventCount: number; installationId: string } {\n  return {\n    enabled: isAnalyticsEnabled(),\n    eventCount: readEvents().length,\n    installationId: getInstallationId(),\n  };\n}\n\nexport function getSummary(): string {\n  const events = readEvents();\n  if (events.length === 0) {\n    return 'No analytics data collected yet.';\n  }\n\n  const total = events.length;\n  const successful = events.filter((e) => e.success).length;\n  const successRate = ((successful / total) * 100).toFixed(1);\n\n  // Count commands\n  const commandCounts: Record<string, number> = {};\n  events.forEach((e) => {\n    commandCounts[e.command] = (commandCounts[e.command] || 0) + 1;\n  });\n\n  // Sort by count\n  const topCommands = Object.entries(commandCounts)\n    .sort((a, b) => b[1] - a[1])\n    .slice(0, 5);\n\n  // Count recent failures (last 7 days)\n  const weekAgo = Date.now() - 7 * 24 * 60 * 60 * 1000;\n  const recentFailures = events.filter(\n    (e) => !e.success && new Date(e.timestamp).getTime() > weekAgo\n  ).length;\n\n  let output = `\nAnalytics Summary\n─────────────────\nTotal commands:  ${total}\nSuccess rate:    ${successRate}%\n\nTop commands:`;\n\n  topCommands.forEach(([cmd, count]) => {\n    output += `\\n  ${cmd.padEnd(15)} ${count}`;\n  });\n\n  output += `\\n\\nRecent failures:  ${recentFailures} in last 7 days`;\n\n  return output;\n}\n"]}
|