@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.
- package/.env.test +5 -5
- package/README.md +441 -441
- package/connector/developerPortal.js +31 -42
- package/connector/mock.js +84 -77
- package/connector/proxy/engine.js +164 -163
- package/connector/proxy/index.js +500 -500
- package/connector/registry.js +252 -252
- package/docs/README.md +50 -50
- package/docs/architecture.md +93 -93
- package/docs/connectors.md +116 -117
- package/docs/handlers.md +125 -125
- package/docs/libraries.md +101 -101
- package/docs/models.md +144 -144
- package/docs/routes.md +115 -115
- package/docs/tests.md +73 -73
- package/handlers/admin.js +523 -523
- package/handlers/appointment.js +193 -0
- package/handlers/auth.js +296 -296
- package/handlers/calldown.js +99 -99
- package/handlers/contact.js +280 -280
- package/handlers/disposition.js +82 -80
- package/handlers/log.js +984 -973
- package/handlers/managedAuth.js +446 -446
- package/handlers/plugin.js +208 -208
- package/handlers/user.js +142 -142
- package/index.js +3140 -2652
- package/jest.config.js +56 -56
- package/lib/analytics.js +54 -54
- package/lib/authSession.js +109 -109
- package/lib/cacheCleanup.js +21 -0
- package/lib/callLogComposer.js +898 -898
- package/lib/callLogLookup.js +34 -0
- package/lib/constants.js +8 -8
- package/lib/debugTracer.js +177 -177
- package/lib/encode.js +30 -30
- package/lib/errorHandler.js +218 -206
- package/lib/generalErrorMessage.js +41 -41
- package/lib/jwt.js +18 -18
- package/lib/logger.js +190 -190
- package/lib/migrateCallLogsSchema.js +116 -0
- package/lib/ringcentral.js +266 -266
- package/lib/s3ErrorLogReport.js +65 -65
- package/lib/sharedSMSComposer.js +471 -471
- package/lib/util.js +67 -67
- package/mcp/README.md +412 -395
- package/mcp/lib/validator.js +91 -91
- package/mcp/mcpHandler.js +425 -425
- package/mcp/tools/cancelAppointment.js +101 -0
- package/mcp/tools/checkAuthStatus.js +105 -105
- package/mcp/tools/confirmAppointment.js +101 -0
- package/mcp/tools/createAppointment.js +157 -0
- package/mcp/tools/createCallLog.js +327 -316
- package/mcp/tools/createContact.js +117 -117
- package/mcp/tools/createMessageLog.js +287 -287
- package/mcp/tools/doAuth.js +60 -60
- package/mcp/tools/findContactByName.js +93 -93
- package/mcp/tools/findContactByPhone.js +101 -101
- package/mcp/tools/getCallLog.js +111 -102
- package/mcp/tools/getGoogleFilePicker.js +99 -99
- package/mcp/tools/getHelp.js +43 -43
- package/mcp/tools/getPublicConnectors.js +94 -94
- package/mcp/tools/getSessionInfo.js +90 -90
- package/mcp/tools/index.js +51 -41
- package/mcp/tools/listAppointments.js +163 -0
- package/mcp/tools/logout.js +96 -96
- package/mcp/tools/rcGetCallLogs.js +65 -65
- package/mcp/tools/updateAppointment.js +154 -0
- package/mcp/tools/updateCallLog.js +130 -126
- package/mcp/ui/App/App.tsx +358 -358
- package/mcp/ui/App/components/AuthInfoForm.tsx +113 -113
- package/mcp/ui/App/components/AuthSuccess.tsx +22 -22
- package/mcp/ui/App/components/ConnectorList.tsx +82 -82
- package/mcp/ui/App/components/DebugPanel.tsx +43 -43
- package/mcp/ui/App/components/OAuthConnect.tsx +270 -270
- package/mcp/ui/App/lib/callTool.ts +130 -130
- package/mcp/ui/App/lib/debugLog.ts +41 -41
- package/mcp/ui/App/lib/developerPortal.ts +111 -111
- package/mcp/ui/App/main.css +5 -5
- package/mcp/ui/App/root.tsx +13 -13
- package/mcp/ui/index.html +13 -13
- package/mcp/ui/package-lock.json +6356 -6356
- package/mcp/ui/package.json +25 -25
- package/mcp/ui/tsconfig.json +26 -26
- package/mcp/ui/vite.config.ts +16 -16
- package/models/accountDataModel.js +33 -33
- package/models/adminConfigModel.js +35 -35
- package/models/cacheModel.js +30 -26
- package/models/callDownListModel.js +34 -34
- package/models/callLogModel.js +33 -27
- package/models/dynamo/connectorSchema.js +146 -146
- package/models/dynamo/lockSchema.js +24 -24
- package/models/dynamo/noteCacheSchema.js +29 -29
- package/models/llmSessionModel.js +17 -17
- package/models/messageLogModel.js +25 -25
- package/models/sequelize.js +16 -16
- package/models/userModel.js +45 -45
- package/package.json +72 -72
- package/releaseNotes.json +1093 -1073
- package/test/connector/proxy/engine.test.js +126 -93
- package/test/connector/proxy/index.test.js +279 -279
- package/test/connector/proxy/sample.json +161 -161
- package/test/connector/registry.test.js +415 -415
- package/test/handlers/admin.test.js +616 -616
- package/test/handlers/auth.test.js +1018 -1015
- package/test/handlers/contact.test.js +1014 -1014
- package/test/handlers/log.test.js +1298 -1160
- package/test/handlers/managedAuth.test.js +458 -458
- package/test/handlers/plugin.test.js +380 -380
- package/test/index.test.js +105 -105
- package/test/lib/cacheCleanup.test.js +42 -0
- package/test/lib/callLogComposer.test.js +1231 -1231
- package/test/lib/debugTracer.test.js +328 -328
- package/test/lib/jwt.test.js +176 -176
- package/test/lib/logger.test.js +206 -206
- package/test/lib/oauth.test.js +359 -359
- package/test/lib/ringcentral.test.js +467 -467
- package/test/lib/sharedSMSComposer.test.js +1084 -1084
- package/test/lib/util.test.js +329 -329
- package/test/mcp/tools/checkAuthStatus.test.js +83 -82
- package/test/mcp/tools/createCallLog.test.js +436 -436
- package/test/mcp/tools/createContact.test.js +58 -58
- package/test/mcp/tools/createMessageLog.test.js +595 -595
- package/test/mcp/tools/doAuth.test.js +113 -113
- package/test/mcp/tools/findContactByName.test.js +275 -275
- package/test/mcp/tools/findContactByPhone.test.js +296 -296
- package/test/mcp/tools/getCallLog.test.js +298 -298
- package/test/mcp/tools/getGoogleFilePicker.test.js +281 -281
- package/test/mcp/tools/getPublicConnectors.test.js +107 -107
- package/test/mcp/tools/getSessionInfo.test.js +127 -127
- package/test/mcp/tools/logout.test.js +233 -233
- package/test/mcp/tools/rcGetCallLogs.test.js +56 -56
- package/test/mcp/tools/updateCallLog.test.js +360 -360
- package/test/models/accountDataModel.test.js +98 -98
- package/test/models/dynamo/connectorSchema.test.js +189 -189
- package/test/models/models.test.js +568 -539
- package/test/routes/managedAuthRoutes.test.js +104 -129
- 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
|
|
16
|
-
try {
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
}
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
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
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
}
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
}
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
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
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
const
|
|
102
|
-
const
|
|
103
|
-
const
|
|
104
|
-
const
|
|
105
|
-
const
|
|
106
|
-
const
|
|
107
|
-
const
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
const
|
|
116
|
-
const
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
}
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
const
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
}
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
|
|
142
|
-
const
|
|
143
|
-
const
|
|
144
|
-
const
|
|
145
|
-
const
|
|
146
|
-
const
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
}
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
|
|
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
|
+
|