|
| 1 | +/** |
| 2 | + * @license |
| 3 | + * Copyright 2026 Google LLC |
| 4 | + * SPDX-License-Identifier: Apache-2.0 |
| 5 | + */ |
| 6 | + |
| 7 | +import crypto from 'node:crypto'; |
| 8 | +import sinon from 'sinon'; |
| 9 | +import {describe, it, afterEach, beforeEach} from 'node:test'; |
| 10 | +import assert from 'node:assert'; |
| 11 | +import {OsType} from '../../../src/telemetry/types.js'; |
| 12 | +import {ClearcutSender} from '../../../src/telemetry/watchdog/clearcut-sender.js'; |
| 13 | + |
| 14 | +describe('ClearcutSender', () => { |
| 15 | + let clock: sinon.SinonFakeTimers; |
| 16 | + let randomUUIDStub: sinon.SinonStub; |
| 17 | + |
| 18 | + beforeEach(() => { |
| 19 | + clock = sinon.useFakeTimers(); |
| 20 | + let uuidCounter = 0; |
| 21 | + randomUUIDStub = sinon.stub(crypto, 'randomUUID').callsFake(() => { |
| 22 | + return `uuid-${++uuidCounter}` as any; |
| 23 | + }); |
| 24 | + }); |
| 25 | + |
| 26 | + afterEach(() => { |
| 27 | + clock.restore(); |
| 28 | + randomUUIDStub.restore(); |
| 29 | + sinon.restore(); |
| 30 | + }); |
| 31 | + |
| 32 | + it('enriches events with app version, os type, and session id', async () => { |
| 33 | + const sender = new ClearcutSender('1.0.0', OsType.OS_TYPE_MACOS); |
| 34 | + const transportStub = sinon.stub(sender, 'transport'); |
| 35 | + |
| 36 | + await sender.send({mcp_client: undefined}); |
| 37 | + |
| 38 | + assert.strictEqual(transportStub.callCount, 1); |
| 39 | + const event = transportStub.firstCall.args[0]; |
| 40 | + |
| 41 | + assert.strictEqual(event.session_id, 'uuid-1'); |
| 42 | + assert.strictEqual(event.app_version, '1.0.0'); |
| 43 | + assert.strictEqual(event.os_type, OsType.OS_TYPE_MACOS); |
| 44 | + }); |
| 45 | + |
| 46 | + it('rotates session ID after 24 hours', async () => { |
| 47 | + const sender = new ClearcutSender('1.0.0', OsType.OS_TYPE_MACOS); |
| 48 | + const transportStub = sinon.stub(sender, 'transport'); |
| 49 | + |
| 50 | + // First call -> session uuid-1 |
| 51 | + await sender.send({}); |
| 52 | + assert.strictEqual(transportStub.lastCall.args[0].session_id, 'uuid-1'); |
| 53 | + |
| 54 | + // Advance time by 23 hours -> should NOT rotate |
| 55 | + clock.tick(23 * 60 * 60 * 1000); |
| 56 | + await sender.send({}); |
| 57 | + assert.strictEqual(transportStub.lastCall.args[0].session_id, 'uuid-1'); |
| 58 | + |
| 59 | + // Advance time by 2 hours (total 25h) -> SHOULD rotate |
| 60 | + clock.tick(2 * 60 * 60 * 1000); |
| 61 | + await sender.send({}); |
| 62 | + assert.strictEqual(transportStub.lastCall.args[0].session_id, 'uuid-2'); |
| 63 | + }); |
| 64 | + |
| 65 | + it('sendShutdownEvent sends a server_shutdown event without flag_usage', async () => { |
| 66 | + const sender = new ClearcutSender('1.0.0', OsType.OS_TYPE_MACOS); |
| 67 | + const transportStub = sinon.stub(sender, 'transport'); |
| 68 | + |
| 69 | + await sender.sendShutdownEvent(); |
| 70 | + |
| 71 | + const event = transportStub.firstCall.args[0]; |
| 72 | + assert.ok(event.server_shutdown); |
| 73 | + assert.strictEqual(event.server_start, undefined); |
| 74 | + }); |
| 75 | +}); |
0 commit comments