@app-connect/core 1.7.24 → 1.7.26

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 (137) hide show
  1. package/.env.test +5 -5
  2. package/README.md +441 -441
  3. package/connector/developerPortal.js +31 -42
  4. package/connector/mock.js +84 -77
  5. package/connector/proxy/engine.js +164 -163
  6. package/connector/proxy/index.js +500 -500
  7. package/connector/registry.js +252 -252
  8. package/docs/README.md +50 -50
  9. package/docs/architecture.md +93 -93
  10. package/docs/connectors.md +116 -117
  11. package/docs/handlers.md +125 -125
  12. package/docs/libraries.md +101 -101
  13. package/docs/models.md +144 -144
  14. package/docs/routes.md +115 -115
  15. package/docs/tests.md +73 -73
  16. package/handlers/admin.js +523 -523
  17. package/handlers/appointment.js +193 -0
  18. package/handlers/auth.js +296 -296
  19. package/handlers/calldown.js +99 -99
  20. package/handlers/contact.js +280 -280
  21. package/handlers/disposition.js +82 -80
  22. package/handlers/log.js +984 -973
  23. package/handlers/managedAuth.js +446 -446
  24. package/handlers/plugin.js +208 -208
  25. package/handlers/user.js +142 -142
  26. package/index.js +3140 -2652
  27. package/jest.config.js +56 -56
  28. package/lib/analytics.js +54 -54
  29. package/lib/authSession.js +109 -109
  30. package/lib/cacheCleanup.js +21 -0
  31. package/lib/callLogComposer.js +898 -898
  32. package/lib/callLogLookup.js +34 -0
  33. package/lib/constants.js +8 -8
  34. package/lib/debugTracer.js +177 -177
  35. package/lib/encode.js +30 -30
  36. package/lib/errorHandler.js +218 -206
  37. package/lib/generalErrorMessage.js +41 -41
  38. package/lib/jwt.js +18 -18
  39. package/lib/logger.js +190 -190
  40. package/lib/migrateCallLogsSchema.js +116 -0
  41. package/lib/ringcentral.js +266 -266
  42. package/lib/s3ErrorLogReport.js +65 -65
  43. package/lib/sharedSMSComposer.js +471 -471
  44. package/lib/util.js +67 -67
  45. package/mcp/README.md +412 -395
  46. package/mcp/lib/validator.js +91 -91
  47. package/mcp/mcpHandler.js +425 -425
  48. package/mcp/tools/cancelAppointment.js +101 -0
  49. package/mcp/tools/checkAuthStatus.js +105 -105
  50. package/mcp/tools/confirmAppointment.js +101 -0
  51. package/mcp/tools/createAppointment.js +157 -0
  52. package/mcp/tools/createCallLog.js +327 -316
  53. package/mcp/tools/createContact.js +117 -117
  54. package/mcp/tools/createMessageLog.js +287 -287
  55. package/mcp/tools/doAuth.js +60 -60
  56. package/mcp/tools/findContactByName.js +93 -93
  57. package/mcp/tools/findContactByPhone.js +101 -101
  58. package/mcp/tools/getCallLog.js +111 -102
  59. package/mcp/tools/getGoogleFilePicker.js +99 -99
  60. package/mcp/tools/getHelp.js +43 -43
  61. package/mcp/tools/getPublicConnectors.js +94 -94
  62. package/mcp/tools/getSessionInfo.js +90 -90
  63. package/mcp/tools/index.js +51 -41
  64. package/mcp/tools/listAppointments.js +163 -0
  65. package/mcp/tools/logout.js +96 -96
  66. package/mcp/tools/rcGetCallLogs.js +65 -65
  67. package/mcp/tools/updateAppointment.js +154 -0
  68. package/mcp/tools/updateCallLog.js +130 -126
  69. package/mcp/ui/App/App.tsx +358 -358
  70. package/mcp/ui/App/components/AuthInfoForm.tsx +113 -113
  71. package/mcp/ui/App/components/AuthSuccess.tsx +22 -22
  72. package/mcp/ui/App/components/ConnectorList.tsx +82 -82
  73. package/mcp/ui/App/components/DebugPanel.tsx +43 -43
  74. package/mcp/ui/App/components/OAuthConnect.tsx +270 -270
  75. package/mcp/ui/App/lib/callTool.ts +130 -130
  76. package/mcp/ui/App/lib/debugLog.ts +41 -41
  77. package/mcp/ui/App/lib/developerPortal.ts +111 -111
  78. package/mcp/ui/App/main.css +5 -5
  79. package/mcp/ui/App/root.tsx +13 -13
  80. package/mcp/ui/index.html +13 -13
  81. package/mcp/ui/package-lock.json +6356 -6356
  82. package/mcp/ui/package.json +25 -25
  83. package/mcp/ui/tsconfig.json +26 -26
  84. package/mcp/ui/vite.config.ts +16 -16
  85. package/models/accountDataModel.js +33 -33
  86. package/models/adminConfigModel.js +35 -35
  87. package/models/cacheModel.js +30 -26
  88. package/models/callDownListModel.js +34 -34
  89. package/models/callLogModel.js +33 -27
  90. package/models/dynamo/connectorSchema.js +146 -146
  91. package/models/dynamo/lockSchema.js +24 -24
  92. package/models/dynamo/noteCacheSchema.js +29 -29
  93. package/models/llmSessionModel.js +17 -17
  94. package/models/messageLogModel.js +25 -25
  95. package/models/sequelize.js +16 -16
  96. package/models/userModel.js +45 -45
  97. package/package.json +72 -72
  98. package/releaseNotes.json +1093 -1073
  99. package/test/connector/proxy/engine.test.js +126 -93
  100. package/test/connector/proxy/index.test.js +279 -279
  101. package/test/connector/proxy/sample.json +161 -161
  102. package/test/connector/registry.test.js +415 -415
  103. package/test/handlers/admin.test.js +616 -616
  104. package/test/handlers/auth.test.js +1018 -1015
  105. package/test/handlers/contact.test.js +1014 -1014
  106. package/test/handlers/log.test.js +1298 -1160
  107. package/test/handlers/managedAuth.test.js +458 -458
  108. package/test/handlers/plugin.test.js +380 -380
  109. package/test/index.test.js +105 -105
  110. package/test/lib/cacheCleanup.test.js +42 -0
  111. package/test/lib/callLogComposer.test.js +1231 -1231
  112. package/test/lib/debugTracer.test.js +328 -328
  113. package/test/lib/jwt.test.js +176 -176
  114. package/test/lib/logger.test.js +206 -206
  115. package/test/lib/oauth.test.js +359 -359
  116. package/test/lib/ringcentral.test.js +467 -467
  117. package/test/lib/sharedSMSComposer.test.js +1084 -1084
  118. package/test/lib/util.test.js +329 -329
  119. package/test/mcp/tools/checkAuthStatus.test.js +83 -82
  120. package/test/mcp/tools/createCallLog.test.js +436 -436
  121. package/test/mcp/tools/createContact.test.js +58 -58
  122. package/test/mcp/tools/createMessageLog.test.js +595 -595
  123. package/test/mcp/tools/doAuth.test.js +113 -113
  124. package/test/mcp/tools/findContactByName.test.js +275 -275
  125. package/test/mcp/tools/findContactByPhone.test.js +296 -296
  126. package/test/mcp/tools/getCallLog.test.js +298 -298
  127. package/test/mcp/tools/getGoogleFilePicker.test.js +281 -281
  128. package/test/mcp/tools/getPublicConnectors.test.js +107 -107
  129. package/test/mcp/tools/getSessionInfo.test.js +127 -127
  130. package/test/mcp/tools/logout.test.js +233 -233
  131. package/test/mcp/tools/rcGetCallLogs.test.js +56 -56
  132. package/test/mcp/tools/updateCallLog.test.js +360 -360
  133. package/test/models/accountDataModel.test.js +98 -98
  134. package/test/models/dynamo/connectorSchema.test.js +189 -189
  135. package/test/models/models.test.js +568 -539
  136. package/test/routes/managedAuthRoutes.test.js +104 -129
  137. package/test/setup.js +178 -178
