@webex/calling 3.0.0-next.2 → 3.0.0-next.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 (283) hide show
  1. package/CHANGELOG.md +1403 -0
  2. package/README.md +74 -74
  3. package/dist/module/CallHistory/CallHistory.js +81 -81
  4. package/dist/module/CallHistory/callHistoryFixtures.js +307 -307
  5. package/dist/module/CallHistory/constants.js +9 -9
  6. package/dist/module/CallHistory/types.js +1 -1
  7. package/dist/module/CallSettings/CallSettings.js +65 -65
  8. package/dist/module/CallSettings/UcmBackendConnector.js +100 -100
  9. package/dist/module/CallSettings/WxCallBackendConnector.js +287 -287
  10. package/dist/module/CallSettings/constants.js +11 -11
  11. package/dist/module/CallSettings/testFixtures.js +62 -62
  12. package/dist/module/CallSettings/types.js +1 -1
  13. package/dist/module/CallingClient/CallingClient.js +248 -248
  14. package/dist/module/CallingClient/callRecordFixtures.js +93 -93
  15. package/dist/module/CallingClient/calling/CallerId/index.js +169 -169
  16. package/dist/module/CallingClient/calling/CallerId/types.js +1 -1
  17. package/dist/module/CallingClient/calling/call.js +1658 -1658
  18. package/dist/module/CallingClient/calling/callManager.js +284 -284
  19. package/dist/module/CallingClient/calling/index.js +2 -2
  20. package/dist/module/CallingClient/calling/types.js +53 -53
  21. package/dist/module/CallingClient/callingClientFixtures.js +38 -38
  22. package/dist/module/CallingClient/constants.js +122 -122
  23. package/dist/module/CallingClient/line/index.js +158 -158
  24. package/dist/module/CallingClient/line/types.js +15 -15
  25. package/dist/module/CallingClient/registration/index.js +1 -1
  26. package/dist/module/CallingClient/registration/register.js +507 -507
  27. package/dist/module/CallingClient/registration/registerFixtures.js +28 -28
  28. package/dist/module/CallingClient/registration/types.js +1 -1
  29. package/dist/module/CallingClient/types.js +1 -1
  30. package/dist/module/Contacts/ContactsClient.js +484 -484
  31. package/dist/module/Contacts/constants.js +20 -20
  32. package/dist/module/Contacts/contactFixtures.js +284 -284
  33. package/dist/module/Contacts/types.js +10 -10
  34. package/dist/module/Errors/catalog/CallError.js +26 -26
  35. package/dist/module/Errors/catalog/CallingDeviceError.js +18 -18
  36. package/dist/module/Errors/catalog/ExtendedError.js +10 -10
  37. package/dist/module/Errors/catalog/LineError.js +24 -24
  38. package/dist/module/Errors/index.js +2 -2
  39. package/dist/module/Errors/types.js +48 -48
  40. package/dist/module/Events/impl/index.js +19 -19
  41. package/dist/module/Events/types.js +83 -83
  42. package/dist/module/Logger/index.js +114 -114
  43. package/dist/module/Logger/types.js +25 -25
  44. package/dist/module/Metrics/index.js +232 -232
  45. package/dist/module/Metrics/types.js +37 -37
  46. package/dist/module/SDKConnector/index.js +39 -39
  47. package/dist/module/SDKConnector/types.js +1 -1
  48. package/dist/module/SDKConnector/utils.js +12 -12
  49. package/dist/module/Voicemail/BroadworksBackendConnector.js +291 -291
  50. package/dist/module/Voicemail/UcmBackendConnector.js +279 -279
  51. package/dist/module/Voicemail/Voicemail.js +110 -110
  52. package/dist/module/Voicemail/WxCallBackendConnector.js +279 -279
  53. package/dist/module/Voicemail/constants.js +29 -29
  54. package/dist/module/Voicemail/types.js +1 -1
  55. package/dist/module/Voicemail/voicemailFixture.js +510 -510
  56. package/dist/module/api.js +6 -6
  57. package/dist/module/common/Utils.js +802 -802
  58. package/dist/module/common/constants.js +41 -41
  59. package/dist/module/common/index.js +1 -1
  60. package/dist/module/common/testUtil.js +938 -938
  61. package/dist/module/common/types.js +57 -57
  62. package/dist/module/index.js +8 -8
  63. package/dist/types/CallHistory/CallHistory.d.ts +17 -17
  64. package/dist/types/CallHistory/callHistoryFixtures.d.ts +94 -94
  65. package/dist/types/CallHistory/constants.d.ts +9 -9
  66. package/dist/types/CallHistory/types.d.ts +18 -18
  67. package/dist/types/CallSettings/CallSettings.d.ts +19 -19
  68. package/dist/types/CallSettings/UcmBackendConnector.d.ts +19 -19
  69. package/dist/types/CallSettings/WxCallBackendConnector.d.ts +21 -21
  70. package/dist/types/CallSettings/constants.d.ts +11 -11
  71. package/dist/types/CallSettings/testFixtures.d.ts +15 -15
  72. package/dist/types/CallSettings/types.d.ts +107 -107
  73. package/dist/types/CallingClient/CallingClient.d.ts +35 -35
  74. package/dist/types/CallingClient/callRecordFixtures.d.ts +3 -3
  75. package/dist/types/CallingClient/calling/CallerId/index.d.ts +17 -17
  76. package/dist/types/CallingClient/calling/CallerId/types.d.ts +41 -41
  77. package/dist/types/CallingClient/calling/call.d.ts +96 -96
  78. package/dist/types/CallingClient/calling/callManager.d.ts +25 -25
  79. package/dist/types/CallingClient/calling/index.d.ts +2 -2
  80. package/dist/types/CallingClient/calling/types.d.ts +206 -206
  81. package/dist/types/CallingClient/callingClientFixtures.d.ts +18 -18
  82. package/dist/types/CallingClient/constants.d.ts +122 -122
  83. package/dist/types/CallingClient/line/index.d.ts +43 -43
  84. package/dist/types/CallingClient/line/types.d.ts +55 -55
  85. package/dist/types/CallingClient/registration/index.d.ts +1 -1
  86. package/dist/types/CallingClient/registration/register.d.ts +64 -64
  87. package/dist/types/CallingClient/registration/registerFixtures.d.ts +28 -28
  88. package/dist/types/CallingClient/registration/types.d.ts +20 -20
  89. package/dist/types/CallingClient/types.d.ts +29 -29
  90. package/dist/types/Contacts/ContactsClient.d.ts +26 -26
  91. package/dist/types/Contacts/constants.d.ts +19 -19
  92. package/dist/types/Contacts/contactFixtures.d.ts +280 -280
  93. package/dist/types/Contacts/types.d.ts +72 -72
  94. package/dist/types/Errors/catalog/CallError.d.ts +11 -11
  95. package/dist/types/Errors/catalog/CallingDeviceError.d.ts +10 -10
  96. package/dist/types/Errors/catalog/ExtendedError.d.ts +6 -6
  97. package/dist/types/Errors/catalog/LineError.d.ts +10 -10
  98. package/dist/types/Errors/index.d.ts +2 -2
  99. package/dist/types/Errors/types.d.ts +60 -60
  100. package/dist/types/Events/impl/index.d.ts +8 -8
  101. package/dist/types/Events/types.d.ts +288 -288
  102. package/dist/types/Logger/index.d.ts +12 -12
  103. package/dist/types/Logger/types.d.ts +25 -25
  104. package/dist/types/Metrics/index.d.ts +5 -5
  105. package/dist/types/Metrics/types.d.ts +42 -42
  106. package/dist/types/SDKConnector/index.d.ts +12 -12
  107. package/dist/types/SDKConnector/types.d.ts +128 -128
  108. package/dist/types/SDKConnector/utils.d.ts +5 -5
  109. package/dist/types/Voicemail/BroadworksBackendConnector.d.ts +27 -27
  110. package/dist/types/Voicemail/UcmBackendConnector.d.ts +34 -34
  111. package/dist/types/Voicemail/Voicemail.d.ts +27 -27
  112. package/dist/types/Voicemail/WxCallBackendConnector.d.ts +23 -23
  113. package/dist/types/Voicemail/constants.d.ts +29 -29
  114. package/dist/types/Voicemail/types.d.ts +133 -133
  115. package/dist/types/Voicemail/voicemailFixture.d.ts +417 -417
  116. package/dist/types/api.d.ts +16 -16
  117. package/dist/types/common/Utils.d.ts +34 -34
  118. package/dist/types/common/constants.d.ts +41 -41
  119. package/dist/types/common/index.d.ts +1 -1
  120. package/dist/types/common/testUtil.d.ts +3611 -3611
  121. package/dist/types/common/types.d.ts +191 -191
  122. package/dist/types/index.d.ts +8 -8
  123. package/package.json +4 -4
  124. package/dist/CallHistory/CallHistory.js +0 -203
  125. package/dist/CallHistory/CallHistory.js.map +0 -1
  126. package/dist/CallHistory/CallHistory.test.js +0 -133
  127. package/dist/CallHistory/CallHistory.test.js.map +0 -1
  128. package/dist/CallHistory/callHistoryFixtures.js +0 -310
  129. package/dist/CallHistory/callHistoryFixtures.js.map +0 -1
  130. package/dist/CallHistory/constants.js +0 -26
  131. package/dist/CallHistory/constants.js.map +0 -1
  132. package/dist/CallHistory/types.js +0 -7
  133. package/dist/CallHistory/types.js.map +0 -1
  134. package/dist/CallSettings/CallSettings.js +0 -276
  135. package/dist/CallSettings/CallSettings.js.map +0 -1
  136. package/dist/CallSettings/CallSettings.test.js +0 -122
  137. package/dist/CallSettings/CallSettings.test.js.map +0 -1
  138. package/dist/CallSettings/UcmBackendConnector.js +0 -217
  139. package/dist/CallSettings/UcmBackendConnector.js.map +0 -1
  140. package/dist/CallSettings/UcmBackendConnector.test.js +0 -227
  141. package/dist/CallSettings/UcmBackendConnector.test.js.map +0 -1
  142. package/dist/CallSettings/WxCallBackendConnector.js +0 -541
  143. package/dist/CallSettings/WxCallBackendConnector.js.map +0 -1
  144. package/dist/CallSettings/WxCallBackendConnector.test.js +0 -802
  145. package/dist/CallSettings/WxCallBackendConnector.test.js.map +0 -1
  146. package/dist/CallSettings/constants.js +0 -30
  147. package/dist/CallSettings/constants.js.map +0 -1
  148. package/dist/CallSettings/testFixtures.js +0 -72
  149. package/dist/CallSettings/testFixtures.js.map +0 -1
  150. package/dist/CallSettings/types.js +0 -7
  151. package/dist/CallSettings/types.js.map +0 -1
  152. package/dist/CallingClient/CallingClient.js +0 -572
  153. package/dist/CallingClient/CallingClient.js.map +0 -1
  154. package/dist/CallingClient/CallingClient.test.js +0 -931
  155. package/dist/CallingClient/CallingClient.test.js.map +0 -1
  156. package/dist/CallingClient/callRecordFixtures.js +0 -103
  157. package/dist/CallingClient/callRecordFixtures.js.map +0 -1
  158. package/dist/CallingClient/calling/CallerId/index.js +0 -287
  159. package/dist/CallingClient/calling/CallerId/index.js.map +0 -1
  160. package/dist/CallingClient/calling/CallerId/index.test.js +0 -274
  161. package/dist/CallingClient/calling/CallerId/index.test.js.map +0 -1
  162. package/dist/CallingClient/calling/CallerId/types.js +0 -7
  163. package/dist/CallingClient/calling/CallerId/types.js.map +0 -1
  164. package/dist/CallingClient/calling/call.js +0 -2995
  165. package/dist/CallingClient/calling/call.js.map +0 -1
  166. package/dist/CallingClient/calling/call.test.js +0 -2614
  167. package/dist/CallingClient/calling/call.test.js.map +0 -1
  168. package/dist/CallingClient/calling/callManager.js +0 -443
  169. package/dist/CallingClient/calling/callManager.js.map +0 -1
  170. package/dist/CallingClient/calling/callManager.test.js +0 -683
  171. package/dist/CallingClient/calling/callManager.test.js.map +0 -1
  172. package/dist/CallingClient/calling/index.js +0 -30
  173. package/dist/CallingClient/calling/index.js.map +0 -1
  174. package/dist/CallingClient/calling/types.js +0 -71
  175. package/dist/CallingClient/calling/types.js.map +0 -1
  176. package/dist/CallingClient/callingClientFixtures.js +0 -53
  177. package/dist/CallingClient/callingClientFixtures.js.map +0 -1
  178. package/dist/CallingClient/constants.js +0 -226
  179. package/dist/CallingClient/constants.js.map +0 -1
  180. package/dist/CallingClient/line/index.js +0 -312
  181. package/dist/CallingClient/line/index.js.map +0 -1
  182. package/dist/CallingClient/line/line.test.js +0 -270
  183. package/dist/CallingClient/line/line.test.js.map +0 -1
  184. package/dist/CallingClient/line/types.js +0 -25
  185. package/dist/CallingClient/line/types.js.map +0 -1
  186. package/dist/CallingClient/registration/index.js +0 -19
  187. package/dist/CallingClient/registration/index.js.map +0 -1
  188. package/dist/CallingClient/registration/register.js +0 -1295
  189. package/dist/CallingClient/registration/register.js.map +0 -1
  190. package/dist/CallingClient/registration/register.test.js +0 -838
  191. package/dist/CallingClient/registration/register.test.js.map +0 -1
  192. package/dist/CallingClient/registration/registerFixtures.js +0 -41
  193. package/dist/CallingClient/registration/registerFixtures.js.map +0 -1
  194. package/dist/CallingClient/registration/types.js +0 -7
  195. package/dist/CallingClient/registration/types.js.map +0 -1
  196. package/dist/CallingClient/types.js +0 -7
  197. package/dist/CallingClient/types.js.map +0 -1
  198. package/dist/Contacts/ContactsClient.js +0 -1139
  199. package/dist/Contacts/ContactsClient.js.map +0 -1
  200. package/dist/Contacts/ContactsClient.test.js +0 -725
  201. package/dist/Contacts/ContactsClient.test.js.map +0 -1
  202. package/dist/Contacts/constants.js +0 -36
  203. package/dist/Contacts/constants.js.map +0 -1
  204. package/dist/Contacts/contactFixtures.js +0 -340
  205. package/dist/Contacts/contactFixtures.js.map +0 -1
  206. package/dist/Contacts/types.js +0 -20
  207. package/dist/Contacts/types.js.map +0 -1
  208. package/dist/Errors/catalog/CallError.js +0 -94
  209. package/dist/Errors/catalog/CallError.js.map +0 -1
  210. package/dist/Errors/catalog/CallingDeviceError.js +0 -88
  211. package/dist/Errors/catalog/CallingDeviceError.js.map +0 -1
  212. package/dist/Errors/catalog/ExtendedError.js +0 -45
  213. package/dist/Errors/catalog/ExtendedError.js.map +0 -1
  214. package/dist/Errors/catalog/LineError.js +0 -90
  215. package/dist/Errors/catalog/LineError.js.map +0 -1
  216. package/dist/Errors/index.js +0 -21
  217. package/dist/Errors/index.js.map +0 -1
  218. package/dist/Errors/types.js +0 -62
  219. package/dist/Errors/types.js.map +0 -1
  220. package/dist/Events/impl/index.js +0 -82
  221. package/dist/Events/impl/index.js.map +0 -1
  222. package/dist/Events/types.js +0 -111
  223. package/dist/Events/types.js.map +0 -1
  224. package/dist/Logger/index.js +0 -221
  225. package/dist/Logger/index.js.map +0 -1
  226. package/dist/Logger/index.test.js +0 -62
  227. package/dist/Logger/index.test.js.map +0 -1
  228. package/dist/Logger/types.js +0 -37
  229. package/dist/Logger/types.js.map +0 -1
  230. package/dist/Metrics/index.js +0 -329
  231. package/dist/Metrics/index.js.map +0 -1
  232. package/dist/Metrics/index.test.js +0 -291
  233. package/dist/Metrics/index.test.js.map +0 -1
  234. package/dist/Metrics/types.js +0 -50
  235. package/dist/Metrics/types.js.map +0 -1
  236. package/dist/SDKConnector/index.js +0 -105
  237. package/dist/SDKConnector/index.js.map +0 -1
  238. package/dist/SDKConnector/index.test.js +0 -9
  239. package/dist/SDKConnector/index.test.js.map +0 -1
  240. package/dist/SDKConnector/types.js +0 -7
  241. package/dist/SDKConnector/types.js.map +0 -1
  242. package/dist/SDKConnector/utils.js +0 -40
  243. package/dist/SDKConnector/utils.js.map +0 -1
  244. package/dist/SDKConnector/utils.test.js +0 -9
  245. package/dist/SDKConnector/utils.test.js.map +0 -1
  246. package/dist/Voicemail/BroadworksBackendConnector.js +0 -632
  247. package/dist/Voicemail/BroadworksBackendConnector.js.map +0 -1
  248. package/dist/Voicemail/BroadworksBackendConnector.test.js +0 -733
  249. package/dist/Voicemail/BroadworksBackendConnector.test.js.map +0 -1
  250. package/dist/Voicemail/UcmBackendConnector.js +0 -566
  251. package/dist/Voicemail/UcmBackendConnector.js.map +0 -1
  252. package/dist/Voicemail/UcmBackendConnector.test.js +0 -628
  253. package/dist/Voicemail/UcmBackendConnector.test.js.map +0 -1
  254. package/dist/Voicemail/Voicemail.js +0 -378
  255. package/dist/Voicemail/Voicemail.js.map +0 -1
  256. package/dist/Voicemail/Voicemail.test.js +0 -226
  257. package/dist/Voicemail/Voicemail.test.js.map +0 -1
  258. package/dist/Voicemail/WxCallBackendConnector.js +0 -540
  259. package/dist/Voicemail/WxCallBackendConnector.js.map +0 -1
  260. package/dist/Voicemail/WxCallBackendConnector.test.js +0 -942
  261. package/dist/Voicemail/WxCallBackendConnector.test.js.map +0 -1
  262. package/dist/Voicemail/constants.js +0 -66
  263. package/dist/Voicemail/constants.js.map +0 -1
  264. package/dist/Voicemail/types.js +0 -7
  265. package/dist/Voicemail/types.js.map +0 -1
  266. package/dist/Voicemail/voicemailFixture.js +0 -559
  267. package/dist/Voicemail/voicemailFixture.js.map +0 -1
  268. package/dist/api.js +0 -134
  269. package/dist/api.js.map +0 -1
  270. package/dist/common/Utils.js +0 -1264
  271. package/dist/common/Utils.js.map +0 -1
  272. package/dist/common/Utils.test.js +0 -1227
  273. package/dist/common/Utils.test.js.map +0 -1
  274. package/dist/common/constants.js +0 -90
  275. package/dist/common/constants.js.map +0 -1
  276. package/dist/common/index.js +0 -19
  277. package/dist/common/index.js.map +0 -1
  278. package/dist/common/testUtil.js +0 -968
  279. package/dist/common/testUtil.js.map +0 -1
  280. package/dist/common/types.js +0 -75
  281. package/dist/common/types.js.map +0 -1
  282. package/dist/index.js +0 -63
  283. package/dist/index.js.map +0 -1
