@wrcb/cb-common 1.0.837 → 1.0.838

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.
@@ -0,0 +1,23 @@
1
+ import { Subjects } from '../subjects';
2
+ export type AIProvider = 'openai' | 'cerebras';
3
+ export type AIRequestType = 'chat' | 'tts' | 'transcribe' | 'image-generate' | 'image-edit';
4
+ export interface AIRequestMadeEvent {
5
+ subject: Subjects.AIRequestMade;
6
+ data: {
7
+ service: string;
8
+ provider: AIProvider;
9
+ type: AIRequestType;
10
+ model: string;
11
+ promptTokens: number;
12
+ completionTokens: number;
13
+ totalTokens: number;
14
+ costUSD: number;
15
+ durationMs: number;
16
+ requestedAt: string;
17
+ prompt: string;
18
+ inputChars?: number;
19
+ imageSize?: string;
20
+ success: boolean;
21
+ errorMessage?: string;
22
+ };
23
+ }
@@ -0,0 +1,2 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
@@ -0,0 +1,6 @@
1
+ import { Publisher } from '../basePublisher';
2
+ import { AIRequestMadeEvent } from './aiRequestMadeEvent';
3
+ import { Subjects } from '../subjects';
4
+ export declare class AIRequestMadePublisher extends Publisher<AIRequestMadeEvent> {
5
+ subject: Subjects.AIRequestMade;
6
+ }
@@ -0,0 +1,12 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.AIRequestMadePublisher = void 0;
4
+ const basePublisher_1 = require("../basePublisher");
5
+ const subjects_1 = require("../subjects");
6
+ class AIRequestMadePublisher extends basePublisher_1.Publisher {
7
+ constructor() {
8
+ super(...arguments);
9
+ this.subject = subjects_1.Subjects.AIRequestMade;
10
+ }
11
+ }
12
+ exports.AIRequestMadePublisher = AIRequestMadePublisher;
@@ -0,0 +1,2 @@
1
+ export * from './aiRequestMadeEvent';
2
+ export * from './aiRequestMadePublisher';
@@ -0,0 +1,18 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __exportStar = (this && this.__exportStar) || function(m, exports) {
14
+ for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
15
+ };
16
+ Object.defineProperty(exports, "__esModule", { value: true });
17
+ __exportStar(require("./aiRequestMadeEvent"), exports);
18
+ __exportStar(require("./aiRequestMadePublisher"), exports);
@@ -61,5 +61,6 @@ export declare enum Subjects {
61
61
  SMATopicApproved = "sma:topic:approved",
62
62
  SMAVideoReady = "sma:video:ready",
63
63
  SMAPostPublished = "sma:post:published",
64
- SMAPostFailed = "sma:post:failed"
64
+ SMAPostFailed = "sma:post:failed",
65
+ AIRequestMade = "ai:request:made"
65
66
  }
@@ -73,4 +73,6 @@ var Subjects;
73
73
  Subjects["SMAVideoReady"] = "sma:video:ready";
74
74
  Subjects["SMAPostPublished"] = "sma:post:published";
75
75
  Subjects["SMAPostFailed"] = "sma:post:failed";
76
+ // ai tracking
77
+ Subjects["AIRequestMade"] = "ai:request:made";
76
78
  })(Subjects || (exports.Subjects = Subjects = {}));
package/build/index.d.ts CHANGED
@@ -56,6 +56,8 @@ export * from './events/baseListener';
56
56
  export * from './events/basePublisher';
57
57
  export * from './events/youtube-dark';
58
58
  export * from './events/social-media-automator';
59
+ export * from './events/ai';
60
+ export * from './services/ai/AIClient';
59
61
  export * from './services/TenantDataService';
60
62
  export * from './services/CurrencyService';
61
63
  export * from './helpers/calculateBonus';
package/build/index.js CHANGED
@@ -78,6 +78,8 @@ __exportStar(require("./events/baseListener"), exports);
78
78
  __exportStar(require("./events/basePublisher"), exports);
79
79
  __exportStar(require("./events/youtube-dark"), exports);
80
80
  __exportStar(require("./events/social-media-automator"), exports);