@@ -1,43 +1,32 @@
1
- const axios = require('axios');
2
- const { logger } = require('../lib/logger');
3
-
4
-
5
- async function getPublicConnectorList() {
6
- try {
7
- const response = await axios.get('https://appconnect.labs.ringcentral.com/public-api/connectors');
8
- return response.data;
9
- } catch (error) {
10
- logger.error('Error getting public connector list:', error);
11
- return null;
12
- }
13
- }
14
-
15
- async function getPrivateConnectorList() {
16
- try {
17
- const response = await axios.get(`https://appconnect.labs.ringcentral.com/public-api/connectors/internal?accountId=${process.env.RC_ACCOUNT_ID}`);
18
- return response.data;
19
- } catch (error) {
20
- logger.error('Error getting private connector list:', error);
21
- return null;
22
- }
23
- }
24
-
25
- async function getConnectorManifest({ connectorId, isPrivate = false }) {
26
- try {
27
- let response = null;
28
- if (isPrivate) {
29
- response = await axios.get(`https://appconnect.labs.ringcentral.com/public-api/connectors/${connectorId}/manifest?access=internal&type=connector&accountId=${process.env.RC_ACCOUNT_ID}`);
30
- }
31
- else {
32
- response = await axios.get(`https://appconnect.labs.ringcentral.com/public-api/connectors/${connectorId}/manifest`);
33
- }
34
- return response.data;
35
- } catch (error) {
36
- logger.error('Error getting connector manifest:', error);
37
- return null;
38
- }
39
- }
40
-
41
- exports.getPublicConnectorList = getPublicConnectorList;
42
- exports.getPrivateConnectorList = getPrivateConnectorList;
1
+ const axios = require('axios');
2
+ const { logger } = require('../lib/logger');
3
+
4
+
5
+ async function getPublicConnectorList() {
6
+ try {
7
+ const response = await axios.get('https://appconnect.labs.ringcentral.com/public-api/connectors');
8
+ return response.data;
9
+ } catch (error) {
10
+ logger.error('Error getting public connector list:', error);
11
+ return null;
12
+ }
13
+ }
14
+
15
+ async function getConnectorManifest({ rcAccountId, connectorId, isPrivate = false }) {
16
+ try {
17
+ let response = null;
18
+ if (isPrivate) {
19
+ response = await axios.get(`https://appconnect.labs.ringcentral.com/public-api/connectors/${connectorId}/manifest?access=internal&type=connector&accountId=${rcAccountId}`);
20
+ }
21
+ else {
22
+ response = await axios.get(`https://appconnect.labs.ringcentral.com/public-api/connectors/${connectorId}/manifest`);
23
+ }
24
+ return response.data;
25
+ } catch (error) {
26
+ logger.error('Error getting connector manifest:', error);
27
+ return null;
28
+ }
29
+ }
30
+
31
+ exports.getPublicConnectorList = getPublicConnectorList;
43
32
  exports.getConnectorManifest = getConnectorManifest;
