UNPKG

37.1 kBJavaScriptView Raw
1"use strict";
2var _a;
3Object.defineProperty(exports, "__esModule", { value: true });
4exports.LogGroup = exports.RetentionDays = void 0;
5const jsiiDeprecationWarnings = require("../.warnings.jsii.js");
6const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
7const cloudwatch = require("@aws-cdk/aws-cloudwatch");
8const iam = require("@aws-cdk/aws-iam");
9const core_1 = require("@aws-cdk/core");
10const log_stream_1 = require("./log-stream");
11const logs_generated_1 = require("./logs.generated");
12const metric_filter_1 = require("./metric-filter");
13const pattern_1 = require("./pattern");
14const policy_1 = require("./policy");
15const subscription_filter_1 = require("./subscription-filter");
16/**
17 * An CloudWatch Log Group
18 */
19class 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 */
143var 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 */
237class 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}
311exports.LogGroup = LogGroup;
312_a = JSII_RTTI_SYMBOL_1;
313LogGroup[_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