81
+ __exportStar(require("./events/ai"), exports);
82
+ __exportStar(require("./services/ai/AIClient"), exports);
81
83
  __exportStar(require("./services/TenantDataService"), exports);
82
84
  __exportStar(require("./services/CurrencyService"), exports);
83
85
  __exportStar(require("./helpers/calculateBonus"), exports);
@@ -0,0 +1,63 @@
1
+ import OpenAI from 'openai';
2
+ import { Stan } from 'node-nats-streaming';
3
+ import { AIProvider } from '../../events/ai/aiRequestMadeEvent';
4
+ interface ChatParams {
5
+ provider: AIProvider;
6
+ model: string;
7
+ messages: OpenAI.ChatCompletionMessageParam[];
8
+ maxTokens?: number;
9
+ temperature?: number;
10
+ responseFormat?: {
11
+ type: 'json_object' | 'text';
12
+ };
13
+ openaiClient: OpenAI;
14
+ cerebrasClient?: OpenAI;
15
+ }
16
+ interface TTSParams {
17
+ model: string;
18
+ voice: string;
19
+ input: string;
20
+ speed?: number;
21
+ instructions?: string;
22
+ responseFormat?: string;
23
+ openaiClient: OpenAI;
24
+ }
25
+ interface TranscribeParams {
26
+ audioBuffer: Buffer;
27
+ mimeType?: string;
28
+ openaiClient: OpenAI;
29
+ }
30
+ interface ImageGenerateParams {
31
+ model: string;
32
+ prompt: string;
33
+ size: string;
34
+ quality?: string;
35
+ n?: number;
36
+ openaiClient: OpenAI;
37
+ }
38
+ interface ImageEditParams {
39
+ model: string;
40
+ imageBuffer: Buffer;
41
+ prompt: string;
42
+ size: string;
43
+ n?: number;
44
+ openaiClient: OpenAI;
45
+ }
46
+ export interface ChatResult {
47
+ content: string;
48
+ promptTokens: number;
49
+ completionTokens: number;
50
+ totalTokens: number;
51
+ }
52
+ export declare class AIClient {
53
+ private readonly natsClient;
54
+ private readonly service;
55
+ constructor(natsClient: Stan, service: string);
56
+ private emit;
57
+ chat(params: ChatParams): Promise<ChatResult>;
58
+ tts(params: TTSParams): Promise<Buffer>;
59
+ transcribe(params: TranscribeParams): Promise<string>;
60
+ generateImage(params: ImageGenerateParams): Promise<string>;
61
+ editImage(params: ImageEditParams): Promise<string>;
62
+ }
63
+ export {};
@@ -0,0 +1,215 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || function (mod) {
19
+ if (mod && mod.__esModule) return mod;
20
+ var result = {};
21
+ if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
22
+ __setModuleDefault(result, mod);
23
+ return result;
24
+ };
25
+ var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
26
+ function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
27
+ return new (P || (P = Promise))(function (resolve, reject) {
28
+ function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
29
+ function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
30
+ function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
31
+ step((generator = generator.apply(thisArg, _arguments || [])).next());
32
+ });
33
+ };
34
+ Object.defineProperty(exports, "__esModule", { value: true });
35
+ exports.AIClient = void 0;
36
+ const openai_1 = require("openai");
37
+ const aiRequestMadePublisher_1 = require("../../events/ai/aiRequestMadePublisher");
38
+ // Preços em USD por 1K tokens (ou por unidade para TTS/imagens)
39
+ const COST_PER_1K = {
40
+ 'gpt-4o': { input: 0.0025, output: 0.010 },
41
+ 'gpt-4o-mini': { input: 0.00015, output: 0.0006 },
42
+ 'gpt-oss-120b': { input: 0, output: 0 },
43
+ 'whisper-1': { input: 0.006, output: 0 }, // por minuto de áudio
44
+ 'tts-1': { input: 0.015, output: 0 }, // por 1K chars
45
+ 'tts-1-hd': { input: 0.030, output: 0 },
46
+ 'gpt-4o-mini-tts': { input: 0.012, output: 0 },
47
+ 'dall-e-3': { input: 0.040, output: 0 }, // por imagem standard 1024x1024
48
+ 'dall-e-2': { input: 0.020, output: 0 },
49
+ 'gpt-image-1': { input: 0.067, output: 0 }, // medium quality
50
+ };
51
+ function calcCost(model, promptTokens, completionTokens) {
52
+ var _a;
53
+ const rates = (_a = COST_PER_1K[model]) !== null && _a !== void 0 ? _a : { input: 0, output: 0 };
54
+ return (promptTokens / 1000) * rates.input + (completionTokens / 1000) * rates.output;
55
+ }
56
+ class AIClient {
57
+ constructor(natsClient, service) {
58
+ this.natsClient = natsClient;
59
+ this.service = service;
60
+ }
61
+ emit(data) {
62
+ return __awaiter(this, void 0, void 0, function* () {
63
+ try {
64
+ yield new aiRequestMadePublisher_1.AIRequestMadePublisher(this.natsClient).publish({
65
+ service: this.service,
66
+ provider: data.provider,
67
+ type: data.type,
68
+ model: data.model,
69
+ promptTokens: data.promptTokens,
70
+ completionTokens: data.completionTokens,
71
+ totalTokens: data.promptTokens + data.completionTokens,
72
+ costUSD: data.costUSD,
73
+ durationMs: data.durationMs,
74
+ requestedAt: data.requestedAt,
75
+ prompt: data.prompt,
76
+ inputChars: data.inputChars,
77
+ imageSize: data.imageSize,
78
+ success: data.success,
79
+ errorMessage: data.errorMessage,
80
+ });
81
+ }
82
+ catch (_a) {
83
+ // nunca deixa o emit de evento quebrar a requisição principal
84
+ }
85
+ });
86
+ }
87
+ chat(params) {
88
+ return __awaiter(this, void 0, void 0, function* () {
89
+ var _a, _b, _c, _d, _e, _f, _g;
90
+ const { provider, model, messages, maxTokens, temperature, responseFormat, openaiClient, cerebrasClient } = params;
91
+ const client = provider === 'cerebras' ? (cerebrasClient !== null && cerebrasClient !== void 0 ? cerebrasClient : openaiClient) : openaiClient;
92
+ const requestedAt = new Date().toISOString();
93
+ const t0 = Date.now();
94
+ const promptSummary = messages.map(m => { var _a; return `[${m.role}] ${String((_a = m.content) !== null && _a !== void 0 ? _a : '').slice(0, 200)}`; }).join(' | ').slice(0, 500);
95
+ try {
96
+ const response = yield client.chat.completions.create(Object.assign(Object.assign(Object.assign({ model,
97
+ messages }, (maxTokens !== undefined && { max_tokens: maxTokens })), (temperature !== undefined && { temperature })), (responseFormat !== undefined && { response_format: responseFormat })));
98
+ const promptTokens = (_b = (_a = response.usage) === null || _a === void 0 ? void 0 : _a.prompt_tokens) !== null && _b !== void 0 ? _b : 0;
99
+ const completionTokens = (_d = (_c = response.usage) === null || _c === void 0 ? void 0 : _c.completion_tokens) !== null && _d !== void 0 ? _d : 0;
100
+ const costUSD = calcCost(model, promptTokens, completionTokens);
101
+ const durationMs = Date.now() - t0;
102
+ yield this.emit({ provider, type: 'chat', model, promptTokens, completionTokens, costUSD, durationMs, requestedAt, prompt: promptSummary, success: true });
103
+ return {
104
+ content: (_g = (_f = (_e = response.choices[0]) === null || _e === void 0 ? void 0 : _e.message) === null || _f === void 0 ? void 0 : _f.content) !== null && _g !== void 0 ? _g : '',
105
+ promptTokens,
106
+ completionTokens,
107
+ totalTokens: promptTokens + completionTokens,
108
+ };
109
+ }
110
+ catch (err) {
111
+ const errorMessage = err instanceof Error ? err.message : String(err);
112
+ yield this.emit({ provider, type: 'chat', model, promptTokens: 0, completionTokens: 0, costUSD: 0, durationMs: Date.now() - t0, requestedAt, prompt: promptSummary, success: false, errorMessage });
113
+ throw err;
114
+ }
115
+ });
116
+ }
117
+ tts(params) {
118
+ return __awaiter(this, void 0, void 0, function* () {
119
+ var _a, _b;
120
+ const { model, voice, input, speed, instructions, responseFormat, openaiClient } = params;
121
+ const requestedAt = new Date().toISOString();
122
+ const t0 = Date.now();
123
+ const costUSD = (input.length / 1000) * ((_b = (_a = COST_PER_1K[model]) === null || _a === void 0 ? void 0 : _a.input) !== null && _b !== void 0 ? _b : 0.030);
124
+ try {
125
+ const response = yield openaiClient.audio.speech.create(Object.assign(Object.assign({ model, voice: voice, input, response_format: (responseFormat !== null && responseFormat !== void 0 ? responseFormat : 'mp3') }, (speed !== undefined && { speed })), (instructions !== undefined && { instructions })));
126
+ const buffer = Buffer.from(yield response.arrayBuffer());
127
+ const durationMs = Date.now() - t0;
128
+ yield this.emit({ provider: 'openai', type: 'tts', model, promptTokens: 0, completionTokens: 0, costUSD, durationMs, requestedAt, prompt: input.slice(0, 200), inputChars: input.length, success: true });
129
+ return buffer;
130
+ }
131
+ catch (err) {
132
+ const errorMessage = err instanceof Error ? err.message : String(err);
133
+ yield this.emit({ provider: 'openai', type: 'tts', model, promptTokens: 0, completionTokens: 0, costUSD: 0, durationMs: Date.now() - t0, requestedAt, prompt: input.slice(0, 200), inputChars: input.length, success: false, errorMessage });
134
+ throw err;
135
+ }
136
+ });
137
+ }
138
+ transcribe(params) {
139
+ return __awaiter(this, void 0, void 0, function* () {
140
+ const { audioBuffer, mimeType, openaiClient } = params;
141
+ const requestedAt = new Date().toISOString();
142
+ const t0 = Date.now();
143
+ const estimatedMinutes = audioBuffer.length / 40000 / 60;
144
+ const costUSD = estimatedMinutes * 0.006;
145
+ try {
146
+ const file = yield (0, openai_1.toFile)(audioBuffer, 'audio.m4a', { type: mimeType !== null && mimeType !== void 0 ? mimeType : 'audio/m4a' });
147
+ const response = yield openaiClient.audio.transcriptions.create({ model: 'whisper-1', file });
148
+ const durationMs = Date.now() - t0;
149
+ yield this.emit({ provider: 'openai', type: 'transcribe', model: 'whisper-1', promptTokens: 0, completionTokens: 0, costUSD, durationMs, requestedAt, prompt: `audio ${audioBuffer.length}B`, success: true });
150
+ return response.text;
151
+ }
152
+ catch (err) {
153
+ const errorMessage = err instanceof Error ? err.message : String(err);
154
+ yield this.emit({ provider: 'openai', type: 'transcribe', model: 'whisper-1', promptTokens: 0, completionTokens: 0, costUSD: 0, durationMs: Date.now() - t0, requestedAt, prompt: `audio ${audioBuffer.length}B`, success: false, errorMessage });
155
+ throw err;
156
+ }
157
+ });
158
+ }
159
+ generateImage(params) {
160
+ return __awaiter(this, void 0, void 0, function* () {
161
+ var _a, _b;
162
+ const { model, prompt, size, quality, n, openaiClient } = params;
163
+ const requestedAt = new Date().toISOString();
164
+ const t0 = Date.now();
165
+ const costUSD = (_b = (_a = COST_PER_1K[model]) === null || _a === void 0 ? void 0 : _a.input) !== null && _b !== void 0 ? _b : 0.040;
166
+ try {
167
+ const response = yield openaiClient.images.generate(Object.assign(Object.assign({ model,
168
+ prompt, size: size }, (quality !== undefined && { quality: quality })), { n: n !== null && n !== void 0 ? n : 1, response_format: 'b64_json' }));
169
+ const b64 = response.data[0].b64_json;
170
+ const durationMs = Date.now() - t0;
171
+ yield this.emit({ provider: 'openai', type: 'image-generate', model, promptTokens: 0, completionTokens: 0, costUSD, durationMs, requestedAt, prompt: prompt.slice(0, 300), imageSize: size, success: true });
172
+ return b64;
173
+ }
174
+ catch (err) {
175
+ const errorMessage = err instanceof Error ? err.message : String(err);
176
+ yield this.emit({ provider: 'openai', type: 'image-generate', model, promptTokens: 0, completionTokens: 0, costUSD: 0, durationMs: Date.now() - t0, requestedAt, prompt: prompt.slice(0, 300), imageSize: size, success: false, errorMessage });
177
+ throw err;
178
+ }
179
+ });
180
+ }
181
+ editImage(params) {
182
+ return __awaiter(this, void 0, void 0, function* () {
183
+ var _a, _b;
184
+ const { model, imageBuffer, prompt, size, n, openaiClient } = params;
185
+ const requestedAt = new Date().toISOString();
186
+ const t0 = Date.now();
187
+ const costUSD = (_b = (_a = COST_PER_1K[model]) === null || _a === void 0 ? void 0 : _a.input) !== null && _b !== void 0 ? _b : 0.020;
188
+ try {
189
+ if (!globalThis.File) {
190
+ const { File: NodeFile } = yield Promise.resolve().then(() => __importStar(require('node:buffer')));
191
+ globalThis.File = NodeFile;
192
+ }
193
+ const imageFile = yield (0, openai_1.toFile)(imageBuffer, 'image.png', { type: 'image/png' });
194
+ const response = yield openaiClient.images.edit({
195
+ model,
196
+ image: imageFile,
197
+ prompt,
198
+ size: size,
199
+ n: n !== null && n !== void 0 ? n : 1,
200
+ response_format: 'b64_json',
201
+ });
202
+ const b64 = response.data[0].b64_json;
203
+ const durationMs = Date.now() - t0;
204
+ yield this.emit({ provider: 'openai', type: 'image-edit', model, promptTokens: 0, completionTokens: 0, costUSD, durationMs, requestedAt, prompt: prompt.slice(0, 300), imageSize: size, success: true });
205
+ return b64;
206
+ }
207
+ catch (err) {
208
+ const errorMessage = err instanceof Error ? err.message : String(err);
209
+ yield this.emit({ provider: 'openai', type: 'image-edit', model, promptTokens: 0, completionTokens: 0, costUSD: 0, durationMs: Date.now() - t0, requestedAt, prompt: prompt.slice(0, 300), imageSize: size, success: false, errorMessage });
210
+ throw err;
211
+ }
212
+ });
213
+ }
214
+ }
215
+ exports.AIClient = AIClient;
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@wrcb/cb-common",
3
- "version": "1.0.837",
3
+ "version": "1.0.838",
4
4
  "description": "Common resources between services",
