@gugananuvem/aws-local-simulator 1.0.0 → 1.0.1

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 (38) hide show
  1. package/README.md +192 -192
  2. package/bin/aws-local-simulator.js +62 -62
  3. package/package.json +8 -5
  4. package/src/config/config-loader.js +113 -0
  5. package/src/config/default-config.js +65 -0
  6. package/src/config/env-loader.js +67 -0
  7. package/src/index.js +130 -130
  8. package/src/index.mjs +124 -0
  9. package/src/server.js +219 -0
  10. package/src/services/apigateway/index.js +67 -0
  11. package/src/services/apigateway/server.js +435 -0
  12. package/src/services/apigateway/simulator.js +1252 -0
  13. package/src/services/cognito/index.js +66 -0
  14. package/src/services/cognito/server.js +229 -0
  15. package/src/services/cognito/simulator.js +848 -0
  16. package/src/services/dynamodb/index.js +71 -0
  17. package/src/services/dynamodb/server.js +122 -0
  18. package/src/services/dynamodb/simulator.js +614 -0
  19. package/src/services/eventbridge/index.js +85 -0
  20. package/src/services/index.js +19 -0
  21. package/src/services/lambda/handler-loader.js +173 -0
  22. package/src/services/lambda/index.js +73 -0
  23. package/src/services/lambda/route-registry.js +275 -0
  24. package/src/services/lambda/server.js +153 -0
  25. package/src/services/lambda/simulator.js +278 -0
  26. package/src/services/s3/index.js +70 -0
  27. package/src/services/s3/server.js +239 -0
  28. package/src/services/s3/simulator.js +740 -0
  29. package/src/services/sns/index.js +76 -0
  30. package/src/services/sqs/index.js +96 -0
  31. package/src/services/sqs/server.js +274 -0
  32. package/src/services/sqs/simulator.js +660 -0
  33. package/src/template/aws-config-template.js +88 -0
  34. package/src/template/aws-config-template.mjs +91 -0
  35. package/src/template/config-template.json +165 -0
  36. package/src/utils/aws-config.js +92 -0
  37. package/src/utils/local-store.js +68 -0
  38. package/src/utils/logger.js +60 -0
