@aws-cdk/aws-logs
Version:
The CDK Construct Library for AWS::Logs
314 lines • 37.1 kB
JavaScript
;
var _a;
Object.defineProperty(exports, "__esModule", { value: true });
exports.LogGroup = exports.RetentionDays = void 0;
const jsiiDeprecationWarnings = require("../.warnings.jsii.js");
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
const cloudwatch = require("@aws-cdk/aws-cloudwatch");
const iam = require("@aws-cdk/aws-iam");
const core_1 = require("@aws-cdk/core");
const log_stream_1 = require("./log-stream");
const logs_generated_1 = require("./logs.generated");
const metric_filter_1 = require("./metric-filter");
const pattern_1 = require("./pattern");
const policy_1 = require("./policy");
const subscription_filter_1 = require("./subscription-filter");
/**
* An CloudWatch Log Group
*/
class LogGroupBase extends core_1.Resource {
/**
* Create a new Log Stream for this Log Group
*
* @param id Unique identifier for the construct in its parent
* @param props Properties for creating the LogStream
*/
addStream(id, props = {}) {
return new log_stream_1.LogStream(this, id, {
logGroup: this,
...props,
});
}
/**
* Create a new Subscription Filter on this Log Group
*
* @param id Unique identifier for the construct in its parent
* @param props Properties for creating the SubscriptionFilter
*/
addSubscriptionFilter(id, props) {
return new subscription_filter_1.SubscriptionFilter(this, id, {
logGroup: this,
...props,
});
}
/**
* Create a new Metric Filter on this Log Group
*
* @param id Unique identifier for the construct in its parent
* @param props Properties for creating the MetricFilter
*/
addMetricFilter(id, props) {
return new metric_filter_1.MetricFilter(this, id, {
logGroup: this,
...props,
});
}
/**
* Extract a metric from structured log events in the LogGroup
*
* Creates a MetricFilter on this LogGroup that will extract the value
* of the indicated JSON field in all records where it occurs.
*
* The metric will be available in CloudWatch Metrics under the
* indicated namespace and name.
*
* @param jsonField JSON field to extract (example: '$.myfield')
* @param metricNamespace Namespace to emit the metric under
* @param metricName Name to emit the metric under
* @returns A Metric object representing the extracted metric
*/
extractMetric(jsonField, metricNamespace, metricName) {
new metric_filter_1.MetricFilter(this, `${metricNamespace}_${metricName}`, {
logGroup: this,
metricNamespace,
metricName,
filterPattern: pattern_1.FilterPattern.exists(jsonField),
metricValue: jsonField,
});
return new cloudwatch.Metric({ metricName, namespace: metricNamespace }).attachTo(this);
}
/**
* Give permissions to create and write to streams in this log group
*/
grantWrite(grantee) {
return this.grant(grantee, 'logs:CreateLogStream', 'logs:PutLogEvents');
}
/**
* Give the indicated permissions on this log group and all streams
*/
grant(grantee, ...actions) {
return iam.Grant.addToPrincipalOrResource({
grantee,
actions,
// A LogGroup ARN out of CloudFormation already includes a ':*' at the end to include the log streams under the group.
// See https://docs.aws.amazon.com/AWSCloudFormation/latest/UserGuide/aws-resource-logs-loggroup.html#w2ab1c21c10c63c43c11
resourceArns: [this.logGroupArn],
resource: this,
});
}
/**
* Public method to get the physical name of this log group
* @returns Physical name of log group
*/
logGroupPhysicalName() {
return this.physicalName;
}
/**
* Adds a statement to the resource policy associated with this log group.
* A resource policy will be automatically created upon the first call to `addToResourcePolicy`.
*
* Any ARN Principals inside of the statement will be converted into AWS Account ID strings
* because CloudWatch Logs Resource Policies do not accept ARN principals.
*
* @param statement The policy statement to add
*/
addToResourcePolicy(statement) {
if (!this.policy) {
this.policy = new policy_1.ResourcePolicy(this, 'Policy');
}
this.policy.document.addStatements(statement.copy({
principals: statement.principals.map(p => this.convertArnPrincpalToAccountId(p)),
}));
return { statementAdded: true, policyDependable: this.policy };
}
convertArnPrincpalToAccountId(principal) {
if (principal.principalAccount) {
// we use ArnPrincipal here because the constructor inserts the argument
// into the template without mutating it, which means that there is no
// ARN created by this call.
return new iam.ArnPrincipal(principal.principalAccount);
}
if (principal instanceof iam.ArnPrincipal) {
const parsedArn = core_1.Arn.split(principal.arn, core_1.ArnFormat.SLASH_RESOURCE_NAME);
if (parsedArn.account) {
return new iam.ArnPrincipal(parsedArn.account);
}
}
return principal;
}
}
/**
* How long, in days, the log contents will be retained.
*/
var RetentionDays;
(function (RetentionDays) {
/**
* 1 day
*/
RetentionDays[RetentionDays["ONE_DAY"] = 1] = "ONE_DAY";
/**
* 3 days
*/
RetentionDays[RetentionDays["THREE_DAYS"] = 3] = "THREE_DAYS";
/**
* 5 days
*/
RetentionDays[RetentionDays["FIVE_DAYS"] = 5] = "FIVE_DAYS";
/**
* 1 week
*/
RetentionDays[RetentionDays["ONE_WEEK"] = 7] = "ONE_WEEK";
/**
* 2 weeks
*/
RetentionDays[RetentionDays["TWO_WEEKS"] = 14] = "TWO_WEEKS";
/**
* 1 month
*/
RetentionDays[RetentionDays["ONE_MONTH"] = 30] = "ONE_MONTH";
/**
* 2 months
*/
RetentionDays[RetentionDays["TWO_MONTHS"] = 60] = "TWO_MONTHS";
/**
* 3 months
*/
RetentionDays[RetentionDays["THREE_MONTHS"] = 90] = "THREE_MONTHS";
/**
* 4 months
*/
RetentionDays[RetentionDays["FOUR_MONTHS"] = 120] = "FOUR_MONTHS";
/**
* 5 months
*/
RetentionDays[RetentionDays["FIVE_MONTHS"] = 150] = "FIVE_MONTHS";
/**
* 6 months
*/
RetentionDays[RetentionDays["SIX_MONTHS"] = 180] = "SIX_MONTHS";
/**
* 1 year
*/
RetentionDays[RetentionDays["ONE_YEAR"] = 365] = "ONE_YEAR";
/**
* 13 months
*/
RetentionDays[RetentionDays["THIRTEEN_MONTHS"] = 400] = "THIRTEEN_MONTHS";
/**
* 18 months
*/
RetentionDays[RetentionDays["EIGHTEEN_MONTHS"] = 545] = "EIGHTEEN_MONTHS";
/**
* 2 years
*/
RetentionDays[RetentionDays["TWO_YEARS"] = 731] = "TWO_YEARS";
/**
* 5 years
*/
RetentionDays[RetentionDays["FIVE_YEARS"] = 1827] = "FIVE_YEARS";
/**
* 6 years
*/
RetentionDays[RetentionDays["SIX_YEARS"] = 2192] = "SIX_YEARS";
/**
* 7 years
*/
RetentionDays[RetentionDays["SEVEN_YEARS"] = 2557] = "SEVEN_YEARS";
/**
* 8 years
*/
RetentionDays[RetentionDays["EIGHT_YEARS"] = 2922] = "EIGHT_YEARS";
/**
* 9 years
*/
RetentionDays[RetentionDays["NINE_YEARS"] = 3288] = "NINE_YEARS";
/**
* 10 years
*/
RetentionDays[RetentionDays["TEN_YEARS"] = 3653] = "TEN_YEARS";
/**
* Retain logs forever
*/
RetentionDays[RetentionDays["INFINITE"] = 9999] = "INFINITE";
})(RetentionDays = exports.RetentionDays || (exports.RetentionDays = {}));
/**
* Define a CloudWatch Log Group
*/
class LogGroup extends LogGroupBase {
constructor(scope, id, props = {}) {
super(scope, id, {
physicalName: props.logGroupName,
});
try {
jsiiDeprecationWarnings._aws_cdk_aws_logs_LogGroupProps(props);
}
catch (error) {
if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
Error.captureStackTrace(error, LogGroup);
}
throw error;
}
let retentionInDays = props.retention;
if (retentionInDays === undefined) {
retentionInDays = RetentionDays.TWO_YEARS;
}
if (retentionInDays === Infinity || retentionInDays === RetentionDays.INFINITE) {
retentionInDays = undefined;
}
if (retentionInDays !== undefined && !core_1.Token.isUnresolved(retentionInDays) && retentionInDays <= 0) {
throw new Error(`retentionInDays must be positive, got ${retentionInDays}`);
}
const resource = new logs_generated_1.CfnLogGroup(this, 'Resource', {
kmsKeyId: props.encryptionKey?.keyArn,
logGroupName: this.physicalName,
retentionInDays,
});
resource.applyRemovalPolicy(props.removalPolicy);
this.logGroupArn = this.getResourceArnAttribute(resource.attrArn, {
service: 'logs',
resource: 'log-group',
resourceName: this.physicalName,
arnFormat: core_1.ArnFormat.COLON_RESOURCE_NAME,
});
this.logGroupName = this.getResourceNameAttribute(resource.ref);
}
/**
* Import an existing LogGroup given its ARN
*/
static fromLogGroupArn(scope, id, logGroupArn) {
const baseLogGroupArn = logGroupArn.replace(/:\*$/, '');
class Import extends LogGroupBase {
constructor() {
super(...arguments);
this.logGroupArn = `${baseLogGroupArn}:*`;
this.logGroupName = core_1.Stack.of(scope).splitArn(baseLogGroupArn, core_1.ArnFormat.COLON_RESOURCE_NAME).resourceName;
}
}
return new Import(scope, id, {
environmentFromArn: baseLogGroupArn,
});
}
/**
* Import an existing LogGroup given its name
*/
static fromLogGroupName(scope, id, logGroupName) {
const baseLogGroupName = logGroupName.replace(/:\*$/, '');
class Import extends LogGroupBase {
constructor() {
super(...arguments);
this.logGroupName = baseLogGroupName;
this.logGroupArn = core_1.Stack.of(scope).formatArn({
service: 'logs',
resource: 'log-group',
arnFormat: core_1.ArnFormat.COLON_RESOURCE_NAME,
resourceName: baseLogGroupName + ':*',
});
}
}
return new Import(scope, id);
}
}
exports.LogGroup = LogGroup;
_a = JSII_RTTI_SYMBOL_1;
LogGroup[_a] = { fqn: "@aws-cdk/aws-logs.LogGroup", version: "1.204.0" };
//# 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"]}