5
5
  "main": "./build/index.js",
6
6
  "types": "./build/index.d.ts",
@@ -27,16 +27,16 @@
27
27
  "author": "Willian Rios",
28
28
  "license": "ISC",
29
29
  "devDependencies": {
30
+ "@types/cookie-session": "^2.0.49",
31
+ "@types/express": "^4.17.21",
30
32
  "@types/express-rate-limit": "^5.1.3",
31
33
  "@types/fluent-ffmpeg": "^2.1.27",
34
+ "@types/jsonwebtoken": "^9.0.6",
32
35
  "@types/mime-types": "^3.0.1",
33
36
  "@types/multer": "^2.0.0",
37
+ "@types/supertest": "^6.0.2",
34
38
  "@typescript-eslint/eslint-plugin": "^8.29.1",
35
39
  "@typescript-eslint/parser": "^8.29.1",
36
- "@types/cookie-session": "^2.0.49",
37
- "@types/express": "^4.17.21",
38
- "@types/jsonwebtoken": "^9.0.6",
39
- "@types/supertest": "^6.0.2",
40
40
  "eslint": "^9.24.0",
41
41
  "eslint-config-prettier": "^10.1.2",
42
42
  "eslint-plugin-prettier": "^5.2.6",
@@ -59,6 +59,7 @@
59
59
  "mongoose": "^8.6.3",
60
60
  "multer": "^2.0.2",
61
61
  "node-nats-streaming": "^0.3.2",
62
+ "openai": "^6.42.0",
62
63
  "rate-limit-redis": "^4.2.0",
63
64
  "sharp": "^0.34.3",
64
65
  "uuid": "^11.1.0"