Files
oneuptime/Common/Server/Services/LlmProviderService.ts
Nawaz Dhandala 035edaf435 feat: Add AI Logs functionality with LLM logging and management
- Introduced LlmLog model to track AI API calls, including details like provider, tokens used, cost, and status.
- Implemented AILogService to handle AI log creation and management, including billing checks and log updates.
- Created LlmLogsTable component for displaying AI logs in the dashboard with filtering and modal views for request/response details.
- Added new routes and pages for viewing AI logs in the context of incidents, alerts, and settings.
- Updated PageMap and RouteMap to include new AI log views.
- Enhanced error handling and logging for AI API interactions.
2025-12-15 20:13:36 +00:00

159 lines
4.0 KiB
TypeScript

import DatabaseService from "./DatabaseService";
import Model from "../../Models/DatabaseModels/LlmProvider";
import CreateBy from "../Types/Database/CreateBy";
import { OnCreate, OnUpdate } from "../Types/Database/Hooks";
import ObjectID from "../../Types/ObjectID";
import UpdateBy from "../Types/Database/UpdateBy";
import QueryHelper from "../Types/Database/QueryHelper";
import LIMIT_MAX from "../../Types/Database/LimitMax";
import CaptureSpan from "../Utils/Telemetry/CaptureSpan";
export class Service extends DatabaseService<Model> {
public constructor() {
super(Model);
}
protected override async onBeforeCreate(
createBy: CreateBy<Model>,
): Promise<OnCreate<Model>> {
// When creating a new LLM provider, set it as default by default
if (createBy.data.isDefault === undefined) {
createBy.data.isDefault = true;
}
// If this provider is being set as default, unset other defaults in the same project
if (createBy.data.isDefault && createBy.data.projectId) {
await this.updateBy({
query: {
projectId: createBy.data.projectId,
isDefault: true,
},
data: {
isDefault: false,
},
props: {
isRoot: true,
},
limit: LIMIT_MAX,
skip: 0,
});
}
return { createBy, carryForward: null };
}
protected override async onBeforeUpdate(
updateBy: UpdateBy<Model>,
): Promise<OnUpdate<Model>> {
// If setting isDefault to true, we need to unset other defaults in the same project
if (updateBy.data.isDefault === true) {
// Get the items being updated to find their project IDs
const itemsToUpdate: Array<Model> = await this.findBy({
query: updateBy.query,
select: {
_id: true,
projectId: true,
},
props: {
isRoot: true,
},
limit: LIMIT_MAX,
skip: 0,
});
// Collect unique project IDs
const projectIds: Set<string> = new Set();
const itemIds: Set<string> = new Set();
for (const item of itemsToUpdate) {
if (item.projectId) {
projectIds.add(item.projectId.toString());
}
if (item._id) {
itemIds.add(item._id);
}
}
// For each project, unset the default on other providers
for (const projectIdStr of projectIds) {
const projectId: ObjectID = new ObjectID(projectIdStr);
await this.updateBy({
query: {
projectId: projectId,
isDefault: true,
_id: QueryHelper.notInOrNull(Array.from(itemIds)),
},
data: {
isDefault: false,
},
props: {
isRoot: true,
},
limit: LIMIT_MAX,
skip: 0,
});
}
}
return { updateBy, carryForward: null };
}
@CaptureSpan()
public async getLLMProviderForProject(
projectId: ObjectID,
): Promise<Model | null> {
// First try to get the default provider for the project
let provider: Model | null = await this.findOneBy({
query: {
projectId: projectId,
isDefault: true,
},
select: {
_id: true,
name: true,
llmType: true,
apiKey: true,
baseUrl: true,
modelName: true,
isGlobalLlm: true,
costPerMillionTokensInUSDCents: true,
},
props: {
isRoot: true,
},
});
if (provider) {
return provider;
}
// If no default provider, get any global provider for the project.
provider = await this.findOneBy({
query: {
projectId: QueryHelper.isNull(),
isGlobalLlm: true,
},
select: {
_id: true,
name: true,
llmType: true,
apiKey: true,
baseUrl: true,
modelName: true,
isGlobalLlm: true,
costPerMillionTokensInUSDCents: true,
},
props: {
isRoot: true,
},
});
if (provider) {
return provider;
}
return null;
}
}
export default new Service();