@gugananuvem/aws-local-simulator 1.0.9 → 1.0.11
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/README.md +204 -192
- package/bin/aws-local-simulator.js +62 -62
- package/package.json +2 -2
- package/src/config/config-loader.js +112 -112
- package/src/config/default-config.js +65 -65
- package/src/config/env-loader.js +68 -66
- package/src/index.js +130 -130
- package/src/index.mjs +123 -123
- package/src/server.js +221 -219
- package/src/services/apigateway/index.js +66 -66
- package/src/services/apigateway/server.js +434 -434
- package/src/services/apigateway/simulator.js +1251 -1251
- package/src/services/cognito/index.js +65 -65
- package/src/services/cognito/server.js +228 -228
- package/src/services/cognito/simulator.js +847 -847
- package/src/services/dynamodb/index.js +70 -70
- package/src/services/dynamodb/server.js +121 -121
- package/src/services/dynamodb/simulator.js +620 -614
- package/src/services/ecs/index.js +66 -0
- package/src/services/ecs/server.js +234 -0
- package/src/services/ecs/simulator.js +845 -0
- package/src/services/eventbridge/index.js +84 -84
- package/src/services/index.js +18 -18
- package/src/services/lambda/handler-loader.js +172 -172
- package/src/services/lambda/index.js +72 -72
- package/src/services/lambda/route-registry.js +274 -274
- package/src/services/lambda/server.js +152 -152
- package/src/services/lambda/simulator.js +284 -277
- package/src/services/s3/index.js +69 -69
- package/src/services/s3/server.js +238 -238
- package/src/services/s3/simulator.js +740 -740
- package/src/services/sns/index.js +75 -75
- package/src/services/sqs/index.js +95 -95
- package/src/services/sqs/server.js +273 -273
- package/src/services/sqs/simulator.js +659 -659
- package/src/template/aws-config-template.js +87 -87
- package/src/template/aws-config-template.mjs +90 -90
- package/src/template/config-template.json +203 -165
- package/src/utils/aws-config.js +91 -91
- package/src/utils/local-store.js +67 -67
- package/src/utils/logger.js +59 -59
|
@@ -1,278 +1,285 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Lambda Simulator - Simula execução de funções Lambda
|
|
3
|
-
*/
|
|
4
|
-
|
|
5
|
-
const HandlerLoader = require('./handler-loader');
|
|
6
|
-
const RouteRegistry = require('./route-registry');
|
|
7
|
-
const logger = require('../../utils/logger');
|
|
8
|
-
|
|
9
|
-
class LambdaSimulator {
|
|
10
|
-
constructor(config) {
|
|
11
|
-
this.config = config;
|
|
12
|
-
this.routeRegistry = new RouteRegistry();
|
|
13
|
-
this.lambdas = new Map();
|
|
14
|
-
this.environment = { ...process.env };
|
|
15
|
-
}
|
|
16
|
-
|
|
17
|
-
async initialize() {
|
|
18
|
-
logger.debug('Inicializando Lambda Simulator...');
|
|
19
|
-
|
|
20
|
-
if (this.config.lambdas && this.config.lambdas.length > 0) {
|
|
21
|
-
for (const lambdaConfig of this.config.lambdas) {
|
|
22
|
-
await this.registerLambda(lambdaConfig);
|
|
23
|
-
}
|
|
24
|
-
}
|
|
25
|
-
|
|
26
|
-
logger.debug(`✅ ${this.lambdas.size} Lambdas registradas`);
|
|
27
|
-
}
|
|
28
|
-
|
|
29
|
-
async registerLambda(lambdaConfig) {
|
|
30
|
-
try {
|
|
31
|
-
const { path, handler: handlerPath, env = {}, type = 'auto' } = lambdaConfig;
|
|
32
|
-
|
|
33
|
-
// Carrega o handler
|
|
34
|
-
const handler = await HandlerLoader.load(handlerPath, type);
|
|
35
|
-
|
|
36
|
-
// Registra no route registry
|
|
37
|
-
this.routeRegistry.register(path, handler, env);
|
|
38
|
-
|
|
39
|
-
// Armazena metadata
|
|
40
|
-
this.lambdas.set(path, {
|
|
41
|
-
path,
|
|
42
|
-
handler,
|
|
43
|
-
handlerPath,
|
|
44
|
-
handlerName: handler.name || 'anonymous',
|
|
45
|
-
env,
|
|
46
|
-
type,
|
|
47
|
-
registeredAt: new Date().toISOString()
|
|
48
|
-
});
|
|
49
|
-
|
|
50
|
-
logger.debug(`✅ Lambda registrada: ${path} -> ${handler.name || 'anonymous'}`);
|
|
51
|
-
|
|
52
|
-
} catch (error) {
|
|
53
|
-
logger.error(`❌ Erro ao registrar Lambda ${lambdaConfig.path}:`, error);
|
|
54
|
-
throw error;
|
|
55
|
-
}
|
|
56
|
-
}
|
|
57
|
-
|
|
58
|
-
async handleRequest(req, res) {
|
|
59
|
-
const matchedRoute = this.routeRegistry.find(req.path);
|
|
60
|
-
|
|
61
|
-
if (!matchedRoute) {
|
|
62
|
-
return {
|
|
63
|
-
error: {
|
|
64
|
-
statusCode: 404,
|
|
65
|
-
message: `Route not found: ${req.path}`,
|
|
66
|
-
availableRoutes: this.listRoutes()
|
|
67
|
-
},
|
|
68
|
-
status: 404
|
|
69
|
-
};
|
|
70
|
-
}
|
|
71
|
-
|
|
72
|
-
// Aplica variáveis de ambiente específicas da rota
|
|
73
|
-
this.applyEnvironment(matchedRoute.env);
|
|
74
|
-
|
|
75
|
-
// Prepara evento Lambda
|
|
76
|
-
const event = this.toLambdaEvent(req, matchedRoute.params);
|
|
77
|
-
|
|
78
|
-
logger.debug(`🎯 Executando: ${matchedRoute.path} -> ${matchedRoute.handler.name || 'anonymous'}`);
|
|
79
|
-
|
|
80
|
-
// Executa middlewares
|
|
81
|
-
const middlewares = this.routeRegistry.getMiddlewares(matchedRoute);
|
|
82
|
-
let handled = false;
|
|
83
|
-
let result = null;
|
|
84
|
-
|
|
85
|
-
const runMiddlewares = async (index) => {
|
|
86
|
-
if (index >= middlewares.length) {
|
|
87
|
-
// Executa handler
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
|
|
142
|
-
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
|
|
146
|
-
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
|
|
163
|
-
|
|
164
|
-
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
|
|
172
|
-
|
|
173
|
-
};
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
|
|
179
|
-
|
|
180
|
-
|
|
181
|
-
|
|
182
|
-
|
|
183
|
-
|
|
184
|
-
|
|
185
|
-
|
|
186
|
-
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
|
|
191
|
-
|
|
192
|
-
|
|
193
|
-
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
}
|
|
197
|
-
}
|
|
198
|
-
|
|
199
|
-
|
|
200
|
-
|
|
201
|
-
|
|
202
|
-
|
|
203
|
-
|
|
204
|
-
|
|
205
|
-
|
|
206
|
-
|
|
207
|
-
|
|
208
|
-
|
|
209
|
-
|
|
210
|
-
|
|
211
|
-
|
|
212
|
-
|
|
213
|
-
|
|
214
|
-
|
|
215
|
-
|
|
216
|
-
|
|
217
|
-
|
|
218
|
-
|
|
219
|
-
|
|
220
|
-
|
|
221
|
-
|
|
222
|
-
|
|
223
|
-
|
|
224
|
-
|
|
225
|
-
|
|
226
|
-
|
|
227
|
-
|
|
228
|
-
|
|
229
|
-
|
|
230
|
-
|
|
231
|
-
|
|
232
|
-
|
|
233
|
-
|
|
234
|
-
|
|
235
|
-
|
|
236
|
-
|
|
237
|
-
|
|
238
|
-
|
|
239
|
-
|
|
240
|
-
|
|
241
|
-
|
|
242
|
-
|
|
243
|
-
|
|
244
|
-
|
|
245
|
-
|
|
246
|
-
|
|
247
|
-
|
|
248
|
-
|
|
249
|
-
|
|
250
|
-
|
|
251
|
-
|
|
252
|
-
|
|
253
|
-
|
|
254
|
-
|
|
255
|
-
|
|
256
|
-
|
|
257
|
-
|
|
258
|
-
|
|
259
|
-
|
|
260
|
-
|
|
261
|
-
|
|
262
|
-
|
|
263
|
-
|
|
264
|
-
|
|
265
|
-
|
|
266
|
-
|
|
267
|
-
|
|
268
|
-
|
|
269
|
-
|
|
270
|
-
|
|
271
|
-
|
|
272
|
-
|
|
273
|
-
|
|
274
|
-
|
|
275
|
-
|
|
276
|
-
|
|
277
|
-
|
|
1
|
+
/**
|
|
2
|
+
* Lambda Simulator - Simula execução de funções Lambda
|
|
3
|
+
*/
|
|
4
|
+
|
|
5
|
+
const HandlerLoader = require('./handler-loader');
|
|
6
|
+
const RouteRegistry = require('./route-registry');
|
|
7
|
+
const logger = require('../../utils/logger');
|
|
8
|
+
|
|
9
|
+
class LambdaSimulator {
|
|
10
|
+
constructor(config) {
|
|
11
|
+
this.config = config;
|
|
12
|
+
this.routeRegistry = new RouteRegistry();
|
|
13
|
+
this.lambdas = new Map();
|
|
14
|
+
this.environment = { ...process.env };
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
async initialize() {
|
|
18
|
+
logger.debug('Inicializando Lambda Simulator...');
|
|
19
|
+
|
|
20
|
+
if (this.config.lambdas && this.config.lambdas.length > 0) {
|
|
21
|
+
for (const lambdaConfig of this.config.lambdas) {
|
|
22
|
+
await this.registerLambda(lambdaConfig);
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
logger.debug(`✅ ${this.lambdas.size} Lambdas registradas`);
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
async registerLambda(lambdaConfig) {
|
|
30
|
+
try {
|
|
31
|
+
const { path, handler: handlerPath, env = {}, type = 'auto' } = lambdaConfig;
|
|
32
|
+
|
|
33
|
+
// Carrega o handler
|
|
34
|
+
const handler = await HandlerLoader.load(handlerPath, type);
|
|
35
|
+
|
|
36
|
+
// Registra no route registry
|
|
37
|
+
this.routeRegistry.register(path, handler, env);
|
|
38
|
+
|
|
39
|
+
// Armazena metadata
|
|
40
|
+
this.lambdas.set(path, {
|
|
41
|
+
path,
|
|
42
|
+
handler,
|
|
43
|
+
handlerPath,
|
|
44
|
+
handlerName: handler.name || 'anonymous',
|
|
45
|
+
env,
|
|
46
|
+
type,
|
|
47
|
+
registeredAt: new Date().toISOString()
|
|
48
|
+
});
|
|
49
|
+
|
|
50
|
+
logger.debug(`✅ Lambda registrada: ${path} -> ${handler.name || 'anonymous'}`);
|
|
51
|
+
|
|
52
|
+
} catch (error) {
|
|
53
|
+
logger.error(`❌ Erro ao registrar Lambda ${lambdaConfig.path}:`, error);
|
|
54
|
+
throw error;
|
|
55
|
+
}
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
async handleRequest(req, res) {
|
|
59
|
+
const matchedRoute = this.routeRegistry.find(req.path);
|
|
60
|
+
|
|
61
|
+
if (!matchedRoute) {
|
|
62
|
+
return {
|
|
63
|
+
error: {
|
|
64
|
+
statusCode: 404,
|
|
65
|
+
message: `Route not found: ${req.path}`,
|
|
66
|
+
availableRoutes: this.listRoutes()
|
|
67
|
+
},
|
|
68
|
+
status: 404
|
|
69
|
+
};
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
// Aplica variáveis de ambiente específicas da rota
|
|
73
|
+
this.applyEnvironment(matchedRoute.env);
|
|
74
|
+
|
|
75
|
+
// Prepara evento Lambda
|
|
76
|
+
const event = this.toLambdaEvent(req, matchedRoute.params);
|
|
77
|
+
|
|
78
|
+
logger.debug(`🎯 Executando: ${matchedRoute.path} -> ${matchedRoute.handler.name || 'anonymous'}`);
|
|
79
|
+
|
|
80
|
+
// Executa middlewares
|
|
81
|
+
const middlewares = this.routeRegistry.getMiddlewares(matchedRoute);
|
|
82
|
+
let handled = false;
|
|
83
|
+
let result = null;
|
|
84
|
+
|
|
85
|
+
const runMiddlewares = async (index) => {
|
|
86
|
+
if (index >= middlewares.length) {
|
|
87
|
+
// Executa handler
|
|
88
|
+
|
|
89
|
+
result = await this.executeHandler(matchedRoute.handler, event);
|
|
90
|
+
handled = true;
|
|
91
|
+
|
|
92
|
+
console.log(`✅ Resposta: ${result.statusCode}`);
|
|
93
|
+
res
|
|
94
|
+
.status(result.statusCode || 200)
|
|
95
|
+
.set(result.headers || {})
|
|
96
|
+
.send(result.body ? JSON.parse(result.body) : null);
|
|
97
|
+
return;
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
const middleware = middlewares[index];
|
|
101
|
+
await new Promise((resolve, reject) => {
|
|
102
|
+
middleware(event, {
|
|
103
|
+
status: (code) => ({ json: (data) => {
|
|
104
|
+
result = { statusCode: code, body: data };
|
|
105
|
+
handled = true;
|
|
106
|
+
resolve();
|
|
107
|
+
}}),
|
|
108
|
+
send: (data) => {
|
|
109
|
+
result = { statusCode: 200, body: data };
|
|
110
|
+
handled = true;
|
|
111
|
+
resolve();
|
|
112
|
+
},
|
|
113
|
+
next: () => {
|
|
114
|
+
runMiddlewares(index + 1).then(resolve).catch(reject);
|
|
115
|
+
}
|
|
116
|
+
});
|
|
117
|
+
});
|
|
118
|
+
};
|
|
119
|
+
|
|
120
|
+
await runMiddlewares(0);
|
|
121
|
+
|
|
122
|
+
if (!handled && result) {
|
|
123
|
+
return this.formatResponse(result);
|
|
124
|
+
}
|
|
125
|
+
|
|
126
|
+
return null;
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
async executeHandler(handler, event) {
|
|
130
|
+
try {
|
|
131
|
+
const context = this.createContext();
|
|
132
|
+
const result = await handler(event, context);
|
|
133
|
+
return result;
|
|
134
|
+
} catch (error) {
|
|
135
|
+
logger.error('❌ Erro no handler:', error);
|
|
136
|
+
return {
|
|
137
|
+
statusCode: 500,
|
|
138
|
+
body: {
|
|
139
|
+
error: 'Internal Server Error',
|
|
140
|
+
message: error.message,
|
|
141
|
+
stack: process.env.NODE_ENV === 'development' ? error.stack : undefined
|
|
142
|
+
}
|
|
143
|
+
};
|
|
144
|
+
}
|
|
145
|
+
}
|
|
146
|
+
|
|
147
|
+
toLambdaEvent(req, params = {}) {
|
|
148
|
+
return {
|
|
149
|
+
httpMethod: req.method,
|
|
150
|
+
path: req.path,
|
|
151
|
+
headers: req.headers,
|
|
152
|
+
queryStringParameters: req.query,
|
|
153
|
+
pathParameters: params,
|
|
154
|
+
body: req.body ? (typeof req.body === 'string' ? req.body : JSON.stringify(req.body)) : null,
|
|
155
|
+
isBase64Encoded: false,
|
|
156
|
+
requestContext: {
|
|
157
|
+
path: req.path,
|
|
158
|
+
stage: process.env.STAGE_NAME || 'dev',
|
|
159
|
+
requestId: Math.random().toString(36).substring(7),
|
|
160
|
+
identity: {
|
|
161
|
+
sourceIp: req.ip,
|
|
162
|
+
userAgent: req.headers['user-agent']
|
|
163
|
+
}
|
|
164
|
+
},
|
|
165
|
+
stageVariables: {},
|
|
166
|
+
resource: req.path
|
|
167
|
+
};
|
|
168
|
+
}
|
|
169
|
+
|
|
170
|
+
formatResponse(result) {
|
|
171
|
+
const statusCode = result.statusCode || 200;
|
|
172
|
+
const body = result.body;
|
|
173
|
+
const headers = result.headers || { 'Content-Type': 'application/json' };
|
|
174
|
+
|
|
175
|
+
return {
|
|
176
|
+
statusCode,
|
|
177
|
+
headers,
|
|
178
|
+
body: typeof body === 'string' ? body : JSON.stringify(body),
|
|
179
|
+
isBase64Encoded: false
|
|
180
|
+
};
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
createContext() {
|
|
184
|
+
return {
|
|
185
|
+
awsRequestId: Math.random().toString(36).substring(7),
|
|
186
|
+
functionName: 'local-lambda',
|
|
187
|
+
functionVersion: '$LATEST',
|
|
188
|
+
invokedFunctionArn: 'arn:aws:lambda:local:function',
|
|
189
|
+
memoryLimitInMB: '1024',
|
|
190
|
+
logGroupName: '/aws/lambda/local-lambda',
|
|
191
|
+
logStreamName: 'local-stream',
|
|
192
|
+
getRemainingTimeInMillis: () => 30000,
|
|
193
|
+
callbackWaitsForEmptyEventLoop: true,
|
|
194
|
+
identity: null,
|
|
195
|
+
clientContext: null
|
|
196
|
+
};
|
|
197
|
+
}
|
|
198
|
+
|
|
199
|
+
applyEnvironment(env) {
|
|
200
|
+
for (const [key, value] of Object.entries(env)) {
|
|
201
|
+
process.env[key] = value;
|
|
202
|
+
this.environment[key] = value;
|
|
203
|
+
}
|
|
204
|
+
}
|
|
205
|
+
|
|
206
|
+
setEnvironmentVariable(key, value) {
|
|
207
|
+
process.env[key] = value;
|
|
208
|
+
this.environment[key] = value;
|
|
209
|
+
}
|
|
210
|
+
|
|
211
|
+
getEnvironmentVariables() {
|
|
212
|
+
return { ...this.environment };
|
|
213
|
+
}
|
|
214
|
+
|
|
215
|
+
listLambdas() {
|
|
216
|
+
return Array.from(this.lambdas.values()).map(l => ({
|
|
217
|
+
path: l.path,
|
|
218
|
+
handlerName: l.handlerName,
|
|
219
|
+
handlerPath: l.handlerPath,
|
|
220
|
+
type: l.type,
|
|
221
|
+
env: l.env,
|
|
222
|
+
registeredAt: l.registeredAt
|
|
223
|
+
}));
|
|
224
|
+
}
|
|
225
|
+
|
|
226
|
+
getLambda(path) {
|
|
227
|
+
return this.lambdas.get(path);
|
|
228
|
+
}
|
|
229
|
+
|
|
230
|
+
listRoutes() {
|
|
231
|
+
return this.routeRegistry.list();
|
|
232
|
+
}
|
|
233
|
+
|
|
234
|
+
getLambdasCount() {
|
|
235
|
+
return this.lambdas.size;
|
|
236
|
+
}
|
|
237
|
+
|
|
238
|
+
async reloadLambdas() {
|
|
239
|
+
logger.info('🔄 Recarregando Lambdas...');
|
|
240
|
+
|
|
241
|
+
for (const [path, lambda] of this.lambdas.entries()) {
|
|
242
|
+
try {
|
|
243
|
+
const newHandler = await HandlerLoader.reload(lambda.handlerPath, lambda.type);
|
|
244
|
+
this.routeRegistry.register(path, newHandler, lambda.env);
|
|
245
|
+
lambda.handler = newHandler;
|
|
246
|
+
lambda.handlerName = newHandler.name || 'anonymous';
|
|
247
|
+
logger.debug(`✅ Lambda recarregada: ${path}`);
|
|
248
|
+
} catch (error) {
|
|
249
|
+
logger.error(`❌ Erro ao recarregar Lambda ${path}:`, error);
|
|
250
|
+
}
|
|
251
|
+
}
|
|
252
|
+
|
|
253
|
+
logger.info(`✅ ${this.lambdas.size} Lambdas recarregadas`);
|
|
254
|
+
}
|
|
255
|
+
|
|
256
|
+
getStats() {
|
|
257
|
+
const lambdas = this.listLambdas();
|
|
258
|
+
return {
|
|
259
|
+
totalLambdas: lambdas.length,
|
|
260
|
+
lambdas: lambdas.map(l => ({
|
|
261
|
+
path: l.path,
|
|
262
|
+
handler: l.handlerName
|
|
263
|
+
})),
|
|
264
|
+
routes: this.routeRegistry.getStats(),
|
|
265
|
+
environment: Object.keys(this.environment).length
|
|
266
|
+
};
|
|
267
|
+
}
|
|
268
|
+
|
|
269
|
+
async reset() {
|
|
270
|
+
// Recarrega Lambdas
|
|
271
|
+
await this.reloadLambdas();
|
|
272
|
+
|
|
273
|
+
// Limpa variáveis de ambiente customizadas
|
|
274
|
+
for (const key of Object.keys(this.environment)) {
|
|
275
|
+
if (!process.env.hasOwnProperty(key) || key.startsWith('AWS_LOCAL_SIMULATOR_')) {
|
|
276
|
+
delete process.env[key];
|
|
277
|
+
}
|
|
278
|
+
}
|
|
279
|
+
|
|
280
|
+
this.environment = { ...process.env };
|
|
281
|
+
logger.debug('Lambda: Estado resetado');
|
|
282
|
+
}
|
|
283
|
+
}
|
|
284
|
+
|
|
278
285
|
module.exports = LambdaSimulator;
|