@app-connect/core 1.7.25 → 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 -31
- package/connector/mock.js +84 -77
- package/connector/proxy/engine.js +164 -164
- 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 -116
- 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 +1 -1
- package/releaseNotes.json +1093 -1081
- package/test/connector/proxy/engine.test.js +126 -126
- 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 -1018
- package/test/handlers/contact.test.js +1014 -1014
- package/test/handlers/log.test.js +1298 -1160
- package/test/handlers/managedAuth.test.js +457 -457
- 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 -83
- 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 -104
- package/test/setup.js +178 -178
package/test/lib/jwt.test.js
CHANGED
|
@@ -1,176 +1,176 @@
|
|
|
1
|
-
const jwt = require('../../lib/jwt');
|
|
2
|
-
|
|
3
|
-
describe('JWT Utility', () => {
|
|
4
|
-
beforeEach(() => {
|
|
5
|
-
// Reset environment
|
|
6
|
-
process.env.APP_SERVER_SECRET_KEY = 'test-secret-key';
|
|
7
|
-
});
|
|
8
|
-
|
|
9
|
-
describe('generateJwt', () => {
|
|
10
|
-
test('should generate JWT token from payload', () => {
|
|
11
|
-
// Arrange
|
|
12
|
-
const payload = {
|
|
13
|
-
id: 'test-user-id',
|
|
14
|
-
platform: 'testCRM'
|
|
15
|
-
};
|
|
16
|
-
|
|
17
|
-
// Act
|
|
18
|
-
const token = jwt.generateJwt(payload);
|
|
19
|
-
|
|
20
|
-
// Assert
|
|
21
|
-
expect(token).toBeDefined();
|
|
22
|
-
expect(typeof token).toBe('string');
|
|
23
|
-
expect(token.split('.')).toHaveLength(3); // JWT has 3 parts
|
|
24
|
-
});
|
|
25
|
-
|
|
26
|
-
test('should generate different tokens for different payloads', () => {
|
|
27
|
-
// Arrange
|
|
28
|
-
const payload1 = { id: 'user1', platform: 'testCRM' };
|
|
29
|
-
const payload2 = { id: 'user2', platform: 'testCRM' };
|
|
30
|
-
|
|
31
|
-
// Act
|
|
32
|
-
const token1 = jwt.generateJwt(payload1);
|
|
33
|
-
const token2 = jwt.generateJwt(payload2);
|
|
34
|
-
|
|
35
|
-
// Assert
|
|
36
|
-
expect(token1).not.toBe(token2);
|
|
37
|
-
});
|
|
38
|
-
|
|
39
|
-
test('should generate token with about 2 weeks lifetime', () => {
|
|
40
|
-
// Arrange
|
|
41
|
-
const payload = { id: 'user-ttl', platform: 'testCRM' };
|
|
42
|
-
|
|
43
|
-
// Act
|
|
44
|
-
const token = jwt.generateJwt(payload);
|
|
45
|
-
const decoded = jwt.decodeJwt(token);
|
|
46
|
-
const lifetimeSeconds = decoded.exp - decoded.iat;
|
|
47
|
-
|
|
48
|
-
// Assert
|
|
49
|
-
// Keep a tiny tolerance to avoid timing flakiness.
|
|
50
|
-
expect(lifetimeSeconds).toBeGreaterThanOrEqual((14 * 24 * 60 * 60) - 2);
|
|
51
|
-
expect(lifetimeSeconds).toBeLessThanOrEqual((14 * 24 * 60 * 60) + 2);
|
|
52
|
-
});
|
|
53
|
-
});
|
|
54
|
-
|
|
55
|
-
describe('decodeJwt', () => {
|
|
56
|
-
test('should decode valid JWT token', () => {
|
|
57
|
-
// Arrange
|
|
58
|
-
const payload = {
|
|
59
|
-
id: 'test-user-id',
|
|
60
|
-
platform: 'testCRM'
|
|
61
|
-
};
|
|
62
|
-
const token = jwt.generateJwt(payload);
|
|
63
|
-
|
|
64
|
-
// Act
|
|
65
|
-
const decoded = jwt.decodeJwt(token);
|
|
66
|
-
|
|
67
|
-
// Assert
|
|
68
|
-
expect(decoded).toMatchObject(payload);
|
|
69
|
-
expect(decoded).toHaveProperty('exp');
|
|
70
|
-
expect(decoded).toHaveProperty('iat');
|
|
71
|
-
});
|
|
72
|
-
|
|
73
|
-
test('should return null for invalid token', () => {
|
|
74
|
-
// Arrange
|
|
75
|
-
const invalidToken = 'invalid.jwt.token';
|
|
76
|
-
|
|
77
|
-
// Act
|
|
78
|
-
const decoded = jwt.decodeJwt(invalidToken);
|
|
79
|
-
|
|
80
|
-
// Assert
|
|
81
|
-
expect(decoded).toBeNull();
|
|
82
|
-
});
|
|
83
|
-
|
|
84
|
-
test('should return null for malformed token', () => {
|
|
85
|
-
// Arrange
|
|
86
|
-
const malformedToken = 'not-a-jwt-token';
|
|
87
|
-
|
|
88
|
-
// Act
|
|
89
|
-
const decoded = jwt.decodeJwt(malformedToken);
|
|
90
|
-
|
|
91
|
-
// Assert
|
|
92
|
-
expect(decoded).toBeNull();
|
|
93
|
-
});
|
|
94
|
-
|
|
95
|
-
test('should return null for token with wrong secret', () => {
|
|
96
|
-
// Arrange
|
|
97
|
-
const payload = { id: 'test-user-id', platform: 'testCRM' };
|
|
98
|
-
const token = jwt.generateJwt(payload);
|
|
99
|
-
|
|
100
|
-
// Change secret temporarily
|
|
101
|
-
const originalSecret = process.env.APP_SERVER_SECRET_KEY;
|
|
102
|
-
process.env.APP_SERVER_SECRET_KEY = 'different-secret';
|
|
103
|
-
|
|
104
|
-
// Act
|
|
105
|
-
const decoded = jwt.decodeJwt(token);
|
|
106
|
-
|
|
107
|
-
// Restore secret
|
|
108
|
-
process.env.APP_SERVER_SECRET_KEY = originalSecret;
|
|
109
|
-
|
|
110
|
-
// Assert
|
|
111
|
-
expect(decoded).toBeNull();
|
|
112
|
-
});
|
|
113
|
-
});
|
|
114
|
-
|
|
115
|
-
describe('generateJwt and decodeJwt round trip', () => {
|
|
116
|
-
test('should successfully generate and decode complex payload', () => {
|
|
117
|
-
// Arrange
|
|
118
|
-
const complexPayload = {
|
|
119
|
-
id: 'test-user-id',
|
|
120
|
-
platform: 'testCRM',
|
|
121
|
-
timestamp: Date.now(),
|
|
122
|
-
metadata: {
|
|
123
|
-
timezone: 'America/Los_Angeles',
|
|
124
|
-
preferences: {
|
|
125
|
-
autoLog: true,
|
|
126
|
-
callPop: false
|
|
127
|
-
}
|
|
128
|
-
}
|
|
129
|
-
};
|
|
130
|
-
|
|
131
|
-
// Act
|
|
132
|
-
const token = jwt.generateJwt(complexPayload);
|
|
133
|
-
const decoded = jwt.decodeJwt(token);
|
|
134
|
-
|
|
135
|
-
// Assert
|
|
136
|
-
expect(decoded).toMatchObject(complexPayload);
|
|
137
|
-
expect(decoded).toHaveProperty('exp');
|
|
138
|
-
expect(decoded).toHaveProperty('iat');
|
|
139
|
-
});
|
|
140
|
-
|
|
141
|
-
test('should handle empty payload', () => {
|
|
142
|
-
// Arrange
|
|
143
|
-
const emptyPayload = {};
|
|
144
|
-
|
|
145
|
-
// Act
|
|
146
|
-
const token = jwt.generateJwt(emptyPayload);
|
|
147
|
-
const decoded = jwt.decodeJwt(token);
|
|
148
|
-
|
|
149
|
-
// Assert
|
|
150
|
-
expect(decoded).toMatchObject(emptyPayload);
|
|
151
|
-
expect(decoded).toHaveProperty('exp');
|
|
152
|
-
expect(decoded).toHaveProperty('iat');
|
|
153
|
-
});
|
|
154
|
-
});
|
|
155
|
-
|
|
156
|
-
describe('error handling', () => {
|
|
157
|
-
test('should handle missing secret key', () => {
|
|
158
|
-
// Arrange
|
|
159
|
-
const payload = { id: 'test-user-id' };
|
|
160
|
-
delete process.env.APP_SERVER_SECRET_KEY;
|
|
161
|
-
|
|
162
|
-
// Act & Assert
|
|
163
|
-
expect(() => jwt.generateJwt(payload)).toThrow();
|
|
164
|
-
});
|
|
165
|
-
|
|
166
|
-
test('should handle null payload', () => {
|
|
167
|
-
// Act & Assert
|
|
168
|
-
expect(() => jwt.generateJwt(null)).toThrow();
|
|
169
|
-
});
|
|
170
|
-
|
|
171
|
-
test('should handle undefined payload', () => {
|
|
172
|
-
// Act & Assert
|
|
173
|
-
expect(() => jwt.generateJwt(undefined)).toThrow();
|
|
174
|
-
});
|
|
175
|
-
});
|
|
176
|
-
});
|
|
1
|
+
const jwt = require('../../lib/jwt');
|
|
2
|
+
|
|
3
|
+
describe('JWT Utility', () => {
|
|
4
|
+
beforeEach(() => {
|
|
5
|
+
// Reset environment
|
|
6
|
+
process.env.APP_SERVER_SECRET_KEY = 'test-secret-key';
|
|
7
|
+
});
|
|
8
|
+
|
|
9
|
+
describe('generateJwt', () => {
|
|
10
|
+
test('should generate JWT token from payload', () => {
|
|
11
|
+
// Arrange
|
|
12
|
+
const payload = {
|
|
13
|
+
id: 'test-user-id',
|
|
14
|
+
platform: 'testCRM'
|
|
15
|
+
};
|
|
16
|
+
|
|
17
|
+
// Act
|
|
18
|
+
const token = jwt.generateJwt(payload);
|
|
19
|
+
|
|
20
|
+
// Assert
|
|
21
|
+
expect(token).toBeDefined();
|
|
22
|
+
expect(typeof token).toBe('string');
|
|
23
|
+
expect(token.split('.')).toHaveLength(3); // JWT has 3 parts
|
|
24
|
+
});
|
|
25
|
+
|
|
26
|
+
test('should generate different tokens for different payloads', () => {
|
|
27
|
+
// Arrange
|
|
28
|
+
const payload1 = { id: 'user1', platform: 'testCRM' };
|
|
29
|
+
const payload2 = { id: 'user2', platform: 'testCRM' };
|
|
30
|
+
|
|
31
|
+
// Act
|
|
32
|
+
const token1 = jwt.generateJwt(payload1);
|
|
33
|
+
const token2 = jwt.generateJwt(payload2);
|
|
34
|
+
|
|
35
|
+
// Assert
|
|
36
|
+
expect(token1).not.toBe(token2);
|
|
37
|
+
});
|
|
38
|
+
|
|
39
|
+
test('should generate token with about 2 weeks lifetime', () => {
|
|
40
|
+
// Arrange
|
|
41
|
+
const payload = { id: 'user-ttl', platform: 'testCRM' };
|
|
42
|
+
|
|
43
|
+
// Act
|
|
44
|
+
const token = jwt.generateJwt(payload);
|
|
45
|
+
const decoded = jwt.decodeJwt(token);
|
|
46
|
+
const lifetimeSeconds = decoded.exp - decoded.iat;
|
|
47
|
+
|
|
48
|
+
// Assert
|
|
49
|
+
// Keep a tiny tolerance to avoid timing flakiness.
|
|
50
|
+
expect(lifetimeSeconds).toBeGreaterThanOrEqual((14 * 24 * 60 * 60) - 2);
|
|
51
|
+
expect(lifetimeSeconds).toBeLessThanOrEqual((14 * 24 * 60 * 60) + 2);
|
|
52
|
+
});
|
|
53
|
+
});
|
|
54
|
+
|
|
55
|
+
describe('decodeJwt', () => {
|
|
56
|
+
test('should decode valid JWT token', () => {
|
|
57
|
+
// Arrange
|
|
58
|
+
const payload = {
|
|
59
|
+
id: 'test-user-id',
|
|
60
|
+
platform: 'testCRM'
|
|
61
|
+
};
|
|
62
|
+
const token = jwt.generateJwt(payload);
|
|
63
|
+
|
|
64
|
+
// Act
|
|
65
|
+
const decoded = jwt.decodeJwt(token);
|
|
66
|
+
|
|
67
|
+
// Assert
|
|
68
|
+
expect(decoded).toMatchObject(payload);
|
|
69
|
+
expect(decoded).toHaveProperty('exp');
|
|
70
|
+
expect(decoded).toHaveProperty('iat');
|
|
71
|
+
});
|
|
72
|
+
|
|
73
|
+
test('should return null for invalid token', () => {
|
|
74
|
+
// Arrange
|
|
75
|
+
const invalidToken = 'invalid.jwt.token';
|
|
76
|
+
|
|
77
|
+
// Act
|
|
78
|
+
const decoded = jwt.decodeJwt(invalidToken);
|
|
79
|
+
|
|
80
|
+
// Assert
|
|
81
|
+
expect(decoded).toBeNull();
|
|
82
|
+
});
|
|
83
|
+
|
|
84
|
+
test('should return null for malformed token', () => {
|
|
85
|
+
// Arrange
|
|
86
|
+
const malformedToken = 'not-a-jwt-token';
|
|
87
|
+
|
|
88
|
+
// Act
|
|
89
|
+
const decoded = jwt.decodeJwt(malformedToken);
|
|
90
|
+
|
|
91
|
+
// Assert
|
|
92
|
+
expect(decoded).toBeNull();
|
|
93
|
+
});
|
|
94
|
+
|
|
95
|
+
test('should return null for token with wrong secret', () => {
|
|
96
|
+
// Arrange
|
|
97
|
+
const payload = { id: 'test-user-id', platform: 'testCRM' };
|
|
98
|
+
const token = jwt.generateJwt(payload);
|
|
99
|
+
|
|
100
|
+
// Change secret temporarily
|
|
101
|
+
const originalSecret = process.env.APP_SERVER_SECRET_KEY;
|
|
102
|
+
process.env.APP_SERVER_SECRET_KEY = 'different-secret';
|
|
103
|
+
|
|
104
|
+
// Act
|
|
105
|
+
const decoded = jwt.decodeJwt(token);
|
|
106
|
+
|
|
107
|
+
// Restore secret
|
|
108
|
+
process.env.APP_SERVER_SECRET_KEY = originalSecret;
|
|
109
|
+
|
|
110
|
+
// Assert
|
|
111
|
+
expect(decoded).toBeNull();
|
|
112
|
+
});
|
|
113
|
+
});
|
|
114
|
+
|
|
115
|
+
describe('generateJwt and decodeJwt round trip', () => {
|
|
116
|
+
test('should successfully generate and decode complex payload', () => {
|
|
117
|
+
// Arrange
|
|
118
|
+
const complexPayload = {
|
|
119
|
+
id: 'test-user-id',
|
|
120
|
+
platform: 'testCRM',
|
|
121
|
+
timestamp: Date.now(),
|
|
122
|
+
metadata: {
|
|
123
|
+
timezone: 'America/Los_Angeles',
|
|
124
|
+
preferences: {
|
|
125
|
+
autoLog: true,
|
|
126
|
+
callPop: false
|
|
127
|
+
}
|
|
128
|
+
}
|
|
129
|
+
};
|
|
130
|
+
|
|
131
|
+
// Act
|
|
132
|
+
const token = jwt.generateJwt(complexPayload);
|
|
133
|
+
const decoded = jwt.decodeJwt(token);
|
|
134
|
+
|
|
135
|
+
// Assert
|
|
136
|
+
expect(decoded).toMatchObject(complexPayload);
|
|
137
|
+
expect(decoded).toHaveProperty('exp');
|
|
138
|
+
expect(decoded).toHaveProperty('iat');
|
|
139
|
+
});
|
|
140
|
+
|
|
141
|
+
test('should handle empty payload', () => {
|
|
142
|
+
// Arrange
|
|
143
|
+
const emptyPayload = {};
|
|
144
|
+
|
|
145
|
+
// Act
|
|
146
|
+
const token = jwt.generateJwt(emptyPayload);
|
|
147
|
+
const decoded = jwt.decodeJwt(token);
|
|
148
|
+
|
|
149
|
+
// Assert
|
|
150
|
+
expect(decoded).toMatchObject(emptyPayload);
|
|
151
|
+
expect(decoded).toHaveProperty('exp');
|
|
152
|
+
expect(decoded).toHaveProperty('iat');
|
|
153
|
+
});
|
|
154
|
+
});
|
|
155
|
+
|
|
156
|
+
describe('error handling', () => {
|
|
157
|
+
test('should handle missing secret key', () => {
|
|
158
|
+
// Arrange
|
|
159
|
+
const payload = { id: 'test-user-id' };
|
|
160
|
+
delete process.env.APP_SERVER_SECRET_KEY;
|
|
161
|
+
|
|
162
|
+
// Act & Assert
|
|
163
|
+
expect(() => jwt.generateJwt(payload)).toThrow();
|
|
164
|
+
});
|
|
165
|
+
|
|
166
|
+
test('should handle null payload', () => {
|
|
167
|
+
// Act & Assert
|
|
168
|
+
expect(() => jwt.generateJwt(null)).toThrow();
|
|
169
|
+
});
|
|
170
|
+
|
|
171
|
+
test('should handle undefined payload', () => {
|
|
172
|
+
// Act & Assert
|
|
173
|
+
expect(() => jwt.generateJwt(undefined)).toThrow();
|
|
174
|
+
});
|
|
175
|
+
});
|
|
176
|
+
});
|