@gugananuvem/aws-local-simulator 1.0.10 → 1.0.12

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 (44) hide show
  1. package/README.md +257 -193
  2. package/bin/aws-local-simulator.js +62 -62
  3. package/package.json +2 -2
  4. package/src/config/config-loader.js +114 -112
  5. package/src/config/default-config.js +67 -65
  6. package/src/config/env-loader.js +68 -68
  7. package/src/index.js +130 -130
  8. package/src/index.mjs +123 -123
  9. package/src/server.js +223 -222
  10. package/src/services/apigateway/index.js +68 -66
  11. package/src/services/apigateway/server.js +487 -434
  12. package/src/services/apigateway/simulator.js +1251 -1251
  13. package/src/services/cognito/index.js +65 -65
  14. package/src/services/cognito/server.js +279 -228
  15. package/src/services/cognito/simulator.js +1114 -847
  16. package/src/services/dynamodb/index.js +70 -70
  17. package/src/services/dynamodb/server.js +121 -121
  18. package/src/services/dynamodb/simulator.js +620 -614
  19. package/src/services/ecs/index.js +65 -65
  20. package/src/services/ecs/server.js +233 -233
  21. package/src/services/ecs/simulator.js +844 -844
  22. package/src/services/eventbridge/index.js +84 -84
  23. package/src/services/index.js +18 -18
  24. package/src/services/lambda/handler-loader.js +183 -172
  25. package/src/services/lambda/index.js +73 -72
  26. package/src/services/lambda/route-registry.js +274 -274
  27. package/src/services/lambda/server.js +145 -152
  28. package/src/services/lambda/simulator.js +172 -285
  29. package/src/services/s3/index.js +69 -69
  30. package/src/services/s3/server.js +238 -238
  31. package/src/services/s3/simulator.js +740 -740
  32. package/src/services/sns/index.js +75 -75
  33. package/src/services/sqs/index.js +95 -95
  34. package/src/services/sqs/server.js +345 -273
  35. package/src/services/sqs/simulator.js +441 -660
  36. package/src/services/sts/index.js +37 -0
  37. package/src/services/sts/server.js +142 -0
  38. package/src/services/sts/simulator.js +69 -0
  39. package/src/template/aws-config-template.js +87 -87
  40. package/src/template/aws-config-template.mjs +90 -90
  41. package/src/template/config-template.json +203 -203
  42. package/src/utils/aws-config.js +91 -91
  43. package/src/utils/local-store.js +67 -67
  44. package/src/utils/logger.js +59 -59
