@gammarers/aws-daily-cloud-watch-logs-archive-stack 2.7.0

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/lib/index.js ADDED
@@ -0,0 +1,251 @@
1
+ "use strict";
2
+ var _a;
3
+ Object.defineProperty(exports, "__esModule", { value: true });
4
+ exports.DailyCloudWatchLogsArchiveStack = void 0;
5
+ const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
6
+ const crypto = require("crypto");
7
+ const aws_secure_bucket_1 = require("@gammarers/aws-secure-bucket");
8
+ const aws_secure_log_bucket_1 = require("@gammarers/aws-secure-log-bucket");
9
+ const cdk = require("aws-cdk-lib");
10
+ const aws_cdk_lib_1 = require("aws-cdk-lib");
11
+ const iam = require("aws-cdk-lib/aws-iam");
12
+ const logs = require("aws-cdk-lib/aws-logs");
13
+ const scheduler = require("aws-cdk-lib/aws-scheduler");
14
+ const sfn = require("aws-cdk-lib/aws-stepfunctions");
15
+ const tasks = require("aws-cdk-lib/aws-stepfunctions-tasks");
16
+ const log_archiver_function_1 = require("./funcs/log-archiver-function");
17
+ class DailyCloudWatchLogsArchiveStack extends cdk.Stack {
18
+ constructor(scope, id, props) {
19
+ super(scope, id, props);
20
+ // 👇 Get current account & region
21
+ //const account = cdk.Stack.of(this).account;
22
+ //const stackName: string = cdk.Stack.of(this).stackName;
23
+ //const region = cdk.Stack.of(this).region;
24
+ //const account = this.account;
25
+ const region = this.region;
26
+ const randomNameKey = crypto.createHash('shake256', { outputLength: 4 })
27
+ .update(cdk.Names.uniqueId(this))
28
+ .digest('hex');
29
+ // 👇 Create Backup S3 Bucket
30
+ const logArchiveBucket = new aws_secure_log_bucket_1.SecureLogBucket(this, 'LogArchiveBucket', {
31
+ bucketName: `log-archive-${randomNameKey}`,
32
+ encryption: aws_secure_bucket_1.SecureBucketEncryption.S3_MANAGED,
33
+ });
34
+ logArchiveBucket.addToResourcePolicy(new iam.PolicyStatement({
35
+ effect: iam.Effect.ALLOW,
36
+ principals: [
37
+ new iam.ServicePrincipal(`logs.${region}.amazonaws.com`),
38
+ ],
39
+ actions: [
40
+ 's3:GetBucketAcl',
41
+ ],
42
+ resources: [
43
+ logArchiveBucket.bucketArn,
44
+ ],
45
+ }));
46
+ logArchiveBucket.addToResourcePolicy(new iam.PolicyStatement({
47
+ effect: iam.Effect.ALLOW,
48
+ principals: [
49
+ new iam.ServicePrincipal(`logs.${region}.amazonaws.com`),
50
+ ],
51
+ actions: [
52
+ 's3:PutObject',
53
+ ],
54
+ resources: [
55
+ `${logArchiveBucket.bucketArn}/*`,
56
+ ],
57
+ conditions: {
58
+ StringEquals: {
59
+ 's3:x-amz-acl': 'bucket-owner-full-control',
60
+ },
61
+ },
62
+ }));
63
+ // 👇 Create Lambda Execution role.
64
+ const lambdaExecutionRole = new iam.Role(this, 'LambdaExecutionRole', {
65
+ roleName: `daily-cw-logs-archive-lambda-exec-${randomNameKey}-role`,
66
+ description: 'daily CloudWatch Logs archive machine exec role.',
67
+ assumedBy: new iam.ServicePrincipal('lambda.amazonaws.com'),
68
+ managedPolicies: [
69
+ iam.ManagedPolicy.fromAwsManagedPolicyName('service-role/AWSLambdaBasicExecutionRole'),
70
+ ],
71
+ inlinePolicies: {
72
+ ['log-export-policy']: new iam.PolicyDocument({
73
+ statements: [
74
+ new iam.PolicyStatement({
75
+ effect: iam.Effect.ALLOW,
76
+ actions: [
77
+ 'logs:CreateExportTask',
78
+ ],
79
+ resources: ['*'],
80
+ }),
81
+ ],
82
+ }),
83
+ ['put-bucket-policy']: new iam.PolicyDocument({
84
+ statements: [
85
+ new iam.PolicyStatement({
86
+ effect: iam.Effect.ALLOW,
87
+ actions: [
88
+ 's3:GetBucketAcl',
89
+ 's3:PutObject',
90
+ ],
91
+ resources: [
92
+ logArchiveBucket.bucketArn,
93
+ ],
94
+ }),
95
+ ],
96
+ }),
97
+ },
98
+ });
99
+ // 👇 Create Lambda Function
100
+ const lambdaFunction = new log_archiver_function_1.LogArchiverFunction(this, 'LogArchiveFunction', {
101
+ functionName: `daily-cw-logs-archive-${randomNameKey}-func`,
102
+ description: 'A function to archive logs s3 bucket from CloudWatch Logs.',
103
+ environment: {
104
+ BUCKET_NAME: logArchiveBucket.bucketName,
105
+ },
106
+ role: lambdaExecutionRole,
107
+ });
108
+ // 👇 Create Lambda Function Log Group
109
+ new logs.LogGroup(this, 'LambdaFunctionLogGroup', {
110
+ // logGroupName: lambdaFunction.logGroup.logGroupName, // <- If you specify this line to Custom:LogRotation resource created.
111
+ logGroupName: `/aws/lambda/${lambdaFunction.functionName}`,
112
+ retention: logs.RetentionDays.ONE_MONTH,
113
+ removalPolicy: cdk.RemovalPolicy.DESTROY,
114
+ });
115
+ const succeed = new sfn.Succeed(this, 'Succeed');
116
+ // 👇 Get CloudWatch Logs Resources
117
+ const getLogGroupResources = new tasks.CallAwsService(this, 'GetResources', {
118
+ iamResources: ['*'],
119
+ iamAction: 'tag:GetResources',
120
+ service: 'resourcegroupstaggingapi',
121
+ action: 'getResources',
122
+ parameters: {
123
+ ResourceTypeFilters: [
124
+ 'logs:log-group',
125
+ ],
126
+ TagFilters: [
127
+ {
128
+ 'Key.$': '$.tagKey',
129
+ 'Values.$': '$.tagValues',
130
+ },
131
+ ],
132
+ },
133
+ resultPath: '$.Result',
134
+ resultSelector: {
135
+ 'TargetResources.$': '$..ResourceTagMappingList[*].ResourceARN',
136
+ },
137
+ });
138
+ // Log Group Export Map
139
+ const logGroupExportMap = new sfn.Map(this, 'LogGroupExportMap', {
140
+ itemsPath: sfn.JsonPath.stringAt('$.Result.TargetResources'),
141
+ maxConcurrency: 1,
142
+ });
143
+ // 👇 Get Log Group Name
144
+ const getLogGroupName = new sfn.Pass(this, 'GetLogGroupName', {
145
+ parameters: {
146
+ 'TargetLogGroupName.$': "States.ArrayGetItem(States.StringSplit($, ':'), 6)",
147
+ },
148
+ });
149
+ logGroupExportMap.iterator(getLogGroupName);
150
+ // 👇 Invoke Lambda Function
151
+ const invokeLambdaFunction = new tasks.LambdaInvoke(this, 'InvokeLambdaFunction', {
152
+ lambdaFunction: lambdaFunction,
153
+ outputPath: '$.Payload',
154
+ payload: sfn.TaskInput.fromJsonPathAt('$'),
155
+ retryOnServiceExceptions: true,
156
+ });
157
+ getLogGroupName.next(invokeLambdaFunction);
158
+ // 👇 Describe Export Tasks
159
+ const describeExportTasks = new tasks.CallAwsService(this, 'DescribeExportTasks', {
160
+ iamResources: ['*'],
161
+ iamAction: 'logs:DescribeExportTasks',
162
+ service: 'cloudwatchlogs',
163
+ action: 'describeExportTasks',
164
+ parameters: {
165
+ 'TaskId.$': '$.TaskId',
166
+ },
167
+ resultPath: '$.Result',
168
+ resultSelector: {
169
+ 'DescribeExportTasksStatus.$': '$.ExportTasks[0].Status.Code',
170
+ },
171
+ });
172
+ invokeLambdaFunction.next(describeExportTasks);
173
+ const exportRunningWait = new sfn.Wait(this, 'ExportRunningWait', {
174
+ time: sfn.WaitTime.duration(aws_cdk_lib_1.Duration.seconds(10)),
175
+ });
176
+ const exportPendingWait = new sfn.Wait(this, 'ExportPendingWait', {
177
+ time: sfn.WaitTime.duration(aws_cdk_lib_1.Duration.seconds(3)),
178
+ });
179
+ // 👇 Export Status Check
180
+ const exportTaskStatusCheck = new sfn.Choice(this, 'ExportTaskStatusCheck')
181
+ .when(sfn.Condition.stringEquals('$.Result.DescribeExportTasksStatus', 'FAILED'), getLogGroupName)
182
+ .when(sfn.Condition.stringEquals('$.Result.DescribeExportTasksStatus', 'RUNNING'), exportRunningWait
183
+ .next(describeExportTasks))
184
+ .when(sfn.Condition.stringEquals('$.Result.DescribeExportTasksStatus', 'PENDING'), exportPendingWait
185
+ .next(describeExportTasks))
186
+ .otherwise(succeed);
187
+ describeExportTasks.next(exportTaskStatusCheck);
188
+ getLogGroupResources.next(logGroupExportMap);
189
+ //
190
+ const machine = new sfn.StateMachine(this, 'StateMachine', {
191
+ stateMachineName: `daily-cw-logs-archive-${randomNameKey}-machine`,
192
+ definition: getLogGroupResources,
193
+ });
194
+ // 👇 auto generated role name & description renaming.
195
+ const role = machine.node.findChild('Role');
196
+ const cfnRole = role.node.defaultChild;
197
+ cfnRole.addPropertyOverride('RoleName', `daily-cw-logs-archive-machine-${randomNameKey}-role`);
198
+ cfnRole.addPropertyOverride('Description', 'daily CloudWatch Logs archive machine role.');
199
+ const policy = role.node.findChild('DefaultPolicy');
200
+ const cfnPolicy = policy.node.defaultChild;
201
+ cfnPolicy.addPropertyOverride('PolicyName', `daily-cw-logs-archive-machine-${randomNameKey}-default-policy`);
202
+ // 👇 EventBridge Scheduler IAM Role (StateMachine Start Execution)
203
+ const schedulerExecutionRole = new iam.Role(this, 'SchedulerExecutionRole', {
204
+ roleName: `daily-cw-logs-archive-${randomNameKey}-schedule-role`,
205
+ description: 'daily CloudWatch Log archive schedule',
206
+ assumedBy: new iam.ServicePrincipal('scheduler.amazonaws.com'),
207
+ inlinePolicies: {
208
+ 'state-machine-exec-policy': new iam.PolicyDocument({
209
+ statements: [
210
+ new iam.PolicyStatement({
211
+ effect: iam.Effect.ALLOW,
212
+ actions: [
213
+ 'states:StartExecution',
214
+ ],
215
+ resources: [
216
+ machine.stateMachineArn,
217
+ ],
218
+ }),
219
+ ],
220
+ }),
221
+ },
222
+ });
223
+ // 👇 Schedule
224
+ new scheduler.CfnSchedule(this, 'Schedule', {
225
+ name: `daily-cw-logs-archive-${randomNameKey}-schedule`,
226
+ description: 'daily CloudWatch Logs archive schedule',
227
+ state: 'ENABLED',
228
+ flexibleTimeWindow: {
229
+ mode: 'OFF',
230
+ },
231
+ scheduleExpressionTimezone: 'UTC',
232
+ scheduleExpression: 'cron(1 13 * * ? *)',
233
+ target: {
234
+ arn: machine.stateMachineArn,
235
+ roleArn: schedulerExecutionRole.roleArn,
236
+ input: JSON.stringify({
237
+ tagKey: props.targetResourceTag.key,
238
+ tagValues: props.targetResourceTag.values,
239
+ }),
240
+ retryPolicy: {
241
+ maximumEventAgeInSeconds: 60,
242
+ maximumRetryAttempts: 0,
243
+ },
244
+ },
245
+ });
246
+ }
247
+ }
248
+ exports.DailyCloudWatchLogsArchiveStack = DailyCloudWatchLogsArchiveStack;
249
+ _a = JSII_RTTI_SYMBOL_1;
250
+ DailyCloudWatchLogsArchiveStack[_a] = { fqn: "@gammarers/aws-daily-cloud-watch-logs-archive-stack.DailyCloudWatchLogsArchiveStack", version: "2.7.0" };
251
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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","sourcesContent":["import * as crypto from 'crypto';\nimport { SecureBucketEncryption } from '@gammarers/aws-secure-bucket';\nimport { SecureLogBucket } from '@gammarers/aws-secure-log-bucket';\nimport * as cdk from 'aws-cdk-lib';\nimport { Duration } from 'aws-cdk-lib';\nimport * as iam from 'aws-cdk-lib/aws-iam';\nimport * as logs from 'aws-cdk-lib/aws-logs';\nimport * as scheduler from 'aws-cdk-lib/aws-scheduler';\nimport * as sfn from 'aws-cdk-lib/aws-stepfunctions';\nimport * as tasks from 'aws-cdk-lib/aws-stepfunctions-tasks';\nimport { Construct } from 'constructs';\nimport { LogArchiverFunction } from './funcs/log-archiver-function';\n\nexport interface DailyCloudWatchLogsArchiveStackProps extends cdk.StackProps {\n  readonly targetResourceTag: TargetResourceTagProperty;\n}\n\nexport interface TargetResourceTagProperty {\n  readonly key: string;\n  readonly values: string[];\n}\n\nexport class DailyCloudWatchLogsArchiveStack extends cdk.Stack {\n  constructor(scope: Construct, id: string, props: DailyCloudWatchLogsArchiveStackProps) {\n    super(scope, id, props);\n\n    // 👇 Get current account & region\n    //const account = cdk.Stack.of(this).account;\n    //const stackName: string = cdk.Stack.of(this).stackName;\n    //const region = cdk.Stack.of(this).region;\n    //const account = this.account;\n    const region = this.region;\n\n    const randomNameKey = crypto.createHash('shake256', { outputLength: 4 })\n      .update(cdk.Names.uniqueId(this))\n      .digest('hex');\n\n\n    // 👇 Create Backup S3 Bucket\n    const logArchiveBucket = new SecureLogBucket(this, 'LogArchiveBucket', {\n      bucketName: `log-archive-${randomNameKey}`,\n      encryption: SecureBucketEncryption.S3_MANAGED,\n    });\n    logArchiveBucket.addToResourcePolicy(new iam.PolicyStatement({\n      effect: iam.Effect.ALLOW,\n      principals: [\n        new iam.ServicePrincipal(`logs.${region}.amazonaws.com`),\n      ],\n      actions: [\n        's3:GetBucketAcl',\n      ],\n      resources: [\n        logArchiveBucket.bucketArn,\n      ],\n    }));\n    logArchiveBucket.addToResourcePolicy(new iam.PolicyStatement({\n      effect: iam.Effect.ALLOW,\n      principals: [\n        new iam.ServicePrincipal(`logs.${region}.amazonaws.com`),\n      ],\n      actions: [\n        's3:PutObject',\n      ],\n      resources: [\n        `${logArchiveBucket.bucketArn}/*`,\n      ],\n      conditions: {\n        StringEquals: {\n          's3:x-amz-acl': 'bucket-owner-full-control',\n        },\n      },\n    }));\n\n    // 👇 Create Lambda Execution role.\n    const lambdaExecutionRole = new iam.Role(this, 'LambdaExecutionRole', {\n      roleName: `daily-cw-logs-archive-lambda-exec-${randomNameKey}-role`,\n      description: 'daily CloudWatch Logs archive machine exec role.',\n      assumedBy: new iam.ServicePrincipal('lambda.amazonaws.com'),\n      managedPolicies: [\n        iam.ManagedPolicy.fromAwsManagedPolicyName('service-role/AWSLambdaBasicExecutionRole'),\n      ],\n      inlinePolicies: {\n        ['log-export-policy']: new iam.PolicyDocument({\n          statements: [\n            new iam.PolicyStatement({\n              effect: iam.Effect.ALLOW,\n              actions: [\n                'logs:CreateExportTask',\n              ],\n              resources: ['*'],\n            }),\n          ],\n        }),\n        ['put-bucket-policy']: new iam.PolicyDocument({\n          statements: [\n            new iam.PolicyStatement({\n              effect: iam.Effect.ALLOW,\n              actions: [\n                's3:GetBucketAcl',\n                's3:PutObject',\n              ],\n              resources: [\n                logArchiveBucket.bucketArn,\n              ],\n            }),\n          ],\n        }),\n      },\n    });\n\n    // 👇 Create Lambda Function\n    const lambdaFunction = new LogArchiverFunction(this, 'LogArchiveFunction', {\n      functionName: `daily-cw-logs-archive-${randomNameKey}-func`,\n      description: 'A function to archive logs s3 bucket from CloudWatch Logs.',\n      environment: {\n        BUCKET_NAME: logArchiveBucket.bucketName,\n      },\n      role: lambdaExecutionRole,\n    });\n\n    // 👇 Create Lambda Function Log Group\n    new logs.LogGroup(this, 'LambdaFunctionLogGroup', {\n      // logGroupName: lambdaFunction.logGroup.logGroupName, // <- If you specify this line to Custom:LogRotation resource created.\n      logGroupName: `/aws/lambda/${lambdaFunction.functionName}`,\n      retention: logs.RetentionDays.ONE_MONTH,\n      removalPolicy: cdk.RemovalPolicy.DESTROY,\n    });\n\n    const succeed = new sfn.Succeed(this, 'Succeed');\n\n    // 👇 Get CloudWatch Logs Resources\n    const getLogGroupResources = new tasks.CallAwsService(this, 'GetResources', {\n      iamResources: ['*'],\n      iamAction: 'tag:GetResources',\n      service: 'resourcegroupstaggingapi',\n      action: 'getResources',\n      parameters: {\n        ResourceTypeFilters: [\n          'logs:log-group',\n        ],\n        TagFilters: [\n          {\n            'Key.$': '$.tagKey',\n            'Values.$': '$.tagValues',\n          },\n        ],\n      },\n      resultPath: '$.Result',\n      resultSelector: {\n        'TargetResources.$': '$..ResourceTagMappingList[*].ResourceARN',\n      },\n    });\n\n    // Log Group Export Map\n    const logGroupExportMap = new sfn.Map(this, 'LogGroupExportMap', {\n      itemsPath: sfn.JsonPath.stringAt('$.Result.TargetResources'),\n      maxConcurrency: 1,\n    });\n\n    // 👇 Get Log Group Name\n    const getLogGroupName = new sfn.Pass(this, 'GetLogGroupName', {\n      parameters: {\n        'TargetLogGroupName.$': \"States.ArrayGetItem(States.StringSplit($, ':'), 6)\",\n      },\n    });\n\n    logGroupExportMap.iterator(getLogGroupName);\n\n    // 👇 Invoke Lambda Function\n    const invokeLambdaFunction = new tasks.LambdaInvoke(this, 'InvokeLambdaFunction', {\n      lambdaFunction: lambdaFunction,\n      outputPath: '$.Payload',\n      payload: sfn.TaskInput.fromJsonPathAt('$'),\n      retryOnServiceExceptions: true,\n    });\n\n    getLogGroupName.next(invokeLambdaFunction);\n\n    // 👇 Describe Export Tasks\n    const describeExportTasks = new tasks.CallAwsService(this, 'DescribeExportTasks', {\n      iamResources: ['*'],\n      iamAction: 'logs:DescribeExportTasks',\n      service: 'cloudwatchlogs',\n      action: 'describeExportTasks',\n      parameters: {\n        'TaskId.$': '$.TaskId',\n      },\n      resultPath: '$.Result',\n      resultSelector: {\n        'DescribeExportTasksStatus.$': '$.ExportTasks[0].Status.Code',\n      },\n    });\n\n    invokeLambdaFunction.next(describeExportTasks);\n\n    const exportRunningWait = new sfn.Wait(this, 'ExportRunningWait', {\n      time: sfn.WaitTime.duration(Duration.seconds(10)),\n    });\n\n    const exportPendingWait = new sfn.Wait(this, 'ExportPendingWait', {\n      time: sfn.WaitTime.duration(Duration.seconds(3)),\n    });\n\n    // 👇 Export Status Check\n    const exportTaskStatusCheck = new sfn.Choice(this, 'ExportTaskStatusCheck')\n      .when(\n        sfn.Condition.stringEquals('$.Result.DescribeExportTasksStatus', 'FAILED'),\n        getLogGroupName,\n      )\n      .when(\n        sfn.Condition.stringEquals('$.Result.DescribeExportTasksStatus', 'RUNNING'),\n        exportRunningWait\n          .next(describeExportTasks),\n      )\n      .when(\n        sfn.Condition.stringEquals('$.Result.DescribeExportTasksStatus', 'PENDING'),\n        exportPendingWait\n          .next(describeExportTasks),\n      )\n      .otherwise(succeed);\n\n    describeExportTasks.next(exportTaskStatusCheck);\n\n    getLogGroupResources.next(logGroupExportMap);\n\n    //\n    const machine = new sfn.StateMachine(this, 'StateMachine', {\n      stateMachineName: `daily-cw-logs-archive-${randomNameKey}-machine`,\n      definition: getLogGroupResources,\n    });\n    // 👇 auto generated role name & description renaming.\n    const role = machine.node.findChild('Role') as iam.Role;\n    const cfnRole = role.node.defaultChild as iam.CfnRole;\n    cfnRole.addPropertyOverride('RoleName', `daily-cw-logs-archive-machine-${randomNameKey}-role`);\n    cfnRole.addPropertyOverride('Description', 'daily CloudWatch Logs archive machine role.');\n    const policy = role.node.findChild('DefaultPolicy') as iam.Policy;\n    const cfnPolicy = policy.node.defaultChild as iam.CfnPolicy;\n    cfnPolicy.addPropertyOverride('PolicyName', `daily-cw-logs-archive-machine-${randomNameKey}-default-policy`);\n\n    // 👇 EventBridge Scheduler IAM Role (StateMachine Start Execution)\n    const schedulerExecutionRole = new iam.Role(this, 'SchedulerExecutionRole', {\n      roleName: `daily-cw-logs-archive-${randomNameKey}-schedule-role`,\n      description: 'daily CloudWatch Log archive schedule',\n      assumedBy: new iam.ServicePrincipal('scheduler.amazonaws.com'),\n      inlinePolicies: {\n        'state-machine-exec-policy': new iam.PolicyDocument({\n          statements: [\n            new iam.PolicyStatement({\n              effect: iam.Effect.ALLOW,\n              actions: [\n                'states:StartExecution',\n              ],\n              resources: [\n                machine.stateMachineArn,\n              ],\n            }),\n          ],\n        }),\n      },\n    });\n\n    // 👇 Schedule\n    new scheduler.CfnSchedule(this, 'Schedule', {\n      name: `daily-cw-logs-archive-${randomNameKey}-schedule`,\n      description: 'daily CloudWatch Logs archive schedule',\n      state: 'ENABLED',\n      flexibleTimeWindow: {\n        mode: 'OFF',\n      },\n      scheduleExpressionTimezone: 'UTC',\n      scheduleExpression: 'cron(1 13 * * ? *)',\n      target: {\n        arn: machine.stateMachineArn,\n        roleArn: schedulerExecutionRole.roleArn,\n        input: JSON.stringify({\n          tagKey: props.targetResourceTag.key,\n          tagValues: props.targetResourceTag.values,\n        }),\n        retryPolicy: {\n          maximumEventAgeInSeconds: 60,\n          maximumRetryAttempts: 0,\n        },\n      },\n    });\n  }\n}"]}
package/package.json ADDED
@@ -0,0 +1,165 @@
1
+ {
2
+ "name": "@gammarers/aws-daily-cloud-watch-logs-archive-stack",
3
+ "description": "AWS CloudWatch Logs daily archive to s3 bucket",
4
+ "repository": {
5
+ "type": "git",
6
+ "url": "https://github.com/gammarers/aws-daily-cloud-watch-logs-archive-stack.git"
7
+ },
8
+ "scripts": {
9
+ "build": "npx projen build",
10
+ "bump": "npx projen bump",
11
+ "bundle": "npx projen bundle",
12
+ "bundle:funcs/log-archiver.lambda": "npx projen bundle:funcs/log-archiver.lambda",
13
+ "bundle:funcs/log-archiver.lambda:watch": "npx projen bundle:funcs/log-archiver.lambda:watch",
14
+ "clobber": "npx projen clobber",
15
+ "compat": "npx projen compat",
16
+ "compile": "npx projen compile",
17
+ "default": "npx projen default",
18
+ "docgen": "npx projen docgen",
19
+ "eject": "npx projen eject",
20
+ "eslint": "npx projen eslint",
21
+ "package": "npx projen package",
22
+ "package-all": "npx projen package-all",
23
+ "package:dotnet": "npx projen package:dotnet",
24
+ "package:js": "npx projen package:js",
25
+ "package:python": "npx projen package:python",
26
+ "post-compile": "npx projen post-compile",
27
+ "post-upgrade": "npx projen post-upgrade",
28
+ "pre-compile": "npx projen pre-compile",
29
+ "release": "npx projen release",
30
+ "test": "npx projen test",
31
+ "test:watch": "npx projen test:watch",
32
+ "unbump": "npx projen unbump",
33
+ "upgrade": "npx projen upgrade",
34
+ "watch": "npx projen watch",
35
+ "projen": "npx projen"
36
+ },
37
+ "author": {
38
+ "name": "yicr",
39
+ "email": "yicr@users.noreply.github.com",
40
+ "organization": false
41
+ },
42
+ "devDependencies": {
43
+ "@aws-sdk/client-cloudwatch-logs": "3.556.x",
44
+ "@gammarer/jest-serializer-aws-cdk-asset-filename-replacer": "^0.4.30",
45
+ "@types/aws-lambda": "^8.10.137",
46
+ "@types/jest": "^29.5.12",
47
+ "@types/node": "^18",
48
+ "@typescript-eslint/eslint-plugin": "^6",
49
+ "@typescript-eslint/parser": "^6",
50
+ "aws-cdk-lib": "2.80.0",
51
+ "aws-sdk-client-mock": "^3",
52
+ "aws-sdk-client-mock-jest": "^3",
53
+ "constructs": "10.0.5",
54
+ "esbuild": "^0.20.2",
55
+ "eslint": "^8",
56
+ "eslint-import-resolver-typescript": "^3.6.1",
57
+ "eslint-plugin-import": "^2.29.1",
58
+ "jest": "^29.7.0",
59
+ "jest-junit": "^15",
60
+ "jsii": "5.2.x",
61
+ "jsii-diff": "^1.97.0",
62
+ "jsii-docgen": "^10.4.1",
63
+ "jsii-pacmak": "^1.97.0",
64
+ "jsii-rosetta": "5.2.x",
65
+ "projen": "^0.81.1",
66
+ "standard-version": "^9",
67
+ "ts-jest": "^29.1.2",
68
+ "ts-node": "^10.9.2",
69
+ "typescript": "5.2.x"
70
+ },
71
+ "peerDependencies": {
72
+ "@gammarers/aws-secure-bucket": "~1.3.3",
73
+ "@gammarers/aws-secure-log-bucket": "~1.6.2",
74
+ "aws-cdk-lib": "^2.80.0",
75
+ "constructs": "^10.0.5"
76
+ },
77
+ "dependencies": {
78
+ "@gammarers/aws-secure-bucket": "~1.3.3",
79
+ "@gammarers/aws-secure-log-bucket": "~1.6.2"
80
+ },
81
+ "keywords": [
82
+ "archive",
83
+ "aws",
84
+ "aws-cdk",
85
+ "bucket",
86
+ "cdk",
87
+ "lambda",
88
+ "s3",
89
+ "scheduler"
90
+ ],
91
+ "engines": {
92
+ "node": ">= 18.0.0"
93
+ },
94
+ "main": "lib/index.js",
95
+ "license": "Apache-2.0",
96
+ "publishConfig": {
97
+ "access": "public"
98
+ },
99
+ "version": "2.7.0",
100
+ "jest": {
101
+ "snapshotSerializers": [
102
+ "<rootDir>/node_modules/@gammarer/jest-serializer-aws-cdk-asset-filename-replacer"
103
+ ],
104
+ "testMatch": [
105
+ "<rootDir>/src/**/__tests__/**/*.ts?(x)",
106
+ "<rootDir>/(test|src)/**/*(*.)@(spec|test).ts?(x)"
107
+ ],
108
+ "clearMocks": true,
109
+ "collectCoverage": true,
110
+ "coverageReporters": [
111
+ "json",
112
+ "lcov",
113
+ "clover",
114
+ "cobertura",
115
+ "text"
116
+ ],
117
+ "coverageDirectory": "coverage",
118
+ "coveragePathIgnorePatterns": [
119
+ "/node_modules/"
120
+ ],
121
+ "testPathIgnorePatterns": [
122
+ "/node_modules/"
123
+ ],
124
+ "watchPathIgnorePatterns": [
125
+ "/node_modules/"
126
+ ],
127
+ "reporters": [
128
+ "default",
129
+ [
130
+ "jest-junit",
131
+ {
132
+ "outputDirectory": "test-reports"
133
+ }
134
+ ]
135
+ ],
136
+ "transform": {
137
+ "^.+\\.[t]sx?$": [
138
+ "ts-jest",
139
+ {
140
+ "tsconfig": "tsconfig.dev.json"
141
+ }
142
+ ]
143
+ }
144
+ },
145
+ "types": "lib/index.d.ts",
146
+ "stability": "stable",
147
+ "jsii": {
148
+ "outdir": "dist",
149
+ "targets": {
150
+ "python": {
151
+ "distName": "gammarers.aws-daily-cloud-watch-logs-archive-stack",
152
+ "module": "gammarers.aws_daily_cloud_watch_logs_archive_stack"
153
+ },
154
+ "dotnet": {
155
+ "namespace": "Gammarers.CDK.AWS",
156
+ "packageId": "Gammarers.CDK.AWS.DailyCloudWatchLogsArchiveStack"
157
+ }
158
+ },
159
+ "tsc": {
160
+ "outDir": "lib",
161
+ "rootDir": "src"
162
+ }
163
+ },
164
+ "//": "~~ Generated by projen. To modify, edit .projenrc.ts and run \"npx projen\"."
165
+ }