@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.
- package/README.md +192 -192
- package/bin/aws-local-simulator.js +62 -62
- package/package.json +8 -5
- package/src/config/config-loader.js +113 -0
- package/src/config/default-config.js +65 -0
- package/src/config/env-loader.js +67 -0
- package/src/index.js +130 -130
- package/src/index.mjs +124 -0
- package/src/server.js +219 -0
- package/src/services/apigateway/index.js +67 -0
- package/src/services/apigateway/server.js +435 -0
- package/src/services/apigateway/simulator.js +1252 -0
- package/src/services/cognito/index.js +66 -0
- package/src/services/cognito/server.js +229 -0
- package/src/services/cognito/simulator.js +848 -0
- package/src/services/dynamodb/index.js +71 -0
- package/src/services/dynamodb/server.js +122 -0
- package/src/services/dynamodb/simulator.js +614 -0
- package/src/services/eventbridge/index.js +85 -0
- package/src/services/index.js +19 -0
- package/src/services/lambda/handler-loader.js +173 -0
- package/src/services/lambda/index.js +73 -0
- package/src/services/lambda/route-registry.js +275 -0
- package/src/services/lambda/server.js +153 -0
- package/src/services/lambda/simulator.js +278 -0
- package/src/services/s3/index.js +70 -0
- package/src/services/s3/server.js +239 -0
- package/src/services/s3/simulator.js +740 -0
- package/src/services/sns/index.js +76 -0
- package/src/services/sqs/index.js +96 -0
- package/src/services/sqs/server.js +274 -0
- package/src/services/sqs/simulator.js +660 -0
- package/src/template/aws-config-template.js +88 -0
- package/src/template/aws-config-template.mjs +91 -0
- package/src/template/config-template.json +165 -0
- package/src/utils/aws-config.js +92 -0
- package/src/utils/local-store.js +68 -0
- 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, '&')
|
|
238
|
+
.replace(/</g, '<')
|
|
239
|
+
.replace(/>/g, '>')
|
|
240
|
+
.replace(/"/g, '"')
|
|
241
|
+
.replace(/'/g, ''');
|
|
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;
|