@oneuptime/common 10.3.0 → 10.4.1

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 (115) hide show
  1. package/Models/DatabaseModels/IncomingCallPolicyLabelRule.ts +522 -0
  2. package/Models/DatabaseModels/IncomingCallPolicyOwnerRule.ts +606 -0
  3. package/Models/DatabaseModels/IncomingCallPolicyOwnerTeam.ts +508 -0
  4. package/Models/DatabaseModels/IncomingCallPolicyOwnerUser.ts +507 -0
  5. package/Models/DatabaseModels/Index.ts +24 -0
  6. package/Models/DatabaseModels/OnCallDutyPolicyLabelRule.ts +522 -0
  7. package/Models/DatabaseModels/OnCallDutyPolicyOwnerRule.ts +606 -0
  8. package/Models/DatabaseModels/OnCallDutyPolicyScheduleLabelRule.ts +522 -0
  9. package/Models/DatabaseModels/OnCallDutyPolicyScheduleOwnerRule.ts +606 -0
  10. package/Models/DatabaseModels/OnCallDutyPolicyScheduleOwnerTeam.ts +508 -0
  11. package/Models/DatabaseModels/OnCallDutyPolicyScheduleOwnerUser.ts +507 -0
  12. package/Server/API/TelemetryAPI.ts +69 -0
  13. package/Server/Infrastructure/Postgres/SchemaMigrations/1778929624633-AddOnCallDutyPolicyScheduleOwners.ts +153 -0
  14. package/Server/Infrastructure/Postgres/SchemaMigrations/1778931537020-AddOnCallIncomingCallOwnersAndRules.ts +823 -0
  15. package/Server/Infrastructure/Postgres/SchemaMigrations/Index.ts +4 -0
  16. package/Server/Services/IncomingCallPolicyLabelRuleEngineService.ts +197 -0
  17. package/Server/Services/IncomingCallPolicyLabelRuleService.ts +10 -0
  18. package/Server/Services/IncomingCallPolicyOwnerRuleEngineService.ts +220 -0
  19. package/Server/Services/IncomingCallPolicyOwnerRuleService.ts +10 -0
  20. package/Server/Services/IncomingCallPolicyOwnerTeamService.ts +9 -0
  21. package/Server/Services/IncomingCallPolicyOwnerUserService.ts +9 -0
  22. package/Server/Services/IncomingCallPolicyService.ts +35 -0
  23. package/Server/Services/OnCallDutyPolicyLabelRuleEngineService.ts +191 -0
  24. package/Server/Services/OnCallDutyPolicyLabelRuleService.ts +10 -0
  25. package/Server/Services/OnCallDutyPolicyOwnerRuleEngineService.ts +214 -0
  26. package/Server/Services/OnCallDutyPolicyOwnerRuleService.ts +10 -0
  27. package/Server/Services/OnCallDutyPolicyScheduleLabelRuleEngineService.ts +200 -0
  28. package/Server/Services/OnCallDutyPolicyScheduleLabelRuleService.ts +10 -0
  29. package/Server/Services/OnCallDutyPolicyScheduleOwnerRuleEngineService.ts +223 -0
  30. package/Server/Services/OnCallDutyPolicyScheduleOwnerRuleService.ts +10 -0
  31. package/Server/Services/OnCallDutyPolicyScheduleOwnerTeamService.ts +9 -0
  32. package/Server/Services/OnCallDutyPolicyScheduleOwnerUserService.ts +9 -0
  33. package/Server/Services/OnCallDutyPolicyScheduleService.ts +32 -1
  34. package/Server/Services/OnCallDutyPolicyService.ts +21 -0
  35. package/Server/Services/ProfileAggregationService.ts +119 -0
  36. package/Server/Types/Database/Permissions/QueryPermission.ts +36 -0
  37. package/Types/Monitor/MonitorStep.ts +66 -0
  38. package/Types/Permission.ts +475 -0
  39. package/build/dist/Models/DatabaseModels/IncomingCallPolicyLabelRule.js +532 -0
  40. package/build/dist/Models/DatabaseModels/IncomingCallPolicyLabelRule.js.map +1 -0
  41. package/build/dist/Models/DatabaseModels/IncomingCallPolicyOwnerRule.js +615 -0
  42. package/build/dist/Models/DatabaseModels/IncomingCallPolicyOwnerRule.js.map +1 -0
  43. package/build/dist/Models/DatabaseModels/IncomingCallPolicyOwnerTeam.js +524 -0
  44. package/build/dist/Models/DatabaseModels/IncomingCallPolicyOwnerTeam.js.map +1 -0
  45. package/build/dist/Models/DatabaseModels/IncomingCallPolicyOwnerUser.js +523 -0
  46. package/build/dist/Models/DatabaseModels/IncomingCallPolicyOwnerUser.js.map +1 -0
  47. package/build/dist/Models/DatabaseModels/Index.js +22 -0
  48. package/build/dist/Models/DatabaseModels/Index.js.map +1 -1
  49. package/build/dist/Models/DatabaseModels/OnCallDutyPolicyLabelRule.js +532 -0
  50. package/build/dist/Models/DatabaseModels/OnCallDutyPolicyLabelRule.js.map +1 -0
  51. package/build/dist/Models/DatabaseModels/OnCallDutyPolicyOwnerRule.js +615 -0
  52. package/build/dist/Models/DatabaseModels/OnCallDutyPolicyOwnerRule.js.map +1 -0
  53. package/build/dist/Models/DatabaseModels/OnCallDutyPolicyScheduleLabelRule.js +532 -0
  54. package/build/dist/Models/DatabaseModels/OnCallDutyPolicyScheduleLabelRule.js.map +1 -0
  55. package/build/dist/Models/DatabaseModels/OnCallDutyPolicyScheduleOwnerRule.js +615 -0
  56. package/build/dist/Models/DatabaseModels/OnCallDutyPolicyScheduleOwnerRule.js.map +1 -0
  57. package/build/dist/Models/DatabaseModels/OnCallDutyPolicyScheduleOwnerTeam.js +524 -0
  58. package/build/dist/Models/DatabaseModels/OnCallDutyPolicyScheduleOwnerTeam.js.map +1 -0
  59. package/build/dist/Models/DatabaseModels/OnCallDutyPolicyScheduleOwnerUser.js +523 -0
  60. package/build/dist/Models/DatabaseModels/OnCallDutyPolicyScheduleOwnerUser.js.map +1 -0
  61. package/build/dist/Server/API/TelemetryAPI.js +34 -0
  62. package/build/dist/Server/API/TelemetryAPI.js.map +1 -1
  63. package/build/dist/Server/Infrastructure/Postgres/SchemaMigrations/1778929624633-AddOnCallDutyPolicyScheduleOwners.js +58 -0
  64. package/build/dist/Server/Infrastructure/Postgres/SchemaMigrations/1778929624633-AddOnCallDutyPolicyScheduleOwners.js.map +1 -0
  65. package/build/dist/Server/Infrastructure/Postgres/SchemaMigrations/1778931537020-AddOnCallIncomingCallOwnersAndRules.js +292 -0
  66. package/build/dist/Server/Infrastructure/Postgres/SchemaMigrations/1778931537020-AddOnCallIncomingCallOwnersAndRules.js.map +1 -0
  67. package/build/dist/Server/Infrastructure/Postgres/SchemaMigrations/Index.js +4 -0
  68. package/build/dist/Server/Infrastructure/Postgres/SchemaMigrations/Index.js.map +1 -1
  69. package/build/dist/Server/Services/IncomingCallPolicyLabelRuleEngineService.js +158 -0
  70. package/build/dist/Server/Services/IncomingCallPolicyLabelRuleEngineService.js.map +1 -0
  71. package/build/dist/Server/Services/IncomingCallPolicyLabelRuleService.js +9 -0
  72. package/build/dist/Server/Services/IncomingCallPolicyLabelRuleService.js.map +1 -0
  73. package/build/dist/Server/Services/IncomingCallPolicyOwnerRuleEngineService.js +181 -0
  74. package/build/dist/Server/Services/IncomingCallPolicyOwnerRuleEngineService.js.map +1 -0
  75. package/build/dist/Server/Services/IncomingCallPolicyOwnerRuleService.js +9 -0
  76. package/build/dist/Server/Services/IncomingCallPolicyOwnerRuleService.js.map +1 -0
  77. package/build/dist/Server/Services/IncomingCallPolicyOwnerTeamService.js +9 -0
  78. package/build/dist/Server/Services/IncomingCallPolicyOwnerTeamService.js.map +1 -0
  79. package/build/dist/Server/Services/IncomingCallPolicyOwnerUserService.js +9 -0
  80. package/build/dist/Server/Services/IncomingCallPolicyOwnerUserService.js.map +1 -0
  81. package/build/dist/Server/Services/IncomingCallPolicyService.js +38 -0
  82. package/build/dist/Server/Services/IncomingCallPolicyService.js.map +1 -1
  83. package/build/dist/Server/Services/OnCallDutyPolicyLabelRuleEngineService.js +157 -0
  84. package/build/dist/Server/Services/OnCallDutyPolicyLabelRuleEngineService.js.map +1 -0
  85. package/build/dist/Server/Services/OnCallDutyPolicyLabelRuleService.js +9 -0
  86. package/build/dist/Server/Services/OnCallDutyPolicyLabelRuleService.js.map +1 -0
  87. package/build/dist/Server/Services/OnCallDutyPolicyOwnerRuleEngineService.js +180 -0
  88. package/build/dist/Server/Services/OnCallDutyPolicyOwnerRuleEngineService.js.map +1 -0
  89. package/build/dist/Server/Services/OnCallDutyPolicyOwnerRuleService.js +9 -0
  90. package/build/dist/Server/Services/OnCallDutyPolicyOwnerRuleService.js.map +1 -0
  91. package/build/dist/Server/Services/OnCallDutyPolicyScheduleLabelRuleEngineService.js +158 -0
  92. package/build/dist/Server/Services/OnCallDutyPolicyScheduleLabelRuleEngineService.js.map +1 -0
  93. package/build/dist/Server/Services/OnCallDutyPolicyScheduleLabelRuleService.js +9 -0
  94. package/build/dist/Server/Services/OnCallDutyPolicyScheduleLabelRuleService.js.map +1 -0
  95. package/build/dist/Server/Services/OnCallDutyPolicyScheduleOwnerRuleEngineService.js +181 -0
  96. package/build/dist/Server/Services/OnCallDutyPolicyScheduleOwnerRuleEngineService.js.map +1 -0
  97. package/build/dist/Server/Services/OnCallDutyPolicyScheduleOwnerRuleService.js +9 -0
  98. package/build/dist/Server/Services/OnCallDutyPolicyScheduleOwnerRuleService.js.map +1 -0
  99. package/build/dist/Server/Services/OnCallDutyPolicyScheduleOwnerTeamService.js +9 -0
  100. package/build/dist/Server/Services/OnCallDutyPolicyScheduleOwnerTeamService.js.map +1 -0
  101. package/build/dist/Server/Services/OnCallDutyPolicyScheduleOwnerUserService.js +9 -0
  102. package/build/dist/Server/Services/OnCallDutyPolicyScheduleOwnerUserService.js.map +1 -0
  103. package/build/dist/Server/Services/OnCallDutyPolicyScheduleService.js +21 -0
  104. package/build/dist/Server/Services/OnCallDutyPolicyScheduleService.js.map +1 -1
  105. package/build/dist/Server/Services/OnCallDutyPolicyService.js +17 -3
  106. package/build/dist/Server/Services/OnCallDutyPolicyService.js.map +1 -1
  107. package/build/dist/Server/Services/ProfileAggregationService.js +80 -0
  108. package/build/dist/Server/Services/ProfileAggregationService.js.map +1 -1
  109. package/build/dist/Server/Types/Database/Permissions/QueryPermission.js +25 -0
  110. package/build/dist/Server/Types/Database/Permissions/QueryPermission.js.map +1 -1
  111. package/build/dist/Types/Monitor/MonitorStep.js +42 -0
  112. package/build/dist/Types/Monitor/MonitorStep.js.map +1 -1
  113. package/build/dist/Types/Permission.js +415 -0
  114. package/build/dist/Types/Permission.js.map +1 -1
  115. package/package.json +1 -1