@@ -1,484 +1,484 @@
1
- import { FAILURE_MESSAGE, STATUS_CODE, SUCCESS_MESSAGE } from '../common/constants';
2
- import { HTTP_METHODS } from '../common/types';
3
- import SDKConnector from '../SDKConnector';
4
- import log from '../Logger';
5
- import { CONTACTS_FILE, CONTACTS_SCHEMA, CONTACT_FILTER, DEFAULT_GROUP_NAME, ENCRYPT_FILTER, GROUP_FILTER, USERS, encryptedFields, } from './constants';
6
- import { ContactType, GroupType, } from './types';
7
- import { serviceErrorCodeHandler } from '../common/Utils';
8
- export class ContactsClient {
9
- sdkConnector;
10
- encryptionKeyUrl;
11
- webex;
12
- groups;
13
- contacts;
14
- defaultGroupId;
15
- constructor(webex, logger) {
16
- this.sdkConnector = SDKConnector;
17
- if (!this.sdkConnector.getWebex()) {
18
- SDKConnector.setWebex(webex);
19
- }
20
- this.webex = this.sdkConnector.getWebex();
21
- this.encryptionKeyUrl = '';
22
- this.groups = undefined;
23
- this.contacts = undefined;
24
- this.defaultGroupId = '';
25
- log.setLogger(logger.level, CONTACTS_FILE);
26
- }
27
- async decryptContactDetail(encryptionKeyUrl, contactDetails) {
28
- const decryptedContactDetail = [...contactDetails];
29
- const decryptedValues = await Promise.all(decryptedContactDetail.map((detail) => this.webex.internal.encryption.decryptText(encryptionKeyUrl, detail.value)));
30
- decryptedValues.forEach((decryptedValue, index) => {
31
- decryptedContactDetail[index].value = decryptedValue;
32
- });
33
- return decryptedContactDetail;
34
- }
35
- async encryptContactDetail(encryptionKeyUrl, contactDetails) {
36
- const encryptedContactDetail = [...contactDetails];
37
- const encryptedValues = await Promise.all(encryptedContactDetail.map((detail) => this.webex.internal.encryption.encryptText(encryptionKeyUrl, detail.value)));
38
- encryptedValues.forEach((encryptedValue, index) => {
39
- encryptedContactDetail[index].value = encryptedValue;
40
- });
41
- return encryptedContactDetail;
42
- }
43
- async encryptContact(contact) {
44
- const { encryptionKeyUrl } = contact;
45
- const encryptedContact = { ...contact };
46
- const encryptionPromises = Object.values(encryptedFields).map(async (field) => {
47
- switch (field) {
48
- case encryptedFields.ADDRESS_INFO: {
49
- const plaintextAddressInfo = encryptedContact.addressInfo;
50
- let encryptedAddressInfo;
51
- if (plaintextAddressInfo) {
52
- const encryptedAddressInfoPromises = Object.entries(plaintextAddressInfo).map(async ([key, value]) => [
53
- key,
54
- await this.webex.internal.encryption.encryptText(encryptionKeyUrl, value),
55
- ]);
56
- encryptedAddressInfo = Object.fromEntries(await Promise.all(encryptedAddressInfoPromises));
57
- }
58
- return [field, encryptedAddressInfo];
59
- }
60
- case encryptedFields.EMAILS:
61
- case encryptedFields.PHONE_NUMBERS:
62
- case encryptedFields.SIP_ADDRESSES: {
63
- const plainTextDetails = encryptedContact[field];
64
- let encryptedDetails;
65
- if (plainTextDetails) {
66
- encryptedDetails = await this.encryptContactDetail(encryptionKeyUrl, plainTextDetails);
67
- }
68
- return [field, encryptedDetails];
69
- }
70
- default: {
71
- let encryptedValue;
72
- if (Object.values(encryptedFields).includes(field) && encryptedContact[field]) {
73
- encryptedValue = await this.webex.internal.encryption.encryptText(encryptionKeyUrl, encryptedContact[field]);
74
- }
75
- return [field, encryptedValue];
76
- }
77
- }
78
- });
79
- const encryptedFieldsList = await Promise.all(encryptionPromises);
80
- encryptedFieldsList.forEach(([field, value]) => {
81
- if (value !== undefined) {
82
- encryptedContact[field] = value;
83
- }
84
- });
85
- return encryptedContact;
86
- }
87
- async decryptContact(contact) {
88
- const { encryptionKeyUrl } = contact;
89
- const decryptedContact = { ...contact };
90
- const decryptionPromises = Object.values(encryptedFields).map(async (field) => {
91
- switch (field) {
92
- case encryptedFields.ADDRESS_INFO: {
93
- const plaintextAddressInfo = decryptedContact.addressInfo;
94
- let decryptedAddressInfo;
95
- if (plaintextAddressInfo) {
96
- const decryptedAddressInfoPromises = Object.entries(plaintextAddressInfo).map(async ([key, value]) => [
97
- key,
98
- await this.webex.internal.encryption.decryptText(encryptionKeyUrl, value),
99
- ]);
100
- decryptedAddressInfo = Object.fromEntries(await Promise.all(decryptedAddressInfoPromises));
101
- }
102
- return [field, decryptedAddressInfo];
103
- }
104
- case encryptedFields.EMAILS:
105
- case encryptedFields.PHONE_NUMBERS:
106
- case encryptedFields.SIP_ADDRESSES: {
107
- const plainTextDetails = decryptedContact[field];
108
- let decryptedDetails;
109
- if (plainTextDetails) {
110
- decryptedDetails = await this.decryptContactDetail(encryptionKeyUrl, plainTextDetails);
111
- }
112
- return [field, decryptedDetails];
113
- }
114
- default: {
115
- let decryptedValue;
116
- if (Object.values(encryptedFields).includes(field) && decryptedContact[field]) {
117
- decryptedValue = await this.webex.internal.encryption.decryptText(encryptionKeyUrl, decryptedContact[field]);
118
- }
119
- return [field, decryptedValue];
120
- }
121
- }
122
- });
123
- const decryptedFieldsList = await Promise.all(decryptionPromises);
124
- decryptedFieldsList.forEach(([field, value]) => {
125
- if (value !== undefined) {
126
- decryptedContact[field] = value;
127
- }
128
- });
129
- return decryptedContact;
130
- }
131
- async fetchContactFromDSS(contactsDataMap) {
132
- const contactList = [];
133
- const dssResult = await this.webex.internal.dss.lookup({ ids: Object.keys(contactsDataMap) });
134
- for (let i = 0; i < dssResult.length; i += 1) {
135
- const contact = dssResult[i];
136
- const contactId = contact.identity;
137
- const { displayName, emails, phoneNumbers, sipAddresses, photos } = contact;
138
- const { department, firstName, identityManager, jobTitle, lastName } = contact.additionalInfo;
139
- const manager = identityManager && identityManager.displayName ? identityManager.displayName : undefined;
140
- const { contactType, avatarUrlDomain, encryptionKeyUrl, ownerId, groups } = contactsDataMap[contactId];
141
- let avatarURL = '';
142
- if (photos.length) {
143
- avatarURL = photos[0].value;
144
- }
145
- const addedPhoneNumbers = contactsDataMap[contactId].phoneNumbers;
146
- if (addedPhoneNumbers) {
147
- const decryptedPhoneNumbers = await this.decryptContactDetail(encryptionKeyUrl, addedPhoneNumbers);
148
- decryptedPhoneNumbers.forEach((number) => phoneNumbers.push(number));
149
- }
150
- const addedSipAddresses = contactsDataMap[contactId].sipAddresses;
151
- if (addedSipAddresses) {
152
- const decryptedSipAddresses = await this.decryptContactDetail(encryptionKeyUrl, addedSipAddresses);
153
- decryptedSipAddresses.forEach((address) => sipAddresses.push(address));
154
- }
155
- const cloudContact = {
156
- avatarUrlDomain,
157
- avatarURL,
158
- contactId,
159
- contactType,
160
- department,
161
- displayName,
162
- emails,
163
- encryptionKeyUrl,
164
- firstName,
165
- groups,
166
- lastName,
167
- manager,
168
- ownerId,
169
- phoneNumbers,
170
- sipAddresses,
171
- title: jobTitle,
172
- };
173
- contactList.push(cloudContact);
174
- }
175
- return contactList;
176
- }
177
- async getContacts() {
178
- const loggerContext = {
179
- file: CONTACTS_FILE,
180
- method: 'getContacts',
181
- };
182
- const contactList = [];
183
- const contactsDataMap = {};
184
- try {
185
- const response = await this.webex.request({
186
- uri: `${this.webex.internal.services._serviceUrls.contactsService}/${ENCRYPT_FILTER}/${USERS}/${CONTACT_FILTER}`,
187
- method: HTTP_METHODS.GET,
188
- });
189
- const responseBody = response.body;
190
- if (!responseBody) {
191
- throw new Error(`${response}`);
192
- }
193
- const { contacts, groups } = responseBody;
194
- for (let i = 0; i < contacts.length; i += 1) {
195
- const contact = contacts[i];
196
- if (contact.contactType === ContactType.CUSTOM) {
197
- const decryptedContact = await this.decryptContact(contact);
198
- contactList.push(decryptedContact);
199
- }
200
- else if (contact.contactType === ContactType.CLOUD && contact.contactId) {
201
- contactsDataMap[contact.contactId] = contact;
202
- }
203
- }
204
- if (Object.keys(contactsDataMap).length) {
205
- const cloudContacts = await this.fetchContactFromDSS(contactsDataMap);
206
- contactList.push(...cloudContacts);
207
- }
208
- await Promise.all(groups.map(async (group, idx) => {
209
- groups[idx].displayName = await this.webex.internal.encryption.decryptText(group.encryptionKeyUrl, group.displayName);
210
- }));
211
- this.groups = groups;
212
- this.contacts = contactList;
213
- const contactResponse = {
214
- statusCode: Number(response[STATUS_CODE]),
215
- data: {
216
- contacts: contactList,
217
- groups,
218
- },
219
- message: SUCCESS_MESSAGE,
220
- };
221
- return contactResponse;
222
- }
223
- catch (err) {
224
- const errorInfo = err;
225
- const errorStatus = serviceErrorCodeHandler(errorInfo, loggerContext);
226
- return errorStatus;
227
- }
228
- }
229
- async createNewEncryptionKeyUrl() {
230
- const loggerContext = {
231
- file: CONTACTS_FILE,
232
- method: this.createNewEncryptionKeyUrl.name,
233
- };
234
- let unboundedKeyUri = '';
235
- log.info('Requesting kms for a new KRO and key', loggerContext);
236
- const unboundedKeys = await this.webex.internal.encryption.kms.createUnboundKeys({ count: 1 });
237
- unboundedKeyUri = unboundedKeys[0].uri;
238
- this.webex.internal.encryption.kms.createResource({ keyUris: [unboundedKeyUri] });
239
- return unboundedKeyUri;
240
- }
241
- async fetchEncryptionKeyUrl() {
242
- if (this.encryptionKeyUrl) {
243
- return this.encryptionKeyUrl;
244
- }
245
- if (this.groups === undefined) {
246
- this.getContacts();
247
- }
248
- if (this.groups && this.groups.length) {
249
- return this.groups[0].encryptionKeyUrl;
250
- }
251
- this.encryptionKeyUrl = await this.createNewEncryptionKeyUrl();
252
- log.info(`Creating a default group: ${DEFAULT_GROUP_NAME}`, {
253
- file: CONTACTS_FILE,
254
- method: this.fetchEncryptionKeyUrl.name,
255
- });
256
- const response = await this.createContactGroup(DEFAULT_GROUP_NAME, this.encryptionKeyUrl);
257
- if (response.data.group?.groupId) {
258
- this.defaultGroupId = response.data.group?.groupId;
259
- }
260
- return this.encryptionKeyUrl;
261
- }
262
- async fetchDefaultGroup() {
263
- if (this.defaultGroupId) {
264
- return this.defaultGroupId;
265
- }
266
- if (this.groups && this.groups.length) {
267
- for (let i = 0; i < this.groups.length; i += 1) {
268
- if (this.groups[i].displayName === DEFAULT_GROUP_NAME) {
269
- this.defaultGroupId = this.groups[i].groupId;
270
- return this.defaultGroupId;
271
- }
272
- }
273
- }
274
- log.info('No default group found.', {
275
- file: CONTACTS_FILE,
276
- method: this.fetchDefaultGroup.name,
277
- });
278
- const response = await this.createContactGroup(DEFAULT_GROUP_NAME);
279
- const { group } = response.data;
280
- if (group) {
281
- return group.groupId;
282
- }
283
- return '';
284
- }
285
- async createContactGroup(displayName, encryptionKeyUrl, groupType) {
286
- const loggerContext = {
287
- file: CONTACTS_FILE,
288
- method: this.createContactGroup.name,
289
- };
290
- log.info(`Creating contact group ${displayName}`, loggerContext);
291
- const encryptionKeyUrlFinal = encryptionKeyUrl || (await this.fetchEncryptionKeyUrl());
292
- if (this.groups === undefined) {
293
- await this.getContacts();
294
- }
295
- if (this.groups && this.groups.length) {
296
- const isExistingGroup = this.groups.find((group) => {
297
- return group.displayName === displayName;
298
- });
299
- if (isExistingGroup) {
300
- log.warn(`Group name ${displayName} already exists.`, loggerContext);
301
- return {
302
- statusCode: 400,
303
- data: { error: 'Group displayName already exists' },
304
- message: FAILURE_MESSAGE,
305
- };
306
- }
307
- }
308
- const encryptedDisplayName = await this.webex.internal.encryption.encryptText(encryptionKeyUrlFinal, displayName);
309
- const groupInfo = {
310
- schemas: CONTACTS_SCHEMA,
311
- displayName: encryptedDisplayName,
312
- groupType: groupType || GroupType.NORMAL,
313
- encryptionKeyUrl: encryptionKeyUrlFinal,
314
- };
315
- try {
316
- const response = await this.webex.request({
317
- uri: `${this.webex.internal.services._serviceUrls.contactsService}/${ENCRYPT_FILTER}/${USERS}/${GROUP_FILTER}`,
318
- method: HTTP_METHODS.POST,
319
- body: groupInfo,
320
- });
321
- const group = response.body;
322
- group.displayName = displayName;
323
- const contactResponse = {
324
- statusCode: Number(response[STATUS_CODE]),
325
- data: {
326
- group,
327
- },
328
- message: SUCCESS_MESSAGE,
329
- };
330
- this.groups?.push(group);
331
- return contactResponse;
332
- }
333
- catch (err) {
334
- log.warn('Unable to create contact group.', loggerContext);
335
- const errorInfo = err;
336
- const errorStatus = serviceErrorCodeHandler(errorInfo, loggerContext);
337
- return errorStatus;
338
- }
339
- }
340
- async deleteContactGroup(groupId) {
341
- const loggerContext = {
342
- file: CONTACTS_FILE,
343
- method: this.deleteContactGroup.name,
344
- };
345
- try {
346
- log.info(`Deleting contact group: ${groupId}`, loggerContext);
347
- const response = await this.webex.request({
348
- uri: `${this.webex.internal.services._serviceUrls.contactsService}/${ENCRYPT_FILTER}/${USERS}/${GROUP_FILTER}/${groupId}`,
349
- method: HTTP_METHODS.DELETE,
350
- });
351
- const contactResponse = {
352
- statusCode: Number(response[STATUS_CODE]),
353
- data: {},
354
- message: SUCCESS_MESSAGE,
355
- };
356
- const groupToDelete = this.groups?.findIndex((group) => group.groupId === groupId);
357
- if (groupToDelete !== undefined && groupToDelete !== -1) {
358
- this.groups?.splice(groupToDelete, 1);
359
- }
360
- if (!this.groups?.length) {
361
- this.defaultGroupId = '';
362
- }
363
- return contactResponse;
364
- }
365
- catch (err) {
366
- log.warn(`Unable to delete contact group ${groupId}`, loggerContext);
367
- const errorInfo = err;
368
- const errorStatus = serviceErrorCodeHandler(errorInfo, loggerContext);
369
- return errorStatus;
370
- }
371
- }
372
- async createContact(contactInfo) {
373
- const loggerContext = {
374
- file: CONTACTS_FILE,
375
- method: this.createContact.name,
376
- };
377
- log.info(`Request to create contact: contactType: ${contactInfo.contactType}`, loggerContext);
378
- try {
379
- const contact = { ...contactInfo };
380
- if (!contact.encryptionKeyUrl) {
381
- contact.encryptionKeyUrl = await this.fetchEncryptionKeyUrl();
382
- }
383
- if (!contact.groups || contact.groups.length === 0) {
384
- const defaultGroupId = await this.fetchDefaultGroup();
385
- contact.groups = [defaultGroupId];
386
- }
387
- contact.schemas = CONTACTS_SCHEMA;
388
- let requestBody = {};
389
- switch (contact.contactType) {
390
- case ContactType.CUSTOM: {
391
- const encryptedContact = await this.encryptContact(contact);
392
- requestBody = encryptedContact;
393
- break;
394
- }
395
- case ContactType.CLOUD: {
396
- if (!contact.contactId) {
397
- return {
398
- statusCode: 400,
399
- data: {
400
- error: 'contactId is required for contactType:CLOUD.',
401
- },
402
- message: FAILURE_MESSAGE,
403
- };
404
- }
405
- const encryptedContact = await this.encryptContact(contact);
406
- requestBody = encryptedContact;
407
- break;
408
- }
409
- default: {
410
- return {
411
- statusCode: 400,
412
- data: {
413
- error: 'Unknown contactType received.',
414
- },
415
- message: FAILURE_MESSAGE,
416
- };
417
- }
418
- }
419
- const response = await this.webex.request({
420
- uri: `${this.webex.internal.services._serviceUrls.contactsService}/${ENCRYPT_FILTER}/${USERS}/${CONTACT_FILTER}`,
421
- method: HTTP_METHODS.POST,
422
- body: requestBody,
423
- });
424
- const newContact = response.body;
425
- contact.contactId = newContact.contactId;
426
- const contactResponse = {
427
- statusCode: Number(response[STATUS_CODE]),
428
- data: {
429
- contact,
430
- },
431
- message: SUCCESS_MESSAGE,
432
- };
433
- if (contact.contactType === ContactType.CLOUD) {
434
- const decryptedContacts = await this.fetchContactFromDSS(Object.fromEntries([[newContact.contactId, newContact]]));
435
- if (decryptedContacts.length && decryptedContacts[0]) {
436
- this.contacts?.push(decryptedContacts[0]);
437
- }
438
- }
439
- else {
440
- this.contacts?.push(contact);
441
- }
442
- return contactResponse;
443
- }
444
- catch (err) {
445
- log.warn('Failed to create contact.', {
446
- file: CONTACTS_FILE,
447
- method: this.createContact.name,
448
- });
449
- const errorInfo = err;
450
- const errorStatus = serviceErrorCodeHandler(errorInfo, loggerContext);
451
- return errorStatus;
452
- }
453
- }
454
- async deleteContact(contactId) {
455
- const loggerContext = {
456
- file: CONTACTS_FILE,
457
- method: this.deleteContact.name,
458
- };
459
- try {
460
- log.info(`Deleting contact : ${contactId}`, loggerContext);
461
- const response = await this.webex.request({
462
- uri: `${this.webex.internal.services._serviceUrls.contactsService}/${ENCRYPT_FILTER}/${USERS}/${CONTACT_FILTER}/${contactId}`,
463
- method: HTTP_METHODS.DELETE,
464
- });
465
- const contactResponse = {
466
- statusCode: Number(response[STATUS_CODE]),
467
- data: {},
468
- message: SUCCESS_MESSAGE,
469
- };
470
- const contactToDelete = this.contacts?.findIndex((contact) => contact.contactId === contactId);
471
- if (contactToDelete !== undefined && contactToDelete !== -1) {
472
- this.contacts?.splice(contactToDelete, 1);
473
- }
474
- return contactResponse;
475
- }
476
- catch (err) {
477
- log.warn(`Unable to delete contact ${contactId}`, loggerContext);
478
- const errorInfo = err;
479
- const errorStatus = serviceErrorCodeHandler(errorInfo, loggerContext);
480
- return errorStatus;
481
- }
482
- }
483
- }
484
- export const createContactsClient = (webex, logger) => new ContactsClient(webex, logger);
1
+ import { FAILURE_MESSAGE, STATUS_CODE, SUCCESS_MESSAGE } from '../common/constants';
2
+ import { HTTP_METHODS } from '../common/types';
3
+ import SDKConnector from '../SDKConnector';
4
+ import log from '../Logger';
5
+ import { CONTACTS_FILE, CONTACTS_SCHEMA, CONTACT_FILTER, DEFAULT_GROUP_NAME, ENCRYPT_FILTER, GROUP_FILTER, USERS, encryptedFields, } from './constants';
6
+ import { ContactType, GroupType, } from './types';
7
+ import { serviceErrorCodeHandler } from '../common/Utils';
8
+ export class ContactsClient {
9
+ sdkConnector;
10
+ encryptionKeyUrl;
11
+ webex;
12
+ groups;
13
+ contacts;
14
+ defaultGroupId;
15
+ constructor(webex, logger) {
16
+ this.sdkConnector = SDKConnector;
17
+ if (!this.sdkConnector.getWebex()) {
18
+ SDKConnector.setWebex(webex);
19
+ }
20
+ this.webex = this.sdkConnector.getWebex();
21
+ this.encryptionKeyUrl = '';
22
+ this.groups = undefined;
23
+ this.contacts = undefined;
24
+ this.defaultGroupId = '';
25
+ log.setLogger(logger.level, CONTACTS_FILE);
26
+ }
27
+ async decryptContactDetail(encryptionKeyUrl, contactDetails) {
28
+ const decryptedContactDetail = [...contactDetails];
29
+ const decryptedValues = await Promise.all(decryptedContactDetail.map((detail) => this.webex.internal.encryption.decryptText(encryptionKeyUrl, detail.value)));
30
+ decryptedValues.forEach((decryptedValue, index) => {
31
+ decryptedContactDetail[index].value = decryptedValue;
32
+ });
33
+ return decryptedContactDetail;
34
+ }
35
+ async encryptContactDetail(encryptionKeyUrl, contactDetails) {
36
+ const encryptedContactDetail = [...contactDetails];
37
+ const encryptedValues = await Promise.all(encryptedContactDetail.map((detail) => this.webex.internal.encryption.encryptText(encryptionKeyUrl, detail.value)));
38
+ encryptedValues.forEach((encryptedValue, index) => {
39
+ encryptedContactDetail[index].value = encryptedValue;
40
+ });
41
+ return encryptedContactDetail;
42
+ }
43
+ async encryptContact(contact) {
44
+ const { encryptionKeyUrl } = contact;
45
+ const encryptedContact = { ...contact };
46
+ const encryptionPromises = Object.values(encryptedFields).map(async (field) => {
47
+ switch (field) {
48
+ case encryptedFields.ADDRESS_INFO: {
49
+ const plaintextAddressInfo = encryptedContact.addressInfo;
50
+ let encryptedAddressInfo;
51
+ if (plaintextAddressInfo) {
52
+ const encryptedAddressInfoPromises = Object.entries(plaintextAddressInfo).map(async ([key, value]) => [
53
+ key,
54
+ await this.webex.internal.encryption.encryptText(encryptionKeyUrl, value),
55
+ ]);
56
+ encryptedAddressInfo = Object.fromEntries(await Promise.all(encryptedAddressInfoPromises));
57
+ }
58
+ return [field, encryptedAddressInfo];
59
+ }
60
+ case encryptedFields.EMAILS:
61
+ case encryptedFields.PHONE_NUMBERS:
62
+ case encryptedFields.SIP_ADDRESSES: {
63
+ const plainTextDetails = encryptedContact[field];
64
+ let encryptedDetails;
65
+ if (plainTextDetails) {
66
+ encryptedDetails = await this.encryptContactDetail(encryptionKeyUrl, plainTextDetails);
67
+ }
68
+ return [field, encryptedDetails];
69
+ }
70
+ default: {
71
+ let encryptedValue;
72
+ if (Object.values(encryptedFields).includes(field) && encryptedContact[field]) {
73
+ encryptedValue = await this.webex.internal.encryption.encryptText(encryptionKeyUrl, encryptedContact[field]);
74
+ }
75
+ return [field, encryptedValue];
76
+ }
77
+ }
78
+ });
79
+ const encryptedFieldsList = await Promise.all(encryptionPromises);
80
+ encryptedFieldsList.forEach(([field, value]) => {
81
+ if (value !== undefined) {
82
+ encryptedContact[field] = value;
83
+ }
84
+ });
85
+ return encryptedContact;
86
+ }
87
+ async decryptContact(contact) {
88
+ const { encryptionKeyUrl } = contact;
89
+ const decryptedContact = { ...contact };
90
+ const decryptionPromises = Object.values(encryptedFields).map(async (field) => {
91
+ switch (field) {
92
+ case encryptedFields.ADDRESS_INFO: {
93
+ const plaintextAddressInfo = decryptedContact.addressInfo;
94
+ let decryptedAddressInfo;
95
+ if (plaintextAddressInfo) {
96
+ const decryptedAddressInfoPromises = Object.entries(plaintextAddressInfo).map(async ([key, value]) => [
97
+ key,
98
+ await this.webex.internal.encryption.decryptText(encryptionKeyUrl, value),
99
+ ]);
100
+ decryptedAddressInfo = Object.fromEntries(await Promise.all(decryptedAddressInfoPromises));
101
+ }
102
+ return [field, decryptedAddressInfo];
103
+ }
104
+ case encryptedFields.EMAILS:
105
+ case encryptedFields.PHONE_NUMBERS:
106
+ case encryptedFields.SIP_ADDRESSES: {
107
+ const plainTextDetails = decryptedContact[field];
108
+ let decryptedDetails;
109
+ if (plainTextDetails) {
110
+ decryptedDetails = await this.decryptContactDetail(encryptionKeyUrl, plainTextDetails);
111
+ }
112
+ return [field, decryptedDetails];
113
+ }
114
+ default: {
115
+ let decryptedValue;
116
+ if (Object.values(encryptedFields).includes(field) && decryptedContact[field]) {
117
+ decryptedValue = await this.webex.internal.encryption.decryptText(encryptionKeyUrl, decryptedContact[field]);
118
+ }
119
+ return [field, decryptedValue];
120
+ }
121
+ }
122
+ });
123
+ const decryptedFieldsList = await Promise.all(decryptionPromises);
124
+ decryptedFieldsList.forEach(([field, value]) => {
125
+ if (value !== undefined) {
126
+ decryptedContact[field] = value;
127
+ }
128
+ });
129
+ return decryptedContact;
130
+ }
131
+ async fetchContactFromDSS(contactsDataMap) {
132
+ const contactList = [];
133
+ const dssResult = await this.webex.internal.dss.lookup({ ids: Object.keys(contactsDataMap) });
134
+ for (let i = 0; i < dssResult.length; i += 1) {
135
+ const contact = dssResult[i];
136
+ const contactId = contact.identity;
137
+ const { displayName, emails, phoneNumbers, sipAddresses, photos } = contact;
138
+ const { department, firstName, identityManager, jobTitle, lastName } = contact.additionalInfo;
139
+ const manager = identityManager && identityManager.displayName ? identityManager.displayName : undefined;
140
+ const { contactType, avatarUrlDomain, encryptionKeyUrl, ownerId, groups } = contactsDataMap[contactId];
141
+ let avatarURL = '';
142
+ if (photos.length) {
143
+ avatarURL = photos[0].value;
144
+ }
145
+ const addedPhoneNumbers = contactsDataMap[contactId].phoneNumbers;
146
+ if (addedPhoneNumbers) {
147
+ const decryptedPhoneNumbers = await this.decryptContactDetail(encryptionKeyUrl, addedPhoneNumbers);
148
+ decryptedPhoneNumbers.forEach((number) => phoneNumbers.push(number));
149
+ }
150
+ const addedSipAddresses = contactsDataMap[contactId].sipAddresses;
151
+ if (addedSipAddresses) {
152
+ const decryptedSipAddresses = await this.decryptContactDetail(encryptionKeyUrl, addedSipAddresses);
153
+ decryptedSipAddresses.forEach((address) => sipAddresses.push(address));
154
+ }
155
+ const cloudContact = {
156
+ avatarUrlDomain,
157
+ avatarURL,
158
+ contactId,
159
+ contactType,
160
+ department,
161
+ displayName,
162
+ emails,
163
+ encryptionKeyUrl,
164
+ firstName,
165
+ groups,
166
+ lastName,
167
+ manager,
168
+ ownerId,
169
+ phoneNumbers,
170
+ sipAddresses,
171
+ title: jobTitle,
172
+ };
173
+ contactList.push(cloudContact);
174
+ }
175
+ return contactList;
176
+ }
177
+ async getContacts() {
178
+ const loggerContext = {
179
+ file: CONTACTS_FILE,
180
+ method: 'getContacts',
181
+ };
182
+ const contactList = [];
183
+ const contactsDataMap = {};
184
+ try {
185
+ const response = await this.webex.request({
186
+ uri: `${this.webex.internal.services._serviceUrls.contactsService}/${ENCRYPT_FILTER}/${USERS}/${CONTACT_FILTER}`,
187
+ method: HTTP_METHODS.GET,
188
+ });
189
+ const responseBody = response.body;
190
+ if (!responseBody) {
191
+ throw new Error(`${response}`);
192
+ }
193
+ const { contacts, groups } = responseBody;
194
+ for (let i = 0; i < contacts.length; i += 1) {
195
+ const contact = contacts[i];
196
+ if (contact.contactType === ContactType.CUSTOM) {
197
+ const decryptedContact = await this.decryptContact(contact);
198
+ contactList.push(decryptedContact);
199
+ }
200
+ else if (contact.contactType === ContactType.CLOUD && contact.contactId) {
201
+ contactsDataMap[contact.contactId] = contact;
202
+ }
203
+ }
204
+ if (Object.keys(contactsDataMap).length) {
205
+ const cloudContacts = await this.fetchContactFromDSS(contactsDataMap);
206
+ contactList.push(...cloudContacts);
207
+ }
208
+ await Promise.all(groups.map(async (group, idx) => {
209
+ groups[idx].displayName = await this.webex.internal.encryption.decryptText(group.encryptionKeyUrl, group.displayName);
210
+ }));
211
+ this.groups = groups;
212
+ this.contacts = contactList;
213
+ const contactResponse = {
214
+ statusCode: Number(response[STATUS_CODE]),
215
+ data: {
216
+ contacts: contactList,
217
+ groups,
218
+ },
219
+ message: SUCCESS_MESSAGE,
220
+ };
221
+ return contactResponse;
222
+ }
223
+ catch (err) {
224
+ const errorInfo = err;
225
+ const errorStatus = serviceErrorCodeHandler(errorInfo, loggerContext);
226
+ return errorStatus;
227
+ }
228
+ }
229
+ async createNewEncryptionKeyUrl() {
230
+ const loggerContext = {
231
+ file: CONTACTS_FILE,
232
+ method: this.createNewEncryptionKeyUrl.name,
233
+ };
234
+ let unboundedKeyUri = '';
235
+ log.info('Requesting kms for a new KRO and key', loggerContext);
236
+ const unboundedKeys = await this.webex.internal.encryption.kms.createUnboundKeys({ count: 1 });
237
+ unboundedKeyUri = unboundedKeys[0].uri;
238
+ this.webex.internal.encryption.kms.createResource({ keyUris: [unboundedKeyUri] });
239
+ return unboundedKeyUri;
240
+ }
241
+ async fetchEncryptionKeyUrl() {
242
+ if (this.encryptionKeyUrl) {
243
+ return this.encryptionKeyUrl;
244
+ }
245
+ if (this.groups === undefined) {
246
+ this.getContacts();
247
+ }
248
+ if (this.groups && this.groups.length) {
249
+ return this.groups[0].encryptionKeyUrl;
250
+ }
251
+ this.encryptionKeyUrl = await this.createNewEncryptionKeyUrl();
252
+ log.info(`Creating a default group: ${DEFAULT_GROUP_NAME}`, {
253
+ file: CONTACTS_FILE,
254
+ method: this.fetchEncryptionKeyUrl.name,
255
+ });
256
+ const response = await this.createContactGroup(DEFAULT_GROUP_NAME, this.encryptionKeyUrl);
257
+ if (response.data.group?.groupId) {
258
+ this.defaultGroupId = response.data.group?.groupId;
259
+ }
260
+ return this.encryptionKeyUrl;
261
+ }
262
+ async fetchDefaultGroup() {
263
+ if (this.defaultGroupId) {
264
+ return this.defaultGroupId;
265
+ }
266
+ if (this.groups && this.groups.length) {
267
+ for (let i = 0; i < this.groups.length; i += 1) {
268
+ if (this.groups[i].displayName === DEFAULT_GROUP_NAME) {
269
+ this.defaultGroupId = this.groups[i].groupId;
270
+ return this.defaultGroupId;
271
+ }
272
+ }
273
+ }
274
+ log.info('No default group found.', {
275
+ file: CONTACTS_FILE,
276
+ method: this.fetchDefaultGroup.name,
277
+ });
278
+ const response = await this.createContactGroup(DEFAULT_GROUP_NAME);
279
+ const { group } = response.data;
280
+ if (group) {
281
+ return group.groupId;
282
+ }
283
+ return '';
284
+ }
285
+ async createContactGroup(displayName, encryptionKeyUrl, groupType) {
286
+ const loggerContext = {
287
+ file: CONTACTS_FILE,
288
+ method: this.createContactGroup.name,
289
+ };
290
+ log.info(`Creating contact group ${displayName}`, loggerContext);
291
+ const encryptionKeyUrlFinal = encryptionKeyUrl || (await this.fetchEncryptionKeyUrl());
292
+ if (this.groups === undefined) {
293
+ await this.getContacts();
294
+ }
295
+ if (this.groups && this.groups.length) {
296
+ const isExistingGroup = this.groups.find((group) => {
297
+ return group.displayName === displayName;
298
+ });
299
+ if (isExistingGroup) {
300
+ log.warn(`Group name ${displayName} already exists.`, loggerContext);
301
+ return {
302
+ statusCode: 400,
303
+ data: { error: 'Group displayName already exists' },
304
+ message: FAILURE_MESSAGE,
305
+ };
306
+ }
307
+ }
308
+ const encryptedDisplayName = await this.webex.internal.encryption.encryptText(encryptionKeyUrlFinal, displayName);
309
+ const groupInfo = {
310
+ schemas: CONTACTS_SCHEMA,
311
+ displayName: encryptedDisplayName,
312
+ groupType: groupType || GroupType.NORMAL,
313
+ encryptionKeyUrl: encryptionKeyUrlFinal,
314
+ };
315
+ try {
316
+ const response = await this.webex.request({
317
+ uri: `${this.webex.internal.services._serviceUrls.contactsService}/${ENCRYPT_FILTER}/${USERS}/${GROUP_FILTER}`,
318
+ method: HTTP_METHODS.POST,
319
+ body: groupInfo,
320
+ });
321
+ const group = response.body;
322
+ group.displayName = displayName;
323
+ const contactResponse = {
324
+ statusCode: Number(response[STATUS_CODE]),
325
+ data: {
326
+ group,
327
+ },
328
+ message: SUCCESS_MESSAGE,
329
+ };
330
+ this.groups?.push(group);
331
+ return contactResponse;
332
+ }
333
+ catch (err) {
334
+ log.warn('Unable to create contact group.', loggerContext);
335
+ const errorInfo = err;
336
+ const errorStatus = serviceErrorCodeHandler(errorInfo, loggerContext);
337
+ return errorStatus;
338
+ }
339
+ }
340
+ async deleteContactGroup(groupId) {
341
+ const loggerContext = {
342
+ file: CONTACTS_FILE,
343
+ method: this.deleteContactGroup.name,
344
+ };
345
+ try {
346
+ log.info(`Deleting contact group: ${groupId}`, loggerContext);
347
+ const response = await this.webex.request({
348
+ uri: `${this.webex.internal.services._serviceUrls.contactsService}/${ENCRYPT_FILTER}/${USERS}/${GROUP_FILTER}/${groupId}`,
349
+ method: HTTP_METHODS.DELETE,
350
+ });
351
+ const contactResponse = {
352
+ statusCode: Number(response[STATUS_CODE]),
353
+ data: {},
354
+ message: SUCCESS_MESSAGE,
355
+ };
356
+ const groupToDelete = this.groups?.findIndex((group) => group.groupId === groupId);
357
+ if (groupToDelete !== undefined && groupToDelete !== -1) {
358
+ this.groups?.splice(groupToDelete, 1);
359
+ }
360
+ if (!this.groups?.length) {
361
+ this.defaultGroupId = '';
362
+ }
363
+ return contactResponse;
364
+ }
365
+ catch (err) {
366
+ log.warn(`Unable to delete contact group ${groupId}`, loggerContext);
367
+ const errorInfo = err;
368
+ const errorStatus = serviceErrorCodeHandler(errorInfo, loggerContext);
369
+ return errorStatus;
370
+ }
371
+ }
372
+ async createContact(contactInfo) {
373
+ const loggerContext = {
374
+ file: CONTACTS_FILE,
375
+ method: this.createContact.name,
376
+ };
377
+ log.info(`Request to create contact: contactType: ${contactInfo.contactType}`, loggerContext);
378
+ try {
379
+ const contact = { ...contactInfo };
380
+ if (!contact.encryptionKeyUrl) {
381
+ contact.encryptionKeyUrl = await this.fetchEncryptionKeyUrl();
382
+ }
383
+ if (!contact.groups || contact.groups.length === 0) {
384
+ const defaultGroupId = await this.fetchDefaultGroup();
385
+ contact.groups = [defaultGroupId];
386
+ }
387
+ contact.schemas = CONTACTS_SCHEMA;
388
+ let requestBody = {};
389
+ switch (contact.contactType) {
390
+ case ContactType.CUSTOM: {
391
+ const encryptedContact = await this.encryptContact(contact);
392
+ requestBody = encryptedContact;
393
+ break;
394
+ }
395
+ case ContactType.CLOUD: {
396
+ if (!contact.contactId) {
397
+ return {
398
+ statusCode: 400,
399
+ data: {
400
+ error: 'contactId is required for contactType:CLOUD.',
401
+ },
402
+ message: FAILURE_MESSAGE,
403
+ };
404
+ }
405
+ const encryptedContact = await this.encryptContact(contact);
406
+ requestBody = encryptedContact;
407
+ break;
408
+ }
409
+ default: {
410
+ return {
411
+ statusCode: 400,
412
+ data: {
413
+ error: 'Unknown contactType received.',
414
+ },
415
+ message: FAILURE_MESSAGE,
416
+ };
417
+ }
418
+ }
419
+ const response = await this.webex.request({
420
+ uri: `${this.webex.internal.services._serviceUrls.contactsService}/${ENCRYPT_FILTER}/${USERS}/${CONTACT_FILTER}`,
421
+ method: HTTP_METHODS.POST,
422
+ body: requestBody,
423
+ });
424
+ const newContact = response.body;
425
+ contact.contactId = newContact.contactId;
426
+ const contactResponse = {
427
+ statusCode: Number(response[STATUS_CODE]),
428
+ data: {
429
+ contact,
430
+ },
431
+ message: SUCCESS_MESSAGE,
432
+ };
433
+ if (contact.contactType === ContactType.CLOUD) {
434
+ const decryptedContacts = await this.fetchContactFromDSS(Object.fromEntries([[newContact.contactId, newContact]]));
435
+ if (decryptedContacts.length && decryptedContacts[0]) {
436
+ this.contacts?.push(decryptedContacts[0]);
437
+ }
438
+ }
439
+ else {
440
+ this.contacts?.push(contact);
441
+ }
442
+ return contactResponse;
443
+ }
444
+ catch (err) {
445
+ log.warn('Failed to create contact.', {
446
+ file: CONTACTS_FILE,
447
+ method: this.createContact.name,
448
+ });
449
+ const errorInfo = err;
450
+ const errorStatus = serviceErrorCodeHandler(errorInfo, loggerContext);
451
+ return errorStatus;
452
+ }
453
+ }
454
+ async deleteContact(contactId) {
455
+ const loggerContext = {
456
+ file: CONTACTS_FILE,
457
+ method: this.deleteContact.name,
458
+ };
459
+ try {
460
+ log.info(`Deleting contact : ${contactId}`, loggerContext);
461
+ const response = await this.webex.request({
462
+ uri: `${this.webex.internal.services._serviceUrls.contactsService}/${ENCRYPT_FILTER}/${USERS}/${CONTACT_FILTER}/${contactId}`,
463
+ method: HTTP_METHODS.DELETE,
464
+ });
465
+ const contactResponse = {
466
+ statusCode: Number(response[STATUS_CODE]),
467
+ data: {},
468
+ message: SUCCESS_MESSAGE,
469
+ };
470
+ const contactToDelete = this.contacts?.findIndex((contact) => contact.contactId === contactId);
471
+ if (contactToDelete !== undefined && contactToDelete !== -1) {
472
+ this.contacts?.splice(contactToDelete, 1);
473
+ }
474
+ return contactResponse;
475
+ }
476
+ catch (err) {
477
+ log.warn(`Unable to delete contact ${contactId}`, loggerContext);
478
+ const errorInfo = err;
479
+ const errorStatus = serviceErrorCodeHandler(errorInfo, loggerContext);
480
+ return errorStatus;
481
+ }
482
+ }
483
+ }
484
+ export const createContactsClient = (webex, logger) => new ContactsClient(webex, logger);