merge: resolve conflicts with develop (telemetry + lockfile)
Keep both Mosaic Telemetry section (from develop) and Matrix Dev Environment section (from feature branch) in .env.example. Regenerate pnpm-lock.yaml with both dependency trees merged. Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
This commit is contained in:
17
apps/api/src/mosaic-telemetry/index.ts
Normal file
17
apps/api/src/mosaic-telemetry/index.ts
Normal file
@@ -0,0 +1,17 @@
|
||||
/**
|
||||
* Mosaic Telemetry module — task completion tracking and crowd-sourced predictions.
|
||||
*
|
||||
* **Not to be confused with the OpenTelemetry (OTEL) TelemetryModule** at
|
||||
* `src/telemetry/`, which handles distributed request tracing.
|
||||
*
|
||||
* @module mosaic-telemetry
|
||||
*/
|
||||
|
||||
export { MosaicTelemetryModule } from "./mosaic-telemetry.module";
|
||||
export { MosaicTelemetryService } from "./mosaic-telemetry.service";
|
||||
export {
|
||||
loadMosaicTelemetryConfig,
|
||||
toSdkConfig,
|
||||
MOSAIC_TELEMETRY_ENV,
|
||||
type MosaicTelemetryModuleConfig,
|
||||
} from "./mosaic-telemetry.config";
|
||||
78
apps/api/src/mosaic-telemetry/mosaic-telemetry.config.ts
Normal file
78
apps/api/src/mosaic-telemetry/mosaic-telemetry.config.ts
Normal file
@@ -0,0 +1,78 @@
|
||||
import type { ConfigService } from "@nestjs/config";
|
||||
import type { TelemetryConfig } from "@mosaicstack/telemetry-client";
|
||||
|
||||
/**
|
||||
* Configuration interface for the Mosaic Telemetry module.
|
||||
* Maps environment variables to SDK configuration.
|
||||
*/
|
||||
export interface MosaicTelemetryModuleConfig {
|
||||
/** Whether telemetry collection is enabled. Default: true */
|
||||
enabled: boolean;
|
||||
/** Base URL of the telemetry server */
|
||||
serverUrl: string;
|
||||
/** API key for authentication (64-char hex string) */
|
||||
apiKey: string;
|
||||
/** Instance UUID for this client */
|
||||
instanceId: string;
|
||||
/** If true, log events instead of sending them. Default: false */
|
||||
dryRun: boolean;
|
||||
}
|
||||
|
||||
/**
|
||||
* Environment variable names used by the Mosaic Telemetry module.
|
||||
*/
|
||||
export const MOSAIC_TELEMETRY_ENV = {
|
||||
ENABLED: "MOSAIC_TELEMETRY_ENABLED",
|
||||
SERVER_URL: "MOSAIC_TELEMETRY_SERVER_URL",
|
||||
API_KEY: "MOSAIC_TELEMETRY_API_KEY",
|
||||
INSTANCE_ID: "MOSAIC_TELEMETRY_INSTANCE_ID",
|
||||
DRY_RUN: "MOSAIC_TELEMETRY_DRY_RUN",
|
||||
} as const;
|
||||
|
||||
/**
|
||||
* Read Mosaic Telemetry configuration from environment variables via NestJS ConfigService.
|
||||
*
|
||||
* @param configService - NestJS ConfigService instance
|
||||
* @returns Parsed module configuration
|
||||
*/
|
||||
export function loadMosaicTelemetryConfig(
|
||||
configService: ConfigService
|
||||
): MosaicTelemetryModuleConfig {
|
||||
const enabledRaw = configService.get<string>(MOSAIC_TELEMETRY_ENV.ENABLED, "true");
|
||||
const dryRunRaw = configService.get<string>(MOSAIC_TELEMETRY_ENV.DRY_RUN, "false");
|
||||
|
||||
return {
|
||||
enabled: enabledRaw.toLowerCase() === "true",
|
||||
serverUrl: configService.get<string>(MOSAIC_TELEMETRY_ENV.SERVER_URL, ""),
|
||||
apiKey: configService.get<string>(MOSAIC_TELEMETRY_ENV.API_KEY, ""),
|
||||
instanceId: configService.get<string>(MOSAIC_TELEMETRY_ENV.INSTANCE_ID, ""),
|
||||
dryRun: dryRunRaw.toLowerCase() === "true",
|
||||
};
|
||||
}
|
||||
|
||||
/**
|
||||
* Convert module config to SDK TelemetryConfig format.
|
||||
* Includes the onError callback for NestJS Logger integration.
|
||||
*
|
||||
* @param config - Module configuration
|
||||
* @param onError - Error callback (typically NestJS Logger)
|
||||
* @returns SDK-compatible TelemetryConfig
|
||||
*/
|
||||
export function toSdkConfig(
|
||||
config: MosaicTelemetryModuleConfig,
|
||||
onError?: (error: Error) => void
|
||||
): TelemetryConfig {
|
||||
const sdkConfig: TelemetryConfig = {
|
||||
serverUrl: config.serverUrl,
|
||||
apiKey: config.apiKey,
|
||||
instanceId: config.instanceId,
|
||||
enabled: config.enabled,
|
||||
dryRun: config.dryRun,
|
||||
};
|
||||
|
||||
if (onError) {
|
||||
sdkConfig.onError = onError;
|
||||
}
|
||||
|
||||
return sdkConfig;
|
||||
}
|
||||
92
apps/api/src/mosaic-telemetry/mosaic-telemetry.controller.ts
Normal file
92
apps/api/src/mosaic-telemetry/mosaic-telemetry.controller.ts
Normal file
@@ -0,0 +1,92 @@
|
||||
import { Controller, Get, Query, UseGuards, BadRequestException } from "@nestjs/common";
|
||||
import { AuthGuard } from "../auth/guards/auth.guard";
|
||||
import { PredictionService } from "./prediction.service";
|
||||
import {
|
||||
TaskType,
|
||||
Complexity,
|
||||
Provider,
|
||||
type PredictionResponse,
|
||||
} from "@mosaicstack/telemetry-client";
|
||||
|
||||
/**
|
||||
* Valid values for query parameter validation.
|
||||
*/
|
||||
const VALID_TASK_TYPES = new Set<string>(Object.values(TaskType));
|
||||
const VALID_COMPLEXITIES = new Set<string>(Object.values(Complexity));
|
||||
const VALID_PROVIDERS = new Set<string>(Object.values(Provider));
|
||||
|
||||
/**
|
||||
* Response DTO for the estimate endpoint.
|
||||
*/
|
||||
interface EstimateResponseDto {
|
||||
data: PredictionResponse | null;
|
||||
}
|
||||
|
||||
/**
|
||||
* Mosaic Telemetry Controller
|
||||
*
|
||||
* Provides API endpoints for accessing telemetry prediction data.
|
||||
* All endpoints require authentication via AuthGuard.
|
||||
*
|
||||
* This controller is intentionally lightweight - it delegates to PredictionService
|
||||
* for the actual prediction logic and returns results directly to the frontend.
|
||||
*/
|
||||
@Controller("telemetry")
|
||||
@UseGuards(AuthGuard)
|
||||
export class MosaicTelemetryController {
|
||||
constructor(private readonly predictionService: PredictionService) {}
|
||||
|
||||
/**
|
||||
* GET /api/telemetry/estimate
|
||||
*
|
||||
* Get a cost/token estimate for a given task configuration.
|
||||
* Returns prediction data including confidence level, or null if
|
||||
* no prediction is available.
|
||||
*
|
||||
* @param taskType - Task type enum value (e.g. "implementation", "planning")
|
||||
* @param model - Model name (e.g. "claude-sonnet-4-5")
|
||||
* @param provider - Provider enum value (e.g. "anthropic", "openai")
|
||||
* @param complexity - Complexity level (e.g. "low", "medium", "high")
|
||||
* @returns Prediction response with estimates and confidence
|
||||
*/
|
||||
@Get("estimate")
|
||||
getEstimate(
|
||||
@Query("taskType") taskType: string,
|
||||
@Query("model") model: string,
|
||||
@Query("provider") provider: string,
|
||||
@Query("complexity") complexity: string
|
||||
): EstimateResponseDto {
|
||||
if (!taskType || !model || !provider || !complexity) {
|
||||
throw new BadRequestException(
|
||||
"Missing query parameters. Required: taskType, model, provider, complexity"
|
||||
);
|
||||
}
|
||||
|
||||
if (!VALID_TASK_TYPES.has(taskType)) {
|
||||
throw new BadRequestException(
|
||||
`Invalid taskType "${taskType}". Valid values: ${[...VALID_TASK_TYPES].join(", ")}`
|
||||
);
|
||||
}
|
||||
|
||||
if (!VALID_PROVIDERS.has(provider)) {
|
||||
throw new BadRequestException(
|
||||
`Invalid provider "${provider}". Valid values: ${[...VALID_PROVIDERS].join(", ")}`
|
||||
);
|
||||
}
|
||||
|
||||
if (!VALID_COMPLEXITIES.has(complexity)) {
|
||||
throw new BadRequestException(
|
||||
`Invalid complexity "${complexity}". Valid values: ${[...VALID_COMPLEXITIES].join(", ")}`
|
||||
);
|
||||
}
|
||||
|
||||
const prediction = this.predictionService.getEstimate(
|
||||
taskType as TaskType,
|
||||
model,
|
||||
provider as Provider,
|
||||
complexity as Complexity
|
||||
);
|
||||
|
||||
return { data: prediction };
|
||||
}
|
||||
}
|
||||
212
apps/api/src/mosaic-telemetry/mosaic-telemetry.module.spec.ts
Normal file
212
apps/api/src/mosaic-telemetry/mosaic-telemetry.module.spec.ts
Normal file
@@ -0,0 +1,212 @@
|
||||
import { describe, it, expect, vi, beforeEach } from "vitest";
|
||||
import { Test, TestingModule } from "@nestjs/testing";
|
||||
import { ConfigModule } from "@nestjs/config";
|
||||
import { MosaicTelemetryModule } from "./mosaic-telemetry.module";
|
||||
import { MosaicTelemetryService } from "./mosaic-telemetry.service";
|
||||
|
||||
// Mock the telemetry client to avoid real HTTP calls
|
||||
vi.mock("@mosaicstack/telemetry-client", async (importOriginal) => {
|
||||
const actual = await importOriginal<typeof import("@mosaicstack/telemetry-client")>();
|
||||
|
||||
class MockTelemetryClient {
|
||||
private _isRunning = false;
|
||||
|
||||
constructor(_config: unknown) {
|
||||
// no-op
|
||||
}
|
||||
|
||||
get eventBuilder() {
|
||||
return { build: vi.fn().mockReturnValue({ event_id: "test-event-id" }) };
|
||||
}
|
||||
|
||||
start(): void {
|
||||
this._isRunning = true;
|
||||
}
|
||||
|
||||
async stop(): Promise<void> {
|
||||
this._isRunning = false;
|
||||
}
|
||||
|
||||
track(_event: unknown): void {
|
||||
// no-op
|
||||
}
|
||||
|
||||
getPrediction(_query: unknown): unknown {
|
||||
return null;
|
||||
}
|
||||
|
||||
async refreshPredictions(_queries: unknown): Promise<void> {
|
||||
// no-op
|
||||
}
|
||||
|
||||
get queueSize(): number {
|
||||
return 0;
|
||||
}
|
||||
|
||||
get isRunning(): boolean {
|
||||
return this._isRunning;
|
||||
}
|
||||
}
|
||||
|
||||
return {
|
||||
...actual,
|
||||
TelemetryClient: MockTelemetryClient,
|
||||
};
|
||||
});
|
||||
|
||||
describe("MosaicTelemetryModule", () => {
|
||||
let module: TestingModule;
|
||||
|
||||
beforeEach(() => {
|
||||
vi.clearAllMocks();
|
||||
});
|
||||
|
||||
describe("module initialization", () => {
|
||||
it("should compile the module successfully", async () => {
|
||||
module = await Test.createTestingModule({
|
||||
imports: [
|
||||
ConfigModule.forRoot({
|
||||
isGlobal: true,
|
||||
envFilePath: [],
|
||||
load: [
|
||||
() => ({
|
||||
MOSAIC_TELEMETRY_ENABLED: "false",
|
||||
}),
|
||||
],
|
||||
}),
|
||||
MosaicTelemetryModule,
|
||||
],
|
||||
}).compile();
|
||||
|
||||
expect(module).toBeDefined();
|
||||
await module.close();
|
||||
});
|
||||
|
||||
it("should provide MosaicTelemetryService", async () => {
|
||||
module = await Test.createTestingModule({
|
||||
imports: [
|
||||
ConfigModule.forRoot({
|
||||
isGlobal: true,
|
||||
envFilePath: [],
|
||||
load: [
|
||||
() => ({
|
||||
MOSAIC_TELEMETRY_ENABLED: "false",
|
||||
}),
|
||||
],
|
||||
}),
|
||||
MosaicTelemetryModule,
|
||||
],
|
||||
}).compile();
|
||||
|
||||
const service = module.get<MosaicTelemetryService>(MosaicTelemetryService);
|
||||
expect(service).toBeDefined();
|
||||
expect(service).toBeInstanceOf(MosaicTelemetryService);
|
||||
|
||||
await module.close();
|
||||
});
|
||||
|
||||
it("should export MosaicTelemetryService for injection in other modules", async () => {
|
||||
module = await Test.createTestingModule({
|
||||
imports: [
|
||||
ConfigModule.forRoot({
|
||||
isGlobal: true,
|
||||
envFilePath: [],
|
||||
load: [
|
||||
() => ({
|
||||
MOSAIC_TELEMETRY_ENABLED: "false",
|
||||
}),
|
||||
],
|
||||
}),
|
||||
MosaicTelemetryModule,
|
||||
],
|
||||
}).compile();
|
||||
|
||||
const service = module.get(MosaicTelemetryService);
|
||||
expect(service).toBeDefined();
|
||||
|
||||
await module.close();
|
||||
});
|
||||
});
|
||||
|
||||
describe("lifecycle integration", () => {
|
||||
it("should initialize service on module init when enabled", async () => {
|
||||
module = await Test.createTestingModule({
|
||||
imports: [
|
||||
ConfigModule.forRoot({
|
||||
isGlobal: true,
|
||||
envFilePath: [],
|
||||
load: [
|
||||
() => ({
|
||||
MOSAIC_TELEMETRY_ENABLED: "true",
|
||||
MOSAIC_TELEMETRY_SERVER_URL: "https://tel.test.local",
|
||||
MOSAIC_TELEMETRY_API_KEY: "a".repeat(64),
|
||||
MOSAIC_TELEMETRY_INSTANCE_ID: "550e8400-e29b-41d4-a716-446655440000",
|
||||
MOSAIC_TELEMETRY_DRY_RUN: "false",
|
||||
}),
|
||||
],
|
||||
}),
|
||||
MosaicTelemetryModule,
|
||||
],
|
||||
}).compile();
|
||||
|
||||
await module.init();
|
||||
|
||||
const service = module.get<MosaicTelemetryService>(MosaicTelemetryService);
|
||||
expect(service.isEnabled).toBe(true);
|
||||
|
||||
await module.close();
|
||||
});
|
||||
|
||||
it("should not start client when disabled via env", async () => {
|
||||
module = await Test.createTestingModule({
|
||||
imports: [
|
||||
ConfigModule.forRoot({
|
||||
isGlobal: true,
|
||||
envFilePath: [],
|
||||
load: [
|
||||
() => ({
|
||||
MOSAIC_TELEMETRY_ENABLED: "false",
|
||||
}),
|
||||
],
|
||||
}),
|
||||
MosaicTelemetryModule,
|
||||
],
|
||||
}).compile();
|
||||
|
||||
await module.init();
|
||||
|
||||
const service = module.get<MosaicTelemetryService>(MosaicTelemetryService);
|
||||
expect(service.isEnabled).toBe(false);
|
||||
|
||||
await module.close();
|
||||
});
|
||||
|
||||
it("should cleanly shut down on module destroy", async () => {
|
||||
module = await Test.createTestingModule({
|
||||
imports: [
|
||||
ConfigModule.forRoot({
|
||||
isGlobal: true,
|
||||
envFilePath: [],
|
||||
load: [
|
||||
() => ({
|
||||
MOSAIC_TELEMETRY_ENABLED: "true",
|
||||
MOSAIC_TELEMETRY_SERVER_URL: "https://tel.test.local",
|
||||
MOSAIC_TELEMETRY_API_KEY: "a".repeat(64),
|
||||
MOSAIC_TELEMETRY_INSTANCE_ID: "550e8400-e29b-41d4-a716-446655440000",
|
||||
MOSAIC_TELEMETRY_DRY_RUN: "false",
|
||||
}),
|
||||
],
|
||||
}),
|
||||
MosaicTelemetryModule,
|
||||
],
|
||||
}).compile();
|
||||
|
||||
await module.init();
|
||||
|
||||
const service = module.get<MosaicTelemetryService>(MosaicTelemetryService);
|
||||
expect(service.isEnabled).toBe(true);
|
||||
|
||||
await expect(module.close()).resolves.not.toThrow();
|
||||
});
|
||||
});
|
||||
});
|
||||
41
apps/api/src/mosaic-telemetry/mosaic-telemetry.module.ts
Normal file
41
apps/api/src/mosaic-telemetry/mosaic-telemetry.module.ts
Normal file
@@ -0,0 +1,41 @@
|
||||
import { Module, Global } from "@nestjs/common";
|
||||
import { ConfigModule } from "@nestjs/config";
|
||||
import { AuthModule } from "../auth/auth.module";
|
||||
import { MosaicTelemetryService } from "./mosaic-telemetry.service";
|
||||
import { PredictionService } from "./prediction.service";
|
||||
import { MosaicTelemetryController } from "./mosaic-telemetry.controller";
|
||||
|
||||
/**
|
||||
* Global module providing Mosaic Telemetry integration via @mosaicstack/telemetry-client.
|
||||
*
|
||||
* Tracks task completion events and provides crowd-sourced predictions for
|
||||
* token usage, cost estimation, and quality metrics.
|
||||
*
|
||||
* **This is separate from the OpenTelemetry (OTEL) TelemetryModule** which
|
||||
* handles distributed request tracing. This module is specifically for
|
||||
* Mosaic Stack's own telemetry aggregation service.
|
||||
*
|
||||
* Configuration via environment variables:
|
||||
* - MOSAIC_TELEMETRY_ENABLED (boolean, default: true)
|
||||
* - MOSAIC_TELEMETRY_SERVER_URL (string)
|
||||
* - MOSAIC_TELEMETRY_API_KEY (string, 64-char hex)
|
||||
* - MOSAIC_TELEMETRY_INSTANCE_ID (string, UUID)
|
||||
* - MOSAIC_TELEMETRY_DRY_RUN (boolean, default: false)
|
||||
*
|
||||
* @example
|
||||
* ```typescript
|
||||
* // In any service (no need to import module — it's global):
|
||||
* @Injectable()
|
||||
* export class MyService {
|
||||
* constructor(private readonly telemetry: MosaicTelemetryService) {}
|
||||
* }
|
||||
* ```
|
||||
*/
|
||||
@Global()
|
||||
@Module({
|
||||
imports: [ConfigModule, AuthModule],
|
||||
controllers: [MosaicTelemetryController],
|
||||
providers: [MosaicTelemetryService, PredictionService],
|
||||
exports: [MosaicTelemetryService, PredictionService],
|
||||
})
|
||||
export class MosaicTelemetryModule {}
|
||||
504
apps/api/src/mosaic-telemetry/mosaic-telemetry.service.spec.ts
Normal file
504
apps/api/src/mosaic-telemetry/mosaic-telemetry.service.spec.ts
Normal file
@@ -0,0 +1,504 @@
|
||||
import { describe, it, expect, vi, beforeEach, afterEach } from "vitest";
|
||||
import { ConfigService } from "@nestjs/config";
|
||||
import { MOSAIC_TELEMETRY_ENV } from "./mosaic-telemetry.config";
|
||||
import type {
|
||||
TaskCompletionEvent,
|
||||
PredictionQuery,
|
||||
PredictionResponse,
|
||||
} from "@mosaicstack/telemetry-client";
|
||||
import { TaskType, Complexity, Provider, Outcome } from "@mosaicstack/telemetry-client";
|
||||
|
||||
// Track mock instances created during tests
|
||||
const mockStartFn = vi.fn();
|
||||
const mockStopFn = vi.fn().mockResolvedValue(undefined);
|
||||
const mockTrackFn = vi.fn();
|
||||
const mockGetPredictionFn = vi.fn().mockReturnValue(null);
|
||||
const mockRefreshPredictionsFn = vi.fn().mockResolvedValue(undefined);
|
||||
const mockBuildFn = vi.fn().mockReturnValue({ event_id: "test-event-id" });
|
||||
|
||||
vi.mock("@mosaicstack/telemetry-client", async (importOriginal) => {
|
||||
const actual = await importOriginal<typeof import("@mosaicstack/telemetry-client")>();
|
||||
|
||||
class MockTelemetryClient {
|
||||
private _isRunning = false;
|
||||
|
||||
constructor(_config: unknown) {
|
||||
// no-op
|
||||
}
|
||||
|
||||
get eventBuilder() {
|
||||
return { build: mockBuildFn };
|
||||
}
|
||||
|
||||
start(): void {
|
||||
this._isRunning = true;
|
||||
mockStartFn();
|
||||
}
|
||||
|
||||
async stop(): Promise<void> {
|
||||
this._isRunning = false;
|
||||
await mockStopFn();
|
||||
}
|
||||
|
||||
track(event: unknown): void {
|
||||
mockTrackFn(event);
|
||||
}
|
||||
|
||||
getPrediction(query: unknown): unknown {
|
||||
return mockGetPredictionFn(query);
|
||||
}
|
||||
|
||||
async refreshPredictions(queries: unknown): Promise<void> {
|
||||
await mockRefreshPredictionsFn(queries);
|
||||
}
|
||||
|
||||
get queueSize(): number {
|
||||
return 0;
|
||||
}
|
||||
|
||||
get isRunning(): boolean {
|
||||
return this._isRunning;
|
||||
}
|
||||
}
|
||||
|
||||
return {
|
||||
...actual,
|
||||
TelemetryClient: MockTelemetryClient,
|
||||
};
|
||||
});
|
||||
|
||||
// Lazy-import the service after the mock is in place
|
||||
const { MosaicTelemetryService } = await import("./mosaic-telemetry.service");
|
||||
|
||||
/**
|
||||
* Create a ConfigService mock that returns environment values from the provided map.
|
||||
*/
|
||||
function createConfigService(envMap: Record<string, string | undefined> = {}): ConfigService {
|
||||
const configService = {
|
||||
get: vi.fn((key: string, defaultValue?: string): string => {
|
||||
const value = envMap[key];
|
||||
if (value !== undefined) {
|
||||
return value;
|
||||
}
|
||||
return defaultValue ?? "";
|
||||
}),
|
||||
} as unknown as ConfigService;
|
||||
return configService;
|
||||
}
|
||||
|
||||
/**
|
||||
* Default env config for an enabled telemetry service.
|
||||
*/
|
||||
const ENABLED_CONFIG: Record<string, string> = {
|
||||
[MOSAIC_TELEMETRY_ENV.ENABLED]: "true",
|
||||
[MOSAIC_TELEMETRY_ENV.SERVER_URL]: "https://tel.test.local",
|
||||
[MOSAIC_TELEMETRY_ENV.API_KEY]: "a".repeat(64),
|
||||
[MOSAIC_TELEMETRY_ENV.INSTANCE_ID]: "550e8400-e29b-41d4-a716-446655440000",
|
||||
[MOSAIC_TELEMETRY_ENV.DRY_RUN]: "false",
|
||||
};
|
||||
|
||||
/**
|
||||
* Create a minimal TaskCompletionEvent for testing.
|
||||
*/
|
||||
function createTestEvent(): TaskCompletionEvent {
|
||||
return {
|
||||
schema_version: "1.0.0",
|
||||
event_id: "test-event-123",
|
||||
timestamp: new Date().toISOString(),
|
||||
instance_id: "550e8400-e29b-41d4-a716-446655440000",
|
||||
task_duration_ms: 5000,
|
||||
task_type: TaskType.FEATURE,
|
||||
complexity: Complexity.MEDIUM,
|
||||
harness: "claude-code" as TaskCompletionEvent["harness"],
|
||||
model: "claude-sonnet-4-20250514",
|
||||
provider: Provider.ANTHROPIC,
|
||||
estimated_input_tokens: 1000,
|
||||
estimated_output_tokens: 500,
|
||||
actual_input_tokens: 1100,
|
||||
actual_output_tokens: 450,
|
||||
estimated_cost_usd_micros: 5000,
|
||||
actual_cost_usd_micros: 4800,
|
||||
quality_gate_passed: true,
|
||||
quality_gates_run: [],
|
||||
quality_gates_failed: [],
|
||||
context_compactions: 0,
|
||||
context_rotations: 0,
|
||||
context_utilization_final: 0.45,
|
||||
outcome: Outcome.SUCCESS,
|
||||
retry_count: 0,
|
||||
};
|
||||
}
|
||||
|
||||
describe("MosaicTelemetryService", () => {
|
||||
let service: InstanceType<typeof MosaicTelemetryService>;
|
||||
|
||||
afterEach(async () => {
|
||||
if (service) {
|
||||
await service.onModuleDestroy();
|
||||
}
|
||||
vi.clearAllMocks();
|
||||
});
|
||||
|
||||
describe("onModuleInit", () => {
|
||||
it("should initialize the client when enabled with valid config", () => {
|
||||
const configService = createConfigService(ENABLED_CONFIG);
|
||||
service = new MosaicTelemetryService(configService);
|
||||
|
||||
service.onModuleInit();
|
||||
|
||||
expect(mockStartFn).toHaveBeenCalledOnce();
|
||||
expect(service.isEnabled).toBe(true);
|
||||
});
|
||||
|
||||
it("should not initialize client when disabled", () => {
|
||||
const configService = createConfigService({
|
||||
...ENABLED_CONFIG,
|
||||
[MOSAIC_TELEMETRY_ENV.ENABLED]: "false",
|
||||
});
|
||||
service = new MosaicTelemetryService(configService);
|
||||
|
||||
service.onModuleInit();
|
||||
|
||||
expect(mockStartFn).not.toHaveBeenCalled();
|
||||
expect(service.isEnabled).toBe(false);
|
||||
});
|
||||
|
||||
it("should disable when server URL is missing", () => {
|
||||
const configService = createConfigService({
|
||||
...ENABLED_CONFIG,
|
||||
[MOSAIC_TELEMETRY_ENV.SERVER_URL]: "",
|
||||
});
|
||||
service = new MosaicTelemetryService(configService);
|
||||
|
||||
service.onModuleInit();
|
||||
|
||||
expect(service.isEnabled).toBe(false);
|
||||
});
|
||||
|
||||
it("should disable when API key is missing", () => {
|
||||
const configService = createConfigService({
|
||||
...ENABLED_CONFIG,
|
||||
[MOSAIC_TELEMETRY_ENV.API_KEY]: "",
|
||||
});
|
||||
service = new MosaicTelemetryService(configService);
|
||||
|
||||
service.onModuleInit();
|
||||
|
||||
expect(service.isEnabled).toBe(false);
|
||||
});
|
||||
|
||||
it("should disable when instance ID is missing", () => {
|
||||
const configService = createConfigService({
|
||||
...ENABLED_CONFIG,
|
||||
[MOSAIC_TELEMETRY_ENV.INSTANCE_ID]: "",
|
||||
});
|
||||
service = new MosaicTelemetryService(configService);
|
||||
|
||||
service.onModuleInit();
|
||||
|
||||
expect(service.isEnabled).toBe(false);
|
||||
});
|
||||
|
||||
it("should log dry-run mode when configured", () => {
|
||||
const configService = createConfigService({
|
||||
...ENABLED_CONFIG,
|
||||
[MOSAIC_TELEMETRY_ENV.DRY_RUN]: "true",
|
||||
});
|
||||
service = new MosaicTelemetryService(configService);
|
||||
|
||||
service.onModuleInit();
|
||||
|
||||
expect(mockStartFn).toHaveBeenCalledOnce();
|
||||
});
|
||||
});
|
||||
|
||||
describe("onModuleDestroy", () => {
|
||||
it("should stop the client on shutdown", async () => {
|
||||
const configService = createConfigService(ENABLED_CONFIG);
|
||||
service = new MosaicTelemetryService(configService);
|
||||
service.onModuleInit();
|
||||
|
||||
await service.onModuleDestroy();
|
||||
|
||||
expect(mockStopFn).toHaveBeenCalledOnce();
|
||||
});
|
||||
|
||||
it("should not throw when client is not initialized (disabled)", async () => {
|
||||
const configService = createConfigService({
|
||||
...ENABLED_CONFIG,
|
||||
[MOSAIC_TELEMETRY_ENV.ENABLED]: "false",
|
||||
});
|
||||
service = new MosaicTelemetryService(configService);
|
||||
service.onModuleInit();
|
||||
|
||||
await expect(service.onModuleDestroy()).resolves.not.toThrow();
|
||||
});
|
||||
|
||||
it("should not throw when called multiple times", async () => {
|
||||
const configService = createConfigService(ENABLED_CONFIG);
|
||||
service = new MosaicTelemetryService(configService);
|
||||
service.onModuleInit();
|
||||
|
||||
await service.onModuleDestroy();
|
||||
await expect(service.onModuleDestroy()).resolves.not.toThrow();
|
||||
});
|
||||
});
|
||||
|
||||
describe("trackTaskCompletion", () => {
|
||||
it("should queue event via client.track() when enabled", () => {
|
||||
const configService = createConfigService(ENABLED_CONFIG);
|
||||
service = new MosaicTelemetryService(configService);
|
||||
service.onModuleInit();
|
||||
|
||||
const event = createTestEvent();
|
||||
service.trackTaskCompletion(event);
|
||||
|
||||
expect(mockTrackFn).toHaveBeenCalledWith(event);
|
||||
});
|
||||
|
||||
it("should be a no-op when disabled", () => {
|
||||
const configService = createConfigService({
|
||||
...ENABLED_CONFIG,
|
||||
[MOSAIC_TELEMETRY_ENV.ENABLED]: "false",
|
||||
});
|
||||
service = new MosaicTelemetryService(configService);
|
||||
service.onModuleInit();
|
||||
|
||||
const event = createTestEvent();
|
||||
service.trackTaskCompletion(event);
|
||||
|
||||
expect(mockTrackFn).not.toHaveBeenCalled();
|
||||
});
|
||||
});
|
||||
|
||||
describe("getPrediction", () => {
|
||||
const testQuery: PredictionQuery = {
|
||||
task_type: TaskType.FEATURE,
|
||||
model: "claude-sonnet-4-20250514",
|
||||
provider: Provider.ANTHROPIC,
|
||||
complexity: Complexity.MEDIUM,
|
||||
};
|
||||
|
||||
it("should return cached prediction when available", () => {
|
||||
const mockPrediction: PredictionResponse = {
|
||||
prediction: {
|
||||
input_tokens: { p10: 100, p25: 200, median: 300, p75: 400, p90: 500 },
|
||||
output_tokens: { p10: 50, p25: 100, median: 150, p75: 200, p90: 250 },
|
||||
cost_usd_micros: { median: 5000 },
|
||||
duration_ms: { median: 10000 },
|
||||
correction_factors: { input: 1.0, output: 1.0 },
|
||||
quality: { gate_pass_rate: 0.95, success_rate: 0.9 },
|
||||
},
|
||||
metadata: {
|
||||
sample_size: 100,
|
||||
fallback_level: 0,
|
||||
confidence: "high",
|
||||
last_updated: new Date().toISOString(),
|
||||
cache_hit: true,
|
||||
},
|
||||
};
|
||||
|
||||
const configService = createConfigService(ENABLED_CONFIG);
|
||||
service = new MosaicTelemetryService(configService);
|
||||
service.onModuleInit();
|
||||
|
||||
mockGetPredictionFn.mockReturnValueOnce(mockPrediction);
|
||||
|
||||
const result = service.getPrediction(testQuery);
|
||||
|
||||
expect(result).toEqual(mockPrediction);
|
||||
expect(mockGetPredictionFn).toHaveBeenCalledWith(testQuery);
|
||||
});
|
||||
|
||||
it("should return null when disabled", () => {
|
||||
const configService = createConfigService({
|
||||
...ENABLED_CONFIG,
|
||||
[MOSAIC_TELEMETRY_ENV.ENABLED]: "false",
|
||||
});
|
||||
service = new MosaicTelemetryService(configService);
|
||||
service.onModuleInit();
|
||||
|
||||
const result = service.getPrediction(testQuery);
|
||||
|
||||
expect(result).toBeNull();
|
||||
});
|
||||
|
||||
it("should return null when no cached prediction exists", () => {
|
||||
const configService = createConfigService(ENABLED_CONFIG);
|
||||
service = new MosaicTelemetryService(configService);
|
||||
service.onModuleInit();
|
||||
|
||||
mockGetPredictionFn.mockReturnValueOnce(null);
|
||||
|
||||
const result = service.getPrediction(testQuery);
|
||||
|
||||
expect(result).toBeNull();
|
||||
});
|
||||
});
|
||||
|
||||
describe("refreshPredictions", () => {
|
||||
const testQueries: PredictionQuery[] = [
|
||||
{
|
||||
task_type: TaskType.FEATURE,
|
||||
model: "claude-sonnet-4-20250514",
|
||||
provider: Provider.ANTHROPIC,
|
||||
complexity: Complexity.MEDIUM,
|
||||
},
|
||||
];
|
||||
|
||||
it("should call client.refreshPredictions when enabled", async () => {
|
||||
const configService = createConfigService(ENABLED_CONFIG);
|
||||
service = new MosaicTelemetryService(configService);
|
||||
service.onModuleInit();
|
||||
|
||||
await service.refreshPredictions(testQueries);
|
||||
|
||||
expect(mockRefreshPredictionsFn).toHaveBeenCalledWith(testQueries);
|
||||
});
|
||||
|
||||
it("should be a no-op when disabled", async () => {
|
||||
const configService = createConfigService({
|
||||
...ENABLED_CONFIG,
|
||||
[MOSAIC_TELEMETRY_ENV.ENABLED]: "false",
|
||||
});
|
||||
service = new MosaicTelemetryService(configService);
|
||||
service.onModuleInit();
|
||||
|
||||
await service.refreshPredictions(testQueries);
|
||||
|
||||
expect(mockRefreshPredictionsFn).not.toHaveBeenCalled();
|
||||
});
|
||||
});
|
||||
|
||||
describe("eventBuilder", () => {
|
||||
it("should return EventBuilder when enabled", () => {
|
||||
const configService = createConfigService(ENABLED_CONFIG);
|
||||
service = new MosaicTelemetryService(configService);
|
||||
service.onModuleInit();
|
||||
|
||||
const builder = service.eventBuilder;
|
||||
|
||||
expect(builder).toBeDefined();
|
||||
expect(builder).not.toBeNull();
|
||||
expect(typeof builder?.build).toBe("function");
|
||||
});
|
||||
|
||||
it("should return null when disabled", () => {
|
||||
const configService = createConfigService({
|
||||
...ENABLED_CONFIG,
|
||||
[MOSAIC_TELEMETRY_ENV.ENABLED]: "false",
|
||||
});
|
||||
service = new MosaicTelemetryService(configService);
|
||||
service.onModuleInit();
|
||||
|
||||
const builder = service.eventBuilder;
|
||||
|
||||
expect(builder).toBeNull();
|
||||
});
|
||||
});
|
||||
|
||||
describe("isEnabled", () => {
|
||||
it("should return true when client is running", () => {
|
||||
const configService = createConfigService(ENABLED_CONFIG);
|
||||
service = new MosaicTelemetryService(configService);
|
||||
service.onModuleInit();
|
||||
|
||||
expect(service.isEnabled).toBe(true);
|
||||
});
|
||||
|
||||
it("should return false when disabled", () => {
|
||||
const configService = createConfigService({
|
||||
...ENABLED_CONFIG,
|
||||
[MOSAIC_TELEMETRY_ENV.ENABLED]: "false",
|
||||
});
|
||||
service = new MosaicTelemetryService(configService);
|
||||
service.onModuleInit();
|
||||
|
||||
expect(service.isEnabled).toBe(false);
|
||||
});
|
||||
});
|
||||
|
||||
describe("queueSize", () => {
|
||||
it("should return 0 when disabled", () => {
|
||||
const configService = createConfigService({
|
||||
...ENABLED_CONFIG,
|
||||
[MOSAIC_TELEMETRY_ENV.ENABLED]: "false",
|
||||
});
|
||||
service = new MosaicTelemetryService(configService);
|
||||
service.onModuleInit();
|
||||
|
||||
expect(service.queueSize).toBe(0);
|
||||
});
|
||||
|
||||
it("should delegate to client.queueSize when enabled", () => {
|
||||
const configService = createConfigService(ENABLED_CONFIG);
|
||||
service = new MosaicTelemetryService(configService);
|
||||
service.onModuleInit();
|
||||
|
||||
expect(service.queueSize).toBe(0);
|
||||
});
|
||||
});
|
||||
|
||||
describe("disabled mode (comprehensive)", () => {
|
||||
beforeEach(() => {
|
||||
const configService = createConfigService({
|
||||
...ENABLED_CONFIG,
|
||||
[MOSAIC_TELEMETRY_ENV.ENABLED]: "false",
|
||||
});
|
||||
service = new MosaicTelemetryService(configService);
|
||||
service.onModuleInit();
|
||||
});
|
||||
|
||||
it("should not make any HTTP calls when disabled", () => {
|
||||
const event = createTestEvent();
|
||||
service.trackTaskCompletion(event);
|
||||
|
||||
expect(mockTrackFn).not.toHaveBeenCalled();
|
||||
expect(mockStartFn).not.toHaveBeenCalled();
|
||||
});
|
||||
|
||||
it("should safely handle all method calls when disabled", async () => {
|
||||
expect(() => service.trackTaskCompletion(createTestEvent())).not.toThrow();
|
||||
expect(
|
||||
service.getPrediction({
|
||||
task_type: TaskType.FEATURE,
|
||||
model: "test",
|
||||
provider: Provider.ANTHROPIC,
|
||||
complexity: Complexity.LOW,
|
||||
})
|
||||
).toBeNull();
|
||||
await expect(service.refreshPredictions([])).resolves.not.toThrow();
|
||||
expect(service.eventBuilder).toBeNull();
|
||||
expect(service.isEnabled).toBe(false);
|
||||
expect(service.queueSize).toBe(0);
|
||||
});
|
||||
});
|
||||
|
||||
describe("dry-run mode", () => {
|
||||
it("should create client in dry-run mode", () => {
|
||||
const configService = createConfigService({
|
||||
...ENABLED_CONFIG,
|
||||
[MOSAIC_TELEMETRY_ENV.DRY_RUN]: "true",
|
||||
});
|
||||
service = new MosaicTelemetryService(configService);
|
||||
service.onModuleInit();
|
||||
|
||||
expect(mockStartFn).toHaveBeenCalledOnce();
|
||||
expect(service.isEnabled).toBe(true);
|
||||
});
|
||||
|
||||
it("should accept events in dry-run mode", () => {
|
||||
const configService = createConfigService({
|
||||
...ENABLED_CONFIG,
|
||||
[MOSAIC_TELEMETRY_ENV.DRY_RUN]: "true",
|
||||
});
|
||||
service = new MosaicTelemetryService(configService);
|
||||
service.onModuleInit();
|
||||
|
||||
const event = createTestEvent();
|
||||
service.trackTaskCompletion(event);
|
||||
|
||||
expect(mockTrackFn).toHaveBeenCalledWith(event);
|
||||
});
|
||||
});
|
||||
});
|
||||
164
apps/api/src/mosaic-telemetry/mosaic-telemetry.service.ts
Normal file
164
apps/api/src/mosaic-telemetry/mosaic-telemetry.service.ts
Normal file
@@ -0,0 +1,164 @@
|
||||
import { Injectable, Logger, OnModuleInit, OnModuleDestroy } from "@nestjs/common";
|
||||
import { ConfigService } from "@nestjs/config";
|
||||
import {
|
||||
TelemetryClient,
|
||||
type TaskCompletionEvent,
|
||||
type PredictionQuery,
|
||||
type PredictionResponse,
|
||||
type EventBuilder,
|
||||
} from "@mosaicstack/telemetry-client";
|
||||
import {
|
||||
loadMosaicTelemetryConfig,
|
||||
toSdkConfig,
|
||||
type MosaicTelemetryModuleConfig,
|
||||
} from "./mosaic-telemetry.config";
|
||||
|
||||
/**
|
||||
* NestJS service wrapping the @mosaicstack/telemetry-client SDK.
|
||||
*
|
||||
* Provides convenience methods for tracking task completions and reading
|
||||
* crowd-sourced predictions. When telemetry is disabled via
|
||||
* MOSAIC_TELEMETRY_ENABLED=false, all methods are safe no-ops.
|
||||
*
|
||||
* This service is provided globally by MosaicTelemetryModule — any service
|
||||
* can inject it without importing the module explicitly.
|
||||
*
|
||||
* @example
|
||||
* ```typescript
|
||||
* @Injectable()
|
||||
* export class TasksService {
|
||||
* constructor(private readonly telemetry: MosaicTelemetryService) {}
|
||||
*
|
||||
* async completeTask(taskId: string): Promise<void> {
|
||||
* // ... complete the task ...
|
||||
* const event = this.telemetry.eventBuilder.build({ ... });
|
||||
* this.telemetry.trackTaskCompletion(event);
|
||||
* }
|
||||
* }
|
||||
* ```
|
||||
*/
|
||||
@Injectable()
|
||||
export class MosaicTelemetryService implements OnModuleInit, OnModuleDestroy {
|
||||
private readonly logger = new Logger(MosaicTelemetryService.name);
|
||||
private client: TelemetryClient | null = null;
|
||||
private config: MosaicTelemetryModuleConfig | null = null;
|
||||
|
||||
constructor(private readonly configService: ConfigService) {}
|
||||
|
||||
/**
|
||||
* Initialize the telemetry client on module startup.
|
||||
* Reads configuration from environment variables and starts background submission.
|
||||
*/
|
||||
onModuleInit(): void {
|
||||
this.config = loadMosaicTelemetryConfig(this.configService);
|
||||
|
||||
if (!this.config.enabled) {
|
||||
this.logger.log("Mosaic Telemetry is disabled");
|
||||
return;
|
||||
}
|
||||
|
||||
if (!this.config.serverUrl || !this.config.apiKey || !this.config.instanceId) {
|
||||
this.logger.warn(
|
||||
"Mosaic Telemetry is enabled but missing configuration " +
|
||||
"(MOSAIC_TELEMETRY_SERVER_URL, MOSAIC_TELEMETRY_API_KEY, or MOSAIC_TELEMETRY_INSTANCE_ID). " +
|
||||
"Telemetry will remain disabled."
|
||||
);
|
||||
this.config = { ...this.config, enabled: false };
|
||||
return;
|
||||
}
|
||||
|
||||
const sdkConfig = toSdkConfig(this.config, (error: Error) => {
|
||||
this.logger.error(`Telemetry client error: ${error.message}`, error.stack);
|
||||
});
|
||||
|
||||
this.client = new TelemetryClient(sdkConfig);
|
||||
this.client.start();
|
||||
|
||||
const mode = this.config.dryRun ? "dry-run" : "live";
|
||||
this.logger.log(`Mosaic Telemetry client started (${mode}) -> ${this.config.serverUrl}`);
|
||||
}
|
||||
|
||||
/**
|
||||
* Stop the telemetry client on module shutdown.
|
||||
* Flushes any remaining queued events before stopping.
|
||||
*/
|
||||
async onModuleDestroy(): Promise<void> {
|
||||
if (this.client) {
|
||||
this.logger.log("Stopping Mosaic Telemetry client...");
|
||||
await this.client.stop();
|
||||
this.client = null;
|
||||
this.logger.log("Mosaic Telemetry client stopped");
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Queue a task completion event for batch submission.
|
||||
* No-op when telemetry is disabled.
|
||||
*
|
||||
* @param event - The task completion event to track
|
||||
*/
|
||||
trackTaskCompletion(event: TaskCompletionEvent): void {
|
||||
if (!this.client) {
|
||||
return;
|
||||
}
|
||||
this.client.track(event);
|
||||
}
|
||||
|
||||
/**
|
||||
* Get a cached prediction for the given query.
|
||||
* Returns null when telemetry is disabled or if not cached/expired.
|
||||
*
|
||||
* @param query - The prediction query parameters
|
||||
* @returns Cached prediction response, or null
|
||||
*/
|
||||
getPrediction(query: PredictionQuery): PredictionResponse | null {
|
||||
if (!this.client) {
|
||||
return null;
|
||||
}
|
||||
return this.client.getPrediction(query);
|
||||
}
|
||||
|
||||
/**
|
||||
* Force-refresh predictions from the telemetry server.
|
||||
* No-op when telemetry is disabled.
|
||||
*
|
||||
* @param queries - Array of prediction queries to refresh
|
||||
*/
|
||||
async refreshPredictions(queries: PredictionQuery[]): Promise<void> {
|
||||
if (!this.client) {
|
||||
return;
|
||||
}
|
||||
await this.client.refreshPredictions(queries);
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the EventBuilder for constructing TaskCompletionEvent objects.
|
||||
* Returns null when telemetry is disabled.
|
||||
*
|
||||
* @returns EventBuilder instance, or null if disabled
|
||||
*/
|
||||
get eventBuilder(): EventBuilder | null {
|
||||
if (!this.client) {
|
||||
return null;
|
||||
}
|
||||
return this.client.eventBuilder;
|
||||
}
|
||||
|
||||
/**
|
||||
* Whether the telemetry client is currently active and running.
|
||||
*/
|
||||
get isEnabled(): boolean {
|
||||
return this.client?.isRunning ?? false;
|
||||
}
|
||||
|
||||
/**
|
||||
* Number of events currently queued for submission.
|
||||
* Returns 0 when telemetry is disabled.
|
||||
*/
|
||||
get queueSize(): number {
|
||||
if (!this.client) {
|
||||
return 0;
|
||||
}
|
||||
return this.client.queueSize;
|
||||
}
|
||||
}
|
||||
297
apps/api/src/mosaic-telemetry/prediction.service.spec.ts
Normal file
297
apps/api/src/mosaic-telemetry/prediction.service.spec.ts
Normal file
@@ -0,0 +1,297 @@
|
||||
import { describe, it, expect, beforeEach, vi } from "vitest";
|
||||
import { Test, TestingModule } from "@nestjs/testing";
|
||||
import { TaskType, Complexity, Provider } from "@mosaicstack/telemetry-client";
|
||||
import type { PredictionResponse, PredictionQuery } from "@mosaicstack/telemetry-client";
|
||||
import { MosaicTelemetryService } from "./mosaic-telemetry.service";
|
||||
import { PredictionService } from "./prediction.service";
|
||||
|
||||
describe("PredictionService", () => {
|
||||
let service: PredictionService;
|
||||
let mockTelemetryService: {
|
||||
isEnabled: boolean;
|
||||
getPrediction: ReturnType<typeof vi.fn>;
|
||||
refreshPredictions: ReturnType<typeof vi.fn>;
|
||||
};
|
||||
|
||||
const mockPredictionResponse: PredictionResponse = {
|
||||
prediction: {
|
||||
input_tokens: {
|
||||
p10: 50,
|
||||
p25: 80,
|
||||
median: 120,
|
||||
p75: 200,
|
||||
p90: 350,
|
||||
},
|
||||
output_tokens: {
|
||||
p10: 100,
|
||||
p25: 150,
|
||||
median: 250,
|
||||
p75: 400,
|
||||
p90: 600,
|
||||
},
|
||||
cost_usd_micros: {
|
||||
p10: 500,
|
||||
p25: 800,
|
||||
median: 1200,
|
||||
p75: 2000,
|
||||
p90: 3500,
|
||||
},
|
||||
duration_ms: {
|
||||
p10: 200,
|
||||
p25: 400,
|
||||
median: 800,
|
||||
p75: 1500,
|
||||
p90: 3000,
|
||||
},
|
||||
correction_factors: {
|
||||
input: 1.0,
|
||||
output: 1.0,
|
||||
},
|
||||
quality: {
|
||||
gate_pass_rate: 0.95,
|
||||
success_rate: 0.92,
|
||||
},
|
||||
},
|
||||
metadata: {
|
||||
sample_size: 150,
|
||||
fallback_level: 0,
|
||||
confidence: "high",
|
||||
last_updated: "2026-02-15T00:00:00Z",
|
||||
cache_hit: true,
|
||||
},
|
||||
};
|
||||
|
||||
const nullPredictionResponse: PredictionResponse = {
|
||||
prediction: null,
|
||||
metadata: {
|
||||
sample_size: 0,
|
||||
fallback_level: 3,
|
||||
confidence: "none",
|
||||
last_updated: null,
|
||||
cache_hit: false,
|
||||
},
|
||||
};
|
||||
|
||||
beforeEach(async () => {
|
||||
mockTelemetryService = {
|
||||
isEnabled: true,
|
||||
getPrediction: vi.fn().mockReturnValue(mockPredictionResponse),
|
||||
refreshPredictions: vi.fn().mockResolvedValue(undefined),
|
||||
};
|
||||
|
||||
const module: TestingModule = await Test.createTestingModule({
|
||||
providers: [
|
||||
PredictionService,
|
||||
{
|
||||
provide: MosaicTelemetryService,
|
||||
useValue: mockTelemetryService,
|
||||
},
|
||||
],
|
||||
}).compile();
|
||||
|
||||
service = module.get<PredictionService>(PredictionService);
|
||||
});
|
||||
|
||||
it("should be defined", () => {
|
||||
expect(service).toBeDefined();
|
||||
});
|
||||
|
||||
// ---------- getEstimate ----------
|
||||
|
||||
describe("getEstimate", () => {
|
||||
it("should return prediction response for valid query", () => {
|
||||
const result = service.getEstimate(
|
||||
TaskType.IMPLEMENTATION,
|
||||
"claude-sonnet-4-5",
|
||||
Provider.ANTHROPIC,
|
||||
Complexity.LOW
|
||||
);
|
||||
|
||||
expect(result).toEqual(mockPredictionResponse);
|
||||
expect(mockTelemetryService.getPrediction).toHaveBeenCalledWith({
|
||||
task_type: TaskType.IMPLEMENTATION,
|
||||
model: "claude-sonnet-4-5",
|
||||
provider: Provider.ANTHROPIC,
|
||||
complexity: Complexity.LOW,
|
||||
});
|
||||
});
|
||||
|
||||
it("should pass correct query parameters to telemetry service", () => {
|
||||
service.getEstimate(TaskType.CODE_REVIEW, "gpt-4o", Provider.OPENAI, Complexity.HIGH);
|
||||
|
||||
expect(mockTelemetryService.getPrediction).toHaveBeenCalledWith({
|
||||
task_type: TaskType.CODE_REVIEW,
|
||||
model: "gpt-4o",
|
||||
provider: Provider.OPENAI,
|
||||
complexity: Complexity.HIGH,
|
||||
});
|
||||
});
|
||||
|
||||
it("should return null when telemetry returns null", () => {
|
||||
mockTelemetryService.getPrediction.mockReturnValue(null);
|
||||
|
||||
const result = service.getEstimate(
|
||||
TaskType.IMPLEMENTATION,
|
||||
"claude-sonnet-4-5",
|
||||
Provider.ANTHROPIC,
|
||||
Complexity.LOW
|
||||
);
|
||||
|
||||
expect(result).toBeNull();
|
||||
});
|
||||
|
||||
it("should return null prediction response when confidence is none", () => {
|
||||
mockTelemetryService.getPrediction.mockReturnValue(nullPredictionResponse);
|
||||
|
||||
const result = service.getEstimate(
|
||||
TaskType.IMPLEMENTATION,
|
||||
"unknown-model",
|
||||
Provider.UNKNOWN,
|
||||
Complexity.LOW
|
||||
);
|
||||
|
||||
expect(result).toEqual(nullPredictionResponse);
|
||||
expect(result?.metadata.confidence).toBe("none");
|
||||
});
|
||||
|
||||
it("should return null and not throw when getPrediction throws", () => {
|
||||
mockTelemetryService.getPrediction.mockImplementation(() => {
|
||||
throw new Error("Prediction fetch failed");
|
||||
});
|
||||
|
||||
const result = service.getEstimate(
|
||||
TaskType.IMPLEMENTATION,
|
||||
"claude-sonnet-4-5",
|
||||
Provider.ANTHROPIC,
|
||||
Complexity.LOW
|
||||
);
|
||||
|
||||
expect(result).toBeNull();
|
||||
});
|
||||
|
||||
it("should handle non-Error thrown objects gracefully", () => {
|
||||
mockTelemetryService.getPrediction.mockImplementation(() => {
|
||||
throw "string error";
|
||||
});
|
||||
|
||||
const result = service.getEstimate(
|
||||
TaskType.IMPLEMENTATION,
|
||||
"claude-sonnet-4-5",
|
||||
Provider.ANTHROPIC,
|
||||
Complexity.LOW
|
||||
);
|
||||
|
||||
expect(result).toBeNull();
|
||||
});
|
||||
});
|
||||
|
||||
// ---------- refreshCommonPredictions ----------
|
||||
|
||||
describe("refreshCommonPredictions", () => {
|
||||
it("should call refreshPredictions with multiple query combinations", async () => {
|
||||
await service.refreshCommonPredictions();
|
||||
|
||||
expect(mockTelemetryService.refreshPredictions).toHaveBeenCalledTimes(1);
|
||||
|
||||
const queries: PredictionQuery[] = mockTelemetryService.refreshPredictions.mock.calls[0][0];
|
||||
|
||||
// Should have queries for cross-product of models, task types, and complexities
|
||||
expect(queries.length).toBeGreaterThan(0);
|
||||
|
||||
// Verify all queries have valid structure
|
||||
for (const query of queries) {
|
||||
expect(query).toHaveProperty("task_type");
|
||||
expect(query).toHaveProperty("model");
|
||||
expect(query).toHaveProperty("provider");
|
||||
expect(query).toHaveProperty("complexity");
|
||||
}
|
||||
});
|
||||
|
||||
it("should include Anthropic model predictions", async () => {
|
||||
await service.refreshCommonPredictions();
|
||||
|
||||
const queries: PredictionQuery[] = mockTelemetryService.refreshPredictions.mock.calls[0][0];
|
||||
|
||||
const anthropicQueries = queries.filter(
|
||||
(q: PredictionQuery) => q.provider === Provider.ANTHROPIC
|
||||
);
|
||||
expect(anthropicQueries.length).toBeGreaterThan(0);
|
||||
});
|
||||
|
||||
it("should include OpenAI model predictions", async () => {
|
||||
await service.refreshCommonPredictions();
|
||||
|
||||
const queries: PredictionQuery[] = mockTelemetryService.refreshPredictions.mock.calls[0][0];
|
||||
|
||||
const openaiQueries = queries.filter((q: PredictionQuery) => q.provider === Provider.OPENAI);
|
||||
expect(openaiQueries.length).toBeGreaterThan(0);
|
||||
});
|
||||
|
||||
it("should not call refreshPredictions when telemetry is disabled", async () => {
|
||||
mockTelemetryService.isEnabled = false;
|
||||
|
||||
await service.refreshCommonPredictions();
|
||||
|
||||
expect(mockTelemetryService.refreshPredictions).not.toHaveBeenCalled();
|
||||
});
|
||||
|
||||
it("should not throw when refreshPredictions rejects", async () => {
|
||||
mockTelemetryService.refreshPredictions.mockRejectedValue(new Error("Server unreachable"));
|
||||
|
||||
// Should not throw
|
||||
await expect(service.refreshCommonPredictions()).resolves.not.toThrow();
|
||||
});
|
||||
|
||||
it("should include common task types in queries", async () => {
|
||||
await service.refreshCommonPredictions();
|
||||
|
||||
const queries: PredictionQuery[] = mockTelemetryService.refreshPredictions.mock.calls[0][0];
|
||||
|
||||
const taskTypes = new Set(queries.map((q: PredictionQuery) => q.task_type));
|
||||
|
||||
expect(taskTypes.has(TaskType.IMPLEMENTATION)).toBe(true);
|
||||
expect(taskTypes.has(TaskType.PLANNING)).toBe(true);
|
||||
expect(taskTypes.has(TaskType.CODE_REVIEW)).toBe(true);
|
||||
});
|
||||
|
||||
it("should include common complexity levels in queries", async () => {
|
||||
await service.refreshCommonPredictions();
|
||||
|
||||
const queries: PredictionQuery[] = mockTelemetryService.refreshPredictions.mock.calls[0][0];
|
||||
|
||||
const complexities = new Set(queries.map((q: PredictionQuery) => q.complexity));
|
||||
|
||||
expect(complexities.has(Complexity.LOW)).toBe(true);
|
||||
expect(complexities.has(Complexity.MEDIUM)).toBe(true);
|
||||
});
|
||||
});
|
||||
|
||||
// ---------- onModuleInit ----------
|
||||
|
||||
describe("onModuleInit", () => {
|
||||
it("should trigger refreshCommonPredictions on init when telemetry is enabled", () => {
|
||||
// refreshPredictions is async, but onModuleInit fires it and forgets
|
||||
service.onModuleInit();
|
||||
|
||||
// Give the promise microtask a chance to execute
|
||||
expect(mockTelemetryService.isEnabled).toBe(true);
|
||||
// refreshPredictions will be called asynchronously
|
||||
});
|
||||
|
||||
it("should not refresh when telemetry is disabled", () => {
|
||||
mockTelemetryService.isEnabled = false;
|
||||
|
||||
service.onModuleInit();
|
||||
|
||||
// refreshPredictions should not be called since we returned early
|
||||
expect(mockTelemetryService.refreshPredictions).not.toHaveBeenCalled();
|
||||
});
|
||||
|
||||
it("should not throw when refresh fails on init", () => {
|
||||
mockTelemetryService.refreshPredictions.mockRejectedValue(new Error("Connection refused"));
|
||||
|
||||
// Should not throw
|
||||
expect(() => service.onModuleInit()).not.toThrow();
|
||||
});
|
||||
});
|
||||
});
|
||||
161
apps/api/src/mosaic-telemetry/prediction.service.ts
Normal file
161
apps/api/src/mosaic-telemetry/prediction.service.ts
Normal file
@@ -0,0 +1,161 @@
|
||||
import { Injectable, Logger, OnModuleInit } from "@nestjs/common";
|
||||
import {
|
||||
TaskType,
|
||||
Complexity,
|
||||
Provider,
|
||||
type PredictionQuery,
|
||||
type PredictionResponse,
|
||||
} from "@mosaicstack/telemetry-client";
|
||||
import { MosaicTelemetryService } from "./mosaic-telemetry.service";
|
||||
|
||||
/**
|
||||
* Common model-provider combinations used for pre-fetching predictions.
|
||||
* These represent the most frequently used LLM configurations.
|
||||
*/
|
||||
const COMMON_MODELS: { model: string; provider: Provider }[] = [
|
||||
{ model: "claude-sonnet-4-5", provider: Provider.ANTHROPIC },
|
||||
{ model: "claude-opus-4", provider: Provider.ANTHROPIC },
|
||||
{ model: "claude-haiku-4-5", provider: Provider.ANTHROPIC },
|
||||
{ model: "gpt-4o", provider: Provider.OPENAI },
|
||||
{ model: "gpt-4o-mini", provider: Provider.OPENAI },
|
||||
];
|
||||
|
||||
/**
|
||||
* Common task types to pre-fetch predictions for.
|
||||
*/
|
||||
const COMMON_TASK_TYPES: TaskType[] = [
|
||||
TaskType.IMPLEMENTATION,
|
||||
TaskType.PLANNING,
|
||||
TaskType.CODE_REVIEW,
|
||||
];
|
||||
|
||||
/**
|
||||
* Common complexity levels to pre-fetch predictions for.
|
||||
*/
|
||||
const COMMON_COMPLEXITIES: Complexity[] = [Complexity.LOW, Complexity.MEDIUM];
|
||||
|
||||
/**
|
||||
* PredictionService
|
||||
*
|
||||
* Provides pre-task cost and token estimates using crowd-sourced prediction data
|
||||
* from the Mosaic Telemetry server. Predictions are cached by the underlying SDK
|
||||
* with a 6-hour TTL.
|
||||
*
|
||||
* This service is intentionally non-blocking: if predictions are unavailable
|
||||
* (telemetry disabled, server unreachable, no data), all methods return null
|
||||
* without throwing errors. Task execution should never be blocked by prediction
|
||||
* failures.
|
||||
*
|
||||
* @example
|
||||
* ```typescript
|
||||
* const estimate = this.predictionService.getEstimate(
|
||||
* TaskType.IMPLEMENTATION,
|
||||
* "claude-sonnet-4-5",
|
||||
* Provider.ANTHROPIC,
|
||||
* Complexity.LOW,
|
||||
* );
|
||||
* if (estimate?.prediction) {
|
||||
* console.log(`Estimated cost: ${estimate.prediction.cost_usd_micros}`);
|
||||
* }
|
||||
* ```
|
||||
*/
|
||||
@Injectable()
|
||||
export class PredictionService implements OnModuleInit {
|
||||
private readonly logger = new Logger(PredictionService.name);
|
||||
|
||||
constructor(private readonly telemetry: MosaicTelemetryService) {}
|
||||
|
||||
/**
|
||||
* Refresh common predictions on startup.
|
||||
* Runs asynchronously and never blocks module initialization.
|
||||
*/
|
||||
onModuleInit(): void {
|
||||
if (!this.telemetry.isEnabled) {
|
||||
this.logger.log("Telemetry disabled - skipping prediction refresh");
|
||||
return;
|
||||
}
|
||||
|
||||
// Fire-and-forget: refresh in the background
|
||||
this.refreshCommonPredictions().catch((error: unknown) => {
|
||||
const msg = error instanceof Error ? error.message : String(error);
|
||||
this.logger.warn(`Failed to refresh common predictions on startup: ${msg}`);
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* Get a cost/token estimate for a given task configuration.
|
||||
*
|
||||
* Returns the cached prediction from the SDK, or null if:
|
||||
* - Telemetry is disabled
|
||||
* - No prediction data exists for this combination
|
||||
* - The prediction has expired
|
||||
*
|
||||
* @param taskType - The type of task to estimate
|
||||
* @param model - The model name (e.g. "claude-sonnet-4-5")
|
||||
* @param provider - The provider enum value
|
||||
* @param complexity - The complexity level
|
||||
* @returns Prediction response with estimates and confidence, or null
|
||||
*/
|
||||
getEstimate(
|
||||
taskType: TaskType,
|
||||
model: string,
|
||||
provider: Provider,
|
||||
complexity: Complexity
|
||||
): PredictionResponse | null {
|
||||
try {
|
||||
const query: PredictionQuery = {
|
||||
task_type: taskType,
|
||||
model,
|
||||
provider,
|
||||
complexity,
|
||||
};
|
||||
|
||||
return this.telemetry.getPrediction(query);
|
||||
} catch (error: unknown) {
|
||||
const msg = error instanceof Error ? error.message : String(error);
|
||||
this.logger.warn(`Failed to get prediction estimate: ${msg}`);
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Refresh predictions for commonly used (taskType, model, provider, complexity) combinations.
|
||||
*
|
||||
* Generates the cross-product of common models, task types, and complexities,
|
||||
* then batch-refreshes them from the telemetry server. The SDK caches the
|
||||
* results with a 6-hour TTL.
|
||||
*
|
||||
* This method is safe to call at any time. If telemetry is disabled or the
|
||||
* server is unreachable, it completes without error.
|
||||
*/
|
||||
async refreshCommonPredictions(): Promise<void> {
|
||||
if (!this.telemetry.isEnabled) {
|
||||
return;
|
||||
}
|
||||
|
||||
const queries: PredictionQuery[] = [];
|
||||
|
||||
for (const { model, provider } of COMMON_MODELS) {
|
||||
for (const taskType of COMMON_TASK_TYPES) {
|
||||
for (const complexity of COMMON_COMPLEXITIES) {
|
||||
queries.push({
|
||||
task_type: taskType,
|
||||
model,
|
||||
provider,
|
||||
complexity,
|
||||
});
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
this.logger.log(`Refreshing ${String(queries.length)} common prediction queries...`);
|
||||
|
||||
try {
|
||||
await this.telemetry.refreshPredictions(queries);
|
||||
this.logger.log(`Successfully refreshed ${String(queries.length)} predictions`);
|
||||
} catch (error: unknown) {
|
||||
const msg = error instanceof Error ? error.message : String(error);
|
||||
this.logger.warn(`Failed to refresh predictions: ${msg}`);
|
||||
}
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user