package/connector/mock.js CHANGED
@@ -1,77 +1,84 @@
1
- // This mock is to run high traffic tests on the server
2
-
3
- const { UserModel } = require("../models/userModel");
4
- const { CallLogModel } = require("../models/callLogModel");
5
- const shortid = require("shortid");
6
- const Op = require('sequelize').Op;
7
-
8
- async function createUser() {
9
- let mockUser = await UserModel.findByPk('mockUser');
10
- if (!mockUser) {
11
- mockUser = await UserModel.create({
12
- id: 'mockUser'
13
- });
14
- }
15
- return mockUser;
16
- }
17
-
18
- async function deleteUser() {
19
- let mockUser = await UserModel.findByPk('mockUser');
20
- if (mockUser) {
21
- await mockUser.destroy();
22
- return true;
23
- }
24
- return false;
25
- }
26
-
27
- async function getCallLog({ sessionIds }) {
28
- const sessionIdsArray = sessionIds.split(',');
29
- const callLogs = await CallLogModel.findAll({
30
- where: {
31
- sessionId: {
32
- [Op.in]: sessionIdsArray
33
- }
34
- }
35
- });
36
- const logs = [];
37
- for (const sId of sessionIdsArray) {
38
- const callLog = callLogs.find(c => c.sessionId === sId);
39
- if (!callLog) {
40
- logs.push({ sessionId: sId, matched: false });
41
- }
42
- else {
43
- logs.push({ sessionId: callLog.sessionId, matched: true, logId: 'mockThirdPartyLogId' });
44
- }
45
- }
46
-
47
- return logs;
48
- }
49
-
50
- async function createCallLog({ sessionId }) {
51
- let callLog = await CallLogModel.findOne({
52
- where: {
53
- sessionId
54
- }
55
- });
56
- if (!callLog) {
57
- callLog = await CallLogModel.create({
58
- id: shortid.generate(),
59
- sessionId,
60
- userId: 'mockUser'
61
- });
62
- }
63
- }
64
-
65
- async function cleanUpMockLogs() {
66
- await CallLogModel.destroy({
67
- where: {
68
- userId: 'mockUser'
69
- }
70
- });
71
- }
72
-
73
- exports.createUser = createUser;
74
- exports.deleteUser = deleteUser;
75
- exports.getCallLog = getCallLog;
76
- exports.createCallLog = createCallLog;
77
- exports.cleanUpMockLogs = cleanUpMockLogs;
1
+ // This mock is to run high traffic tests on the server
2
+
3
+ const { UserModel } = require("../models/userModel");
4
+ const { CallLogModel } = require("../models/callLogModel");
5
+ const shortid = require("shortid");
6
+ const {
7
+ buildCallLogSessionWhere,
8
+ findMatchingCallLog,
9
+ } = require('../lib/callLogLookup');
10
+
11
+ async function createUser() {
12
+ let mockUser = await UserModel.findByPk('mockUser');
13
+ if (!mockUser) {
14
+ mockUser = await UserModel.create({
15
+ id: 'mockUser'
16
+ });
17
+ }
18
+ return mockUser;
19
+ }
20
+
21
+ async function deleteUser() {
22
+ let mockUser = await UserModel.findByPk('mockUser');
23
+ if (mockUser) {
24
+ await mockUser.destroy();
25
+ return true;
26
+ }
27
+ return false;
28
+ }
29
+
30
+ async function getCallLog({ sessionIds, extensionNumber }) {
31
+ const sessionIdsArray = sessionIds.split(',');
32
+ const extensionNumberValue = extensionNumber?.toString() ?? '';
33
+ const callLogs = await CallLogModel.findAll({
34
+ where: buildCallLogSessionWhere({
35
+ sessionIds: sessionIdsArray,
36
+ extensionNumber: extensionNumberValue,
37
+ }),
38
+ order: [['extensionNumber', 'ASC']]
39
+ });
40
+ const logs = [];
41
+ for (const sId of sessionIdsArray) {
42
+ const callLog = findMatchingCallLog(callLogs, sId, extensionNumberValue);
43
+ if (!callLog) {
44
+ logs.push({ sessionId: sId, matched: false });
45
+ }
46
+ else {
47
+ logs.push({ sessionId: callLog.sessionId, matched: true, logId: 'mockThirdPartyLogId' });
48
+ }
49
+ }
50
+
51
+ return logs;
52
+ }
53
+
54
+ async function createCallLog({ sessionId, extensionNumber }) {
55
+ const extensionNumberValue = extensionNumber?.toString() ?? '';
56
+ let callLog = await CallLogModel.findOne({
57
+ where: buildCallLogSessionWhere({
58
+ sessionId,
59
+ extensionNumber: extensionNumberValue,
60
+ })
61
+ });
62
+ if (!callLog) {
63
+ callLog = await CallLogModel.create({
64
+ id: shortid.generate(),
65
+ sessionId,
66
+ extensionNumber: extensionNumberValue,
67
+ userId: 'mockUser'
68
+ });
69
+ }
70
+ }
71
+
72
+ async function cleanUpMockLogs() {
73
+ await CallLogModel.destroy({
74
+ where: {
75
+ userId: 'mockUser'
76
+ }
77
+ });
78
+ }
79
+
80
+ exports.createUser = createUser;
81
+ exports.deleteUser = deleteUser;
82
+ exports.getCallLog = getCallLog;
83
+ exports.createCallLog = createCallLog;
84
+ exports.cleanUpMockLogs = cleanUpMockLogs;
@@ -1,163 +1,164 @@
1
- const axios = require('axios');
2
-
3
- function getByPath(obj, path) {
4
- if (!path || path === '$') return obj;
5
- const parts = path.split('.');
6
- let cur = obj;
7
- for (const p of parts) {
8
- if (cur == null) return undefined;
9
- cur = cur[p];
10
- }
11
- return cur;
12
- }
13
-
14
- function renderTemplateString(template, context) {
15
- if (typeof template !== 'string') return template;
16
- // if only template value, return value with stringify
17
- const onlyVarName = template.match(/^\{\{\s*([^}]+?)\s*\}\}$/);
18
- if (onlyVarName) {
19
- return getByPath(context, onlyVarName[1]);
20
- }
21
- return template.replace(/\{\{\s*([^}]+?)\s*\}\}/g, (_, expr) => {
22
- const value = getByPath(context, expr.trim());
23
- return value == null ? '' : String(value);
24
- });
25
- }
26
-
27
- function renderDeep(input, context) {
28
- if (input == null) return input;
29
- if (typeof input === 'string') return renderTemplateString(input, context);
30
- if (Array.isArray(input)) return input.map(v => renderDeep(v, context));
31
- if (typeof input === 'object') {
32
- const out = {};
33
- for (const [k, v] of Object.entries(input)) {
34
- out[k] = renderDeep(v, context);
35
- }
36
- return out;
37
- }
38
- return input;
39
- }
40
-
41
- function joinUrl(baseUrl, path) {
42
- if (!baseUrl) return path;
43
- if (!path) return baseUrl;
44
- if (path.startsWith('http://') || path.startsWith('https://')) return path;
45
- return `${baseUrl.replace(/\/$/, '')}/${path.replace(/^\//, '')}`;
46
- }
47
-
48
- function getAuthHeaderFromAuthConfig({ auth, context, authHeader }) {
49
- const authHeaders = {};
50
- const headerName = auth?.headerName || 'Authorization';
51
- if (auth && auth.credentialTemplate) {
52
- const credentials = renderTemplateString(auth.credentialTemplate, context);
53
- const encode = auth.encode === 'none' ? false : true;
54
- const token = encode ? Buffer.from(credentials).toString('base64') : credentials;
55
- authHeaders[headerName] = auth.scheme ? `${auth.scheme} ${token}` : token;
56
- } else if (authHeader) {
57
- authHeaders[headerName] = authHeader;
58
- } else if (auth && auth.type === 'oauth' && context.user) {
59
- authHeaders[headerName] = `${auth.scheme || 'Bearer'} ${context.user.accessToken}`;
60
- }
61
- return authHeaders;
62
- }
63
-
64
- function buildHeaders({ config, operation, authHeader, context }) {
65
- const headers = renderDeep(Object.assign({}, config.requestDefaults?.defaultHeaders || {}), context);
66
- const renderedOpHeaders = renderDeep(operation?.headers || {}, context);
67
- for (const [k, v] of Object.entries(renderedOpHeaders)) headers[k] = v;
68
-
69
- // Per-operation auth override
70
- const authHeaders = getAuthHeaderFromAuthConfig({
71
- auth: operation?.auth || config.auth,
72
- context,
73
- authHeader
74
- });
75
- for (const [k, v] of Object.entries(authHeaders)) headers[k] = v;
76
- return headers;
77
- }
78
-
79
- async function performRequest({ config, opName, inputs, user, authHeader }) {
80
- const op = config.operations?.[opName];
81
- if (!op) return null;
82
- const context = Object.assign({}, inputs, {
83
- user: user ? {
84
- accessToken: user.accessToken,
85
- id: user.id?.split('-')[0],
86
- hostname: user.hostname,
87
- timezoneName: user.timezoneName,
88
- timezoneOffset: user.timezoneOffset,
89
- platform: user.platform,
90
- platformAdditionalInfo: user.platformAdditionalInfo,
91
- refreshToken: user.refreshToken,
92
- tokenExpiry: user.tokenExpiry,
93
- } : {
94
- accessToken: '',
95
- },
96
- authHeader,
97
- apiKey: user?.accessToken,
98
- secretKey: config.secretKey,
99
- });
100
- const url = joinUrl(config.requestDefaults?.baseUrl, renderTemplateString(op.url, context));
101
- const method = (op.method || 'GET').toUpperCase();
102
- const headers = buildHeaders({ config, operation: op, authHeader, context });
103
- const params = renderDeep(op.query || {}, context);
104
- const data = renderDeep(op.body || {}, context);
105
- const timeout = (config.requestDefaults?.timeoutSeconds || 30) * 1000;
106
- const axiosParams = { url, method, headers, params, data, timeout };
107
- const response = await axios(axiosParams);
108
- return response;
109
- }
110
-
111
- function mapFindContactResponse({ config, response }) {
112
- const map = config.operations?.findContact?.responseMapping;
113
- if (!map) return [];
114
- const __ctx = { body: response.data };
115
- const list = getByPath(__ctx, map.listPath || 'body') || [];
116
- const itemMap = map.item || {};
117
- return list.map(it => {
118
- return {
119
- id: getByPath(it, itemMap.idPath || 'id'),
120
- name: getByPath(it, itemMap.namePath || 'name') || '',
121
- phone: getByPath(it, itemMap.phonePath || 'phone') || undefined,
122
- type: getByPath(it, itemMap.typePath || 'type') || 'Contact',
123
- title: getByPath(it, itemMap.titlePath || 'title') || "",
124
- company: getByPath(it, itemMap.companyPath || 'company') || "",
125
- mostRecentActivityDate: getByPath(it, itemMap.mostRecentActivityDatePath || 'mostRecentActivityDate') || undefined,
126
- additionalInfo: getByPath(it, itemMap.additionalInfoPath || 'additionalInfo') || null
127
- };
128
- });
129
- }
130
-
131
- function mapCreateCallLogResponse({ config, response }) {
132
- const map = config.operations?.createCallLog?.responseMapping;
133
- if (!map) return { logId: undefined };
134
- const __ctx = { body: response.data };
135
- const logId = getByPath(__ctx, map.idPath || 'body.id');
136
-
137
- return { logId: logId ? String(logId) : undefined };
138
- }
139
-
140
- function mapGetCallLogResponse({ config, response }) {
141
- const map = config.operations?.getCallLog?.responseMapping || {};
142
- const __ctx = { body: response.data };
143
- const subject = getByPath(__ctx, map.subjectPath || 'body.subject');
144
- const note = getByPath(__ctx, map.notePath || 'body.note');
145
- const fullBody = getByPath(__ctx, map.fullBodyPath || 'body.note');
146
- const fullLogResponse = response.data;
147
- return {
148
- callLogInfo: { subject, note, fullBody, fullLogResponse }
149
- };
150
- }
151
-
152
- module.exports = {
153
- getByPath,
154
- renderTemplateString,
155
- renderDeep,
156
- joinUrl,
157
- performRequest,
158
- mapFindContactResponse,
159
- mapCreateCallLogResponse,
160
- mapGetCallLogResponse
161
- };
162
-
163
-
1
+ const axios = require('axios');
2
+
3
+ function getByPath(obj, path) {
4
+ if (!path || path === '$') return obj;
5
+ const parts = path.split('.');
6
+ let cur = obj;
7
+ for (const p of parts) {
8
+ if (cur == null) return undefined;
9
+ cur = cur[p];
10
+ }
11
+ return cur;
12
+ }
13
+
14
+ function renderTemplateString(template, context) {
15
+ if (typeof template !== 'string') return template;
16
+ // if only template value, return value with stringify
17
+ const onlyVarName = template.match(/^\{\{\s*([^}]+?)\s*\}\}$/);
18
+ if (onlyVarName) {
19
+ return getByPath(context, onlyVarName[1]);
20
+ }
21
+ return template.replace(/\{\{\s*([^}]+?)\s*\}\}/g, (_, expr) => {
22
+ const value = getByPath(context, expr.trim());
23
+ return value == null ? '' : String(value);
24
+ });
25
+ }
26
+
27
+ function renderDeep(input, context) {
28
+ if (input == null) return input;
29
+ if (typeof input === 'string') return renderTemplateString(input, context);
30
+ if (Array.isArray(input)) return input.map(v => renderDeep(v, context));
31
+ if (typeof input === 'object') {
32
+ const out = {};
33
+ for (const [k, v] of Object.entries(input)) {
34
+ out[k] = renderDeep(v, context);
35
+ }
36
+ return out;
37
+ }
38
+ return input;
39
+ }
40
+
41
+ function joinUrl(baseUrl, path) {
42
+ if (!baseUrl) return path;
43
+ if (!path) return baseUrl;
44
+ if (path.startsWith('http://') || path.startsWith('https://')) return path;
45
+ return `${baseUrl.replace(/\/$/, '')}/${path.replace(/^\//, '')}`;
46
+ }
47
+
48
+ function getAuthHeaderFromAuthConfig({ auth, context, authHeader }) {
49
+ const authHeaders = {};
50
+ const headerName = auth?.headerName || 'Authorization';
51
+ if (auth && auth.credentialTemplate) {
52
+ const credentials = renderTemplateString(auth.credentialTemplate, context);
53
+ const encode = auth.encode === 'none' ? false : true;
54
+ const token = encode ? Buffer.from(credentials).toString('base64') : credentials;
55
+ authHeaders[headerName] = auth.scheme ? `${auth.scheme} ${token}` : token;
56
+ } else if (authHeader) {
57
+ authHeaders[headerName] = authHeader;
58
+ } else if (auth && auth.type === 'oauth' && context.user) {
59
+ authHeaders[headerName] = `${auth.scheme || 'Bearer'} ${context.user.accessToken}`;
60
+ }
61
+ return authHeaders;
62
+ }
63
+
64
+ function buildHeaders({ config, operation, authHeader, context }) {
65
+ const headers = renderDeep(Object.assign({}, config.requestDefaults?.defaultHeaders || {}), context);
66
+ const renderedOpHeaders = renderDeep(operation?.headers || {}, context);
67
+ for (const [k, v] of Object.entries(renderedOpHeaders)) headers[k] = v;
68
+
69
+ // Per-operation auth override
70
+ const authHeaders = getAuthHeaderFromAuthConfig({
71
+ auth: operation?.auth || config.auth,
72
+ context,
73
+ authHeader
74
+ });
75
+ for (const [k, v] of Object.entries(authHeaders)) headers[k] = v;
76
+ return headers;
77
+ }
78
+
79
+ async function performRequest({ config, opName, inputs, user, authHeader }) {
80
+ const op = config.operations?.[opName];
81
+ if (!op) return null;
82
+ const accessToken = user?.accessToken ?? inputs?.apiKey ?? '';
83
+ const context = Object.assign({}, inputs, {
84
+ user: user ? {
85
+ accessToken,
86
+ id: user.id?.split('-')[0],
87
+ hostname: user.hostname,
88
+ timezoneName: user.timezoneName,
89
+ timezoneOffset: user.timezoneOffset,
90
+ platform: user.platform,
91
+ platformAdditionalInfo: user.platformAdditionalInfo,
92
+ refreshToken: user.refreshToken,
93
+ tokenExpiry: user.tokenExpiry,
94
+ } : {
95
+ accessToken,
96
+ },
97
+ authHeader,
98
+ apiKey: accessToken,
99
+ secretKey: config.secretKey,
100
+ });
101
+ const url = joinUrl(config.requestDefaults?.baseUrl, renderTemplateString(op.url, context));
102
+ const method = (op.method || 'GET').toUpperCase();
103
+ const headers = buildHeaders({ config, operation: op, authHeader, context });
104
+ const params = renderDeep(op.query || {}, context);
105
+ const data = renderDeep(op.body || {}, context);
106
+ const timeout = (config.requestDefaults?.timeoutSeconds || 30) * 1000;
107
+ const axiosParams = { url, method, headers, params, data, timeout };
108
+ const response = await axios(axiosParams);
109
+ return response;
110
+ }
111
+
112
+ function mapFindContactResponse({ config, response }) {
113
+ const map = config.operations?.findContact?.responseMapping;
114
+ if (!map) return [];
115
+ const __ctx = { body: response.data };
116
+ const list = getByPath(__ctx, map.listPath || 'body') || [];
117
+ const itemMap = map.item || {};
118
+ return list.map(it => {
119
+ return {
120
+ id: getByPath(it, itemMap.idPath || 'id'),
121
+ name: getByPath(it, itemMap.namePath || 'name') || '',
122
+ phone: getByPath(it, itemMap.phonePath || 'phone') || undefined,
123
+ type: getByPath(it, itemMap.typePath || 'type') || 'Contact',
124
+ title: getByPath(it, itemMap.titlePath || 'title') || "",
125
+ company: getByPath(it, itemMap.companyPath || 'company') || "",
126
+ mostRecentActivityDate: getByPath(it, itemMap.mostRecentActivityDatePath || 'mostRecentActivityDate') || undefined,
127
+ additionalInfo: getByPath(it, itemMap.additionalInfoPath || 'additionalInfo') || null
128
+ };
129
+ });
130
+ }
131
+
132
+ function mapCreateCallLogResponse({ config, response }) {
133
+ const map = config.operations?.createCallLog?.responseMapping;
134
+ if (!map) return { logId: undefined };
135
+ const __ctx = { body: response.data };
136
+ const logId = getByPath(__ctx, map.idPath || 'body.id');
137
+
138
+ return { logId: logId ? String(logId) : undefined };
139
+ }
140
+
141
+ function mapGetCallLogResponse({ config, response }) {
142
+ const map = config.operations?.getCallLog?.responseMapping || {};
143
+ const __ctx = { body: response.data };
144
+ const subject = getByPath(__ctx, map.subjectPath || 'body.subject');
145
+ const note = getByPath(__ctx, map.notePath || 'body.note');
146
+ const fullBody = getByPath(__ctx, map.fullBodyPath || 'body.note');
147
+ const fullLogResponse = response.data;
148
+ return {
149
+ callLogInfo: { subject, note, fullBody, fullLogResponse }
150
+ };
151
+ }
152
+
153
+ module.exports = {
154
+ getByPath,
155
+ renderTemplateString,
156
+ renderDeep,
157
+ joinUrl,
158
+ performRequest,
159
+ mapFindContactResponse,
160
+ mapCreateCallLogResponse,
161
+ mapGetCallLogResponse
162
+ };
163
+
164
+