tychat-contracts 1.3.2 → 1.3.4

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.
Files changed (44) hide show
  1. package/dist/ai/ai-integration-response.dto.d.ts +3 -0
  2. package/dist/ai/ai-integration-response.dto.d.ts.map +1 -1
  3. package/dist/ai/ai-integration-response.dto.js +27 -0
  4. package/dist/analytics/analytics-emitter.helper.d.ts +1 -1
  5. package/dist/analytics/analytics-emitter.helper.js +1 -1
  6. package/dist/configurations/ai-tool-policy.dto.d.ts +26 -0
  7. package/dist/configurations/ai-tool-policy.dto.d.ts.map +1 -0
  8. package/dist/configurations/ai-tool-policy.dto.js +117 -0
  9. package/dist/configurations/ai-tool-policy.enums.d.ts +7 -0
  10. package/dist/configurations/ai-tool-policy.enums.d.ts.map +1 -0
  11. package/dist/configurations/ai-tool-policy.enums.js +22 -0
  12. package/dist/configurations/index.d.ts +2 -0
  13. package/dist/configurations/index.d.ts.map +1 -1
  14. package/dist/configurations/index.js +2 -0
  15. package/dist/configurations/plan-summary.dto.d.ts +1 -0
  16. package/dist/configurations/plan-summary.dto.d.ts.map +1 -1
  17. package/dist/configurations/plan-summary.dto.js +5 -0
  18. package/dist/tenants/consume-tenant-ai-tokens.dto.d.ts +14 -0
  19. package/dist/tenants/consume-tenant-ai-tokens.dto.d.ts.map +1 -0
  20. package/dist/tenants/consume-tenant-ai-tokens.dto.js +64 -0
  21. package/dist/tenants/index.d.ts +4 -0
  22. package/dist/tenants/index.d.ts.map +1 -1
  23. package/dist/tenants/index.js +4 -0
  24. package/dist/tenants/recharge-tenant-ai-tokens.dto.d.ts +10 -0
  25. package/dist/tenants/recharge-tenant-ai-tokens.dto.d.ts.map +1 -0
  26. package/dist/tenants/recharge-tenant-ai-tokens.dto.js +49 -0
  27. package/dist/tenants/renew-tenant-ai-tokens.dto.d.ts +7 -0
  28. package/dist/tenants/renew-tenant-ai-tokens.dto.d.ts.map +1 -0
  29. package/dist/tenants/renew-tenant-ai-tokens.dto.js +34 -0
  30. package/dist/tenants/tenant-ai-token-state.dto.d.ts +13 -0
  31. package/dist/tenants/tenant-ai-token-state.dto.d.ts.map +1 -0
  32. package/dist/tenants/tenant-ai-token-state.dto.js +83 -0
  33. package/package.json +1 -1
  34. package/src/ai/ai-integration-response.dto.ts +21 -0
  35. package/src/analytics/analytics-emitter.helper.ts +1 -1
  36. package/src/configurations/ai-tool-policy.dto.ts +92 -0
  37. package/src/configurations/ai-tool-policy.enums.ts +25 -0
  38. package/src/configurations/index.ts +2 -0
  39. package/src/configurations/plan-summary.dto.ts +3 -0
  40. package/src/tenants/consume-tenant-ai-tokens.dto.ts +41 -0
  41. package/src/tenants/index.ts +4 -0
  42. package/src/tenants/recharge-tenant-ai-tokens.dto.ts +27 -0
  43. package/src/tenants/renew-tenant-ai-tokens.dto.ts +16 -0
  44. package/src/tenants/tenant-ai-token-state.dto.ts +50 -0
@@ -4,5 +4,8 @@ export declare class AiIntegrationResponseDto {
4
4
  response: AiStructuredResponseDto;
5
5
  output?: string;
6
6
  aiUsageId?: string;
7
+ promptTokens?: number;
8
+ completionTokens?: number;
9
+ totalTokens?: number;
7
10
  }
8
11
  //# sourceMappingURL=ai-integration-response.dto.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"ai-integration-response.dto.d.ts","sourceRoot":"","sources":["../../src/ai/ai-integration-response.dto.ts"],"names":[],"mappings":"AAGA,OAAO,EAAE,uBAAuB,EAAE,MAAM,iBAAiB,CAAC;AAE1D,qBAAa,wBAAwB;IAOnC,MAAM,EAAE,MAAM,CAAC;IAQf,QAAQ,EAAE,uBAAuB,CAAC;IAUlC,MAAM,CAAC,EAAE,MAAM,CAAC;IAQhB,SAAS,CAAC,EAAE,MAAM,CAAC;CACpB"}
1
+ {"version":3,"file":"ai-integration-response.dto.d.ts","sourceRoot":"","sources":["../../src/ai/ai-integration-response.dto.ts"],"names":[],"mappings":"AAGA,OAAO,EAAE,uBAAuB,EAAE,MAAM,iBAAiB,CAAC;AAE1D,qBAAa,wBAAwB;IAOnC,MAAM,EAAE,MAAM,CAAC;IAQf,QAAQ,EAAE,uBAAuB,CAAC;IAUlC,MAAM,CAAC,EAAE,MAAM,CAAC;IAQhB,SAAS,CAAC,EAAE,MAAM,CAAC;IAOnB,YAAY,CAAC,EAAE,MAAM,CAAC;IAOtB,gBAAgB,CAAC,EAAE,MAAM,CAAC;IAO1B,WAAW,CAAC,EAAE,MAAM,CAAC;CACtB"}
@@ -19,6 +19,9 @@ class AiIntegrationResponseDto {
19
19
  response;
20
20
  output;
21
21
  aiUsageId;
22
+ promptTokens;
23
+ completionTokens;
24
+ totalTokens;
22
25
  }
23
26
  exports.AiIntegrationResponseDto = AiIntegrationResponseDto;
