@gugananuvem/aws-local-simulator 1.0.12 → 1.0.14
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 +235 -11
- package/package.json +12 -2
- package/src/config/default-config.js +1 -0
- package/src/index.js +18 -2
- package/src/server.js +36 -32
- package/src/services/apigateway/index.js +5 -0
- package/src/services/apigateway/server.js +20 -0
- package/src/services/apigateway/simulator.js +13 -3
- package/src/services/athena/index.js +75 -0
- package/src/services/athena/server.js +101 -0
- package/src/services/athena/simulador.js +998 -0
- package/src/services/athena/simulator.js +346 -0
- package/src/services/cloudformation/index.js +106 -0
- package/src/services/cloudformation/server.js +417 -0
- package/src/services/cloudformation/simulador.js +1045 -0
- package/src/services/cloudtrail/index.js +84 -0
- package/src/services/cloudtrail/server.js +235 -0
- package/src/services/cloudtrail/simulador.js +719 -0
- package/src/services/cloudwatch/index.js +84 -0
- package/src/services/cloudwatch/server.js +366 -0
- package/src/services/cloudwatch/simulador.js +1173 -0
- package/src/services/cognito/index.js +5 -0
- package/src/services/cognito/simulator.js +4 -0
- package/src/services/config/index.js +96 -0
- package/src/services/config/server.js +215 -0
- package/src/services/config/simulador.js +1260 -0
- package/src/services/dynamodb/index.js +7 -3
- package/src/services/dynamodb/server.js +4 -2
- package/src/services/dynamodb/simulator.js +39 -29
- package/src/services/eventbridge/index.js +55 -51
- package/src/services/eventbridge/server.js +209 -0
- package/src/services/eventbridge/simulator.js +684 -0
- package/src/services/index.js +30 -4
- package/src/services/kms/index.js +75 -0
- package/src/services/kms/server.js +67 -0
- package/src/services/kms/simulator.js +324 -0
- package/src/services/lambda/index.js +5 -0
- package/src/services/lambda/simulator.js +48 -38
- package/src/services/parameter-store/index.js +80 -0
- package/src/services/parameter-store/server.js +50 -0
- package/src/services/parameter-store/simulator.js +201 -0
- package/src/services/s3/index.js +7 -3
- package/src/services/s3/server.js +20 -13
- package/src/services/s3/simulator.js +163 -407
- package/src/services/secret-manager/index.js +80 -0
- package/src/services/secret-manager/server.js +50 -0
- package/src/services/secret-manager/simulator.js +171 -0
- package/src/services/sns/index.js +55 -42
- package/src/services/sns/server.js +580 -0
- package/src/services/sns/simulator.js +1482 -0
- package/src/services/sqs/index.js +2 -4
- package/src/services/sqs/server.js +4 -2
- package/src/services/xray/index.js +83 -0
- package/src/services/xray/server.js +308 -0
- package/src/services/xray/simulador.js +994 -0
- package/src/utils/cloudtrail-audit.js +129 -0
- package/src/utils/local-store.js +18 -2
|
@@ -0,0 +1,129 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* CloudTrail Audit Helper
|
|
5
|
+
*
|
|
6
|
+
* Utilitário para registrar eventos de API nos simuladores.
|
|
7
|
+
* Cada simulador recebe uma instância via injectDependencies.
|
|
8
|
+
*
|
|
9
|
+
* Uso:
|
|
10
|
+
* this.audit = new CloudTrailAudit('s3.amazonaws.com');
|
|
11
|
+
* // após injeção:
|
|
12
|
+
* this.audit.setTrail(cloudtrailSimulator);
|
|
13
|
+
* // ao executar operação:
|
|
14
|
+
* this.audit.record({ eventName: 'PutObject', readOnly: false, ... });
|
|
15
|
+
*/
|
|
16
|
+
|
|
17
|
+
class CloudTrailAudit {
|
|
18
|
+
/**
|
|
19
|
+
* @param {string} eventSource - ex: 's3.amazonaws.com'
|
|
20
|
+
*/
|
|
21
|
+
constructor(eventSource) {
|
|
22
|
+
this.eventSource = eventSource;
|
|
23
|
+
this.trail = null;
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
/** @param {Object|null} cloudtrailSimulator */
|
|
27
|
+
setTrail(cloudtrailSimulator) {
|
|
28
|
+
this.trail = cloudtrailSimulator || null;
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
/**
|
|
32
|
+
* Registra um evento se o CloudTrail estiver ativo e o tipo de evento
|
|
33
|
+
* estiver habilitado nos event selectors do trail.
|
|
34
|
+
*
|
|
35
|
+
* @param {Object} params
|
|
36
|
+
* @param {string} params.eventName
|
|
37
|
+
* @param {boolean} [params.readOnly=false]
|
|
38
|
+
* @param {Array} [params.resources=[]]
|
|
39
|
+
* @param {Object} [params.requestParameters=null]
|
|
40
|
+
* @param {Object} [params.responseElements=null]
|
|
41
|
+
* @param {string} [params.username='local-user']
|
|
42
|
+
* @param {string} [params.sourceIPAddress='127.0.0.1']
|
|
43
|
+
*/
|
|
44
|
+
record(params = {}) {
|
|
45
|
+
if (!this.trail) return;
|
|
46
|
+
|
|
47
|
+
try {
|
|
48
|
+
// Verifica se há algum trail com logging ativo
|
|
49
|
+
const hasActiveTrail = this._hasActiveTrail();
|
|
50
|
+
if (!hasActiveTrail) return;
|
|
51
|
+
|
|
52
|
+
// Verifica se o evento deve ser gravado (management vs data event)
|
|
53
|
+
if (!this._shouldRecord(params)) return;
|
|
54
|
+
|
|
55
|
+
this.trail.recordEvent({
|
|
56
|
+
eventSource: this.eventSource,
|
|
57
|
+
eventName: params.eventName,
|
|
58
|
+
readOnly: params.readOnly !== undefined ? params.readOnly : false,
|
|
59
|
+
resources: params.resources || [],
|
|
60
|
+
requestParameters: params.requestParameters || null,
|
|
61
|
+
responseElements: params.responseElements || null,
|
|
62
|
+
username: params.username || 'local-user',
|
|
63
|
+
sourceIPAddress: params.sourceIPAddress || '127.0.0.1',
|
|
64
|
+
});
|
|
65
|
+
} catch (_) {
|
|
66
|
+
// Nunca deixa o audit quebrar o fluxo principal
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
_hasActiveTrail() {
|
|
71
|
+
if (!this.trail || !this.trail.trailStatus) return false;
|
|
72
|
+
for (const status of this.trail.trailStatus.values()) {
|
|
73
|
+
if (status.isLogging) return true;
|
|
74
|
+
}
|
|
75
|
+
return false;
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
/**
|
|
79
|
+
* Verifica event selectors para decidir se o evento deve ser gravado.
|
|
80
|
+
* Management events são sempre gravados quando logging está ativo.
|
|
81
|
+
* Data events só são gravados se houver um selector configurado para o recurso.
|
|
82
|
+
*/
|
|
83
|
+
_shouldRecord(params) {
|
|
84
|
+
if (!this.trail || !this.trail.eventSelectors) return true;
|
|
85
|
+
|
|
86
|
+
const isDataEvent = params.isDataEvent === true;
|
|
87
|
+
|
|
88
|
+
// Management events: gravados por padrão
|
|
89
|
+
if (!isDataEvent) return true;
|
|
90
|
+
|
|
91
|
+
// Data events: verifica se algum trail tem selector para este eventSource
|
|
92
|
+
for (const [trailName, selectors] of this.trail.eventSelectors.entries()) {
|
|
93
|
+
const status = this.trail.trailStatus.get(trailName);
|
|
94
|
+
if (!status || !status.isLogging) continue;
|
|
95
|
+
|
|
96
|
+
for (const selector of selectors) {
|
|
97
|
+
const dataResources = selector.DataResources || [];
|
|
98
|
+
for (const dr of dataResources) {
|
|
99
|
+
if (this._matchesDataResource(dr)) return true;
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
return false;
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
_matchesDataResource(dataResource) {
|
|
108
|
+
const typeMap = {
|
|
109
|
+
's3.amazonaws.com': 'AWS::S3::Object',
|
|
110
|
+
'dynamodb.amazonaws.com': 'AWS::DynamoDB::Table',
|
|
111
|
+
'kms.amazonaws.com': 'AWS::KMS::Key',
|
|
112
|
+
'secretsmanager.amazonaws.com': 'AWS::SecretsManager::Secret',
|
|
113
|
+
'ssm.amazonaws.com': 'AWS::SSM::Parameter',
|
|
114
|
+
'cognito-idp.amazonaws.com': 'AWS::Cognito::UserPool',
|
|
115
|
+
'execute-api.amazonaws.com': 'AWS::APIGateway::Stage',
|
|
116
|
+
};
|
|
117
|
+
|
|
118
|
+
const expectedType = typeMap[this.eventSource];
|
|
119
|
+
if (!expectedType) return false;
|
|
120
|
+
|
|
121
|
+
if (dataResource.Type !== expectedType) return false;
|
|
122
|
+
|
|
123
|
+
// Values: wildcard ou ARN específico
|
|
124
|
+
const values = dataResource.Values || [];
|
|
125
|
+
return values.some((v) => v.endsWith('*') || v === expectedType);
|
|
126
|
+
}
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
module.exports = { CloudTrailAudit };
|
package/src/utils/local-store.js
CHANGED
|
@@ -19,6 +19,10 @@ class LocalStore {
|
|
|
19
19
|
}
|
|
20
20
|
|
|
21
21
|
getFilePath(entity) {
|
|
22
|
+
// Suporta subpaths como 'parameter-store/parameters'
|
|
23
|
+
const parts = entity.split('/');
|
|
24
|
+
const dir = parts.length > 1 ? path.join(this.dataDir, ...parts.slice(0, -1)) : this.dataDir;
|
|
25
|
+
if (!fs.existsSync(dir)) mkdirp.sync(dir);
|
|
22
26
|
return path.join(this.dataDir, `${entity}.json`);
|
|
23
27
|
}
|
|
24
28
|
|
|
@@ -35,10 +39,12 @@ class LocalStore {
|
|
|
35
39
|
}
|
|
36
40
|
}
|
|
37
41
|
|
|
38
|
-
write(entity, data) {
|
|
42
|
+
write(entity, data, _data2) {
|
|
43
|
+
// Suporta chamada com 3 args: write(entity, null, data) usado por alguns simuladores
|
|
44
|
+
const payload = (data === null || data === undefined) ? _data2 : data;
|
|
39
45
|
const filePath = this.getFilePath(entity);
|
|
40
46
|
try {
|
|
41
|
-
fs.writeFileSync(filePath, JSON.stringify(
|
|
47
|
+
fs.writeFileSync(filePath, JSON.stringify(payload, null, 2));
|
|
42
48
|
} catch (error) {
|
|
43
49
|
console.error(`Erro ao escrever ${entity}:`, error);
|
|
44
50
|
throw error;
|
|
@@ -63,6 +69,16 @@ class LocalStore {
|
|
|
63
69
|
.filter(f => f.endsWith('.json'))
|
|
64
70
|
.map(f => f.replace('.json', ''));
|
|
65
71
|
}
|
|
72
|
+
|
|
73
|
+
// Aliases async para compatibilidade com simuladores que usam store.load/store.save
|
|
74
|
+
async load(entity) {
|
|
75
|
+
const data = this.read(entity);
|
|
76
|
+
return Array.isArray(data) && data.length === 0 ? null : data;
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
async save(entity, data) {
|
|
80
|
+
this.write(entity, data);
|
|
81
|
+
}
|
|
66
82
|
}
|
|
67
83
|
|
|
68
84
|
module.exports = LocalStore;
|