1 | ;
|
2 | var _a;
|
3 | Object.defineProperty(exports, "__esModule", { value: true });
|
4 | exports.LogGroup = exports.RetentionDays = void 0;
|
5 | const jsiiDeprecationWarnings = require("../.warnings.jsii.js");
|
6 | const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
7 | const cloudwatch = require("@aws-cdk/aws-cloudwatch");
|
8 | const iam = require("@aws-cdk/aws-iam");
|
9 | const core_1 = require("@aws-cdk/core");
|
10 | const log_stream_1 = require("./log-stream");
|
11 | const logs_generated_1 = require("./logs.generated");
|
12 | const metric_filter_1 = require("./metric-filter");
|
13 | const pattern_1 = require("./pattern");
|
14 | const policy_1 = require("./policy");
|
15 | const subscription_filter_1 = require("./subscription-filter");
|
16 | /**
|
17 | * An CloudWatch Log Group
|
18 | */
|
19 | class LogGroupBase extends core_1.Resource {
|
20 | /**
|
21 | * Create a new Log Stream for this Log Group
|
22 | *
|
23 | * @param id Unique identifier for the construct in its parent
|
24 | * @param props Properties for creating the LogStream
|
25 | */
|
26 | addStream(id, props = {}) {
|
27 | return new log_stream_1.LogStream(this, id, {
|
28 | logGroup: this,
|
29 | ...props,
|
30 | });
|
31 | }
|
32 | /**
|
33 | * Create a new Subscription Filter on this Log Group
|
34 | *
|
35 | * @param id Unique identifier for the construct in its parent
|
36 | * @param props Properties for creating the SubscriptionFilter
|
37 | */
|
38 | addSubscriptionFilter(id, props) {
|
39 | return new subscription_filter_1.SubscriptionFilter(this, id, {
|
40 | logGroup: this,
|
41 | ...props,
|
42 | });
|
43 | }
|
44 | /**
|
45 | * Create a new Metric Filter on this Log Group
|
46 | *
|
47 | * @param id Unique identifier for the construct in its parent
|
48 | * @param props Properties for creating the MetricFilter
|
49 | */
|
50 | addMetricFilter(id, props) {
|
51 | return new metric_filter_1.MetricFilter(this, id, {
|
52 | logGroup: this,
|
53 | ...props,
|
54 | });
|
55 | }
|
56 | /**
|
57 | * Extract a metric from structured log events in the LogGroup
|
58 | *
|
59 | * Creates a MetricFilter on this LogGroup that will extract the value
|
60 | * of the indicated JSON field in all records where it occurs.
|
61 | *
|
62 | * The metric will be available in CloudWatch Metrics under the
|
63 | * indicated namespace and name.
|
64 | *
|
65 | * @param jsonField JSON field to extract (example: '$.myfield')
|
66 | * @param metricNamespace Namespace to emit the metric under
|
67 | * @param metricName Name to emit the metric under
|
68 | * @returns A Metric object representing the extracted metric
|
69 | */
|
70 | extractMetric(jsonField, metricNamespace, metricName) {
|
71 | new metric_filter_1.MetricFilter(this, `${metricNamespace}_${metricName}`, {
|
72 | logGroup: this,
|
73 | metricNamespace,
|
74 | metricName,
|
75 | filterPattern: pattern_1.FilterPattern.exists(jsonField),
|
76 | metricValue: jsonField,
|
77 | });
|
78 | return new cloudwatch.Metric({ metricName, namespace: metricNamespace }).attachTo(this);
|
79 | }
|
80 | /**
|
81 | * Give permissions to create and write to streams in this log group
|
82 | */
|
83 | grantWrite(grantee) {
|
84 | return this.grant(grantee, 'logs:CreateLogStream', 'logs:PutLogEvents');
|
85 | }
|
86 | /**
|
87 | * Give the indicated permissions on this log group and all streams
|
88 | */
|
89 | grant(grantee, ...actions) {
|
90 | return iam.Grant.addToPrincipalOrResource({
|
91 | grantee,
|
92 | actions,
|
93 | // A LogGroup ARN out of CloudFormation already includes a ':*' at the end to include the log streams under the group.
|
94 | // See https://docs.aws.amazon.com/AWSCloudFormation/latest/UserGuide/aws-resource-logs-loggroup.html#w2ab1c21c10c63c43c11
|
95 | resourceArns: [this.logGroupArn],
|
96 | resource: this,
|
97 | });
|
98 | }
|
99 | /**
|
100 | * Public method to get the physical name of this log group
|
101 | * @returns Physical name of log group
|
102 | */
|
103 | logGroupPhysicalName() {
|
104 | return this.physicalName;
|
105 | }
|
106 | /**
|
107 | * Adds a statement to the resource policy associated with this log group.
|
108 | * A resource policy will be automatically created upon the first call to `addToResourcePolicy`.
|
109 | *
|
110 | * Any ARN Principals inside of the statement will be converted into AWS Account ID strings
|
111 | * because CloudWatch Logs Resource Policies do not accept ARN principals.
|
112 | *
|
113 | * @param statement The policy statement to add
|
114 | */
|
115 | addToResourcePolicy(statement) {
|
116 | if (!this.policy) {
|
117 | this.policy = new policy_1.ResourcePolicy(this, 'Policy');
|
118 | }
|
119 | this.policy.document.addStatements(statement.copy({
|
120 | principals: statement.principals.map(p => this.convertArnPrincpalToAccountId(p)),
|
121 | }));
|
122 | return { statementAdded: true, policyDependable: this.policy };
|
123 | }
|
124 | convertArnPrincpalToAccountId(principal) {
|
125 | if (principal.principalAccount) {
|
126 | // we use ArnPrincipal here because the constructor inserts the argument
|
127 | // into the template without mutating it, which means that there is no
|
128 | // ARN created by this call.
|
129 | return new iam.ArnPrincipal(principal.principalAccount);
|
130 | }
|
131 | if (principal instanceof iam.ArnPrincipal) {
|
132 | const parsedArn = core_1.Arn.split(principal.arn, core_1.ArnFormat.SLASH_RESOURCE_NAME);
|
133 | if (parsedArn.account) {
|
134 | return new iam.ArnPrincipal(parsedArn.account);
|
135 | }
|
136 | }
|
137 | return principal;
|
138 | }
|
139 | }
|
140 | /**
|
141 | * How long, in days, the log contents will be retained.
|
142 | */
|
143 | var RetentionDays;
|
144 | (function (RetentionDays) {
|
145 | /**
|
146 | * 1 day
|
147 | */
|
148 | RetentionDays[RetentionDays["ONE_DAY"] = 1] = "ONE_DAY";
|
149 | /**
|
150 | * 3 days
|
151 | */
|
152 | RetentionDays[RetentionDays["THREE_DAYS"] = 3] = "THREE_DAYS";
|
153 | /**
|
154 | * 5 days
|
155 | */
|
156 | RetentionDays[RetentionDays["FIVE_DAYS"] = 5] = "FIVE_DAYS";
|
157 | /**
|
158 | * 1 week
|
159 | */
|
160 | RetentionDays[RetentionDays["ONE_WEEK"] = 7] = "ONE_WEEK";
|
161 | /**
|
162 | * 2 weeks
|
163 | */
|
164 | RetentionDays[RetentionDays["TWO_WEEKS"] = 14] = "TWO_WEEKS";
|
165 | /**
|
166 | * 1 month
|
167 | */
|
168 | RetentionDays[RetentionDays["ONE_MONTH"] = 30] = "ONE_MONTH";
|
169 | /**
|
170 | * 2 months
|
171 | */
|
172 | RetentionDays[RetentionDays["TWO_MONTHS"] = 60] = "TWO_MONTHS";
|
173 | /**
|
174 | * 3 months
|
175 | */
|
176 | RetentionDays[RetentionDays["THREE_MONTHS"] = 90] = "THREE_MONTHS";
|
177 | /**
|
178 | * 4 months
|
179 | */
|
180 | RetentionDays[RetentionDays["FOUR_MONTHS"] = 120] = "FOUR_MONTHS";
|
181 | /**
|
182 | * 5 months
|
183 | */
|
184 | RetentionDays[RetentionDays["FIVE_MONTHS"] = 150] = "FIVE_MONTHS";
|
185 | /**
|
186 | * 6 months
|
187 | */
|
188 | RetentionDays[RetentionDays["SIX_MONTHS"] = 180] = "SIX_MONTHS";
|
189 | /**
|
190 | * 1 year
|
191 | */
|
192 | RetentionDays[RetentionDays["ONE_YEAR"] = 365] = "ONE_YEAR";
|
193 | /**
|
194 | * 13 months
|
195 | */
|
196 | RetentionDays[RetentionDays["THIRTEEN_MONTHS"] = 400] = "THIRTEEN_MONTHS";
|
197 | /**
|
198 | * 18 months
|
199 | */
|
200 | RetentionDays[RetentionDays["EIGHTEEN_MONTHS"] = 545] = "EIGHTEEN_MONTHS";
|
201 | /**
|
202 | * 2 years
|
203 | */
|
204 | RetentionDays[RetentionDays["TWO_YEARS"] = 731] = "TWO_YEARS";
|
205 | /**
|
206 | * 5 years
|
207 | */
|
208 | RetentionDays[RetentionDays["FIVE_YEARS"] = 1827] = "FIVE_YEARS";
|
209 | /**
|
210 | * 6 years
|
211 | */
|
212 | RetentionDays[RetentionDays["SIX_YEARS"] = 2192] = "SIX_YEARS";
|
213 | /**
|
214 | * 7 years
|
215 | */
|
216 | RetentionDays[RetentionDays["SEVEN_YEARS"] = 2557] = "SEVEN_YEARS";
|
217 | /**
|
218 | * 8 years
|
219 | */
|
220 | RetentionDays[RetentionDays["EIGHT_YEARS"] = 2922] = "EIGHT_YEARS";
|
221 | /**
|
222 | * 9 years
|
223 | */
|
224 | RetentionDays[RetentionDays["NINE_YEARS"] = 3288] = "NINE_YEARS";
|
225 | /**
|
226 | * 10 years
|
227 | */
|
228 | RetentionDays[RetentionDays["TEN_YEARS"] = 3653] = "TEN_YEARS";
|
229 | /**
|
230 | * Retain logs forever
|
231 | */
|
232 | RetentionDays[RetentionDays["INFINITE"] = 9999] = "INFINITE";
|
233 | })(RetentionDays = exports.RetentionDays || (exports.RetentionDays = {}));
|
234 | /**
|
235 | * Define a CloudWatch Log Group
|
236 | */
|
237 | class LogGroup extends LogGroupBase {
|
238 | constructor(scope, id, props = {}) {
|
239 | super(scope, id, {
|
240 | physicalName: props.logGroupName,
|
241 | });
|
242 | try {
|
243 | jsiiDeprecationWarnings._aws_cdk_aws_logs_LogGroupProps(props);
|
244 | }
|
245 | catch (error) {
|
246 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
247 | Error.captureStackTrace(error, LogGroup);
|
248 | }
|
249 | throw error;
|
250 | }
|
251 | let retentionInDays = props.retention;
|
252 | if (retentionInDays === undefined) {
|
253 | retentionInDays = RetentionDays.TWO_YEARS;
|
254 | }
|
255 | if (retentionInDays === Infinity || retentionInDays === RetentionDays.INFINITE) {
|
256 | retentionInDays = undefined;
|
257 | }
|
258 | if (retentionInDays !== undefined && !core_1.Token.isUnresolved(retentionInDays) && retentionInDays <= 0) {
|
259 | throw new Error(`retentionInDays must be positive, got ${retentionInDays}`);
|
260 | }
|
261 | const resource = new logs_generated_1.CfnLogGroup(this, 'Resource', {
|
262 | kmsKeyId: props.encryptionKey?.keyArn,
|
263 | logGroupName: this.physicalName,
|
264 | retentionInDays,
|
265 | });
|
266 | resource.applyRemovalPolicy(props.removalPolicy);
|
267 | this.logGroupArn = this.getResourceArnAttribute(resource.attrArn, {
|
268 | service: 'logs',
|
269 | resource: 'log-group',
|
270 | resourceName: this.physicalName,
|
271 | arnFormat: core_1.ArnFormat.COLON_RESOURCE_NAME,
|
272 | });
|
273 | this.logGroupName = this.getResourceNameAttribute(resource.ref);
|
274 | }
|
275 | /**
|
276 | * Import an existing LogGroup given its ARN
|
277 | */
|
278 | static fromLogGroupArn(scope, id, logGroupArn) {
|
279 | const baseLogGroupArn = logGroupArn.replace(/:\*$/, '');
|
280 | class Import extends LogGroupBase {
|
281 | constructor() {
|
282 | super(...arguments);
|
283 | this.logGroupArn = `${baseLogGroupArn}:*`;
|
284 | this.logGroupName = core_1.Stack.of(scope).splitArn(baseLogGroupArn, core_1.ArnFormat.COLON_RESOURCE_NAME).resourceName;
|
285 | }
|
286 | }
|
287 | return new Import(scope, id, {
|
288 | environmentFromArn: baseLogGroupArn,
|
289 | });
|
290 | }
|
291 | /**
|
292 | * Import an existing LogGroup given its name
|
293 | */
|
294 | static fromLogGroupName(scope, id, logGroupName) {
|
295 | const baseLogGroupName = logGroupName.replace(/:\*$/, '');
|
296 | class Import extends LogGroupBase {
|
297 | constructor() {
|
298 | super(...arguments);
|
299 | this.logGroupName = baseLogGroupName;
|
300 | this.logGroupArn = core_1.Stack.of(scope).formatArn({
|
301 | service: 'logs',
|
302 | resource: 'log-group',
|
303 | arnFormat: core_1.ArnFormat.COLON_RESOURCE_NAME,
|
304 | resourceName: baseLogGroupName + ':*',
|
305 | });
|
306 | }
|
307 | }
|
308 | return new Import(scope, id);
|
309 | }
|
310 | }
|
311 | exports.LogGroup = LogGroup;
|
312 | _a = JSII_RTTI_SYMBOL_1;
|
313 | LogGroup[_a] = { fqn: "@aws-cdk/aws-logs.LogGroup", version: "1.204.0" };
|
314 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"log-group.js","sourceRoot":"","sources":["log-group.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cloudwatch from '@aws-cdk/aws-cloudwatch';\nimport * as iam from '@aws-cdk/aws-iam';\nimport * as kms from '@aws-cdk/aws-kms';\nimport { Arn, ArnFormat, RemovalPolicy, Resource, Stack, Token } from '@aws-cdk/core';\nimport { Construct } from 'constructs';\nimport { LogStream } from './log-stream';\nimport { CfnLogGroup } from './logs.generated';\nimport { MetricFilter } from './metric-filter';\nimport { FilterPattern, IFilterPattern } from './pattern';\nimport { ResourcePolicy } from './policy';\nimport { ILogSubscriptionDestination, SubscriptionFilter } from './subscription-filter';\n\nexport interface ILogGroup extends iam.IResourceWithPolicy {\n  /**\n   * The ARN of this log group, with ':*' appended\n   *\n   * @attribute\n   */\n  readonly logGroupArn: string;\n\n  /**\n   * The name of this log group\n   * @attribute\n   */\n  readonly logGroupName: string;\n\n  /**\n   * Create a new Log Stream for this Log Group\n   *\n   * @param id Unique identifier for the construct in its parent\n   * @param props Properties for creating the LogStream\n   */\n  addStream(id: string, props?: StreamOptions): LogStream;\n\n  /**\n   * Create a new Subscription Filter on this Log Group\n   *\n   * @param id Unique identifier for the construct in its parent\n   * @param props Properties for creating the SubscriptionFilter\n   */\n  addSubscriptionFilter(id: string, props: SubscriptionFilterOptions): SubscriptionFilter;\n\n  /**\n   * Create a new Metric Filter on this Log Group\n   *\n   * @param id Unique identifier for the construct in its parent\n   * @param props Properties for creating the MetricFilter\n   */\n  addMetricFilter(id: string, props: MetricFilterOptions): MetricFilter;\n\n  /**\n   * Extract a metric from structured log events in the LogGroup\n   *\n   * Creates a MetricFilter on this LogGroup that will extract the value\n   * of the indicated JSON field in all records where it occurs.\n   *\n   * The metric will be available in CloudWatch Metrics under the\n   * indicated namespace and name.\n   *\n   * @param jsonField JSON field to extract (example: '$.myfield')\n   * @param metricNamespace Namespace to emit the metric under\n   * @param metricName Name to emit the metric under\n   * @returns A Metric object representing the extracted metric\n   */\n  extractMetric(jsonField: string, metricNamespace: string, metricName: string): cloudwatch.Metric;\n\n  /**\n   * Give permissions to write to create and write to streams in this log group\n   */\n  grantWrite(grantee: iam.IGrantable): iam.Grant;\n\n  /**\n   * Give the indicated permissions on this log group and all streams\n   */\n  grant(grantee: iam.IGrantable, ...actions: string[]): iam.Grant;\n\n  /**\n   * Public method to get the physical name of this log group\n   */\n  logGroupPhysicalName(): string;\n}\n\n/**\n * An CloudWatch Log Group\n */\nabstract class LogGroupBase extends Resource implements ILogGroup {\n  /**\n   * The ARN of this log group, with ':*' appended\n   */\n  public abstract readonly logGroupArn: string;\n\n  /**\n   * The name of this log group\n   */\n  public abstract readonly logGroupName: string;\n\n\n  private policy?: ResourcePolicy;\n\n  /**\n   * Create a new Log Stream for this Log Group\n   *\n   * @param id Unique identifier for the construct in its parent\n   * @param props Properties for creating the LogStream\n   */\n  public addStream(id: string, props: StreamOptions = {}): LogStream {\n    return new LogStream(this, id, {\n      logGroup: this,\n      ...props,\n    });\n  }\n\n  /**\n   * Create a new Subscription Filter on this Log Group\n   *\n   * @param id Unique identifier for the construct in its parent\n   * @param props Properties for creating the SubscriptionFilter\n   */\n  public addSubscriptionFilter(id: string, props: SubscriptionFilterOptions): SubscriptionFilter {\n    return new SubscriptionFilter(this, id, {\n      logGroup: this,\n      ...props,\n    });\n  }\n\n  /**\n   * Create a new Metric Filter on this Log Group\n   *\n   * @param id Unique identifier for the construct in its parent\n   * @param props Properties for creating the MetricFilter\n   */\n  public addMetricFilter(id: string, props: MetricFilterOptions): MetricFilter {\n    return new MetricFilter(this, id, {\n      logGroup: this,\n      ...props,\n    });\n  }\n\n  /**\n   * Extract a metric from structured log events in the LogGroup\n   *\n   * Creates a MetricFilter on this LogGroup that will extract the value\n   * of the indicated JSON field in all records where it occurs.\n   *\n   * The metric will be available in CloudWatch Metrics under the\n   * indicated namespace and name.\n   *\n   * @param jsonField JSON field to extract (example: '$.myfield')\n   * @param metricNamespace Namespace to emit the metric under\n   * @param metricName Name to emit the metric under\n   * @returns A Metric object representing the extracted metric\n   */\n  public extractMetric(jsonField: string, metricNamespace: string, metricName: string) {\n    new MetricFilter(this, `${metricNamespace}_${metricName}`, {\n      logGroup: this,\n      metricNamespace,\n      metricName,\n      filterPattern: FilterPattern.exists(jsonField),\n      metricValue: jsonField,\n    });\n\n    return new cloudwatch.Metric({ metricName, namespace: metricNamespace }).attachTo(this);\n  }\n\n  /**\n   * Give permissions to create and write to streams in this log group\n   */\n  public grantWrite(grantee: iam.IGrantable) {\n    return this.grant(grantee, 'logs:CreateLogStream', 'logs:PutLogEvents');\n  }\n\n  /**\n   * Give the indicated permissions on this log group and all streams\n   */\n  public grant(grantee: iam.IGrantable, ...actions: string[]) {\n    return iam.Grant.addToPrincipalOrResource({\n      grantee,\n      actions,\n      // A LogGroup ARN out of CloudFormation already includes a ':*' at the end to include the log streams under the group.\n      // See https://docs.aws.amazon.com/AWSCloudFormation/latest/UserGuide/aws-resource-logs-loggroup.html#w2ab1c21c10c63c43c11\n      resourceArns: [this.logGroupArn],\n      resource: this,\n    });\n  }\n\n  /**\n   * Public method to get the physical name of this log group\n   * @returns Physical name of log group\n   */\n  public logGroupPhysicalName(): string {\n    return this.physicalName;\n  }\n\n  /**\n   * Adds a statement to the resource policy associated with this log group.\n   * A resource policy will be automatically created upon the first call to `addToResourcePolicy`.\n   *\n   * Any ARN Principals inside of the statement will be converted into AWS Account ID strings\n   * because CloudWatch Logs Resource Policies do not accept ARN principals.\n   *\n   * @param statement The policy statement to add\n   */\n  public addToResourcePolicy(statement: iam.PolicyStatement): iam.AddToResourcePolicyResult {\n    if (!this.policy) {\n      this.policy = new ResourcePolicy(this, 'Policy');\n    }\n    this.policy.document.addStatements(statement.copy({\n      principals: statement.principals.map(p => this.convertArnPrincpalToAccountId(p)),\n    }));\n    return { statementAdded: true, policyDependable: this.policy };\n  }\n\n  private convertArnPrincpalToAccountId(principal: iam.IPrincipal) {\n    if (principal.principalAccount) {\n      // we use ArnPrincipal here because the constructor inserts the argument\n      // into the template without mutating it, which means that there is no\n      // ARN created by this call.\n      return new iam.ArnPrincipal(principal.principalAccount);\n    }\n\n    if (principal instanceof iam.ArnPrincipal) {\n      const parsedArn = Arn.split(principal.arn, ArnFormat.SLASH_RESOURCE_NAME);\n      if (parsedArn.account) {\n        return new iam.ArnPrincipal(parsedArn.account);\n      }\n    }\n\n    return principal;\n  }\n}\n\n/**\n * How long, in days, the log contents will be retained.\n */\nexport enum RetentionDays {\n  /**\n   * 1 day\n   */\n  ONE_DAY = 1,\n\n  /**\n   * 3 days\n   */\n  THREE_DAYS = 3,\n\n  /**\n   * 5 days\n   */\n  FIVE_DAYS = 5,\n\n  /**\n   * 1 week\n   */\n  ONE_WEEK = 7,\n\n  /**\n   * 2 weeks\n   */\n  TWO_WEEKS = 14,\n\n  /**\n   * 1 month\n   */\n  ONE_MONTH = 30,\n\n  /**\n   * 2 months\n   */\n  TWO_MONTHS = 60,\n\n  /**\n   * 3 months\n   */\n  THREE_MONTHS = 90,\n\n  /**\n   * 4 months\n   */\n  FOUR_MONTHS = 120,\n\n  /**\n   * 5 months\n   */\n  FIVE_MONTHS = 150,\n\n  /**\n   * 6 months\n   */\n  SIX_MONTHS = 180,\n\n  /**\n   * 1 year\n   */\n  ONE_YEAR = 365,\n\n  /**\n   * 13 months\n   */\n  THIRTEEN_MONTHS = 400,\n\n  /**\n   * 18 months\n   */\n  EIGHTEEN_MONTHS = 545,\n\n  /**\n   * 2 years\n   */\n  TWO_YEARS = 731,\n\n  /**\n   * 5 years\n   */\n  FIVE_YEARS = 1827,\n\n  /**\n   * 6 years\n   */\n  SIX_YEARS = 2192,\n\n  /**\n   * 7 years\n   */\n  SEVEN_YEARS = 2557,\n\n  /**\n   * 8 years\n   */\n  EIGHT_YEARS = 2922,\n\n  /**\n   * 9 years\n   */\n  NINE_YEARS = 3288,\n\n  /**\n   * 10 years\n   */\n  TEN_YEARS = 3653,\n\n  /**\n   * Retain logs forever\n   */\n  INFINITE = 9999,\n}\n\n/**\n * Properties for a LogGroup\n */\nexport interface LogGroupProps {\n  /**\n   * The KMS Key to encrypt the log group with.\n   *\n   * @default - log group is encrypted with the default master key\n   */\n  readonly encryptionKey?: kms.IKey;\n\n  /**\n   * Name of the log group.\n   *\n   * @default Automatically generated\n   */\n  readonly logGroupName?: string;\n\n  /**\n   * How long, in days, the log contents will be retained.\n   *\n   * To retain all logs, set this value to RetentionDays.INFINITE.\n   *\n   * @default RetentionDays.TWO_YEARS\n   */\n  readonly retention?: RetentionDays;\n\n  /**\n   * Determine the removal policy of this log group.\n   *\n   * Normally you want to retain the log group so you can diagnose issues\n   * from logs even after a deployment that no longer includes the log group.\n   * In that case, use the normal date-based retention policy to age out your\n   * logs.\n   *\n   * @default RemovalPolicy.Retain\n   */\n  readonly removalPolicy?: RemovalPolicy;\n}\n\n/**\n * Define a CloudWatch Log Group\n */\nexport class LogGroup extends LogGroupBase {\n  /**\n   * Import an existing LogGroup given its ARN\n   */\n  public static fromLogGroupArn(scope: Construct, id: string, logGroupArn: string): ILogGroup {\n    const baseLogGroupArn = logGroupArn.replace(/:\\*$/, '');\n\n    class Import extends LogGroupBase {\n      public readonly logGroupArn = `${baseLogGroupArn}:*`;\n      public readonly logGroupName = Stack.of(scope).splitArn(baseLogGroupArn, ArnFormat.COLON_RESOURCE_NAME).resourceName!;\n    }\n\n    return new Import(scope, id, {\n      environmentFromArn: baseLogGroupArn,\n    });\n  }\n\n  /**\n   * Import an existing LogGroup given its name\n   */\n  public static fromLogGroupName(scope: Construct, id: string, logGroupName: string): ILogGroup {\n    const baseLogGroupName = logGroupName.replace(/:\\*$/, '');\n\n    class Import extends LogGroupBase {\n      public readonly logGroupName = baseLogGroupName;\n      public readonly logGroupArn = Stack.of(scope).formatArn({\n        service: 'logs',\n        resource: 'log-group',\n        arnFormat: ArnFormat.COLON_RESOURCE_NAME,\n        resourceName: baseLogGroupName + ':*',\n      });\n    }\n\n    return new Import(scope, id);\n  }\n\n  /**\n   * The ARN of this log group\n   */\n  public readonly logGroupArn: string;\n\n  /**\n   * The name of this log group\n   */\n  public readonly logGroupName: string;\n\n  constructor(scope: Construct, id: string, props: LogGroupProps = {}) {\n    super(scope, id, {\n      physicalName: props.logGroupName,\n    });\n\n    let retentionInDays = props.retention;\n    if (retentionInDays === undefined) { retentionInDays = RetentionDays.TWO_YEARS; }\n    if (retentionInDays === Infinity || retentionInDays === RetentionDays.INFINITE) { retentionInDays = undefined; }\n\n    if (retentionInDays !== undefined && !Token.isUnresolved(retentionInDays) && retentionInDays <= 0) {\n      throw new Error(`retentionInDays must be positive, got ${retentionInDays}`);\n    }\n\n    const resource = new CfnLogGroup(this, 'Resource', {\n      kmsKeyId: props.encryptionKey?.keyArn,\n      logGroupName: this.physicalName,\n      retentionInDays,\n    });\n\n    resource.applyRemovalPolicy(props.removalPolicy);\n\n    this.logGroupArn = this.getResourceArnAttribute(resource.attrArn, {\n      service: 'logs',\n      resource: 'log-group',\n      resourceName: this.physicalName,\n      arnFormat: ArnFormat.COLON_RESOURCE_NAME,\n    });\n    this.logGroupName = this.getResourceNameAttribute(resource.ref);\n  }\n}\n\n/**\n * Properties for a new LogStream created from a LogGroup\n */\nexport interface StreamOptions {\n  /**\n   * The name of the log stream to create.\n   *\n   * The name must be unique within the log group.\n   *\n   * @default Automatically generated\n   */\n  readonly logStreamName?: string;\n}\n\n/**\n * Properties for a new SubscriptionFilter created from a LogGroup\n */\nexport interface SubscriptionFilterOptions {\n  /**\n   * The destination to send the filtered events to.\n   *\n   * For example, a Kinesis stream or a Lambda function.\n   */\n  readonly destination: ILogSubscriptionDestination;\n\n  /**\n   * Log events matching this pattern will be sent to the destination.\n   */\n  readonly filterPattern: IFilterPattern;\n}\n\n/**\n * Properties for a MetricFilter created from a LogGroup\n */\nexport interface MetricFilterOptions {\n  /**\n   * Pattern to search for log events.\n   */\n  readonly filterPattern: IFilterPattern;\n\n  /**\n   * The namespace of the metric to emit.\n   */\n  readonly metricNamespace: string;\n\n  /**\n   * The name of the metric to emit.\n   */\n  readonly metricName: string;\n\n  /**\n   * The value to emit for the metric.\n   *\n   * Can either be a literal number (typically \"1\"), or the name of a field in the structure\n   * to take the value from the matched event. If you are using a field value, the field\n   * value must have been matched using the pattern.\n   *\n   * If you want to specify a field from a matched JSON structure, use '$.fieldName',\n   * and make sure the field is in the pattern (if only as '$.fieldName = *').\n   *\n   * If you want to specify a field from a matched space-delimited structure,\n   * use '$fieldName'.\n   *\n   * @default \"1\"\n   */\n  readonly metricValue?: string;\n\n  /**\n   * The value to emit if the pattern does not match a particular event.\n   *\n   * @default No metric emitted.\n   */\n  readonly defaultValue?: number;\n}\n"]} |
\ | No newline at end of file |