24
27
  __decorate([
@@ -59,3 +62,27 @@ __decorate([
59
62
  (0, class_validator_1.IsString)(),
60
63
  __metadata("design:type", String)
61
64
  ], AiIntegrationResponseDto.prototype, "aiUsageId", void 0);
65
+ __decorate([
66
+ (0, swagger_1.ApiPropertyOptional)({
67
+ description: 'Prompt tokens consumed in this AI request',
68
+ example: 120,
69
+ }),
70
+ (0, class_validator_1.IsOptional)(),
71
+ __metadata("design:type", Number)
72
+ ], AiIntegrationResponseDto.prototype, "promptTokens", void 0);
73
+ __decorate([
74
+ (0, swagger_1.ApiPropertyOptional)({
75
+ description: 'Completion tokens generated in this AI request',
76
+ example: 90,
77
+ }),
78
+ (0, class_validator_1.IsOptional)(),
79
+ __metadata("design:type", Number)
80
+ ], AiIntegrationResponseDto.prototype, "completionTokens", void 0);
81
+ __decorate([
82
+ (0, swagger_1.ApiPropertyOptional)({
83
+ description: 'Total tokens consumed in this AI request',
84
+ example: 210,
85
+ }),
86
+ (0, class_validator_1.IsOptional)(),
87
+ __metadata("design:type", Number)
88
+ ], AiIntegrationResponseDto.prototype, "totalTokens", void 0);
@@ -14,7 +14,7 @@ export interface EmitAnalyticEventOptions {
14
14
  /**
15
15
  * Builds the Kafka payload to be sent to the analytics service.
16
16
  *
17
- * Usage (fire-and-forget via ClientKafka.emit):
17
+ * Usage (fire-and-forget via ClientProxy.emit):
18
18
  * ```ts
19
19
  * this.analyticsClient.emit(
20
20
  * TOPIC_ANALYTICS_EVENT,
@@ -8,7 +8,7 @@ Object.defineProperty(exports, "TOPIC_ANALYTICS_EVENT", { enumerable: true, get:
8
8
  /**
9
9
  * Builds the Kafka payload to be sent to the analytics service.
10
10
  *
11
- * Usage (fire-and-forget via ClientKafka.emit):
11
+ * Usage (fire-and-forget via ClientProxy.emit):
12
12
  * ```ts
13
13
  * this.analyticsClient.emit(
14
14
  * TOPIC_ANALYTICS_EVENT,
@@ -0,0 +1,26 @@
1
+ import { type AiToolFallbackMode, type AiToolKey, type AiToolPolicyLevel } from './ai-tool-policy.enums';
2
+ export declare class AiToolPolicyDto {
3
+ toolKey: AiToolKey;
4
+ enabled: boolean;
5
+ fallbackMode?: AiToolFallbackMode;
6
+ level?: AiToolPolicyLevel;
7
+ }
8
+ export declare class UpdateAiToolPolicyDto {
9
+ toolKey: AiToolKey;
10
+ enabled: boolean;
11
+ fallbackMode?: AiToolFallbackMode;
12
+ }
13
+ export declare class UpdateAiToolPoliciesDto {
14
+ policies: UpdateAiToolPolicyDto[];
15
+ }
16
+ export declare class ResolveAiToolPoliciesQueryDto {
17
+ tenantId: string;
18
+ unitId?: number;
19
+ }
20
+ export declare class EffectiveAiToolPolicyDto extends AiToolPolicyDto {
21
+ reasonCode?: string;
22
+ }
23
+ export declare class EffectiveAiToolPoliciesDto {
24
+ policies: EffectiveAiToolPolicyDto[];
25
+ }
26
+ //# sourceMappingURL=ai-tool-policy.dto.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ai-tool-policy.dto.d.ts","sourceRoot":"","sources":["../../src/configurations/ai-tool-policy.dto.ts"],"names":[],"mappings":"AAaA,OAAO,EAIL,KAAK,kBAAkB,EACvB,KAAK,SAAS,EACd,KAAK,iBAAiB,EACvB,MAAM,wBAAwB,CAAC;AAEhC,qBAAa,eAAe;IAG1B,OAAO,EAAE,SAAS,CAAC;IAInB,OAAO,EAAE,OAAO,CAAC;IAKjB,YAAY,CAAC,EAAE,kBAAkB,CAAC;IAKlC,KAAK,CAAC,EAAE,iBAAiB,CAAC;CAC3B;AAED,qBAAa,qBAAqB;IAGhC,OAAO,EAAE,SAAS,CAAC;IAInB,OAAO,EAAE,OAAO,CAAC;IAKjB,YAAY,CAAC,EAAE,kBAAkB,CAAC;CACnC;AAED,qBAAa,uBAAuB;IAKlC,QAAQ,EAAE,qBAAqB,EAAE,CAAC;CACnC;AAED,qBAAa,6BAA6B;IAIxC,QAAQ,EAAE,MAAM,CAAC;IAMjB,MAAM,CAAC,EAAE,MAAM,CAAC;CACjB;AAED,qBAAa,wBAAyB,SAAQ,eAAe;IAI3D,UAAU,CAAC,EAAE,MAAM,CAAC;CACrB;AAED,qBAAa,0BAA0B;IAKrC,QAAQ,EAAE,wBAAwB,EAAE,CAAC;CACtC"}
@@ -0,0 +1,117 @@
1
+ "use strict";
2
+ var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {
3
+ var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
4
+ if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
5
+ else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
6
+ return c > 3 && r && Object.defineProperty(target, key, r), r;
7
+ };
8
+ var __metadata = (this && this.__metadata) || function (k, v) {
9
+ if (typeof Reflect === "object" && typeof Reflect.metadata === "function") return Reflect.metadata(k, v);
10
+ };
11
+ Object.defineProperty(exports, "__esModule", { value: true });
12
+ exports.EffectiveAiToolPoliciesDto = exports.EffectiveAiToolPolicyDto = exports.ResolveAiToolPoliciesQueryDto = exports.UpdateAiToolPoliciesDto = exports.UpdateAiToolPolicyDto = exports.AiToolPolicyDto = void 0;
13
+ const swagger_1 = require("@nestjs/swagger");
14
+ const class_transformer_1 = require("class-transformer");
15
+ const class_validator_1 = require("class-validator");
16
+ const ai_tool_policy_enums_1 = require("./ai-tool-policy.enums");
17
+ class AiToolPolicyDto {
18
+ toolKey;
19
+ enabled;
20
+ fallbackMode;
21
+ level;
22
+ }
23
+ exports.AiToolPolicyDto = AiToolPolicyDto;
24
+ __decorate([
25
+ (0, swagger_1.ApiProperty)({ enum: ai_tool_policy_enums_1.AI_TOOL_KEYS }),
26
+ (0, class_validator_1.IsEnum)(ai_tool_policy_enums_1.AI_TOOL_KEYS),
27
+ __metadata("design:type", String)
28
+ ], AiToolPolicyDto.prototype, "toolKey", void 0);
29
+ __decorate([
30
+ (0, swagger_1.ApiProperty)(),
31
+ (0, class_validator_1.IsBoolean)(),
32
+ __metadata("design:type", Boolean)
33
+ ], AiToolPolicyDto.prototype, "enabled", void 0);
34
+ __decorate([
35
+ (0, swagger_1.ApiPropertyOptional)({ enum: ai_tool_policy_enums_1.AI_TOOL_FALLBACK_MODES }),
36
+ (0, class_validator_1.IsOptional)(),
37
+ (0, class_validator_1.IsEnum)(ai_tool_policy_enums_1.AI_TOOL_FALLBACK_MODES),
38
+ __metadata("design:type", String)
39
+ ], AiToolPolicyDto.prototype, "fallbackMode", void 0);
40
+ __decorate([
41
+ (0, swagger_1.ApiPropertyOptional)({ enum: ai_tool_policy_enums_1.AI_TOOL_POLICY_LEVELS }),
42
+ (0, class_validator_1.IsOptional)(),
43
+ (0, class_validator_1.IsEnum)(ai_tool_policy_enums_1.AI_TOOL_POLICY_LEVELS),
44
+ __metadata("design:type", String)
45
+ ], AiToolPolicyDto.prototype, "level", void 0);
46
+ class UpdateAiToolPolicyDto {
47
+ toolKey;
48
+ enabled;
49
+ fallbackMode;
50
+ }
51
+ exports.UpdateAiToolPolicyDto = UpdateAiToolPolicyDto;
52
+ __decorate([
53
+ (0, swagger_1.ApiProperty)({ enum: ai_tool_policy_enums_1.AI_TOOL_KEYS }),
54
+ (0, class_validator_1.IsEnum)(ai_tool_policy_enums_1.AI_TOOL_KEYS),
55
+ __metadata("design:type", String)
56
+ ], UpdateAiToolPolicyDto.prototype, "toolKey", void 0);
57
+ __decorate([
58
+ (0, swagger_1.ApiProperty)(),
59
+ (0, class_validator_1.IsBoolean)(),
60
+ __metadata("design:type", Boolean)
61
+ ], UpdateAiToolPolicyDto.prototype, "enabled", void 0);
62
+ __decorate([
63
+ (0, swagger_1.ApiPropertyOptional)({ enum: ai_tool_policy_enums_1.AI_TOOL_FALLBACK_MODES }),
64
+ (0, class_validator_1.IsOptional)(),
65
+ (0, class_validator_1.IsEnum)(ai_tool_policy_enums_1.AI_TOOL_FALLBACK_MODES),
66
+ __metadata("design:type", String)
67
+ ], UpdateAiToolPolicyDto.prototype, "fallbackMode", void 0);
68
+ class UpdateAiToolPoliciesDto {
69
+ policies;
70
+ }
71
+ exports.UpdateAiToolPoliciesDto = UpdateAiToolPoliciesDto;
72
+ __decorate([
73
+ (0, swagger_1.ApiProperty)({ type: () => [UpdateAiToolPolicyDto] }),
74
+ (0, class_validator_1.IsArray)(),
75
+ (0, class_validator_1.ValidateNested)({ each: true }),
76
+ (0, class_transformer_1.Type)(() => UpdateAiToolPolicyDto),
77
+ __metadata("design:type", Array)
78
+ ], UpdateAiToolPoliciesDto.prototype, "policies", void 0);
79
+ class ResolveAiToolPoliciesQueryDto {
80
+ tenantId;
81
+ unitId;
82
+ }
83
+ exports.ResolveAiToolPoliciesQueryDto = ResolveAiToolPoliciesQueryDto;
84
+ __decorate([
85
+ (0, swagger_1.ApiProperty)(),
86
+ (0, class_validator_1.IsString)(),
87
+ (0, class_validator_1.IsNotEmpty)(),
88
+ __metadata("design:type", String)
89
+ ], ResolveAiToolPoliciesQueryDto.prototype, "tenantId", void 0);
90
+ __decorate([
91
+ (0, swagger_1.ApiPropertyOptional)({ minimum: 1 }),
92
+ (0, class_validator_1.IsOptional)(),
93
+ (0, class_validator_1.IsInt)(),
94
+ (0, class_validator_1.Min)(1),
95
+ __metadata("design:type", Number)
96
+ ], ResolveAiToolPoliciesQueryDto.prototype, "unitId", void 0);
97
+ class EffectiveAiToolPolicyDto extends AiToolPolicyDto {
98
+ reasonCode;
99
+ }
100
+ exports.EffectiveAiToolPolicyDto = EffectiveAiToolPolicyDto;
101
+ __decorate([
102
+ (0, swagger_1.ApiPropertyOptional)(),
103
+ (0, class_validator_1.IsOptional)(),
104
+ (0, class_validator_1.IsString)(),
105
+ __metadata("design:type", String)
106
+ ], EffectiveAiToolPolicyDto.prototype, "reasonCode", void 0);
107
+ class EffectiveAiToolPoliciesDto {
108
+ policies;
109
+ }
110
+ exports.EffectiveAiToolPoliciesDto = EffectiveAiToolPoliciesDto;
111
+ __decorate([
112
+ (0, swagger_1.ApiProperty)({ type: () => [EffectiveAiToolPolicyDto] }),
113
+ (0, class_validator_1.IsArray)(),
114
+ (0, class_validator_1.ValidateNested)({ each: true }),
115
+ (0, class_transformer_1.Type)(() => EffectiveAiToolPolicyDto),
116
+ __metadata("design:type", Array)
117
+ ], EffectiveAiToolPoliciesDto.prototype, "policies", void 0);
@@ -0,0 +1,7 @@
1
+ export declare const AI_TOOL_KEYS: readonly ["get_procedures_clinic", "get_clinic_opening_hours", "get_available_slots", "search_health_professionals", "update_patient", "search_patients_by_phone", "create_patient", "create_appointment", "add_appointment_procedures", "list_patient_appointments", "cancel_appointment", "submit_satisfaction"];
2
+ export type AiToolKey = (typeof AI_TOOL_KEYS)[number];
3
+ export declare const AI_TOOL_POLICY_LEVELS: readonly ["global", "plan", "tenant", "unit"];
4
+ export type AiToolPolicyLevel = (typeof AI_TOOL_POLICY_LEVELS)[number];
5
+ export declare const AI_TOOL_FALLBACK_MODES: readonly ["transfer_to_human", "unavailable_message"];
6
+ export type AiToolFallbackMode = (typeof AI_TOOL_FALLBACK_MODES)[number];
7
+ //# sourceMappingURL=ai-tool-policy.enums.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ai-tool-policy.enums.d.ts","sourceRoot":"","sources":["../../src/configurations/ai-tool-policy.enums.ts"],"names":[],"mappings":"AAAA,eAAO,MAAM,YAAY,oTAaf,CAAC;AAEX,MAAM,MAAM,SAAS,GAAG,CAAC,OAAO,YAAY,CAAC,CAAC,MAAM,CAAC,CAAC;AAEtD,eAAO,MAAM,qBAAqB,+CAAgD,CAAC;AACnF,MAAM,MAAM,iBAAiB,GAAG,CAAC,OAAO,qBAAqB,CAAC,CAAC,MAAM,CAAC,CAAC;AAEvE,eAAO,MAAM,sBAAsB,uDAGzB,CAAC;AACX,MAAM,MAAM,kBAAkB,GAAG,CAAC,OAAO,sBAAsB,CAAC,CAAC,MAAM,CAAC,CAAC"}
@@ -0,0 +1,22 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.AI_TOOL_FALLBACK_MODES = exports.AI_TOOL_POLICY_LEVELS = exports.AI_TOOL_KEYS = void 0;
4
+ exports.AI_TOOL_KEYS = [
5
+ 'get_procedures_clinic',
6
+ 'get_clinic_opening_hours',
7
+ 'get_available_slots',
8
+ 'search_health_professionals',
9
+ 'update_patient',
10
+ 'search_patients_by_phone',
11
+ 'create_patient',
12
+ 'create_appointment',
13
+ 'add_appointment_procedures',
14
+ 'list_patient_appointments',
15
+ 'cancel_appointment',
16
+ 'submit_satisfaction',
17
+ ];
18
+ exports.AI_TOOL_POLICY_LEVELS = ['global', 'plan', 'tenant', 'unit'];
19
+ exports.AI_TOOL_FALLBACK_MODES = [
20
+ 'transfer_to_human',
21
+ 'unavailable_message',
22
+ ];
@@ -1,6 +1,8 @@
1
1
  export * from './opening-hours-slot.dto';
2
2
  export * from './update-clinic-configuration.dto';
3
3
  export * from './clinic-configuration.dto';
4
+ export * from './ai-tool-policy.enums';
5
+ export * from './ai-tool-policy.dto';
4
6
  export * from './plan-summary.dto';
5
7
  export * from './tenant-limits-summary.dto';
6
8
  export * from './configuration-tenant-summary.dto';
@@ -1 +1 @@
1
- {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/configurations/index.ts"],"names":[],"mappings":"AAAA,cAAc,0BAA0B,CAAC;AACzC,cAAc,mCAAmC,CAAC;AAClD,cAAc,4BAA4B,CAAC;AAC3C,cAAc,oBAAoB,CAAC;AACnC,cAAc,6BAA6B,CAAC;AAC5C,cAAc,oCAAoC,CAAC;AACnD,cAAc,wCAAwC,CAAC"}
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/configurations/index.ts"],"names":[],"mappings":"AAAA,cAAc,0BAA0B,CAAC;AACzC,cAAc,mCAAmC,CAAC;AAClD,cAAc,4BAA4B,CAAC;AAC3C,cAAc,wBAAwB,CAAC;AACvC,cAAc,sBAAsB,CAAC;AACrC,cAAc,oBAAoB,CAAC;AACnC,cAAc,6BAA6B,CAAC;AAC5C,cAAc,oCAAoC,CAAC;AACnD,cAAc,wCAAwC,CAAC"}
@@ -17,6 +17,8 @@ Object.defineProperty(exports, "__esModule", { value: true });
17
17
  __exportStar(require("./opening-hours-slot.dto"), exports);
18
18
  __exportStar(require("./update-clinic-configuration.dto"), exports);
19
19
  __exportStar(require("./clinic-configuration.dto"), exports);
20
+ __exportStar(require("./ai-tool-policy.enums"), exports);
21
+ __exportStar(require("./ai-tool-policy.dto"), exports);
20
22
  __exportStar(require("./plan-summary.dto"), exports);
21
23
  __exportStar(require("./tenant-limits-summary.dto"), exports);
22
24
  __exportStar(require("./configuration-tenant-summary.dto"), exports);
@@ -14,6 +14,7 @@ export declare class PlanSummaryDto {
14
14
  maxAttendants: number;
15
15
  maxAdministrators: number;
16
16
  maxProcedures: number;
17
+ monthlyAiTokens: number;
17
18
  createdAt?: Date;
18
19
  updatedAt?: Date;
19
20
  }
@@ -1 +1 @@
1
- {"version":3,"file":"plan-summary.dto.d.ts","sourceRoot":"","sources":["../../src/configurations/plan-summary.dto.ts"],"names":[],"mappings":"AAEA;;GAEG;AACH,qBAAa,cAAc;IAEzB,EAAE,EAAE,MAAM,CAAC;IAGX,IAAI,EAAE,MAAM,CAAC;IAGb,KAAK,EAAE,MAAM,CAAC;IAGd,cAAc,EAAE,MAAM,CAAC;IAGvB,QAAQ,EAAE,MAAM,CAAC;IAGjB,MAAM,EAAE,QAAQ,GAAG,UAAU,CAAC;IAG9B,oBAAoB,EAAE,MAAM,CAAC;IAG7B,WAAW,EAAE,MAAM,CAAC;IAGpB,UAAU,EAAE,MAAM,CAAC;IAGnB,aAAa,EAAE,MAAM,CAAC;IAGtB,iBAAiB,EAAE,MAAM,CAAC;IAG1B,aAAa,EAAE,MAAM,CAAC;IAGtB,SAAS,CAAC,EAAE,IAAI,CAAC;IAGjB,SAAS,CAAC,EAAE,IAAI,CAAC;CAClB"}
1
+ {"version":3,"file":"plan-summary.dto.d.ts","sourceRoot":"","sources":["../../src/configurations/plan-summary.dto.ts"],"names":[],"mappings":"AAEA;;GAEG;AACH,qBAAa,cAAc;IAEzB,EAAE,EAAE,MAAM,CAAC;IAGX,IAAI,EAAE,MAAM,CAAC;IAGb,KAAK,EAAE,MAAM,CAAC;IAGd,cAAc,EAAE,MAAM,CAAC;IAGvB,QAAQ,EAAE,MAAM,CAAC;IAGjB,MAAM,EAAE,QAAQ,GAAG,UAAU,CAAC;IAG9B,oBAAoB,EAAE,MAAM,CAAC;IAG7B,WAAW,EAAE,MAAM,CAAC;IAGpB,UAAU,EAAE,MAAM,CAAC;IAGnB,aAAa,EAAE,MAAM,CAAC;IAGtB,iBAAiB,EAAE,MAAM,CAAC;IAG1B,aAAa,EAAE,MAAM,CAAC;IAGtB,eAAe,EAAE,MAAM,CAAC;IAGxB,SAAS,CAAC,EAAE,IAAI,CAAC;IAGjB,SAAS,CAAC,EAAE,IAAI,CAAC;CAClB"}
@@ -27,6 +27,7 @@ class PlanSummaryDto {
27
27
  maxAttendants;
28
28
  maxAdministrators;
29
29
  maxProcedures;
30
+ monthlyAiTokens;
30
31
  createdAt;
31
32
  updatedAt;
32
33
  }
@@ -79,6 +80,10 @@ __decorate([
79
80
  (0, swagger_1.ApiProperty)({ description: 'Max procedures', example: 200 }),
80
81
  __metadata("design:type", Number)
81
82
  ], PlanSummaryDto.prototype, "maxProcedures", void 0);
83
+ __decorate([
84
+ (0, swagger_1.ApiProperty)({ description: 'Monthly AI token allowance', example: 5000 }),
85
+ __metadata("design:type", Number)
86
+ ], PlanSummaryDto.prototype, "monthlyAiTokens", void 0);
82
87
  __decorate([
83
88
  (0, swagger_1.ApiPropertyOptional)(),
84
89
  __metadata("design:type", Date)
@@ -0,0 +1,14 @@
1
+ import { TenantAiTokenStateDto } from './tenant-ai-token-state.dto';
2
+ export declare const TENANT_AI_TOKEN_CONSUMPTION_SOURCES: readonly ["ai.request", "ai.followup.generate"];
3
+ export type TenantAiTokenConsumptionSourceDto = (typeof TENANT_AI_TOKEN_CONSUMPTION_SOURCES)[number];
4
+ export declare class ConsumeTenantAiTokensDto {
5
+ tenant: string;
6
+ tokens: number;
7
+ source: TenantAiTokenConsumptionSourceDto;
8
+ correlationId?: string;
9
+ }
10
+ export declare class ConsumeTenantAiTokensResponseDto extends TenantAiTokenStateDto {
11
+ consumed: number;
12
+ becameZero: boolean;
13
+ }
14
+ //# sourceMappingURL=consume-tenant-ai-tokens.dto.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"consume-tenant-ai-tokens.dto.d.ts","sourceRoot":"","sources":["../../src/tenants/consume-tenant-ai-tokens.dto.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,qBAAqB,EAAE,MAAM,6BAA6B,CAAC;AAEpE,eAAO,MAAM,mCAAmC,iDAAkD,CAAC;AACnG,MAAM,MAAM,iCAAiC,GAC3C,CAAC,OAAO,mCAAmC,CAAC,CAAC,MAAM,CAAC,CAAC;AAEvD,qBAAa,wBAAwB;IAInC,MAAM,EAAE,MAAM,CAAC;IAKf,MAAM,EAAE,MAAM,CAAC;IAOf,MAAM,EAAE,iCAAiC,CAAC;IAK1C,aAAa,CAAC,EAAE,MAAM,CAAC;CACxB;AAED,qBAAa,gCAAiC,SAAQ,qBAAqB;IAIzE,QAAQ,EAAE,MAAM,CAAC;IAGjB,UAAU,EAAE,OAAO,CAAC;CACrB"}
@@ -0,0 +1,64 @@
1
+ "use strict";
2
+ var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {
3
+ var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
4
+ if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
5
+ else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
6
+ return c > 3 && r && Object.defineProperty(target, key, r), r;
7
+ };
8
+ var __metadata = (this && this.__metadata) || function (k, v) {
9
+ if (typeof Reflect === "object" && typeof Reflect.metadata === "function") return Reflect.metadata(k, v);
10
+ };
11
+ Object.defineProperty(exports, "__esModule", { value: true });
12
+ exports.ConsumeTenantAiTokensResponseDto = exports.ConsumeTenantAiTokensDto = exports.TENANT_AI_TOKEN_CONSUMPTION_SOURCES = void 0;
13
+ const swagger_1 = require("@nestjs/swagger");
14
+ const class_validator_1 = require("class-validator");
15
+ const tenant_ai_token_state_dto_1 = require("./tenant-ai-token-state.dto");
16
+ exports.TENANT_AI_TOKEN_CONSUMPTION_SOURCES = ['ai.request', 'ai.followup.generate'];
17
+ class ConsumeTenantAiTokensDto {
18
+ tenant;
19
+ tokens;
20
+ source;
21
+ correlationId;
22
+ }
23
+ exports.ConsumeTenantAiTokensDto = ConsumeTenantAiTokensDto;
24
+ __decorate([
25
+ (0, swagger_1.ApiProperty)({ description: 'Tenant slug', example: 'clinic-alpha' }),
26
+ (0, class_validator_1.IsString)(),
27
+ (0, class_validator_1.IsNotEmpty)(),
28
+ __metadata("design:type", String)
29
+ ], ConsumeTenantAiTokensDto.prototype, "tenant", void 0);
30
+ __decorate([
31
+ (0, swagger_1.ApiProperty)({ description: 'Tokens to consume', example: 420 }),
32
+ (0, class_validator_1.IsInt)(),
33
+ (0, class_validator_1.Min)(0),
34
+ __metadata("design:type", Number)
35
+ ], ConsumeTenantAiTokensDto.prototype, "tokens", void 0);
36
+ __decorate([
37
+ (0, swagger_1.ApiProperty)({
38
+ description: 'Deterministic source of consumption',
39
+ enum: exports.TENANT_AI_TOKEN_CONSUMPTION_SOURCES,
40
+ }),
41
+ (0, class_validator_1.IsIn)(exports.TENANT_AI_TOKEN_CONSUMPTION_SOURCES),
42
+ __metadata("design:type", String)
43
+ ], ConsumeTenantAiTokensDto.prototype, "source", void 0);
44
+ __decorate([
45
+ (0, swagger_1.ApiProperty)({ description: 'Optional operation correlation id', required: false }),
46
+ (0, class_validator_1.IsOptional)(),
47
+ (0, class_validator_1.IsString)(),
48
+ __metadata("design:type", String)
49
+ ], ConsumeTenantAiTokensDto.prototype, "correlationId", void 0);
50
+ class ConsumeTenantAiTokensResponseDto extends tenant_ai_token_state_dto_1.TenantAiTokenStateDto {
51
+ consumed;
52
+ becameZero;
53
+ }
54
+ exports.ConsumeTenantAiTokensResponseDto = ConsumeTenantAiTokensResponseDto;
55
+ __decorate([
56
+ (0, swagger_1.ApiProperty)({ description: 'How many tokens were actually consumed', example: 420 }),
57
+ (0, class_validator_1.IsInt)(),
58
+ (0, class_validator_1.Min)(0),
59
+ __metadata("design:type", Number)
60
+ ], ConsumeTenantAiTokensResponseDto.prototype, "consumed", void 0);
61
+ __decorate([
62
+ (0, swagger_1.ApiProperty)({ description: 'True when the balance reached zero in this operation', example: false }),
63
+ __metadata("design:type", Boolean)
64
+ ], ConsumeTenantAiTokensResponseDto.prototype, "becameZero", void 0);
@@ -3,4 +3,8 @@ export * from './get-tenant-by-address.query.dto';
3
3
  export * from './tenant-public-by-address-response.dto';
4
4
  export * from './whatsapp-provider-kind.dto';
5
5
  export * from './tenant-slug.util';
6
+ export * from './tenant-ai-token-state.dto';
7
+ export * from './consume-tenant-ai-tokens.dto';
8
+ export * from './recharge-tenant-ai-tokens.dto';
9
+ export * from './renew-tenant-ai-tokens.dto';
6
10
  //# sourceMappingURL=index.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/tenants/index.ts"],"names":[],"mappings":"AAAA,cAAc,qCAAqC,CAAC;AACpD,cAAc,mCAAmC,CAAC;AAClD,cAAc,yCAAyC,CAAC;AACxD,cAAc,8BAA8B,CAAC;AAC7C,cAAc,oBAAoB,CAAC"}
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/tenants/index.ts"],"names":[],"mappings":"AAAA,cAAc,qCAAqC,CAAC;AACpD,cAAc,mCAAmC,CAAC;AAClD,cAAc,yCAAyC,CAAC;AACxD,cAAc,8BAA8B,CAAC;AAC7C,cAAc,oBAAoB,CAAC;AACnC,cAAc,6BAA6B,CAAC;AAC5C,cAAc,gCAAgC,CAAC;AAC/C,cAAc,iCAAiC,CAAC;AAChD,cAAc,8BAA8B,CAAC"}
@@ -19,3 +19,7 @@ __exportStar(require("./get-tenant-by-address.query.dto"), exports);
19
19
  __exportStar(require("./tenant-public-by-address-response.dto"), exports);
20
20
  __exportStar(require("./whatsapp-provider-kind.dto"), exports);
21
21
  __exportStar(require("./tenant-slug.util"), exports);
22
+ __exportStar(require("./tenant-ai-token-state.dto"), exports);
23
+ __exportStar(require("./consume-tenant-ai-tokens.dto"), exports);
24
+ __exportStar(require("./recharge-tenant-ai-tokens.dto"), exports);
25
+ __exportStar(require("./renew-tenant-ai-tokens.dto"), exports);
@@ -0,0 +1,10 @@
1
+ import { TenantAiTokenStateDto } from './tenant-ai-token-state.dto';
2
+ export declare class RechargeTenantAiTokensDto {
3
+ tenantId: string;
4
+ amount: number;
5
+ reason?: string;
6
+ }
7
+ export declare class RechargeTenantAiTokensResponseDto extends TenantAiTokenStateDto {
8
+ added: number;
9
+ }
10
+ //# sourceMappingURL=recharge-tenant-ai-tokens.dto.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"recharge-tenant-ai-tokens.dto.d.ts","sourceRoot":"","sources":["../../src/tenants/recharge-tenant-ai-tokens.dto.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,qBAAqB,EAAE,MAAM,6BAA6B,CAAC;AAEpE,qBAAa,yBAAyB;IAIpC,QAAQ,EAAE,MAAM,CAAC;IAKjB,MAAM,EAAE,MAAM,CAAC;IAKf,MAAM,CAAC,EAAE,MAAM,CAAC;CACjB;AAED,qBAAa,iCAAkC,SAAQ,qBAAqB;IAI1E,KAAK,EAAE,MAAM,CAAC;CACf"}
@@ -0,0 +1,49 @@
1
+ "use strict";
2
+ var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {
3
+ var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
4
+ if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
5
+ else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
6
+ return c > 3 && r && Object.defineProperty(target, key, r), r;
7
+ };
8
+ var __metadata = (this && this.__metadata) || function (k, v) {
9
+ if (typeof Reflect === "object" && typeof Reflect.metadata === "function") return Reflect.metadata(k, v);
10
+ };
11
+ Object.defineProperty(exports, "__esModule", { value: true });
12
+ exports.RechargeTenantAiTokensResponseDto = exports.RechargeTenantAiTokensDto = void 0;
13
+ const swagger_1 = require("@nestjs/swagger");
14
+ const class_validator_1 = require("class-validator");
15
+ const tenant_ai_token_state_dto_1 = require("./tenant-ai-token-state.dto");
16
+ class RechargeTenantAiTokensDto {
17
+ tenantId;
18
+ amount;
19
+ reason;
20
+ }
21
+ exports.RechargeTenantAiTokensDto = RechargeTenantAiTokensDto;
22
+ __decorate([
23
+ (0, swagger_1.ApiProperty)({ description: 'Tenant id (UUID)', example: 'a1b2c3d4-e5f6-7890-abcd-ef1234567890' }),
24
+ (0, class_validator_1.IsString)(),
25
+ (0, class_validator_1.IsNotEmpty)(),
26
+ __metadata("design:type", String)
27
+ ], RechargeTenantAiTokensDto.prototype, "tenantId", void 0);
28
+ __decorate([
29
+ (0, swagger_1.ApiProperty)({ description: 'Amount of tokens to add', example: 10000 }),
30
+ (0, class_validator_1.IsInt)(),
31
+ (0, class_validator_1.Min)(1),
32
+ __metadata("design:type", Number)
33
+ ], RechargeTenantAiTokensDto.prototype, "amount", void 0);
34
+ __decorate([
35
+ (0, swagger_1.ApiPropertyOptional)({ description: 'Free text reason for audit', example: 'Recarga manual realizada pelo financeiro' }),
36
+ (0, class_validator_1.IsOptional)(),
37
+ (0, class_validator_1.IsString)(),
38
+ __metadata("design:type", String)
39
+ ], RechargeTenantAiTokensDto.prototype, "reason", void 0);
40
+ class RechargeTenantAiTokensResponseDto extends tenant_ai_token_state_dto_1.TenantAiTokenStateDto {
41
+ added;
42
+ }
43
+ exports.RechargeTenantAiTokensResponseDto = RechargeTenantAiTokensResponseDto;
44
+ __decorate([
45
+ (0, swagger_1.ApiProperty)({ description: 'Recharge amount applied', example: 10000 }),
46
+ (0, class_validator_1.IsInt)(),
47
+ (0, class_validator_1.Min)(1),
48
+ __metadata("design:type", Number)
49
+ ], RechargeTenantAiTokensResponseDto.prototype, "added", void 0);
@@ -0,0 +1,7 @@
1
+ export declare class RenewTenantAiTokensDto {
2
+ tenantId?: string;
3
+ }
4
+ export declare class RenewTenantAiTokensResponseDto {
5
+ renewedCount: number;
6
+ }
7
+ //# sourceMappingURL=renew-tenant-ai-tokens.dto.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"renew-tenant-ai-tokens.dto.d.ts","sourceRoot":"","sources":["../../src/tenants/renew-tenant-ai-tokens.dto.ts"],"names":[],"mappings":"AAGA,qBAAa,sBAAsB;IAIjC,QAAQ,CAAC,EAAE,MAAM,CAAC;CACnB;AAED,qBAAa,8BAA8B;IAIzC,YAAY,EAAE,MAAM,CAAC;CACtB"}
@@ -0,0 +1,34 @@
1
+ "use strict";
2
+ var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {
3
+ var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
4
+ if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
5
+ else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
6
+ return c > 3 && r && Object.defineProperty(target, key, r), r;
7
+ };
8
+ var __metadata = (this && this.__metadata) || function (k, v) {
9
+ if (typeof Reflect === "object" && typeof Reflect.metadata === "function") return Reflect.metadata(k, v);
10
+ };
11
+ Object.defineProperty(exports, "__esModule", { value: true });
12
+ exports.RenewTenantAiTokensResponseDto = exports.RenewTenantAiTokensDto = void 0;
13
+ const swagger_1 = require("@nestjs/swagger");
14
+ const class_validator_1 = require("class-validator");
15
+ class RenewTenantAiTokensDto {
16
+ tenantId;
17
+ }
18
+ exports.RenewTenantAiTokensDto = RenewTenantAiTokensDto;
19
+ __decorate([
20
+ (0, swagger_1.ApiPropertyOptional)({ description: 'Optional tenant id to renew one tenant only' }),
21
+ (0, class_validator_1.IsOptional)(),
22
+ (0, class_validator_1.IsString)(),
23
+ __metadata("design:type", String)
24
+ ], RenewTenantAiTokensDto.prototype, "tenantId", void 0);
25
+ class RenewTenantAiTokensResponseDto {
26
+ renewedCount;
27
+ }
28
+ exports.RenewTenantAiTokensResponseDto = RenewTenantAiTokensResponseDto;
29
+ __decorate([
30
+ (0, swagger_1.ApiPropertyOptional)({ description: 'How many tenants were renewed', example: 12 }),
31
+ (0, class_validator_1.IsInt)(),
32
+ (0, class_validator_1.Min)(0),
33
+ __metadata("design:type", Number)
34
+ ], RenewTenantAiTokensResponseDto.prototype, "renewedCount", void 0);
@@ -0,0 +1,13 @@
1
+ export declare class TenantAiTokenStateDto {
2
+ tenant: string;
3
+ currentBalance: number;
4
+ monthlyAllowance: number;
5
+ blocked: boolean;
6
+ cycleStartedAt: string;
7
+ nextRenewalAt: string;
8
+ recurrenceDays: number;
9
+ canUseAi: boolean;
10
+ lastConsumedAt?: string;
11
+ lastRechargedAt?: string;
12
+ }
13
+ //# sourceMappingURL=tenant-ai-token-state.dto.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"tenant-ai-token-state.dto.d.ts","sourceRoot":"","sources":["../../src/tenants/tenant-ai-token-state.dto.ts"],"names":[],"mappings":"AAGA,qBAAa,qBAAqB;IAIhC,MAAM,EAAE,MAAM,CAAC;IAKf,cAAc,EAAE,MAAM,CAAC;IAKvB,gBAAgB,EAAE,MAAM,CAAC;IAIzB,OAAO,EAAE,OAAO,CAAC;IAIjB,cAAc,EAAE,MAAM,CAAC;IAIvB,aAAa,EAAE,MAAM,CAAC;IAKtB,cAAc,EAAE,MAAM,CAAC;IAIvB,QAAQ,EAAE,OAAO,CAAC;IAKlB,cAAc,CAAC,EAAE,MAAM,CAAC;IAKxB,eAAe,CAAC,EAAE,MAAM,CAAC;CAC1B"}
@@ -0,0 +1,83 @@
1
+ "use strict";
2
+ var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {
3
+ var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
4
+ if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
5
+ else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
6
+ return c > 3 && r && Object.defineProperty(target, key, r), r;
7
+ };
8
+ var __metadata = (this && this.__metadata) || function (k, v) {
9
+ if (typeof Reflect === "object" && typeof Reflect.metadata === "function") return Reflect.metadata(k, v);
10
+ };
11
+ Object.defineProperty(exports, "__esModule", { value: true });
12
+ exports.TenantAiTokenStateDto = void 0;
13
+ const swagger_1 = require("@nestjs/swagger");
14
+ const class_validator_1 = require("class-validator");
15
+ class TenantAiTokenStateDto {
16
+ tenant;
17
+ currentBalance;
18
+ monthlyAllowance;
19
+ blocked;
20
+ cycleStartedAt;
21
+ nextRenewalAt;
22
+ recurrenceDays;
23
+ canUseAi;
24
+ lastConsumedAt;
25
+ lastRechargedAt;
26
+ }
27
+ exports.TenantAiTokenStateDto = TenantAiTokenStateDto;
28
+ __decorate([
29
+ (0, swagger_1.ApiProperty)({ description: 'Tenant slug', example: 'clinic-alpha' }),
30
+ (0, class_validator_1.IsString)(),
31
+ (0, class_validator_1.IsNotEmpty)(),
32
+ __metadata("design:type", String)
33
+ ], TenantAiTokenStateDto.prototype, "tenant", void 0);
34
+ __decorate([
35
+ (0, swagger_1.ApiProperty)({ description: 'Current remaining AI tokens', example: 2400 }),
36
+ (0, class_validator_1.IsInt)(),
37
+ (0, class_validator_1.Min)(0),
38
+ __metadata("design:type", Number)
39
+ ], TenantAiTokenStateDto.prototype, "currentBalance", void 0);
40
+ __decorate([
41
+ (0, swagger_1.ApiProperty)({ description: 'Monthly allowance configured for current cycle', example: 5000 }),
42
+ (0, class_validator_1.IsInt)(),
43
+ (0, class_validator_1.Min)(0),
44
+ __metadata("design:type", Number)
45
+ ], TenantAiTokenStateDto.prototype, "monthlyAllowance", void 0);
46
+ __decorate([
47
+ (0, swagger_1.ApiProperty)({ description: 'Whether IA is blocked for this tenant', example: false }),
48
+ (0, class_validator_1.IsBoolean)(),
49
+ __metadata("design:type", Boolean)
50
+ ], TenantAiTokenStateDto.prototype, "blocked", void 0);
51
+ __decorate([
52
+ (0, swagger_1.ApiProperty)({ description: 'Current cycle start timestamp', example: '2026-04-01T00:00:00.000Z' }),
53
+ (0, class_validator_1.IsDateString)(),
54
+ __metadata("design:type", String)
55
+ ], TenantAiTokenStateDto.prototype, "cycleStartedAt", void 0);
56
+ __decorate([
57
+ (0, swagger_1.ApiProperty)({ description: 'Next monthly renewal timestamp', example: '2026-05-01T00:00:00.000Z' }),
58
+ (0, class_validator_1.IsDateString)(),
59
+ __metadata("design:type", String)
60
+ ], TenantAiTokenStateDto.prototype, "nextRenewalAt", void 0);
61
+ __decorate([
62
+ (0, swagger_1.ApiProperty)({ description: 'Plan recurrence in days used for renewal', example: 30 }),
63
+ (0, class_validator_1.IsInt)(),
64
+ (0, class_validator_1.Min)(1),
65
+ __metadata("design:type", Number)
66
+ ], TenantAiTokenStateDto.prototype, "recurrenceDays", void 0);
67
+ __decorate([
68
+ (0, swagger_1.ApiProperty)({ description: 'True when balance is currently greater than zero', example: true }),
69
+ (0, class_validator_1.IsBoolean)(),
70
+ __metadata("design:type", Boolean)
71
+ ], TenantAiTokenStateDto.prototype, "canUseAi", void 0);
72
+ __decorate([
73
+ (0, swagger_1.ApiProperty)({ description: 'Last consumption timestamp', required: false }),
74
+ (0, class_validator_1.IsOptional)(),
75
+ (0, class_validator_1.IsDateString)(),
76
+ __metadata("design:type", String)
77
+ ], TenantAiTokenStateDto.prototype, "lastConsumedAt", void 0);
78
+ __decorate([
79
+ (0, swagger_1.ApiProperty)({ description: 'Last recharge timestamp', required: false }),
80
+ (0, class_validator_1.IsOptional)(),
81
+ (0, class_validator_1.IsDateString)(),
82
+ __metadata("design:type", String)
83
+ ], TenantAiTokenStateDto.prototype, "lastRechargedAt", void 0);
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "tychat-contracts",
3
- "version": "1.3.2",
3
+ "version": "1.3.4",
4
4
  "description": "DTOs compartilhados com class-validator (API e microserviços)",
5
5
  "main": "dist/index.js",
6
6
  "types": "dist/index.d.ts",
@@ -37,4 +37,25 @@ export class AiIntegrationResponseDto {
37
37
  @IsOptional()
38
38
  @IsString()
39
39
  aiUsageId?: string;
40
+
41
+ @ApiPropertyOptional({
42
+ description: 'Prompt tokens consumed in this AI request',
43
+ example: 120,
44
+ })
45
+ @IsOptional()
46
+ promptTokens?: number;
47
+
48
+ @ApiPropertyOptional({
49
+ description: 'Completion tokens generated in this AI request',
50
+ example: 90,
51
+ })
52
+ @IsOptional()
53
+ completionTokens?: number;
54
+
55
+ @ApiPropertyOptional({
56
+ description: 'Total tokens consumed in this AI request',
57
+ example: 210,
58
+ })
59
+ @IsOptional()
60
+ totalTokens?: number;
40
61
  }
@@ -17,7 +17,7 @@ export interface EmitAnalyticEventOptions {
17
17
  /**
18
18
  * Builds the Kafka payload to be sent to the analytics service.
19
19
  *
20
- * Usage (fire-and-forget via ClientKafka.emit):
20
+ * Usage (fire-and-forget via ClientProxy.emit):
21
21
  * ```ts
22
22
  * this.analyticsClient.emit(
23
23
  * TOPIC_ANALYTICS_EVENT,
@@ -0,0 +1,92 @@
1
+ import { ApiProperty, ApiPropertyOptional } from '@nestjs/swagger';
2
+ import { Type } from 'class-transformer';
3
+ import {
4
+ IsArray,
5
+ IsBoolean,
6
+ IsEnum,
7
+ IsInt,
8
+ IsNotEmpty,
9
+ IsOptional,
10
+ IsString,
11
+ Min,
12
+ ValidateNested,
13
+ } from 'class-validator';
14
+ import {
15
+ AI_TOOL_FALLBACK_MODES,
16
+ AI_TOOL_KEYS,
17
+ AI_TOOL_POLICY_LEVELS,
18
+ type AiToolFallbackMode,
19
+ type AiToolKey,
20
+ type AiToolPolicyLevel,
21
+ } from './ai-tool-policy.enums';
22
+
23
+ export class AiToolPolicyDto {
24
+ @ApiProperty({ enum: AI_TOOL_KEYS })
25
+ @IsEnum(AI_TOOL_KEYS)
26
+ toolKey: AiToolKey;
27
+
28
+ @ApiProperty()
29
+ @IsBoolean()
30
+ enabled: boolean;
31
+
32
+ @ApiPropertyOptional({ enum: AI_TOOL_FALLBACK_MODES })
33
+ @IsOptional()
34
+ @IsEnum(AI_TOOL_FALLBACK_MODES)
35
+ fallbackMode?: AiToolFallbackMode;
36
+
37
+ @ApiPropertyOptional({ enum: AI_TOOL_POLICY_LEVELS })
38
+ @IsOptional()
39
+ @IsEnum(AI_TOOL_POLICY_LEVELS)
40
+ level?: AiToolPolicyLevel;
41
+ }
42
+
43
+ export class UpdateAiToolPolicyDto {
44
+ @ApiProperty({ enum: AI_TOOL_KEYS })
45
+ @IsEnum(AI_TOOL_KEYS)
46
+ toolKey: AiToolKey;
47
+
48
+ @ApiProperty()
49
+ @IsBoolean()
50
+ enabled: boolean;
51
+
52
+ @ApiPropertyOptional({ enum: AI_TOOL_FALLBACK_MODES })
53
+ @IsOptional()
54
+ @IsEnum(AI_TOOL_FALLBACK_MODES)
55
+ fallbackMode?: AiToolFallbackMode;
56
+ }
57
+
58
+ export class UpdateAiToolPoliciesDto {
59
+ @ApiProperty({ type: () => [UpdateAiToolPolicyDto] })
60
+ @IsArray()
61
+ @ValidateNested({ each: true })
62
+ @Type(() => UpdateAiToolPolicyDto)
63
+ policies: UpdateAiToolPolicyDto[];
64
+ }
65
+
66
+ export class ResolveAiToolPoliciesQueryDto {
67
+ @ApiProperty()
68
+ @IsString()
69
+ @IsNotEmpty()
70
+ tenantId: string;
71
+
72
+ @ApiPropertyOptional({ minimum: 1 })
73
+ @IsOptional()
74
+ @IsInt()
75
+ @Min(1)
76
+ unitId?: number;
77
+ }
78
+
79
+ export class EffectiveAiToolPolicyDto extends AiToolPolicyDto {
80
+ @ApiPropertyOptional()
81
+ @IsOptional()
82
+ @IsString()
83
+ reasonCode?: string;
84
+ }
85
+
86
+ export class EffectiveAiToolPoliciesDto {
87
+ @ApiProperty({ type: () => [EffectiveAiToolPolicyDto] })
88
+ @IsArray()
89
+ @ValidateNested({ each: true })
90
+ @Type(() => EffectiveAiToolPolicyDto)
91
+ policies: EffectiveAiToolPolicyDto[];
92
+ }
@@ -0,0 +1,25 @@
1
+ export const AI_TOOL_KEYS = [
2
+ 'get_procedures_clinic',
3
+ 'get_clinic_opening_hours',
4
+ 'get_available_slots',
5
+ 'search_health_professionals',
6
+ 'update_patient',
7
+ 'search_patients_by_phone',
8
+ 'create_patient',
9
+ 'create_appointment',
10
+ 'add_appointment_procedures',
11
+ 'list_patient_appointments',
12
+ 'cancel_appointment',
13
+ 'submit_satisfaction',
14
+ ] as const;
15
+
16
+ export type AiToolKey = (typeof AI_TOOL_KEYS)[number];
17
+
18
+ export const AI_TOOL_POLICY_LEVELS = ['global', 'plan', 'tenant', 'unit'] as const;
19
+ export type AiToolPolicyLevel = (typeof AI_TOOL_POLICY_LEVELS)[number];
20
+
21
+ export const AI_TOOL_FALLBACK_MODES = [
22
+ 'transfer_to_human',
23
+ 'unavailable_message',
24
+ ] as const;
25
+ export type AiToolFallbackMode = (typeof AI_TOOL_FALLBACK_MODES)[number];
@@ -1,6 +1,8 @@
1
1
  export * from './opening-hours-slot.dto';
2
2
  export * from './update-clinic-configuration.dto';
3
3
  export * from './clinic-configuration.dto';
4
+ export * from './ai-tool-policy.enums';
5
+ export * from './ai-tool-policy.dto';
4
6
  export * from './plan-summary.dto';
5
7
  export * from './tenant-limits-summary.dto';
6
8
  export * from './configuration-tenant-summary.dto';
@@ -40,6 +40,9 @@ export class PlanSummaryDto {
40
40
  @ApiProperty({ description: 'Max procedures', example: 200 })
41
41
  maxProcedures: number;
42
42
 
43
+ @ApiProperty({ description: 'Monthly AI token allowance', example: 5000 })
44
+ monthlyAiTokens: number;
45
+
43
46
  @ApiPropertyOptional()
44
47
  createdAt?: Date;
45
48
 
@@ -0,0 +1,41 @@
1
+ import { ApiProperty } from '@nestjs/swagger';
2
+ import { IsIn, IsInt, IsNotEmpty, IsOptional, IsString, Min } from 'class-validator';
3
+ import { TenantAiTokenStateDto } from './tenant-ai-token-state.dto';
4
+
5
+ export const TENANT_AI_TOKEN_CONSUMPTION_SOURCES = ['ai.request', 'ai.followup.generate'] as const;
6
+ export type TenantAiTokenConsumptionSourceDto =
7
+ (typeof TENANT_AI_TOKEN_CONSUMPTION_SOURCES)[number];
8
+
9
+ export class ConsumeTenantAiTokensDto {
10
+ @ApiProperty({ description: 'Tenant slug', example: 'clinic-alpha' })
11
+ @IsString()
12
+ @IsNotEmpty()
13
+ tenant: string;
14
+
15
+ @ApiProperty({ description: 'Tokens to consume', example: 420 })
16
+ @IsInt()
17
+ @Min(0)
18
+ tokens: number;
19
+
20
+ @ApiProperty({
21
+ description: 'Deterministic source of consumption',
22
+ enum: TENANT_AI_TOKEN_CONSUMPTION_SOURCES,
23
+ })
24
+ @IsIn(TENANT_AI_TOKEN_CONSUMPTION_SOURCES)
25
+ source: TenantAiTokenConsumptionSourceDto;
26
+
27
+ @ApiProperty({ description: 'Optional operation correlation id', required: false })
28
+ @IsOptional()
29
+ @IsString()
30
+ correlationId?: string;
31
+ }
32
+
33
+ export class ConsumeTenantAiTokensResponseDto extends TenantAiTokenStateDto {
34
+ @ApiProperty({ description: 'How many tokens were actually consumed', example: 420 })
35
+ @IsInt()
36
+ @Min(0)
37
+ consumed: number;
38
+
39
+ @ApiProperty({ description: 'True when the balance reached zero in this operation', example: false })
40
+ becameZero: boolean;
41
+ }
@@ -3,3 +3,7 @@ export * from './get-tenant-by-address.query.dto';
3
3
  export * from './tenant-public-by-address-response.dto';
4
4
  export * from './whatsapp-provider-kind.dto';
5
5
  export * from './tenant-slug.util';
6
+ export * from './tenant-ai-token-state.dto';
7
+ export * from './consume-tenant-ai-tokens.dto';
8
+ export * from './recharge-tenant-ai-tokens.dto';
9
+ export * from './renew-tenant-ai-tokens.dto';
@@ -0,0 +1,27 @@
1
+ import { ApiProperty, ApiPropertyOptional } from '@nestjs/swagger';
2
+ import { IsInt, IsNotEmpty, IsOptional, IsString, Min } from 'class-validator';
3
+ import { TenantAiTokenStateDto } from './tenant-ai-token-state.dto';
4
+
5
+ export class RechargeTenantAiTokensDto {
6
+ @ApiProperty({ description: 'Tenant id (UUID)', example: 'a1b2c3d4-e5f6-7890-abcd-ef1234567890' })
7
+ @IsString()
8
+ @IsNotEmpty()
9
+ tenantId: string;
10
+
11
+ @ApiProperty({ description: 'Amount of tokens to add', example: 10000 })
12
+ @IsInt()
13
+ @Min(1)
14
+ amount: number;
15
+
16
+ @ApiPropertyOptional({ description: 'Free text reason for audit', example: 'Recarga manual realizada pelo financeiro' })
17
+ @IsOptional()
18
+ @IsString()
19
+ reason?: string;
20
+ }
21
+
22
+ export class RechargeTenantAiTokensResponseDto extends TenantAiTokenStateDto {
23
+ @ApiProperty({ description: 'Recharge amount applied', example: 10000 })
24
+ @IsInt()
25
+ @Min(1)
26
+ added: number;
27
+ }
@@ -0,0 +1,16 @@
1
+ import { ApiPropertyOptional } from '@nestjs/swagger';
2
+ import { IsInt, IsOptional, IsString, Min } from 'class-validator';
3
+
4
+ export class RenewTenantAiTokensDto {
5
+ @ApiPropertyOptional({ description: 'Optional tenant id to renew one tenant only' })
6
+ @IsOptional()
7
+ @IsString()
8
+ tenantId?: string;
9
+ }
10
+
11
+ export class RenewTenantAiTokensResponseDto {
12
+ @ApiPropertyOptional({ description: 'How many tenants were renewed', example: 12 })
13
+ @IsInt()
14
+ @Min(0)
15
+ renewedCount: number;
16
+ }
@@ -0,0 +1,50 @@
1
+ import { ApiProperty } from '@nestjs/swagger';
2
+ import { IsBoolean, IsDateString, IsInt, IsNotEmpty, IsOptional, IsString, Min } from 'class-validator';
3
+
4
+ export class TenantAiTokenStateDto {
5
+ @ApiProperty({ description: 'Tenant slug', example: 'clinic-alpha' })
6
+ @IsString()
7
+ @IsNotEmpty()
8
+ tenant: string;
9
+
10
+ @ApiProperty({ description: 'Current remaining AI tokens', example: 2400 })
11
+ @IsInt()
12
+ @Min(0)
13
+ currentBalance: number;
14
+
15
+ @ApiProperty({ description: 'Monthly allowance configured for current cycle', example: 5000 })
16
+ @IsInt()
17
+ @Min(0)
18
+ monthlyAllowance: number;
19
+
20
+ @ApiProperty({ description: 'Whether IA is blocked for this tenant', example: false })
21
+ @IsBoolean()
22
+ blocked: boolean;
23
+
24
+ @ApiProperty({ description: 'Current cycle start timestamp', example: '2026-04-01T00:00:00.000Z' })
25
+ @IsDateString()
26
+ cycleStartedAt: string;
27
+
28
+ @ApiProperty({ description: 'Next monthly renewal timestamp', example: '2026-05-01T00:00:00.000Z' })
29
+ @IsDateString()
30
+ nextRenewalAt: string;
31
+
32
+ @ApiProperty({ description: 'Plan recurrence in days used for renewal', example: 30 })
33
+ @IsInt()
34
+ @Min(1)
35
+ recurrenceDays: number;
36
+
37
+ @ApiProperty({ description: 'True when balance is currently greater than zero', example: true })
38
+ @IsBoolean()
39
+ canUseAi: boolean;
40
+
41
+ @ApiProperty({ description: 'Last consumption timestamp', required: false })
42
+ @IsOptional()
43
+ @IsDateString()
44
+ lastConsumedAt?: string;
45
+
46
+ @ApiProperty({ description: 'Last recharge timestamp', required: false })
47
+ @IsOptional()
48
+ @IsDateString()
49
+ lastRechargedAt?: string;
50
+ }