@@ -334,6 +334,8 @@ import { AddRunbookAgentOwnersAndLabels1778842348388 } from "./1778842348388-Add
334
334
  import { AddRunbookSecret1778851431287 } from "./1778851431287-AddRunbookSecret";
335
335
  import { UpdateTeamPermissionScopeDefault1778877094014 } from "./1778877094014-UpdateTeamPermissionScopeDefault";
336
336
  import { MigrateLegacyManagerRolesToAdmin1778900000000 } from "./1778900000000-MigrateLegacyManagerRolesToAdmin";
337
+ import { AddOnCallDutyPolicyScheduleOwners1778929624633 } from "./1778929624633-AddOnCallDutyPolicyScheduleOwners";
338
+ import { AddOnCallIncomingCallOwnersAndRules1778931537020 } from "./1778931537020-AddOnCallIncomingCallOwnersAndRules";
337
339
  export default [
338
340
  InitialMigration,
339
341
  MigrationName1717678334852,
@@ -671,4 +673,6 @@ export default [
671
673
  AddRunbookSecret1778851431287,
672
674
  UpdateTeamPermissionScopeDefault1778877094014,
673
675
  MigrateLegacyManagerRolesToAdmin1778900000000,
676
+ AddOnCallDutyPolicyScheduleOwners1778929624633,
677
+ AddOnCallIncomingCallOwnersAndRules1778931537020,
674
678
  ];
@@ -0,0 +1,197 @@
1
+ import Label from "../../Models/DatabaseModels/Label";
2
+ import IncomingCallPolicy from "../../Models/DatabaseModels/IncomingCallPolicy";
3
+ import IncomingCallPolicyLabelRule from "../../Models/DatabaseModels/IncomingCallPolicyLabelRule";
4
+ import IncomingCallPolicyLabelRuleService from "./IncomingCallPolicyLabelRuleService";
5
+ import IncomingCallPolicyService from "./IncomingCallPolicyService";
6
+ import ObjectID from "../../Types/ObjectID";
7
+ import CaptureSpan from "../Utils/Telemetry/CaptureSpan";
8
+ import logger, { LogAttributes } from "../Utils/Logger";
9
+
10
+ class IncomingCallPolicyLabelRuleEngineServiceClass {
11
+ @CaptureSpan()
12
+ public async applyRulesToIncomingCallPolicy(
13
+ policy: IncomingCallPolicy,
14
+ ): Promise<void> {
15
+ if (!policy.id || !policy.projectId) {
16
+ return;
17
+ }
18
+
19
+ try {
20
+ const rules: Array<IncomingCallPolicyLabelRule> =
21
+ await IncomingCallPolicyLabelRuleService.findBy({
22
+ query: {
23
+ projectId: policy.projectId,
24
+ isEnabled: true,
25
+ },
26
+ props: { isRoot: true },
27
+ select: {
28
+ _id: true,
29
+ name: true,
30
+ incomingCallPolicyLabels: { _id: true },
31
+ incomingCallPolicyNamePattern: true,
32
+ incomingCallPolicyDescriptionPattern: true,
33
+ labelsToAdd: { _id: true },
34
+ },
35
+ limit: 100,
36
+ skip: 0,
37
+ });
38
+
39
+ if (rules.length === 0) {
40
+ return;
41
+ }
42
+
43
+ const policyWithDetails: IncomingCallPolicy | null =
44
+ await IncomingCallPolicyService.findOneById({
45
+ id: policy.id,
46
+ select: {
47
+ name: true,
48
+ description: true,
49
+ labels: { _id: true },
50
+ },
51
+ props: { isRoot: true },
52
+ });
53
+
54
+ if (!policyWithDetails) {
55
+ return;
56
+ }
57
+
58
+ const labelIdsToAdd: Set<string> = new Set();
59
+
60
+ for (const rule of rules) {
61
+ const matches: boolean = this.doesPolicyMatchRule(
62
+ policyWithDetails,
63
+ rule,
64
+ );
65
+ if (!matches) {
66
+ continue;
67
+ }
68
+ for (const label of rule.labelsToAdd || []) {
69
+ if (label.id) {
70
+ labelIdsToAdd.add(label.id.toString());
71
+ }
72
+ }
73
+ }
74
+
75
+ if (labelIdsToAdd.size === 0) {
76
+ return;
77
+ }
78
+
79
+ const existingLabelIds: Set<string> = new Set(
80
+ (policyWithDetails.labels || [])
81
+ .map((l: Label) => {
82
+ return l.id?.toString() || "";
83
+ })
84
+ .filter((id: string) => {
85
+ return id !== "";
86
+ }),
87
+ );
88
+
89
+ const newLabelIds: Array<string> = Array.from(labelIdsToAdd).filter(
90
+ (id: string) => {
91
+ return !existingLabelIds.has(id);
92
+ },
93
+ );
94
+ if (newLabelIds.length === 0) {
95
+ return;
96
+ }
97
+
98
+ await IncomingCallPolicyService.getRepository()
99
+ .createQueryBuilder()
100
+ .relation(IncomingCallPolicy, "labels")
101
+ .of(policy.id.toString())
102
+ .add(newLabelIds);
103
+
104
+ const mergedLabelIds: Set<string> = new Set([
105
+ ...existingLabelIds,
106
+ ...newLabelIds,
107
+ ]);
108
+ policy.labels = Array.from(mergedLabelIds).map((id: string) => {
109
+ const label: Label = new Label();
110
+ label.id = new ObjectID(id);
111
+ return label;
112
+ });
113
+
114
+ logger.debug(
115
+ `IncomingCallPolicyLabelRuleEngine attached ${newLabelIds.length} labels to policy ${policy.id}`,
116
+ { projectId: policy.projectId.toString() } as LogAttributes,
117
+ );
118
+ } catch (error) {
119
+ logger.error(
120
+ `Error applying incoming call policy label rules: ${error}`,
121
+ {
122
+ projectId: policy.projectId?.toString(),
123
+ incomingCallPolicyId: policy.id?.toString(),
124
+ } as LogAttributes,
125
+ );
126
+ }
127
+ }
128
+
129
+ private doesPolicyMatchRule(
130
+ policy: IncomingCallPolicy,
131
+ rule: IncomingCallPolicyLabelRule,
132
+ ): boolean {
133
+ if (
134
+ rule.incomingCallPolicyLabels &&
135
+ rule.incomingCallPolicyLabels.length > 0
136
+ ) {
137
+ if (!policy.labels || policy.labels.length === 0) {
138
+ return false;
139
+ }
140
+ const ruleLabelIds: Array<string> = rule.incomingCallPolicyLabels.map(
141
+ (l: Label) => {
142
+ return l.id?.toString() || "";
143
+ },
144
+ );
145
+ const labelIds: Array<string> = policy.labels.map((l: Label) => {
146
+ return l.id?.toString() || "";
147
+ });
148
+ if (
149
+ !ruleLabelIds.some((id: string) => {
150
+ return labelIds.includes(id);
151
+ })
152
+ ) {
153
+ return false;
154
+ }
155
+ }
156
+
157
+ if (
158
+ rule.incomingCallPolicyNamePattern &&
159
+ (!policy.name ||
160
+ !this.testRegex(rule.incomingCallPolicyNamePattern, policy.name, rule))
161
+ ) {
162
+ return false;
163
+ }
164
+
165
+ if (
166
+ rule.incomingCallPolicyDescriptionPattern &&
167
+ (!policy.description ||
168
+ !this.testRegex(
169
+ rule.incomingCallPolicyDescriptionPattern,
170
+ policy.description,
171
+ rule,
172
+ ))
173
+ ) {
174
+ return false;
175
+ }
176
+
177
+ return true;
178
+ }
179
+
180
+ private testRegex(
181
+ pattern: string,
182
+ value: string,
183
+ rule: IncomingCallPolicyLabelRule,
184
+ ): boolean {
185
+ try {
186
+ const regex: RegExp = new RegExp(pattern, "i");
187
+ return regex.test(value);
188
+ } catch {
189
+ logger.warn(
190
+ `Invalid regex in incoming call policy label rule ${rule.id}: ${pattern}`,
191
+ );
192
+ return false;
193
+ }
194
+ }
195
+ }
196
+
197
+ export default new IncomingCallPolicyLabelRuleEngineServiceClass();
@@ -0,0 +1,10 @@
1
+ import DatabaseService from "./DatabaseService";
2
+ import Model from "../../Models/DatabaseModels/IncomingCallPolicyLabelRule";
3
+
4
+ export class Service extends DatabaseService<Model> {
5
+ public constructor() {
6
+ super(Model);
7
+ }
8
+ }
9
+
10
+ export default new Service();
@@ -0,0 +1,220 @@
1
+ import Label from "../../Models/DatabaseModels/Label";
2
+ import IncomingCallPolicy from "../../Models/DatabaseModels/IncomingCallPolicy";
3
+ import IncomingCallPolicyOwnerRule from "../../Models/DatabaseModels/IncomingCallPolicyOwnerRule";
4
+ import IncomingCallPolicyOwnerUser from "../../Models/DatabaseModels/IncomingCallPolicyOwnerUser";
5
+ import IncomingCallPolicyOwnerTeam from "../../Models/DatabaseModels/IncomingCallPolicyOwnerTeam";
6
+ import IncomingCallPolicyOwnerRuleService from "./IncomingCallPolicyOwnerRuleService";
7
+ import IncomingCallPolicyOwnerUserService from "./IncomingCallPolicyOwnerUserService";
8
+ import IncomingCallPolicyOwnerTeamService from "./IncomingCallPolicyOwnerTeamService";
9
+ import IncomingCallPolicyService from "./IncomingCallPolicyService";
10
+ import ObjectID from "../../Types/ObjectID";
11
+ import CaptureSpan from "../Utils/Telemetry/CaptureSpan";
12
+ import logger, { LogAttributes } from "../Utils/Logger";
13
+
14
+ class IncomingCallPolicyOwnerRuleEngineServiceClass {
15
+ @CaptureSpan()
16
+ public async applyRulesToIncomingCallPolicy(
17
+ policy: IncomingCallPolicy,
18
+ ): Promise<void> {
19
+ if (!policy.id || !policy.projectId) {
20
+ return;
21
+ }
22
+
23
+ try {
24
+ const rules: Array<IncomingCallPolicyOwnerRule> =
25
+ await IncomingCallPolicyOwnerRuleService.findBy({
26
+ query: {
27
+ projectId: policy.projectId,
28
+ isEnabled: true,
29
+ },
30
+ props: { isRoot: true },
31
+ select: {
32
+ _id: true,
33
+ name: true,
34
+ notifyOwners: true,
35
+ incomingCallPolicyLabels: { _id: true },
36
+ incomingCallPolicyNamePattern: true,
37
+ incomingCallPolicyDescriptionPattern: true,
38
+ ownerUsers: { _id: true },
39
+ ownerTeams: { _id: true },
40
+ },
41
+ limit: 100,
42
+ skip: 0,
43
+ });
44
+
45
+ if (rules.length === 0) {
46
+ return;
47
+ }
48
+
49
+ const policyWithDetails: IncomingCallPolicy | null =
50
+ await IncomingCallPolicyService.findOneById({
51
+ id: policy.id,
52
+ select: {
53
+ name: true,
54
+ description: true,
55
+ labels: { _id: true },
56
+ },
57
+ props: { isRoot: true },
58
+ });
59
+
60
+ if (!policyWithDetails) {
61
+ return;
62
+ }
63
+
64
+ const usersByNotify: Map<boolean, Set<string>> = new Map([
65
+ [true, new Set()],
66
+ [false, new Set()],
67
+ ]);
68
+ const teamsByNotify: Map<boolean, Set<string>> = new Map([
69
+ [true, new Set()],
70
+ [false, new Set()],
71
+ ]);
72
+
73
+ const matchedRules: Array<IncomingCallPolicyOwnerRule> = [];
74
+
75
+ for (const rule of rules) {
76
+ const matches: boolean = this.doesPolicyMatchRule(
77
+ policyWithDetails,
78
+ rule,
79
+ );
80
+ if (!matches) {
81
+ continue;
82
+ }
83
+ let ruleAddedAny: boolean = false;
84
+ const notify: boolean = rule.notifyOwners !== false;
85
+ for (const user of rule.ownerUsers || []) {
86
+ if (user.id) {
87
+ usersByNotify.get(notify)!.add(user.id.toString());
88
+ ruleAddedAny = true;
89
+ }
90
+ }
91
+ for (const team of rule.ownerTeams || []) {
92
+ if (team.id) {
93
+ teamsByNotify.get(notify)!.add(team.id.toString());
94
+ ruleAddedAny = true;
95
+ }
96
+ }
97
+ if (ruleAddedAny) {
98
+ matchedRules.push(rule);
99
+ }
100
+ }
101
+
102
+ if (matchedRules.length === 0) {
103
+ return;
104
+ }
105
+
106
+ for (const notify of [true, false]) {
107
+ const userIds: Set<string> = usersByNotify.get(notify)!;
108
+ const teamIds: Set<string> = teamsByNotify.get(notify)!;
109
+
110
+ for (const userId of userIds) {
111
+ const owner: IncomingCallPolicyOwnerUser =
112
+ new IncomingCallPolicyOwnerUser();
113
+ owner.incomingCallPolicyId = policy.id;
114
+ owner.projectId = policy.projectId;
115
+ owner.userId = new ObjectID(userId);
116
+ owner.isOwnerNotified = !notify;
117
+ await IncomingCallPolicyOwnerUserService.create({
118
+ data: owner,
119
+ props: { isRoot: true },
120
+ });
121
+ }
122
+
123
+ for (const teamId of teamIds) {
124
+ const owner: IncomingCallPolicyOwnerTeam =
125
+ new IncomingCallPolicyOwnerTeam();
126
+ owner.incomingCallPolicyId = policy.id;
127
+ owner.projectId = policy.projectId;
128
+ owner.teamId = new ObjectID(teamId);
129
+ owner.isOwnerNotified = !notify;
130
+ await IncomingCallPolicyOwnerTeamService.create({
131
+ data: owner,
132
+ props: { isRoot: true },
133
+ });
134
+ }
135
+ }
136
+
137
+ logger.debug(
138
+ `IncomingCallPolicyOwnerRuleEngine added owners to policy ${policy.id}`,
139
+ { projectId: policy.projectId.toString() } as LogAttributes,
140
+ );
141
+ } catch (error) {
142
+ logger.error(
143
+ `Error applying incoming call policy owner rules: ${error}`,
144
+ {
145
+ projectId: policy.projectId?.toString(),
146
+ incomingCallPolicyId: policy.id?.toString(),
147
+ } as LogAttributes,
148
+ );
149
+ }
150
+ }
151
+
152
+ private doesPolicyMatchRule(
153
+ policy: IncomingCallPolicy,
154
+ rule: IncomingCallPolicyOwnerRule,
155
+ ): boolean {
156
+ if (
157
+ rule.incomingCallPolicyLabels &&
158
+ rule.incomingCallPolicyLabels.length > 0
159
+ ) {
160
+ if (!policy.labels || policy.labels.length === 0) {
161
+ return false;
162
+ }
163
+ const ruleLabelIds: Array<string> = rule.incomingCallPolicyLabels.map(
164
+ (l: Label) => {
165
+ return l.id?.toString() || "";
166
+ },
167
+ );
168
+ const labelIds: Array<string> = policy.labels.map((l: Label) => {
169
+ return l.id?.toString() || "";
170
+ });
171
+ if (
172
+ !ruleLabelIds.some((id: string) => {
173
+ return labelIds.includes(id);
174
+ })
175
+ ) {
176
+ return false;
177
+ }
178
+ }
179
+
180
+ if (
181
+ rule.incomingCallPolicyNamePattern &&
182
+ (!policy.name ||
183
+ !this.testRegex(rule.incomingCallPolicyNamePattern, policy.name, rule))
184
+ ) {
185
+ return false;
186
+ }
187
+
188
+ if (
189
+ rule.incomingCallPolicyDescriptionPattern &&
190
+ (!policy.description ||
191
+ !this.testRegex(
192
+ rule.incomingCallPolicyDescriptionPattern,
193
+ policy.description,
194
+ rule,
195
+ ))
196
+ ) {
197
+ return false;
198
+ }
199
+
200
+ return true;
201
+ }
202
+
203
+ private testRegex(
204
+ pattern: string,
205
+ value: string,
206
+ rule: IncomingCallPolicyOwnerRule,
207
+ ): boolean {
208
+ try {
209
+ const regex: RegExp = new RegExp(pattern, "i");
210
+ return regex.test(value);
211
+ } catch {
212
+ logger.warn(
213
+ `Invalid regex in incoming call policy owner rule ${rule.id}: ${pattern}`,
214
+ );
215
+ return false;
216
+ }
217
+ }
218
+ }
219
+
220
+ export default new IncomingCallPolicyOwnerRuleEngineServiceClass();
@@ -0,0 +1,10 @@
1
+ import DatabaseService from "./DatabaseService";
2
+ import Model from "../../Models/DatabaseModels/IncomingCallPolicyOwnerRule";
3
+
4
+ export class Service extends DatabaseService<Model> {
5
+ public constructor() {
6
+ super(Model);
7
+ }
8
+ }
9
+
10
+ export default new Service();
@@ -0,0 +1,9 @@
1
+ import DatabaseService from "./DatabaseService";
2
+ import Model from "../../Models/DatabaseModels/IncomingCallPolicyOwnerTeam";
3
+
4
+ export class Service extends DatabaseService<Model> {
5
+ public constructor() {
6
+ super(Model);
7
+ }
8
+ }
9
+ export default new Service();
@@ -0,0 +1,9 @@
1
+ import DatabaseService from "./DatabaseService";
2
+ import Model from "../../Models/DatabaseModels/IncomingCallPolicyOwnerUser";
3
+
4
+ export class Service extends DatabaseService<Model> {
5
+ public constructor() {
6
+ super(Model);
7
+ }
8
+ }
9
+ export default new Service();
@@ -1,10 +1,45 @@
1
1
  import DatabaseService from "./DatabaseService";
2
2
  import IncomingCallPolicy from "../../Models/DatabaseModels/IncomingCallPolicy";
3
+ import IncomingCallPolicyLabelRuleEngineService from "./IncomingCallPolicyLabelRuleEngineService";
4
+ import IncomingCallPolicyOwnerRuleEngineService from "./IncomingCallPolicyOwnerRuleEngineService";
5
+ import { OnCreate } from "../Types/Database/Hooks";
6
+ import CaptureSpan from "../Utils/Telemetry/CaptureSpan";
7
+ import logger, { LogAttributes } from "../Utils/Logger";
3
8
 
4
9
  export class Service extends DatabaseService<IncomingCallPolicy> {
5
10
  public constructor() {
6
11
  super(IncomingCallPolicy);
7
12
  }
13
+
14
+ @CaptureSpan()
15
+ protected override async onCreateSuccess(
16
+ _onCreate: OnCreate<IncomingCallPolicy>,
17
+ createdItem: IncomingCallPolicy,
18
+ ): Promise<IncomingCallPolicy> {
19
+ if (createdItem.projectId && createdItem.id) {
20
+ Promise.resolve()
21
+ .then(async () => {
22
+ await IncomingCallPolicyLabelRuleEngineService.applyRulesToIncomingCallPolicy(
23
+ createdItem,
24
+ );
25
+ })
26
+ .then(async () => {
27
+ await IncomingCallPolicyOwnerRuleEngineService.applyRulesToIncomingCallPolicy(
28
+ createdItem,
29
+ );
30
+ })
31
+ .catch((error: Error) => {
32
+ logger.error(
33
+ `Error applying incoming call policy rules in IncomingCallPolicyService.onCreateSuccess: ${error}`,
34
+ {
35
+ projectId: createdItem.projectId?.toString(),
36
+ incomingCallPolicyId: createdItem.id?.toString(),
37
+ } as LogAttributes,
38
+ );
39
+ });
40
+ }
41
+ return createdItem;
42
+ }
8
43
  }
9
44
 
10
45
  export default new Service();