@@ -1,285 +1,172 @@
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
-
285
- module.exports = LambdaSimulator;
1
+ /**
2
+ * Lambda Simulator - Simula execução de funções Lambda
3
+ */
4
+
5
+ const HandlerLoader = require('./handler-loader');
6
+ const logger = require('../../utils/logger');
7
+
8
+ class LambdaSimulator {
9
+ constructor(config) {
10
+ this.config = config;
11
+ this.lambdas = new Map(); // functionName -> { handler, env, config }
12
+ this.environment = { ...process.env };
13
+ }
14
+
15
+ async initialize() {
16
+ logger.debug('Inicializando Lambda Simulator...');
17
+
18
+ if (this.config.lambdas && this.config.lambdas.length > 0) {
19
+ for (const lambdaConfig of this.config.lambdas) {
20
+ await this.registerLambda(lambdaConfig);
21
+ }
22
+ }
23
+
24
+ logger.debug(`✅ ${this.lambdas.size} Lambdas registradas`);
25
+ }
26
+
27
+ async registerLambda(lambdaConfig) {
28
+ try {
29
+ const { name, handler: handlerPath, env = {}, type = 'auto' } = lambdaConfig;
30
+
31
+ if (!name) {
32
+ logger.warn(`Lambda sem nome ignorada: ${JSON.stringify(lambdaConfig)}`);
33
+ return;
34
+ }
35
+
36
+ const handler = await HandlerLoader.load(handlerPath, type);
37
+
38
+ this.lambdas.set(name, {
39
+ name,
40
+ handler,
41
+ handlerPath,
42
+ handlerName: handler.name || 'anonymous',
43
+ env,
44
+ type,
45
+ registeredAt: new Date().toISOString()
46
+ });
47
+
48
+ logger.debug(`✅ Lambda registrada: ${name} -> ${handlerPath}`);
49
+ } catch (error) {
50
+ logger.error(`❌ Erro ao registrar Lambda ${lambdaConfig.name}:`, error);
51
+ throw error;
52
+ }
53
+ }
54
+
55
+ async invoke(functionName, event, invocationType = 'RequestResponse') {
56
+ const lambda = this.lambdas.get(functionName);
57
+
58
+ if (!lambda) {
59
+ throw new Error(`Function not found: ${functionName}`);
60
+ }
61
+
62
+ this.applyEnvironment(lambda.env);
63
+ logger.debug(`🎯 Invocando Lambda: ${functionName}`);
64
+
65
+ if (invocationType === 'Event') {
66
+ this.executeHandler(lambda.handler, event).catch(err =>
67
+ logger.error(`❌ Async Lambda error (${functionName}):`, err)
68
+ );
69
+ return { StatusCode: 202 };
70
+ }
71
+
72
+ const result = await this.executeHandler(lambda.handler, event);
73
+ return { StatusCode: result.statusCode || 200, Payload: result };
74
+ }
75
+
76
+ async executeHandler(handler, event) {
77
+ try {
78
+ const context = this.createContext();
79
+ const result = await handler(event, context);
80
+ return result;
81
+ } catch (error) {
82
+ logger.error('❌ Erro no handler:', error);
83
+ return {
84
+ statusCode: 500,
85
+ body: JSON.stringify({ error: 'Internal Server Error', message: error.message })
86
+ };
87
+ }
88
+ }
89
+
90
+ createContext() {
91
+ return {
92
+ awsRequestId: Math.random().toString(36).substring(7),
93
+ functionName: 'local-lambda',
94
+ functionVersion: '$LATEST',
95
+ invokedFunctionArn: 'arn:aws:lambda:local:000000000000:function:local-lambda',
96
+ memoryLimitInMB: '1024',
97
+ logGroupName: '/aws/lambda/local-lambda',
98
+ logStreamName: 'local-stream',
99
+ getRemainingTimeInMillis: () => 30000,
100
+ callbackWaitsForEmptyEventLoop: true,
101
+ identity: null,
102
+ clientContext: null
103
+ };
104
+ }
105
+
106
+ applyEnvironment(env) {
107
+ for (const [key, value] of Object.entries(env)) {
108
+ process.env[key] = value;
109
+ this.environment[key] = value;
110
+ }
111
+ }
112
+
113
+ setEnvironmentVariable(key, value) {
114
+ process.env[key] = value;
115
+ this.environment[key] = value;
116
+ }
117
+
118
+ getEnvironmentVariables() {
119
+ return { ...this.environment };
120
+ }
121
+
122
+ listLambdas() {
123
+ return Array.from(this.lambdas.values()).map(l => ({
124
+ name: l.name,
125
+ handlerName: l.handlerName,
126
+ handlerPath: l.handlerPath,
127
+ type: l.type,
128
+ env: l.env,
129
+ registeredAt: l.registeredAt
130
+ }));
131
+ }
132
+
133
+ getLambda(name) {
134
+ return this.lambdas.get(name);
135
+ }
136
+
137
+ getLambdasCount() {
138
+ return this.lambdas.size;
139
+ }
140
+
141
+ async reloadLambdas() {
142
+ logger.info('🔄 Recarregando Lambdas...');
143
+
144
+ for (const [name, lambda] of this.lambdas.entries()) {
145
+ try {
146
+ const newHandler = await HandlerLoader.reload(lambda.handlerPath, lambda.type);
147
+ lambda.handler = newHandler;
148
+ lambda.handlerName = newHandler.name || 'anonymous';
149
+ logger.debug(`✅ Lambda recarregada: ${name}`);
150
+ } catch (error) {
151
+ logger.error(`❌ Erro ao recarregar Lambda ${name}:`, error);
152
+ }
153
+ }
154
+
155
+ logger.info(`✅ ${this.lambdas.size} Lambdas recarregadas`);
156
+ }
157
+
158
+ getStats() {
159
+ return {
160
+ totalLambdas: this.lambdas.size,
161
+ lambdas: this.listLambdas().map(l => ({ name: l.name, handler: l.handlerName }))
162
+ };
163
+ }
164
+
165
+ async reset() {
166
+ await this.reloadLambdas();
167
+ this.environment = { ...process.env };
168
+ logger.debug('Lambda: Estado resetado');
169
+ }
170
+ }
171
+
172
+ module.exports = LambdaSimulator;
@@ -1,70 +1,70 @@
1
- /**
2
- * S3 Service - Ponto de entrada
3
- * Exporta o serviço principal e seus componentes
4
- */
5
-
6
- const S3Server = require('./server');
7
- const S3Simulator = require('./simulator');
8
-
9
- class S3Service {
10
- constructor(config) {
11
- this.config = config;
12
- this.name = 's3';
13
- this.port = config.ports.s3;
14
- this.server = null;
15
- this.simulator = null;
16
- this.isRunning = false;
17
- }
18
-
19
- async initialize() {
20
- const logger = require('../../utils/logger');
21
- logger.debug(`Inicializando S3 Service na porta ${this.port}...`);
22
-
23
- // Cria o simulador
24
- this.simulator = new S3Simulator(this.config);
25
-
26
- // Cria o servidor HTTP
27
- this.server = new S3Server(this.port, this.config);
28
- this.server.simulator = this.simulator;
29
-
30
- await this.server.initialize();
31
-
32
- logger.debug('S3 Service inicializado');
33
- }
34
-
35
- async start() {
36
- if (this.isRunning) return;
37
- await this.server.start();
38
- this.isRunning = true;
39
- }
40
-
41
- async stop() {
42
- if (!this.isRunning) return;
43
- await this.server.stop();
44
- this.isRunning = false;
45
- }
46
-
47
- async reset() {
48
- await this.simulator.reset();
49
- }
50
-
51
- getStatus() {
52
- return {
53
- running: this.isRunning,
54
- port: this.port,
55
- endpoint: `http://localhost:${this.port}`,
56
- bucketsCount: this.simulator?.getBucketsCount() || 0,
57
- objectsCount: this.simulator?.getTotalObjectsCount() || 0
58
- };
59
- }
60
-
61
- getSimulator() {
62
- return this.simulator;
63
- }
64
-
65
- getServer() {
66
- return this.server;
67
- }
68
- }
69
-
1
+ /**
2
+ * S3 Service - Ponto de entrada
3
+ * Exporta o serviço principal e seus componentes
4
+ */
5
+
6
+ const S3Server = require('./server');
7
+ const S3Simulator = require('./simulator');
8
+
9
+ class S3Service {
10
+ constructor(config) {
11
+ this.config = config;
12
+ this.name = 's3';
13
+ this.port = config.ports.s3;
14
+ this.server = null;
15
+ this.simulator = null;
16
+ this.isRunning = false;
17
+ }
18
+
19
+ async initialize() {
20
+ const logger = require('../../utils/logger');
21
+ logger.debug(`Inicializando S3 Service na porta ${this.port}...`);
22
+
23
+ // Cria o simulador
24
+ this.simulator = new S3Simulator(this.config);
25
+
26
+ // Cria o servidor HTTP
27
+ this.server = new S3Server(this.port, this.config);
28
+ this.server.simulator = this.simulator;
29
+
30
+ await this.server.initialize();
31
+
32
+ logger.debug('S3 Service inicializado');
33
+ }
34
+
35
+ async start() {
36
+ if (this.isRunning) return;
37
+ await this.server.start();
38
+ this.isRunning = true;
39
+ }
40
+
41
+ async stop() {
42
+ if (!this.isRunning) return;
43
+ await this.server.stop();
44
+ this.isRunning = false;
45
+ }
46
+
47
+ async reset() {
48
+ await this.simulator.reset();
49
+ }
50
+
51
+ getStatus() {
52
+ return {
53
+ running: this.isRunning,
54
+ port: this.port,
55
+ endpoint: `http://localhost:${this.port}`,
56
+ bucketsCount: this.simulator?.getBucketsCount() || 0,
57
+ objectsCount: this.simulator?.getTotalObjectsCount() || 0
58
+ };
59
+ }
60
+
61
+ getSimulator() {
62
+ return this.simulator;
63
+ }
64
+
65
+ getServer() {
66
+ return this.server;
67
+ }
68
+ }
69
+
70
70
  module.exports = S3Service;