@@ -0,0 +1,76 @@
1
+ /**
2
+ * SNS Service - Ponto de entrada (Stub para implementação futura)
3
+ */
4
+
5
+ const logger = require('../../utils/logger');
6
+
7
+ class SNSService {
8
+ constructor(config) {
9
+ this.config = config;
10
+ this.name = 'sns';
11
+ this.port = config.ports.sns;
12
+ this.isRunning = false;
13
+ this.topics = new Map();
14
+ }
15
+
16
+ async initialize() {
17
+ logger.debug(`Inicializando SNS Service na porta ${this.port}...`);
18
+ logger.warn('⚠️ SNS Service ainda não está completamente implementado');
19
+
20
+ // TODO: Implementar SNS simulator
21
+ // Por enquanto, apenas cria um stub
22
+ this.topics = new Map();
23
+ }
24
+
25
+ async start() {
26
+ if (this.isRunning) return;
27
+
28
+ // TODO: Iniciar servidor HTTP para SNS
29
+ this.isRunning = true;
30
+ logger.info(`📢 SNS Service stub rodando (porta ${this.port}) - Implementação em breve`);
31
+ }
32
+
33
+ async stop() {
34
+ if (!this.isRunning) return;
35
+ this.isRunning = false;
36
+ }
37
+
38
+ async reset() {
39
+ this.topics.clear();
40
+ logger.debug('SNS: Todos os dados resetados');
41
+ }
42
+
43
+ getStatus() {
44
+ return {
45
+ running: this.isRunning,
46
+ port: this.port,
47
+ endpoint: `http://localhost:${this.port}`,
48
+ implemented: false,
49
+ topicsCount: this.topics.size
50
+ };
51
+ }
52
+
53
+ // Métodos stub para compatibilidade
54
+ async createTopic(topicName) {
55
+ if (!this.topics.has(topicName)) {
56
+ this.topics.set(topicName, {
57
+ name: topicName,
58
+ arn: `arn:aws:sns:local:000000000000:${topicName}`,
59
+ subscriptions: [],
60
+ createdAt: new Date().toISOString()
61
+ });
62
+ }
63
+ return this.topics.get(topicName);
64
+ }
65
+
66
+ async publish(topicArn, message) {
67
+ const topic = Array.from(this.topics.values()).find(t => t.arn === topicArn);
68
+ if (!topic) {
69
+ throw new Error(`Topic not found: ${topicArn}`);
70
+ }
71
+ logger.verboso(`SNS: Published message to ${topic.name}`);
72
+ return { MessageId: Math.random().toString(36).substring(7) };
73
+ }
74
+ }
75
+
76
+ module.exports = SNSService;
@@ -0,0 +1,96 @@
1
+ /**
2
+ * SQS Service - Ponto de entrada
3
+ * Exporta o serviço principal e seus componentes
4
+ */
5
+
6
+ const SQSServer = require('./server');
7
+ const SQSSimulator = require('./simulator');
8
+
9
+ class SQSService {
10
+ constructor(config, dependencies = {}) {
11
+ this.config = config;
12
+ this.lambdaService = dependencies.lambda;
13
+ this.name = 'sqs';
14
+ this.port = config.ports.sqs;
15
+ this.server = null;
16
+ this.simulator = null;
17
+ this.isRunning = false;
18
+ }
19
+
20
+ async initialize() {
21
+ const logger = require('../../utils/logger');
22
+ logger.debug(`Inicializando SQS Service na porta ${this.port}...`);
23
+
24
+ // Cria o simulador
25
+ this.simulator = new SQSSimulator(this.config, this.lambdaService);
26
+
27
+ // Cria o servidor HTTP
28
+ this.server = new SQSServer(this.port, this.config, this.lambdaService);
29
+ this.server.simulator = this.simulator;
30
+
31
+ await this.server.initialize();
32
+
33
+ // Configura filas associadas a Lambdas
34
+ await this.setupQueueTriggers();
35
+
36
+ logger.debug('SQS Service inicializado');
37
+ }
38
+
39
+ async setupQueueTriggers() {
40
+ if (!this.config.sqs?.queues) return;
41
+
42
+ const logger = require('../../utils/logger');
43
+
44
+ for (const queueConfig of this.config.sqs.queues) {
45
+ if (queueConfig.lambdaPath && this.lambdaService) {
46
+ const handler = this.lambdaService.getHandler(queueConfig.lambdaPath);
47
+ if (handler) {
48
+ this.simulator.attachLambdaToQueue(
49
+ queueConfig.name,
50
+ handler,
51
+ { batchSize: queueConfig.batchSize || 10 }
52
+ );
53
+ logger.debug(`🔗 Fila ${queueConfig.name} -> Lambda ${queueConfig.lambdaPath}`);
54
+ } else {
55
+ logger.warn(`⚠️ Lambda não encontrada para fila ${queueConfig.name}: ${queueConfig.lambdaPath}`);
56
+ }
57
+ }
58
+ }
59
+ }
60
+
61
+ async start() {
62
+ if (this.isRunning) return;
63
+ await this.server.start();
64
+ this.isRunning = true;
65
+ }
66
+
67
+ async stop() {
68
+ if (!this.isRunning) return;
69
+ await this.server.stop();
70
+ this.isRunning = false;
71
+ }
72
+
73
+ async reset() {
74
+ await this.simulator.reset();
75
+ }
76
+
77
+ getStatus() {
78
+ return {
79
+ running: this.isRunning,
80
+ port: this.port,
81
+ endpoint: `http://localhost:${this.port}`,
82
+ queuesCount: this.simulator?.getQueuesCount() || 0,
83
+ messagesCount: this.simulator?.getTotalMessagesCount() || 0
84
+ };
85
+ }
86
+
87
+ getSimulator() {
88
+ return this.simulator;
89
+ }
90
+
91
+ getServer() {
92
+ return this.server;
93
+ }
94
+ }
95
+
96
+ module.exports = SQSService;
@@ -0,0 +1,274 @@
1
+ /**
2
+ * SQS Server - Servidor HTTP para SQS
3
+ */
4
+
5
+ const express = require('express');
6
+ const SQSSimulator = require('./simulator');
7
+ const logger = require('../../utils/logger');
8
+
9
+ class SQSServer {
10
+ constructor(port, config, lambdaService) {
11
+ this.port = port;
12
+ this.config = config;
13
+ this.lambdaService = lambdaService;
14
+ this.app = express();
15
+ this.simulator = null;
16
+ this.server = null;
17
+ this.setupMiddlewares();
18
+ }
19
+
20
+ setupMiddlewares() {
21
+ this.app.use(express.json());
22
+ this.app.use(express.urlencoded({ extended: true }));
23
+
24
+ // Logging de requisições
25
+ if (logger.currentLogLevel === 'verboso') {
26
+ this.app.use((req, res, next) => {
27
+ const start = Date.now();
28
+ res.on('finish', () => {
29
+ const duration = Date.now() - start;
30
+ logger.verboso(`SQS: ${req.query.Action || req.method} - ${duration}ms`);
31
+ });
32
+ next();
33
+ });
34
+ }
35
+ }
36
+
37
+ async initialize() {
38
+ this.simulator = new SQSSimulator(this.config, this.lambdaService);
39
+ await this.simulator.initialize();
40
+ this.setupRoutes();
41
+ }
42
+
43
+ setupRoutes() {
44
+ // Endpoint principal
45
+ this.app.post('/', (req, res) => {
46
+ const action = req.query.Action || req.body.Action;
47
+ const result = this.simulator.handleRequest(action, req, res);
48
+
49
+ if (result && result.error) {
50
+ res.status(result.status).send(this.simulator.generateErrorResponse(result.error.code, result.error.message));
51
+ } else if (result) {
52
+ // Gera resposta XML
53
+ res.set('Content-Type', 'application/xml');
54
+ res.send(this.generateResponse(action, result));
55
+ }
56
+ });
57
+
58
+ // Endpoint alternativo com nome da fila
59
+ this.app.post('/queue/:queueName', (req, res) => {
60
+ const action = req.query.Action;
61
+ if (action) {
62
+ // Adiciona o nome da fila ao body
63
+ req.body.QueueName = req.params.queueName;
64
+ this.app.handle(req, res);
65
+ } else {
66
+ res.status(400).send(this.simulator.generateErrorResponse('InvalidAction', 'Missing Action parameter'));
67
+ }
68
+ });
69
+
70
+ // Admin endpoints
71
+ this.setupAdminRoutes();
72
+ }
73
+
74
+ setupAdminRoutes() {
75
+ this.app.get('/__admin/queues', (req, res) => {
76
+ res.json(this.simulator.listQueues());
77
+ });
78
+
79
+ this.app.get('/__admin/queues/:queueName', (req, res) => {
80
+ const queue = this.simulator.getQueue(req.params.queueName);
81
+ if (queue) {
82
+ res.json(queue);
83
+ } else {
84
+ res.status(404).json({ error: 'Queue not found' });
85
+ }
86
+ });
87
+
88
+ this.app.get('/__admin/queues/:queueName/messages', (req, res) => {
89
+ const messages = this.simulator.getMessages(req.params.queueName);
90
+ res.json(messages);
91
+ });
92
+
93
+ this.app.delete('/__admin/queues/:queueName', (req, res) => {
94
+ this.simulator.deleteQueue(req.params.queueName);
95
+ res.json({ message: `Queue ${req.params.queueName} deleted` });
96
+ });
97
+
98
+ this.app.delete('/__admin/queues/:queueName/messages', (req, res) => {
99
+ this.simulator.purgeQueue(req.params.queueName);
100
+ res.json({ message: `Queue ${req.params.queueName} purged` });
101
+ });
102
+
103
+ this.app.get('/__admin/stats', (req, res) => {
104
+ res.json(this.simulator.getStats());
105
+ });
106
+ }
107
+
108
+ generateResponse(action, result) {
109
+ switch(action) {
110
+ case 'CreateQueue':
111
+ return this.generateCreateQueueResponse(result.queueUrl);
112
+ case 'SendMessage':
113
+ return this.generateSendMessageResponse(result.messageId, result.md5);
114
+ case 'SendMessageBatch':
115
+ return this.generateSendMessageBatchResponse(result.successful, result.failed);
116
+ case 'ReceiveMessage':
117
+ return this.generateReceiveMessageResponse(result.messages);
118
+ case 'DeleteMessage':
119
+ return this.generateDeleteMessageResponse();
120
+ case 'GetQueueUrl':
121
+ return this.generateGetQueueUrlResponse(result.queueUrl);
122
+ case 'ListQueues':
123
+ return this.generateListQueuesResponse(result.queues);
124
+ default:
125
+ return '';
126
+ }
127
+ }
128
+
129
+ generateCreateQueueResponse(queueUrl) {
130
+ return `<?xml version="1.0" encoding="UTF-8"?>
131
+ <CreateQueueResponse>
132
+ <CreateQueueResult>
133
+ <QueueUrl>${queueUrl}</QueueUrl>
134
+ </CreateQueueResult>
135
+ </CreateQueueResponse>`;
136
+ }
137
+
138
+ generateSendMessageResponse(messageId, md5) {
139
+ return `<?xml version="1.0" encoding="UTF-8"?>
140
+ <SendMessageResponse>
141
+ <SendMessageResult>
142
+ <MD5OfMessageBody>${md5}</MD5OfMessageBody>
143
+ <MessageId>${messageId}</MessageId>
144
+ </SendMessageResult>
145
+ </SendMessageResponse>`;
146
+ }
147
+
148
+ generateSendMessageBatchResponse(successful, failed) {
149
+ let xml = `<?xml version="1.0" encoding="UTF-8"?>
150
+ <SendMessageBatchResponse>
151
+ <SendMessageBatchResult>`;
152
+
153
+ for (const s of successful) {
154
+ xml += `
155
+ <SendMessageBatchResultEntry>
156
+ <Id>${s.Id}</Id>
157
+ <MessageId>${s.MessageId}</MessageId>
158
+ <MD5OfMessageBody>${s.MD5OfMessageBody}</MD5OfMessageBody>
159
+ </SendMessageBatchResultEntry>`;
160
+ }
161
+
162
+ for (const f of failed) {
163
+ xml += `
164
+ <BatchResultErrorEntry>
165
+ <Id>${f.Id}</Id>
166
+ <Code>${f.Code}</Code>
167
+ <Message>${f.Message}</Message>
168
+ </BatchResultErrorEntry>`;
169
+ }
170
+
171
+ xml += `
172
+ </SendMessageBatchResult>
173
+ </SendMessageBatchResponse>`;
174
+
175
+ return xml;
176
+ }
177
+
178
+ generateReceiveMessageResponse(messages) {
179
+ let xml = `<?xml version="1.0" encoding="UTF-8"?>
180
+ <ReceiveMessageResponse>
181
+ <ReceiveMessageResult>`;
182
+
183
+ for (const msg of messages) {
184
+ xml += `
185
+ <Message>
186
+ <MessageId>${msg.MessageId}</MessageId>
187
+ <ReceiptHandle>${msg.ReceiptHandle}</ReceiptHandle>
188
+ <MD5OfBody>${msg.MD5OfBody}</MD5OfBody>
189
+ <Body>${this.escapeXml(msg.Body)}</Body>
190
+ </Message>`;
191
+ }
192
+
193
+ xml += `
194
+ </ReceiveMessageResult>
195
+ </ReceiveMessageResponse>`;
196
+
197
+ return xml;
198
+ }
199
+
200
+ generateDeleteMessageResponse() {
201
+ return `<?xml version="1.0" encoding="UTF-8"?>
202
+ <DeleteMessageResponse>
203
+ <ResponseMetadata>
204
+ <RequestId>${Math.random().toString(36).substring(7)}</RequestId>
205
+ </ResponseMetadata>
206
+ </DeleteMessageResponse>`;
207
+ }
208
+
209
+ generateGetQueueUrlResponse(queueUrl) {
210
+ return `<?xml version="1.0" encoding="UTF-8"?>
211
+ <GetQueueUrlResponse>
212
+ <GetQueueUrlResult>
213
+ <QueueUrl>${queueUrl}</QueueUrl>
214
+ </GetQueueUrlResult>
215
+ </GetQueueUrlResponse>`;
216
+ }
217
+
218
+ generateListQueuesResponse(queues) {
219
+ let xml = `<?xml version="1.0" encoding="UTF-8"?>
220
+ <ListQueuesResponse>
221
+ <ListQueuesResult>`;
222
+
223
+ for (const queue of queues) {
224
+ xml += `<QueueUrl>${queue.url}</QueueUrl>`;
225
+ }
226
+
227
+ xml += `
228
+ </ListQueuesResult>
229
+ </ListQueuesResponse>`;
230
+
231
+ return xml;
232
+ }
233
+
234
+ escapeXml(str) {
235
+ if (!str) return '';
236
+ return str
237
+ .replace(/&/g, '&amp;')
238
+ .replace(/</g, '&lt;')
239
+ .replace(/>/g, '&gt;')
240
+ .replace(/"/g, '&quot;')
241
+ .replace(/'/g, '&apos;');
242
+ }
243
+
244
+ start() {
245
+ return new Promise((resolve) => {
246
+ this.server = this.app.listen(this.port, () => {
247
+ logger.info(`📦 SQS rodando em http://localhost:${this.port}`);
248
+ resolve();
249
+ });
250
+ });
251
+ }
252
+
253
+ stop() {
254
+ return new Promise((resolve) => {
255
+ if (this.server) {
256
+ this.server.close(() => resolve());
257
+ } else {
258
+ resolve();
259
+ }
260
+ });
261
+ }
262
+
263
+ getStatus() {
264
+ return {
265
+ running: !!this.server,
266
+ port: this.port,
267
+ endpoint: `http://localhost:${this.port}`,
268
+ queuesCount: this.simulator?.getQueuesCount() || 0,
269
+ messagesCount: this.simulator?.getTotalMessagesCount() || 0
270
+ };
271
+ }
272
+ }
273
+
274
+ module.exports = SQSServer;