@liflig/cdk 2.14.4 → 2.15.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.
@@ -1,5 +1,7 @@
1
1
  import json
2
+ import logging
2
3
  import os
4
+ import typing as t
3
5
  from urllib.error import HTTPError, URLError
4
6
  from urllib.parse import quote
5
7
  from urllib.request import Request, urlopen
@@ -7,12 +9,12 @@ from urllib.request import Request, urlopen
7
9
  import boto3
8
10
 
9
11
  client = boto3.client("codepipeline")
12
+ s3 = boto3.client("s3")
10
13
 
11
- ACCOUNT_DESC = os.getenv("ACCOUNT_DESC", None)
12
- ACCOUNT_GROUP_NAME = os.getenv("ACCOUNT_GROUP_NAME", None)
14
+ ACCOUNT_FRIENDLY_NAME = os.getenv("ACCOUNT_FRIENDLY_NAME", None)
13
15
  SLACK_URL = os.getenv("SLACK_URL", None)
14
16
  SLACK_CHANNEL = os.getenv("SLACK_CHANNEL", None)
15
- ALWAYS_SHOW_SUCCEEDED = os.getenv("ALWAYS_SHOW_SUCCEEDED", "false") == "true"
17
+ NOTIFICATION_LEVEL = os.getenv("NOTIFICATION_LEVEL", "WARN")
16
18
 
17
19
  # Example event:
18
20
  #
@@ -35,10 +37,38 @@ ALWAYS_SHOW_SUCCEEDED = os.getenv("ALWAYS_SHOW_SUCCEEDED", "false") == "true"
35
37
  # }
36
38
  # }
37
39
 
40
+ STYLES = {
41
+ "FAILED": {"emoji_prefix": ":x:", "message_color": "#ff0000"},
42
+ "SUCCEEDED": {"emoji_prefix": ":white_check_mark:", "message_color": "#008000"},
43
+ "STARTED": {"emoji_prefix": ":rocket:", "message_color": "#00bfff"},
44
+ "SUPERSEDED": {"emoji_prefix": ":arrow_heading_down:", "message_color": "#373737"},
45
+ }
46
+
47
+
48
+ class TriggerMetadataVcs(t.TypedDict):
49
+ branchName: str
50
+ commitAuthor: str
51
+ commitHash: str
52
+ repositoryName: str
53
+ repositoryOwner: str
54
+
55
+
56
+ class TriggerMetadataCi(t.TypedDict):
57
+ type: t.Literal["JENKINS", "GITHUB_ACTIONS"]
58
+ triggeredBy: str
59
+
60
+
61
+ class TriggerMetadata(t.TypedDict):
62
+ version: t.Literal["0.1"]
63
+ ci: TriggerMetadataCi
64
+ vcs: TriggerMetadataVcs
65
+
66
+
67
+ def get_previous_pipeline_execution(
68
+ pipeline_name: str, execution_id: str
69
+ ) -> dict | None:
70
+ """Return the newest past execution that either succeeded or failed"""
38
71
 
39
- def get_previous_pipeline_execution(pipeline_name, execution_id):
40
- # Retrieves executions based on their start timestamp, so
41
- # the previous execution will be next in list.
42
72
  pipeline_executions = client.list_pipeline_executions(
43
73
  pipelineName=pipeline_name,
44
74
  )["pipelineExecutionSummaries"]
@@ -57,12 +87,15 @@ def get_previous_pipeline_execution(pipeline_name, execution_id):
57
87
  return None
58
88
 
59
89
 
60
- def get_blocks_for_failed(pipeline_name, execution_id, state):
90
+ def get_text_for_failed(pipeline_name: str, execution_id: str, state: str) -> str:
91
+ """Return a Slack-formatted string that describes failed pipeline execution actions,
92
+ if any, in a failed execution"""
93
+
61
94
  # We only show details if the pipeline has completed with failed state.
62
95
  # If we were to process this for other events such as started events,
63
96
  # we would include details from after the event took place.
64
97
  if state != "FAILED":
65
- return []
98
+ return ""
66
99
 
67
100
  action_executions = client.list_action_executions(
68
101
  pipelineName=pipeline_name,
@@ -71,7 +104,7 @@ def get_blocks_for_failed(pipeline_name, execution_id, state):
71
104
  },
72
105
  )["actionExecutionDetails"]
73
106
 
74
- result = []
107
+ failures = []
75
108
 
76
109
  for action_execution in action_executions:
77
110
  if action_execution["status"] == "Failed":
@@ -80,32 +113,95 @@ def get_blocks_for_failed(pipeline_name, execution_id, state):
80
113
  summary = action_execution["output"]["executionResult"][
81
114
  "externalExecutionSummary"
82
115
  ]
83
- result.append(
84
- {
85
- "type": "section",
86
- "text": {
87
- "type": "mrkdwn",
88
- "text": f"{stage}.{action} failed:\n```\n{summary}\n```",
89
- },
90
- }
91
- )
116
+ failures.append(f"{stage}.{action} failed:\n{summary}")
117
+
118
+ result = ""
119
+
120
+ if len(failures):
121
+ result = "```\n" + "\n\n".join(failures) + "\n```"
92
122
 
93
123
  return result
94
124
 
95
125
 
126
+ def get_metadata_from_trigger(
127
+ pipeline_name: str, execution_id: str
128
+ ) -> TriggerMetadata | None:
129
+ """Returns a dictionary containing the metadata, if any, stored in the trigger file"""
130
+
131
+ action_response = client.list_action_executions(
132
+ pipelineName=pipeline_name, filter={"pipelineExecutionId": execution_id}
133
+ )
134
+
135
+ action = next(
136
+ (
137
+ action
138
+ for action in action_response["actionExecutionDetails"]
139
+ if action["input"]["actionTypeId"]["category"] == "Source"
140
+ and action["input"]["actionTypeId"]["provider"] == "S3"
141
+ ),
142
+ None,
143
+ )
144
+ if action:
145
+ s3_version_id = action["output"]["outputVariables"]["VersionId"]
146
+ artifacts_bucket = action["input"]["configuration"]["S3Bucket"]
147
+ trigger_file = action["input"]["configuration"]["S3ObjectKey"]
148
+
149
+ try:
150
+ response = s3.get_object(
151
+ Bucket=artifacts_bucket, Key=trigger_file, VersionId=s3_version_id
152
+ )
153
+ file_content = response["Body"].read().decode("utf-8")
154
+ ci_metadata = json.loads(file_content)
155
+ return ci_metadata
156
+ except Exception as e:
157
+ print(f"Could not obtain metadata from trigger file: {e}")
158
+
159
+ return None
160
+
161
+
162
+ def get_footer_text(ci_metadata: TriggerMetadata) -> str:
163
+ """Returns the footer text for the Slack message if the metadata contains the required fields"""
164
+
165
+ footer_text = ""
166
+ if ci_metadata and ci_metadata.get("version", "") == "0.1":
167
+ ci = ci_metadata.get("ci", {})
168
+ vcs = ci_metadata.get("vcs", {})
169
+ triggering_actor = ci.get("triggeredBy", "")
170
+ repository_owner = vcs.get("repositoryOwner", "")
171
+ repository_name = vcs.get("repositoryName", "")
172
+ short_commit_hash = vcs.get("commitHash", "")[:8]
173
+ branch_name = vcs.get("branchName", "")
174
+ if (
175
+ triggering_actor
176
+ and repository_owner
177
+ and repository_name
178
+ and short_commit_hash
179
+ and branch_name
180
+ ):
181
+ commit_link_text = f"{repository_owner}/{repository_name} @ {branch_name} ({short_commit_hash})"
182
+ github_commit_link = f"https://github.com/{repository_owner}/{repository_name}/commit/{short_commit_hash}"
183
+ footer_text = f"Triggered by {triggering_actor} in <{github_commit_link}|{commit_link_text}>"
184
+
185
+ return footer_text
186
+
187
+
96
188
  def handler(event, context):
97
- print("Event: " + json.dumps(event))
98
189
 
99
- if event["detail-type"] != "CodePipeline Pipeline Execution State Change":
100
- print("Ignoring unknown event")
101
- return
190
+ print("Event: " + json.dumps(event))
102
191
 
103
- account = event["account"]
104
192
  region = event["region"]
193
+ account_id = event["account"]
105
194
  pipeline_name = event["detail"]["pipeline"]
106
195
  state = event["detail"]["state"]
107
196
  execution_id = event["detail"]["execution-id"]
108
197
 
198
+ if state in ("STARTED", "SUPERSEDED") and NOTIFICATION_LEVEL != "DEBUG":
199
+ return
200
+
201
+ if event["detail-type"] != "CodePipeline Pipeline Execution State Change":
202
+ print("Ignoring unknown event")
203
+ return
204
+
109
205
  previous_pipeline_execution = get_previous_pipeline_execution(
110
206
  pipeline_name, execution_id
111
207
  )
@@ -117,49 +213,63 @@ def handler(event, context):
117
213
 
118
214
  # We still show succeeded for the first event or when
119
215
  # the previous execution was not success.
120
- if state == "SUCCEEDED" and not ALWAYS_SHOW_SUCCEEDED:
216
+ if state == "SUCCEEDED" and (NOTIFICATION_LEVEL == "WARN"):
121
217
  if previous_pipeline_execution is not None and not previous_failed:
122
218
  print("Ignoring succeeded event")
123
219
  return
124
220
 
125
- emoji_prefix = ""
126
- if state == "FAILED":
127
- emoji_prefix = ":x: "
128
- if state == "SUCCEEDED":
129
- emoji_prefix = ":white_check_mark: "
130
-
131
221
  pipeline_url = f"https://{region}.console.aws.amazon.com/codesuite/codepipeline/pipelines/{quote(pipeline_name, safe='')}/view"
132
222
  execution_url = f"https://{region}.console.aws.amazon.com/codesuite/codepipeline/pipelines/{quote(pipeline_name, safe='')}/executions/{execution_id}/timeline"
133
223
 
134
- account_group_text = ""
135
- if ACCOUNT_GROUP_NAME is not None:
136
- account_group_text += f" ({ACCOUNT_GROUP_NAME})"
224
+ account_friendly_name = f"in {ACCOUNT_FRIENDLY_NAME or account_id}"
137
225
 
138
226
  state_text = state
139
227
  if previous_failed and state == "SUCCEEDED":
140
228
  state_text += " (previously failed)"
141
229
 
142
- blocks_for_failed = get_blocks_for_failed(pipeline_name, execution_id, state)
230
+ ci_metadata = get_metadata_from_trigger(pipeline_name, execution_id)
231
+
232
+ footer_text = get_footer_text(ci_metadata)
143
233
 
144
- account_text = account
145
- if ACCOUNT_DESC is not None:
146
- account_text += f" ({ACCOUNT_DESC})"
234
+ style = STYLES.get(
235
+ state, {"emoji_prefix": ":question:", "message_color": "#ffdf00"}
236
+ )
237
+
238
+ emoji_prefix = style["emoji_prefix"]
239
+ message_color = style["message_color"]
240
+
241
+ text_for_failed = get_text_for_failed(pipeline_name, execution_id, state)
147
242
 
148
- blocks = [
243
+ text = "\n".join(
244
+ s
245
+ for s in [f"*Execution:* <{execution_url}|{execution_id}>", text_for_failed]
246
+ if s
247
+ )
248
+ pretext = " ".join(
249
+ s
250
+ for s in [
251
+ f"{emoji_prefix} Pipeline *<{pipeline_url}|{pipeline_name}>*",
252
+ f"*{state_text}*",
253
+ account_friendly_name,
254
+ ]
255
+ if s
256
+ )
257
+ fallback = f"Pipeline {pipeline_name} {state}"
258
+ attachments = [
149
259
  {
150
- "type": "section",
151
- "text": {
152
- "type": "mrkdwn",
153
- "text": f"{emoji_prefix}Pipeline <{pipeline_url}|{pipeline_name}>{account_group_text} *{state_text}*\n{region} | {account_text}\n<{execution_url}|Execution details>",
154
- },
260
+ "footer": footer_text,
261
+ "color": message_color,
262
+ "text": text,
263
+ "mrkdwn_in": ["text", "pretext"],
264
+ "pretext": pretext,
265
+ "fallback": fallback,
155
266
  },
156
- *blocks_for_failed,
157
267
  ]
158
268
 
159
269
  slack_message = {
160
270
  "channel": SLACK_CHANNEL,
161
- "blocks": blocks,
162
- "username": "Pipeline Status",
271
+ "attachments": attachments,
272
+ "username": "Liflig CDK Pipelines",
163
273
  "icon_emoji": ":traffic_light:",
164
274
  }
165
275
 
@@ -100,9 +100,11 @@ export declare class LifligCdkPipeline extends constructs.Construct {
100
100
  static pipelineS3TriggerKey(pipelineName: string): string;
101
101
  readonly cdkPipeline: pipelines.CodePipeline;
102
102
  readonly codePipeline: codepipeline.Pipeline;
103
+ readonly artifactsBucket: s3.IBucket;
104
+ readonly triggerObjectKey: string;
103
105
  constructor(scope: constructs.Construct, id: string, props: LifligCdkPipelineProps);
104
106
  private static getAwsCdkPackageJsonFile;
105
107
  private cloudAssemblyStage;
106
108
  private cdkSourceStage;
107
- addSlackNotification(props: Omit<SlackNotificationProps, "pipeline">): void;
109
+ addSlackNotification(props: Omit<SlackNotificationProps, "pipeline" | "artifactsBucket" | "triggerObjectKey">): void;
108
110
  }
@@ -73,23 +73,24 @@ class LifligCdkPipeline extends constructs.Construct {
73
73
  constructor(scope, id, props) {
74
74
  var _a, _b;
75
75
  super(scope, id);
76
- const artifactsBucket = (_a = props.artifactsBucket) !== null && _a !== void 0 ? _a : (0, artefact_bucket_1.getGriidArtefactBucket)(this);
76
+ this.artifactsBucket = (_a = props.artifactsBucket) !== null && _a !== void 0 ? _a : (0, artefact_bucket_1.getGriidArtefactBucket)(this);
77
77
  const cloudAssemblyArtifact = new codepipeline.Artifact();
78
78
  let synth;
79
79
  let stages;
80
80
  switch (props.sourceType) {
81
81
  case "cloud-assembly":
82
- const cloudAssembly = this.cloudAssemblyStage(cloudAssemblyArtifact, artifactsBucket, props.pipelineName);
82
+ const cloudAssembly = this.cloudAssemblyStage(cloudAssemblyArtifact, this.artifactsBucket, props.pipelineName);
83
83
  synth = cloudAssembly.synth;
84
84
  stages = cloudAssembly.stages;
85
85
  break;
86
86
  case "cdk-source":
87
- const cdkSource = this.cdkSourceStage(cloudAssemblyArtifact, artifactsBucket, props.pipelineName, (_b = props.parametersNamespace) !== null && _b !== void 0 ? _b : "default");
87
+ const cdkSource = this.cdkSourceStage(cloudAssemblyArtifact, this.artifactsBucket, props.pipelineName, (_b = props.parametersNamespace) !== null && _b !== void 0 ? _b : "default");
88
88
  synth = cdkSource.synth;
89
89
  stages = cdkSource.stages;
90
90
  break;
91
91
  }
92
92
  const dummyArtifact = new codepipeline.Artifact();
93
+ this.triggerObjectKey = LifligCdkPipeline.pipelineS3TriggerKey(props.pipelineName);
93
94
  this.codePipeline = new codepipeline.Pipeline(this, "CodePipeline", {
94
95
  pipelineName: props.pipelineName,
95
96
  stages: [
@@ -98,9 +99,9 @@ class LifligCdkPipeline extends constructs.Construct {
98
99
  actions: [
99
100
  new codepipelineActions.S3SourceAction({
100
101
  actionName: "source",
101
- bucket: artifactsBucket,
102
+ bucket: this.artifactsBucket,
102
103
  trigger: codepipelineActions.S3Trigger.NONE,
103
- bucketKey: LifligCdkPipeline.pipelineS3TriggerKey(props.pipelineName),
104
+ bucketKey: this.triggerObjectKey,
104
105
  output: dummyArtifact,
105
106
  }),
106
107
  ],
@@ -115,10 +116,10 @@ class LifligCdkPipeline extends constructs.Construct {
115
116
  detailType: ["Object Created"],
116
117
  detail: {
117
118
  bucket: {
118
- name: [artifactsBucket.bucketName],
119
+ name: [this.artifactsBucket.bucketName],
119
120
  },
120
121
  object: {
121
- key: [LifligCdkPipeline.pipelineS3TriggerKey(props.pipelineName)],
122
+ key: [this.triggerObjectKey],
122
123
  },
123
124
  },
124
125
  },
@@ -221,9 +222,11 @@ class LifligCdkPipeline extends constructs.Construct {
221
222
  addSlackNotification(props) {
222
223
  new slack_notification_1.SlackNotification(this, "Slack", {
223
224
  pipeline: this.codePipeline,
225
+ artifactsBucket: this.artifactsBucket,
226
+ triggerObjectKey: this.triggerObjectKey,
224
227
  ...props,
225
228
  });
226
229
  }
227
230
  }
228
231
  exports.LifligCdkPipeline = LifligCdkPipeline;
229
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"liflig-cdk-pipeline.js","sourceRoot":"","sources":["../../src/cdk-pipelines/liflig-cdk-pipeline.ts"],"names":[],"mappings":"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","sourcesContent":["import * as constructs from \"constructs\"\nimport * as codepipeline from \"aws-cdk-lib/aws-codepipeline\"\nimport * as codepipelineActions from \"aws-cdk-lib/aws-codepipeline-actions\"\nimport * as iam from \"aws-cdk-lib/aws-iam\"\nimport * as lambda from \"aws-cdk-lib/aws-lambda\"\nimport * as events from \"aws-cdk-lib/aws-events\"\nimport * as targets from \"aws-cdk-lib/aws-events-targets\"\nimport * as s3 from \"aws-cdk-lib/aws-s3\"\nimport * as cdk from \"aws-cdk-lib\"\nimport * as pipelines from \"aws-cdk-lib/pipelines\"\nimport * as fs from \"fs\"\nimport * as path from \"path\"\nimport { getGriidArtefactBucket } from \"../griid/artefact-bucket\"\nimport {\n  cloudAssemblyLookupHandler,\n  CloudAssemblyLookupUserParameters,\n} from \"./cloud-assembly-lookup-handler\"\nimport { SlackNotification, SlackNotificationProps } from \"./slack-notification\"\n\nexport interface LifligCdkPipelineProps {\n  /**\n   * Bucket holding pipeline configuration and trigger file.\n   *\n   * @default - use existing bucket based on Griid conventions\n   */\n  artifactsBucket?: s3.IBucket\n  /**\n   * Name of pipeline. This is used for the path where configuration\n   * is stored in S3.\n   */\n  pipelineName: string\n  /**\n   * Type of uploaded artifact. This changes the behaviour of the\n   * pipeline and what kind of process it performs.\n   *\n   * Two types are supported:\n   *\n   *   - cdk-source: The uploaded artifact represents a CDK application\n   *     as source code. A build step will compile this into a\n   *     CDK Cloud Assembly.\n   *\n   *     As part of synthesizing this into a CDK Cloud Assembly,\n   *     a file \"variables.json\" will be written for the\n   *     CDK application to parameterize the build if any\n   *     variables are found in the pipeline source.\n   *\n   *   - cloud-assembly: The uploaded artifact represents a\n   *     CDK Cloud Assembly which is ready for deployment.\n   *\n   *     This does not support reading variables at the current time\n   *     since CDK Pipelines don't support parameterized deploys.\n   *     See https://github.com/aws/aws-cdk/issues/9560\n   */\n  sourceType: \"cdk-source\" | \"cloud-assembly\"\n  /**\n   * The namespace used for parameters in Parameter Store.\n   *\n   * Only relevant for sourceType of \"cdk-soruce\".\n   *\n   * @default default\n   */\n  parametersNamespace?: string\n}\n\n/**\n * CDK Pipeline for Liflig.\n *\n * Avoid putting multiple pipelines in a stack, since the pipeline\n * will also keep the hosting stack up-to-date.\n *\n * The pipeline is executed by writing an empty file to\n * s3://<artifacts-bucket>/pipelines/<pipeline-name>/trigger\n *\n * Configuration files are read from S3 at the path\n * s3://<artifacts-bucket>/pipelines/<pipeline-name>/\n *\n * For upload type \"cdk-source\":\n *\n *   - cdk-source.json holding a pointer to the active CDK source\n *     that should be used. Schema:\n *\n *     {\n *       bucketName: string\n *       bucketKey: string\n *     }\n *\n *   - variables*.json which can be zero or more files\n *     with string-string map holding variables that will\n *     be written to variables.json and can be read by the\n *     the CDK application during synthesize.\n *\n * For upload type \"cloud-assembly\":\n *\n *   - cloud-assembly.json holding a pointer to the active\n *     CDK Cloud Assembly that should be used: Schema:\n *\n *     {\n *       cloudAssemblyBucketName: string\n *       cloudAssemblyBucketKey: string\n *     }\n *\n * Variables enables separation of IaC code and application code if\n * they are not colocated in the same repository.\n */\nexport class LifligCdkPipeline extends constructs.Construct {\n  /**\n   * Path on S3 for pipeline configuration.\n   */\n  static pipelineS3Prefix(pipelineName: string): string {\n    return `pipelines/${pipelineName}/`\n  }\n\n  /**\n   * Key in S3 bucket used to trigger pipeline.\n   *\n   * This is an empty file within the pipeline path.\n   */\n  static pipelineS3TriggerKey(pipelineName: string): string {\n    return `pipelines/${pipelineName}/trigger`\n  }\n\n  public readonly cdkPipeline: pipelines.CodePipeline\n  public readonly codePipeline: codepipeline.Pipeline\n\n  constructor(\n    scope: constructs.Construct,\n    id: string,\n    props: LifligCdkPipelineProps,\n  ) {\n    super(scope, id)\n\n    const artifactsBucket =\n      props.artifactsBucket ?? getGriidArtefactBucket(this)\n\n    const cloudAssemblyArtifact = new codepipeline.Artifact()\n\n    let synth: pipelines.IFileSetProducer\n    let stages: codepipeline.StageProps[]\n\n    switch (props.sourceType) {\n      case \"cloud-assembly\":\n        const cloudAssembly = this.cloudAssemblyStage(\n          cloudAssemblyArtifact,\n          artifactsBucket,\n          props.pipelineName,\n        )\n        synth = cloudAssembly.synth\n        stages = cloudAssembly.stages\n        break\n      case \"cdk-source\":\n        const cdkSource = this.cdkSourceStage(\n          cloudAssemblyArtifact,\n          artifactsBucket,\n          props.pipelineName,\n          props.parametersNamespace ?? \"default\",\n        )\n        synth = cdkSource.synth\n        stages = cdkSource.stages\n        break\n    }\n\n    const dummyArtifact = new codepipeline.Artifact()\n\n    this.codePipeline = new codepipeline.Pipeline(this, \"CodePipeline\", {\n      pipelineName: props.pipelineName,\n      stages: [\n        {\n          stageName: \"Source\",\n          actions: [\n            new codepipelineActions.S3SourceAction({\n              actionName: \"source\",\n              bucket: artifactsBucket,\n              trigger: codepipelineActions.S3Trigger.NONE,\n              bucketKey: LifligCdkPipeline.pipelineS3TriggerKey(\n                props.pipelineName,\n              ),\n              output: dummyArtifact,\n            }),\n          ],\n        },\n        ...stages,\n      ],\n      restartExecutionOnUpdate: true,\n    })\n\n    new events.Rule(this, \"PipelineTrigger\", {\n      eventPattern: {\n        source: [\"aws.s3\"],\n        detailType: [\"Object Created\"],\n        detail: {\n          bucket: {\n            name: [artifactsBucket.bucketName],\n          },\n          object: {\n            key: [LifligCdkPipeline.pipelineS3TriggerKey(props.pipelineName)],\n          },\n        },\n      },\n      targets: [new targets.CodePipeline(this.codePipeline)],\n    })\n\n    this.cdkPipeline = new pipelines.CodePipeline(this, \"CdkPipeline\", {\n      synth,\n      useChangeSets: false,\n      codePipeline: this.codePipeline,\n    })\n  }\n\n  private static getAwsCdkPackageJsonFile(): string | undefined {\n    // Also look up the tree a bit to handle yarn workspaces.\n    const candidates = [\n      path.join(process.cwd(), \"node_modules/aws-cdk/package.json\"),\n      path.join(process.cwd(), \"../node_modules/aws-cdk/package.json\"),\n      path.join(process.cwd(), \"../../node_modules/aws-cdk/package.json\"),\n      path.join(process.cwd(), \"../../../node_modules/aws-cdk/package.json\"),\n    ]\n\n    for (const candidate of candidates) {\n      if (fs.existsSync(candidate)) {\n        return candidate\n      }\n    }\n\n    return undefined\n  }\n\n  private cloudAssemblyStage(\n    cloudAssemblyArtifact: codepipeline.Artifact,\n    cdkBucket: s3.IBucket,\n    pipelineName: string,\n  ): { stages: codepipeline.StageProps[]; synth: pipelines.IFileSetProducer } {\n    const cloudAssemblyLookupFn = new lambda.Function(\n      this,\n      \"CloudAssemblyLookupFn\",\n      {\n        code: new lambda.InlineCode(\n          `exports.handler = ${cloudAssemblyLookupHandler.toString()};`,\n        ),\n        handler: \"index.handler\",\n        runtime: lambda.Runtime.NODEJS_16_X,\n        timeout: cdk.Duration.minutes(1),\n        memorySize: 512,\n      },\n    )\n\n    cdkBucket.grantReadWrite(cloudAssemblyLookupFn)\n\n    const userParameters: CloudAssemblyLookupUserParameters = {\n      bucketName: cdkBucket.bucketName,\n      objectKey: `pipelines/${pipelineName}/cloud-assembly.json`,\n    }\n\n    const synth = pipelines.CodePipelineFileSet.fromArtifact(\n      cloudAssemblyArtifact,\n    )\n\n    const stages = [\n      {\n        stageName: \"PrepareCloudAssembly\",\n        actions: [\n          new codepipelineActions.LambdaInvokeAction({\n            actionName: \"cloud-assembly-lookup\",\n            lambda: cloudAssemblyLookupFn,\n            outputs: [cloudAssemblyArtifact],\n            userParameters,\n          }),\n        ],\n      },\n    ]\n    return { stages, synth }\n  }\n\n  private cdkSourceStage(\n    cloudAssemblyArtifact: codepipeline.Artifact,\n    cdkBucket: s3.IBucket,\n    pipelineName: string,\n    parametersNamespace: string,\n  ): { stages: codepipeline.StageProps[]; synth: pipelines.IFileSetProducer } {\n    const prepareCdkSourceFn = new lambda.Function(this, \"PrepareCdkSourceFn\", {\n      code: lambda.Code.fromAsset(\n        path.join(__dirname, \"../../assets/prepare-cdk-source-lambda\"),\n      ),\n      handler: \"index.handler\",\n      // Using python instead if NodeJS due to zip-support in stdlib.\n      runtime: lambda.Runtime.PYTHON_3_8,\n      timeout: cdk.Duration.minutes(1),\n      memorySize: 512,\n    })\n\n    const account = cdk.Stack.of(this).account\n    const region = cdk.Stack.of(this).region\n\n    prepareCdkSourceFn.grantPrincipal.addToPrincipalPolicy(\n      new iam.PolicyStatement({\n        actions: [\"ssm:GetParametersByPath\"],\n        resources: [\n          `arn:aws:ssm:${region}:${account}:parameter/liflig-cdk/*/pipeline-variables/*`,\n        ],\n      }),\n    )\n\n    cdkBucket.grantReadWrite(prepareCdkSourceFn)\n\n    const cdkSourceArtifact = new codepipeline.Artifact()\n\n    const userParameters: PrepareCdkSourceUserParameters = {\n      bucketName: cdkBucket.bucketName,\n      prefix: `pipelines/${pipelineName}/`,\n      parametersNamespace: parametersNamespace,\n    }\n\n    const synth = new pipelines.ShellStep(\"GenerateCloudAssembly\", {\n      input: pipelines.CodePipelineFileSet.fromArtifact(cdkSourceArtifact),\n      installCommands: [\"npm ci\"],\n      commands: [\"npx cdk synth\"],\n    })\n    const stages = [\n      {\n        stageName: \"PrepareCdkSource\",\n        actions: [\n          new codepipelineActions.LambdaInvokeAction({\n            actionName: \"prepare-cdk-source\",\n            lambda: prepareCdkSourceFn,\n            outputs: [cdkSourceArtifact],\n            userParameters,\n          }),\n        ],\n      },\n    ]\n    return { stages, synth }\n  }\n\n  addSlackNotification(props: Omit<SlackNotificationProps, \"pipeline\">): void {\n    new SlackNotification(this, \"Slack\", {\n      pipeline: this.codePipeline,\n      ...props,\n    })\n  }\n}\n\ninterface PrepareCdkSourceUserParameters {\n  bucketName: string\n  prefix: string\n  parametersNamespace: string\n}\n"]}
232
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"liflig-cdk-pipeline.js","sourceRoot":"","sources":["../../src/cdk-pipelines/liflig-cdk-pipeline.ts"],"names":[],"mappings":"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","sourcesContent":["import * as constructs from \"constructs\"\nimport * as codepipeline from \"aws-cdk-lib/aws-codepipeline\"\nimport * as codepipelineActions from \"aws-cdk-lib/aws-codepipeline-actions\"\nimport * as iam from \"aws-cdk-lib/aws-iam\"\nimport * as lambda from \"aws-cdk-lib/aws-lambda\"\nimport * as events from \"aws-cdk-lib/aws-events\"\nimport * as targets from \"aws-cdk-lib/aws-events-targets\"\nimport * as s3 from \"aws-cdk-lib/aws-s3\"\nimport * as cdk from \"aws-cdk-lib\"\nimport * as pipelines from \"aws-cdk-lib/pipelines\"\nimport * as fs from \"fs\"\nimport * as path from \"path\"\nimport { getGriidArtefactBucket } from \"../griid/artefact-bucket\"\nimport {\n  cloudAssemblyLookupHandler,\n  CloudAssemblyLookupUserParameters,\n} from \"./cloud-assembly-lookup-handler\"\nimport { SlackNotification, SlackNotificationProps } from \"./slack-notification\"\n\nexport interface LifligCdkPipelineProps {\n  /**\n   * Bucket holding pipeline configuration and trigger file.\n   *\n   * @default - use existing bucket based on Griid conventions\n   */\n  artifactsBucket?: s3.IBucket\n  /**\n   * Name of pipeline. This is used for the path where configuration\n   * is stored in S3.\n   */\n  pipelineName: string\n  /**\n   * Type of uploaded artifact. This changes the behaviour of the\n   * pipeline and what kind of process it performs.\n   *\n   * Two types are supported:\n   *\n   *   - cdk-source: The uploaded artifact represents a CDK application\n   *     as source code. A build step will compile this into a\n   *     CDK Cloud Assembly.\n   *\n   *     As part of synthesizing this into a CDK Cloud Assembly,\n   *     a file \"variables.json\" will be written for the\n   *     CDK application to parameterize the build if any\n   *     variables are found in the pipeline source.\n   *\n   *   - cloud-assembly: The uploaded artifact represents a\n   *     CDK Cloud Assembly which is ready for deployment.\n   *\n   *     This does not support reading variables at the current time\n   *     since CDK Pipelines don't support parameterized deploys.\n   *     See https://github.com/aws/aws-cdk/issues/9560\n   */\n  sourceType: \"cdk-source\" | \"cloud-assembly\"\n  /**\n   * The namespace used for parameters in Parameter Store.\n   *\n   * Only relevant for sourceType of \"cdk-soruce\".\n   *\n   * @default default\n   */\n  parametersNamespace?: string\n}\n\n/**\n * CDK Pipeline for Liflig.\n *\n * Avoid putting multiple pipelines in a stack, since the pipeline\n * will also keep the hosting stack up-to-date.\n *\n * The pipeline is executed by writing an empty file to\n * s3://<artifacts-bucket>/pipelines/<pipeline-name>/trigger\n *\n * Configuration files are read from S3 at the path\n * s3://<artifacts-bucket>/pipelines/<pipeline-name>/\n *\n * For upload type \"cdk-source\":\n *\n *   - cdk-source.json holding a pointer to the active CDK source\n *     that should be used. Schema:\n *\n *     {\n *       bucketName: string\n *       bucketKey: string\n *     }\n *\n *   - variables*.json which can be zero or more files\n *     with string-string map holding variables that will\n *     be written to variables.json and can be read by the\n *     the CDK application during synthesize.\n *\n * For upload type \"cloud-assembly\":\n *\n *   - cloud-assembly.json holding a pointer to the active\n *     CDK Cloud Assembly that should be used: Schema:\n *\n *     {\n *       cloudAssemblyBucketName: string\n *       cloudAssemblyBucketKey: string\n *     }\n *\n * Variables enables separation of IaC code and application code if\n * they are not colocated in the same repository.\n */\nexport class LifligCdkPipeline extends constructs.Construct {\n  /**\n   * Path on S3 for pipeline configuration.\n   */\n  static pipelineS3Prefix(pipelineName: string): string {\n    return `pipelines/${pipelineName}/`\n  }\n\n  /**\n   * Key in S3 bucket used to trigger pipeline.\n   *\n   * This is an empty file within the pipeline path.\n   */\n  static pipelineS3TriggerKey(pipelineName: string): string {\n    return `pipelines/${pipelineName}/trigger`\n  }\n\n  public readonly cdkPipeline: pipelines.CodePipeline\n  public readonly codePipeline: codepipeline.Pipeline\n  public readonly artifactsBucket: s3.IBucket\n  public readonly triggerObjectKey: string\n\n  constructor(\n    scope: constructs.Construct,\n    id: string,\n    props: LifligCdkPipelineProps,\n  ) {\n    super(scope, id)\n\n    this.artifactsBucket = props.artifactsBucket ?? getGriidArtefactBucket(this)\n\n    const cloudAssemblyArtifact = new codepipeline.Artifact()\n\n    let synth: pipelines.IFileSetProducer\n    let stages: codepipeline.StageProps[]\n\n    switch (props.sourceType) {\n      case \"cloud-assembly\":\n        const cloudAssembly = this.cloudAssemblyStage(\n          cloudAssemblyArtifact,\n          this.artifactsBucket,\n          props.pipelineName,\n        )\n        synth = cloudAssembly.synth\n        stages = cloudAssembly.stages\n        break\n      case \"cdk-source\":\n        const cdkSource = this.cdkSourceStage(\n          cloudAssemblyArtifact,\n          this.artifactsBucket,\n          props.pipelineName,\n          props.parametersNamespace ?? \"default\",\n        )\n        synth = cdkSource.synth\n        stages = cdkSource.stages\n        break\n    }\n\n    const dummyArtifact = new codepipeline.Artifact()\n\n    this.triggerObjectKey = LifligCdkPipeline.pipelineS3TriggerKey(\n      props.pipelineName,\n    )\n\n    this.codePipeline = new codepipeline.Pipeline(this, \"CodePipeline\", {\n      pipelineName: props.pipelineName,\n      stages: [\n        {\n          stageName: \"Source\",\n          actions: [\n            new codepipelineActions.S3SourceAction({\n              actionName: \"source\",\n              bucket: this.artifactsBucket,\n              trigger: codepipelineActions.S3Trigger.NONE,\n              bucketKey: this.triggerObjectKey,\n              output: dummyArtifact,\n            }),\n          ],\n        },\n        ...stages,\n      ],\n      restartExecutionOnUpdate: true,\n    })\n\n    new events.Rule(this, \"PipelineTrigger\", {\n      eventPattern: {\n        source: [\"aws.s3\"],\n        detailType: [\"Object Created\"],\n        detail: {\n          bucket: {\n            name: [this.artifactsBucket.bucketName],\n          },\n          object: {\n            key: [this.triggerObjectKey],\n          },\n        },\n      },\n      targets: [new targets.CodePipeline(this.codePipeline)],\n    })\n\n    this.cdkPipeline = new pipelines.CodePipeline(this, \"CdkPipeline\", {\n      synth,\n      useChangeSets: false,\n      codePipeline: this.codePipeline,\n    })\n  }\n\n  private static getAwsCdkPackageJsonFile(): string | undefined {\n    // Also look up the tree a bit to handle yarn workspaces.\n    const candidates = [\n      path.join(process.cwd(), \"node_modules/aws-cdk/package.json\"),\n      path.join(process.cwd(), \"../node_modules/aws-cdk/package.json\"),\n      path.join(process.cwd(), \"../../node_modules/aws-cdk/package.json\"),\n      path.join(process.cwd(), \"../../../node_modules/aws-cdk/package.json\"),\n    ]\n\n    for (const candidate of candidates) {\n      if (fs.existsSync(candidate)) {\n        return candidate\n      }\n    }\n\n    return undefined\n  }\n\n  private cloudAssemblyStage(\n    cloudAssemblyArtifact: codepipeline.Artifact,\n    cdkBucket: s3.IBucket,\n    pipelineName: string,\n  ): { stages: codepipeline.StageProps[]; synth: pipelines.IFileSetProducer } {\n    const cloudAssemblyLookupFn = new lambda.Function(\n      this,\n      \"CloudAssemblyLookupFn\",\n      {\n        code: new lambda.InlineCode(\n          `exports.handler = ${cloudAssemblyLookupHandler.toString()};`,\n        ),\n        handler: \"index.handler\",\n        runtime: lambda.Runtime.NODEJS_16_X,\n        timeout: cdk.Duration.minutes(1),\n        memorySize: 512,\n      },\n    )\n\n    cdkBucket.grantReadWrite(cloudAssemblyLookupFn)\n\n    const userParameters: CloudAssemblyLookupUserParameters = {\n      bucketName: cdkBucket.bucketName,\n      objectKey: `pipelines/${pipelineName}/cloud-assembly.json`,\n    }\n\n    const synth = pipelines.CodePipelineFileSet.fromArtifact(\n      cloudAssemblyArtifact,\n    )\n\n    const stages = [\n      {\n        stageName: \"PrepareCloudAssembly\",\n        actions: [\n          new codepipelineActions.LambdaInvokeAction({\n            actionName: \"cloud-assembly-lookup\",\n            lambda: cloudAssemblyLookupFn,\n            outputs: [cloudAssemblyArtifact],\n            userParameters,\n          }),\n        ],\n      },\n    ]\n    return { stages, synth }\n  }\n\n  private cdkSourceStage(\n    cloudAssemblyArtifact: codepipeline.Artifact,\n    cdkBucket: s3.IBucket,\n    pipelineName: string,\n    parametersNamespace: string,\n  ): { stages: codepipeline.StageProps[]; synth: pipelines.IFileSetProducer } {\n    const prepareCdkSourceFn = new lambda.Function(this, \"PrepareCdkSourceFn\", {\n      code: lambda.Code.fromAsset(\n        path.join(__dirname, \"../../assets/prepare-cdk-source-lambda\"),\n      ),\n      handler: \"index.handler\",\n      // Using python instead if NodeJS due to zip-support in stdlib.\n      runtime: lambda.Runtime.PYTHON_3_8,\n      timeout: cdk.Duration.minutes(1),\n      memorySize: 512,\n    })\n\n    const account = cdk.Stack.of(this).account\n    const region = cdk.Stack.of(this).region\n\n    prepareCdkSourceFn.grantPrincipal.addToPrincipalPolicy(\n      new iam.PolicyStatement({\n        actions: [\"ssm:GetParametersByPath\"],\n        resources: [\n          `arn:aws:ssm:${region}:${account}:parameter/liflig-cdk/*/pipeline-variables/*`,\n        ],\n      }),\n    )\n\n    cdkBucket.grantReadWrite(prepareCdkSourceFn)\n\n    const cdkSourceArtifact = new codepipeline.Artifact()\n\n    const userParameters: PrepareCdkSourceUserParameters = {\n      bucketName: cdkBucket.bucketName,\n      prefix: `pipelines/${pipelineName}/`,\n      parametersNamespace: parametersNamespace,\n    }\n\n    const synth = new pipelines.ShellStep(\"GenerateCloudAssembly\", {\n      input: pipelines.CodePipelineFileSet.fromArtifact(cdkSourceArtifact),\n      installCommands: [\"npm ci\"],\n      commands: [\"npx cdk synth\"],\n    })\n    const stages = [\n      {\n        stageName: \"PrepareCdkSource\",\n        actions: [\n          new codepipelineActions.LambdaInvokeAction({\n            actionName: \"prepare-cdk-source\",\n            lambda: prepareCdkSourceFn,\n            outputs: [cdkSourceArtifact],\n            userParameters,\n          }),\n        ],\n      },\n    ]\n    return { stages, synth }\n  }\n\n  addSlackNotification(\n    props: Omit<\n      SlackNotificationProps,\n      \"pipeline\" | \"artifactsBucket\" | \"triggerObjectKey\"\n    >,\n  ): void {\n    new SlackNotification(this, \"Slack\", {\n      pipeline: this.codePipeline,\n      artifactsBucket: this.artifactsBucket,\n      triggerObjectKey: this.triggerObjectKey,\n      ...props,\n    })\n  }\n}\n\ninterface PrepareCdkSourceUserParameters {\n  bucketName: string\n  prefix: string\n  parametersNamespace: string\n}\n"]}
@@ -1,10 +1,15 @@
1
1
  import * as constructs from "constructs";
2
2
  import * as codepipeline from "aws-cdk-lib/aws-codepipeline";
3
+ import * as s3 from "aws-cdk-lib/aws-s3";
3
4
  export interface SlackNotificationProps {
4
5
  /**
5
6
  * CodePipeline to monitor.
6
7
  */
7
8
  pipeline: codepipeline.IPipeline;
9
+ /**
10
+ * Artifacts bucket used by pipeline
11
+ */
12
+ artifactsBucket: s3.IBucket;
8
13
  /**
9
14
  * Slack webhook URL.
10
15
  */
@@ -14,17 +19,27 @@ export interface SlackNotificationProps {
14
19
  */
15
20
  slackChannel: string;
16
21
  /**
17
- * @default no description
22
+ * An optional friendly name that will be used in the Slack notifications instead of the AWS account ID
18
23
  */
19
- accountGroupName?: string;
24
+ accountFriendlyName?: string;
20
25
  /**
21
- * @default no description
26
+ * Control the amount and types of notifications being sent to Slack.
27
+ * "WARN" is the least verbose, while "DEBUG" is the most verbose.
28
+ *
29
+ * "WARN" - Includes notifications related to the failure of a pipeline execution.
30
+ * "INFO" - Adds notifications for the success of a pipeline execution.
31
+ * "DEBUG" - Adds notifications for the start and superseding of a pipeline execution.
32
+ *
33
+ * @default "WARN"
22
34
  */
23
- accountDescription?: string;
35
+ notificationLevel?: "WARN" | "INFO" | "DEBUG";
24
36
  /**
25
- * @default false
37
+ * The key of the object (e.g., `my-prefix/my-file.json`) that triggers the S3 Source Action associated with the pipeline.
38
+ * By configuring this parameter you can specify which objects the Lambda function that sends messages to Slack can access in the artifacts bucket.
39
+ *
40
+ * @default - the Lambda function can read all objects in the artifacts bucket.
26
41
  */
27
- alwaysShowSucceeded?: boolean;
42
+ triggerObjectKey?: string;
28
43
  /**
29
44
  * Used to control that only one lambda is created in the account.
30
45
  *
@@ -18,19 +18,16 @@ class SlackNotification extends constructs.Construct {
18
18
  const environment = {
19
19
  SLACK_URL: props.slackWebhookUrl,
20
20
  SLACK_CHANNEL: props.slackChannel,
21
- ALWAYS_SHOW_SUCCEEDED: String((_a = props.alwaysShowSucceeded) !== null && _a !== void 0 ? _a : false),
21
+ NOTIFICATION_LEVEL: (_a = props.notificationLevel) !== null && _a !== void 0 ? _a : "WARN",
22
22
  };
23
- if (props.accountGroupName != null) {
24
- environment.ACCOUNT_GROUP_NAME = props.accountGroupName;
25
- }
26
- if (props.accountDescription != null) {
27
- environment.ACCOUNT_DESC = props.accountDescription;
23
+ if (props.accountFriendlyName != null) {
24
+ environment.ACCOUNT_FRIENDLY_NAME = props.accountFriendlyName;
28
25
  }
29
26
  const reportFunction = new lambda.SingletonFunction(this, "Function", {
30
27
  uuid: (_b = props.singletonLambdaUuid) !== null && _b !== void 0 ? _b : "55954fc8-182e-497e-bd60-7af1496dc222",
31
28
  code: lambda.Code.fromAsset(path.join(__dirname, "../../assets/pipeline-slack-notification-lambda")),
32
29
  handler: "index.handler",
33
- runtime: lambda.Runtime.PYTHON_3_8,
30
+ runtime: lambda.Runtime.PYTHON_3_11,
34
31
  timeout: cdk.Duration.seconds(10),
35
32
  environment,
36
33
  description: "Handle CodePipeline pipeline state change and report to Slack",
@@ -42,11 +39,12 @@ class SlackNotification extends constructs.Construct {
42
39
  ],
43
40
  resources: [props.pipeline.pipelineArn],
44
41
  }));
42
+ props.artifactsBucket.grantRead(reportFunction, props.triggerObjectKey);
45
43
  props.pipeline.onStateChange("Event" + ((_c = props.singletonLambdaUuid) !== null && _c !== void 0 ? _c : ""), {
46
44
  eventPattern: {
47
45
  detail: {
48
46
  // Available states: https://docs.aws.amazon.com/codepipeline/latest/userguide/detect-state-changes-cloudwatch-events.html
49
- state: ["SUCCEEDED", "FAILED"],
47
+ state: ["SUCCEEDED", "FAILED", "STARTED", "SUPERSEDED"],
50
48
  },
51
49
  },
52
50
  target: new eventsTargets.LambdaFunction(reportFunction),
@@ -54,4 +52,4 @@ class SlackNotification extends constructs.Construct {
54
52
  }
55
53
  }
56
54
  exports.SlackNotification = SlackNotification;
57
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2xhY2stbm90aWZpY2F0aW9uLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2Nkay1waXBlbGluZXMvc2xhY2stbm90aWZpY2F0aW9uLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUFBLHlDQUF3QztBQUV4QyxnRUFBK0Q7QUFDL0QsMkNBQTBDO0FBQzFDLGlEQUFnRDtBQUNoRCxtQ0FBa0M7QUFDbEMsNkJBQTRCO0FBd0M1Qjs7O0dBR0c7QUFDSCxNQUFhLGlCQUFrQixTQUFRLFVBQVUsQ0FBQyxTQUFTO0lBQ3pELFlBQ0UsS0FBMkIsRUFDM0IsRUFBVSxFQUNWLEtBQTZCOztRQUU3QixLQUFLLENBQUMsS0FBSyxFQUFFLEVBQUUsQ0FBQyxDQUFBO1FBRWhCLE1BQU0sV0FBVyxHQUEyQjtZQUMxQyxTQUFTLEVBQUUsS0FBSyxDQUFDLGVBQWU7WUFDaEMsYUFBYSxFQUFFLEtBQUssQ0FBQyxZQUFZO1lBQ2pDLHFCQUFxQixFQUFFLE1BQU0sQ0FBQyxNQUFBLEtBQUssQ0FBQyxtQkFBbUIsbUNBQUksS0FBSyxDQUFDO1NBQ2xFLENBQUE7UUFFRCxJQUFJLEtBQUssQ0FBQyxnQkFBZ0IsSUFBSSxJQUFJLEVBQUU7WUFDbEMsV0FBVyxDQUFDLGtCQUFrQixHQUFHLEtBQUssQ0FBQyxnQkFBZ0IsQ0FBQTtTQUN4RDtRQUVELElBQUksS0FBSyxDQUFDLGtCQUFrQixJQUFJLElBQUksRUFBRTtZQUNwQyxXQUFXLENBQUMsWUFBWSxHQUFHLEtBQUssQ0FBQyxrQkFBa0IsQ0FBQTtTQUNwRDtRQUVELE1BQU0sY0FBYyxHQUFHLElBQUksTUFBTSxDQUFDLGlCQUFpQixDQUFDLElBQUksRUFBRSxVQUFVLEVBQUU7WUFDcEUsSUFBSSxFQUFFLE1BQUEsS0FBSyxDQUFDLG1CQUFtQixtQ0FBSSxzQ0FBc0M7WUFDekUsSUFBSSxFQUFFLE1BQU0sQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUN6QixJQUFJLENBQUMsSUFBSSxDQUFDLFNBQVMsRUFBRSxpREFBaUQsQ0FBQyxDQUN4RTtZQUNELE9BQU8sRUFBRSxlQUFlO1lBQ3hCLE9BQU8sRUFBRSxNQUFNLENBQUMsT0FBTyxDQUFDLFVBQVU7WUFDbEMsT0FBTyxFQUFFLEdBQUcsQ0FBQyxRQUFRLENBQUMsT0FBTyxDQUFDLEVBQUUsQ0FBQztZQUNqQyxXQUFXO1lBQ1gsV0FBVyxFQUNULCtEQUErRDtTQUNsRSxDQUFDLENBQUE7UUFFRixjQUFjLENBQUMsY0FBYyxDQUFDLG9CQUFvQixDQUNoRCxJQUFJLEdBQUcsQ0FBQyxlQUFlLENBQUM7WUFDdEIsT0FBTyxFQUFFO2dCQUNQLG1DQUFtQztnQkFDbkMscUNBQXFDO2FBQ3RDO1lBQ0QsU0FBUyxFQUFFLENBQUMsS0FBSyxDQUFDLFFBQVEsQ0FBQyxXQUFXLENBQUM7U0FDeEMsQ0FBQyxDQUNILENBQUE7UUFFRCxLQUFLLENBQUMsUUFBUSxDQUFDLGFBQWEsQ0FBQyxPQUFPLEdBQUcsQ0FBQyxNQUFBLEtBQUssQ0FBQyxtQkFBbUIsbUNBQUksRUFBRSxDQUFDLEVBQUU7WUFDeEUsWUFBWSxFQUFFO2dCQUNaLE1BQU0sRUFBRTtvQkFDTiwwSEFBMEg7b0JBQzFILEtBQUssRUFBRSxDQUFDLFdBQVcsRUFBRSxRQUFRLENBQUM7aUJBQy9CO2FBQ0Y7WUFDRCxNQUFNLEVBQUUsSUFBSSxhQUFhLENBQUMsY0FBYyxDQUFDLGNBQWMsQ0FBQztTQUN6RCxDQUFDLENBQUE7SUFDSixDQUFDO0NBQ0Y7QUF2REQsOENBdURDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0ICogYXMgY29uc3RydWN0cyBmcm9tIFwiY29uc3RydWN0c1wiXG5pbXBvcnQgKiBhcyBjb2RlcGlwZWxpbmUgZnJvbSBcImF3cy1jZGstbGliL2F3cy1jb2RlcGlwZWxpbmVcIlxuaW1wb3J0ICogYXMgZXZlbnRzVGFyZ2V0cyBmcm9tIFwiYXdzLWNkay1saWIvYXdzLWV2ZW50cy10YXJnZXRzXCJcbmltcG9ydCAqIGFzIGlhbSBmcm9tIFwiYXdzLWNkay1saWIvYXdzLWlhbVwiXG5pbXBvcnQgKiBhcyBsYW1iZGEgZnJvbSBcImF3cy1jZGstbGliL2F3cy1sYW1iZGFcIlxuaW1wb3J0ICogYXMgY2RrIGZyb20gXCJhd3MtY2RrLWxpYlwiXG5pbXBvcnQgKiBhcyBwYXRoIGZyb20gXCJwYXRoXCJcblxuZXhwb3J0IGludGVyZmFjZSBTbGFja05vdGlmaWNhdGlvblByb3BzIHtcbiAgLyoqXG4gICAqIENvZGVQaXBlbGluZSB0byBtb25pdG9yLlxuICAgKi9cbiAgcGlwZWxpbmU6IGNvZGVwaXBlbGluZS5JUGlwZWxpbmVcbiAgLyoqXG4gICAqIFNsYWNrIHdlYmhvb2sgVVJMLlxuICAgKi9cbiAgc2xhY2tXZWJob29rVXJsOiBzdHJpbmdcbiAgLyoqXG4gICAqIENoYW5uZWwgbmFtZSBpbmNsdWRpbmcgbGVhZGluZyAjLlxuICAgKi9cbiAgc2xhY2tDaGFubmVsOiBzdHJpbmdcbiAgLyoqXG4gICAqIEBkZWZhdWx0IG5vIGRlc2NyaXB0aW9uXG4gICAqL1xuICBhY2NvdW50R3JvdXBOYW1lPzogc3RyaW5nXG4gIC8qKlxuICAgKiBAZGVmYXVsdCBubyBkZXNjcmlwdGlvblxuICAgKi9cbiAgYWNjb3VudERlc2NyaXB0aW9uPzogc3RyaW5nXG4gIC8qKlxuICAgKiBAZGVmYXVsdCBmYWxzZVxuICAgKi9cbiAgYWx3YXlzU2hvd1N1Y2NlZWRlZD86IGJvb2xlYW5cblxuICAvKipcbiAgICogVXNlZCB0byBjb250cm9sIHRoYXQgb25seSBvbmUgbGFtYmRhIGlzIGNyZWF0ZWQgaW4gdGhlIGFjY291bnQuXG4gICAqXG4gICAqIFNwZWNpZnkgYSB2YWx1ZSBoZXJlIHdoZW4geW91IG1hbnVhbGx5IGNyZWF0ZSBhIHtAbGluayBTbGFja05vdGlmaWNhdGlvbn0gZm9yIGEgcGlwZWxpbmVcbiAgICogd2l0aG91dCB1c2luZyB7QGxpbmsgTGlmbGlnQ2RrUGlwZWxpbmUuYWRkU2xhY2tOb3RpZmljYXRpb259LlxuICAgKiBGb3IgZXhhbXBsZTogYFwiNjVmN2E5ZTAtZDBhNC00YmE3LWFkMWYtNmRlYzg1M2JiZGI4XCJgLlxuICAgKlxuICAgKiBAZGVmYXVsdCBcIjU1OTU0ZmM4LTE4MmUtNDk3ZS1iZDYwLTdhZjE0OTZkYzIyMlwiXG4gICAqL1xuICBzaW5nbGV0b25MYW1iZGFVdWlkPzogc3RyaW5nXG59XG5cbi8qKlxuICogTW9uaXRvciBhIENvZGVQaXBlbGluZSBhbmQgc2VuZCBtZXNzYWdlIHRvIFNsYWNrIG9uIGZhaWx1cmVcbiAqIGFuZCBzb21lIHN1Y2NlZWRlZCBldmVudHMuXG4gKi9cbmV4cG9ydCBjbGFzcyBTbGFja05vdGlmaWNhdGlvbiBleHRlbmRzIGNvbnN0cnVjdHMuQ29uc3RydWN0IHtcbiAgY29uc3RydWN0b3IoXG4gICAgc2NvcGU6IGNvbnN0cnVjdHMuQ29uc3RydWN0LFxuICAgIGlkOiBzdHJpbmcsXG4gICAgcHJvcHM6IFNsYWNrTm90aWZpY2F0aW9uUHJvcHMsXG4gICkge1xuICAgIHN1cGVyKHNjb3BlLCBpZClcblxuICAgIGNvbnN0IGVudmlyb25tZW50OiBSZWNvcmQ8c3RyaW5nLCBzdHJpbmc+ID0ge1xuICAgICAgU0xBQ0tfVVJMOiBwcm9wcy5zbGFja1dlYmhvb2tVcmwsXG4gICAgICBTTEFDS19DSEFOTkVMOiBwcm9wcy5zbGFja0NoYW5uZWwsXG4gICAgICBBTFdBWVNfU0hPV19TVUNDRUVERUQ6IFN0cmluZyhwcm9wcy5hbHdheXNTaG93U3VjY2VlZGVkID8/IGZhbHNlKSxcbiAgICB9XG5cbiAgICBpZiAocHJvcHMuYWNjb3VudEdyb3VwTmFtZSAhPSBudWxsKSB7XG4gICAgICBlbnZpcm9ubWVudC5BQ0NPVU5UX0dST1VQX05BTUUgPSBwcm9wcy5hY2NvdW50R3JvdXBOYW1lXG4gICAgfVxuXG4gICAgaWYgKHByb3BzLmFjY291bnREZXNjcmlwdGlvbiAhPSBudWxsKSB7XG4gICAgICBlbnZpcm9ubWVudC5BQ0NPVU5UX0RFU0MgPSBwcm9wcy5hY2NvdW50RGVzY3JpcHRpb25cbiAgICB9XG5cbiAgICBjb25zdCByZXBvcnRGdW5jdGlvbiA9IG5ldyBsYW1iZGEuU2luZ2xldG9uRnVuY3Rpb24odGhpcywgXCJGdW5jdGlvblwiLCB7XG4gICAgICB1dWlkOiBwcm9wcy5zaW5nbGV0b25MYW1iZGFVdWlkID8/IFwiNTU5NTRmYzgtMTgyZS00OTdlLWJkNjAtN2FmMTQ5NmRjMjIyXCIsXG4gICAgICBjb2RlOiBsYW1iZGEuQ29kZS5mcm9tQXNzZXQoXG4gICAgICAgIHBhdGguam9pbihfX2Rpcm5hbWUsIFwiLi4vLi4vYXNzZXRzL3BpcGVsaW5lLXNsYWNrLW5vdGlmaWNhdGlvbi1sYW1iZGFcIiksXG4gICAgICApLFxuICAgICAgaGFuZGxlcjogXCJpbmRleC5oYW5kbGVyXCIsXG4gICAgICBydW50aW1lOiBsYW1iZGEuUnVudGltZS5QWVRIT05fM184LFxuICAgICAgdGltZW91dDogY2RrLkR1cmF0aW9uLnNlY29uZHMoMTApLFxuICAgICAgZW52aXJvbm1lbnQsXG4gICAgICBkZXNjcmlwdGlvbjpcbiAgICAgICAgXCJIYW5kbGUgQ29kZVBpcGVsaW5lIHBpcGVsaW5lIHN0YXRlIGNoYW5nZSBhbmQgcmVwb3J0IHRvIFNsYWNrXCIsXG4gICAgfSlcblxuICAgIHJlcG9ydEZ1bmN0aW9uLmdyYW50UHJpbmNpcGFsLmFkZFRvUHJpbmNpcGFsUG9saWN5KFxuICAgICAgbmV3IGlhbS5Qb2xpY3lTdGF0ZW1lbnQoe1xuICAgICAgICBhY3Rpb25zOiBbXG4gICAgICAgICAgXCJjb2RlcGlwZWxpbmU6TGlzdEFjdGlvbkV4ZWN1dGlvbnNcIixcbiAgICAgICAgICBcImNvZGVwaXBlbGluZTpMaXN0UGlwZWxpbmVFeGVjdXRpb25zXCIsXG4gICAgICAgIF0sXG4gICAgICAgIHJlc291cmNlczogW3Byb3BzLnBpcGVsaW5lLnBpcGVsaW5lQXJuXSxcbiAgICAgIH0pLFxuICAgIClcblxuICAgIHByb3BzLnBpcGVsaW5lLm9uU3RhdGVDaGFuZ2UoXCJFdmVudFwiICsgKHByb3BzLnNpbmdsZXRvbkxhbWJkYVV1aWQgPz8gXCJcIiksIHtcbiAgICAgIGV2ZW50UGF0dGVybjoge1xuICAgICAgICBkZXRhaWw6IHtcbiAgICAgICAgICAvLyBBdmFpbGFibGUgc3RhdGVzOiBodHRwczovL2RvY3MuYXdzLmFtYXpvbi5jb20vY29kZXBpcGVsaW5lL2xhdGVzdC91c2VyZ3VpZGUvZGV0ZWN0LXN0YXRlLWNoYW5nZXMtY2xvdWR3YXRjaC1ldmVudHMuaHRtbFxuICAgICAgICAgIHN0YXRlOiBbXCJTVUNDRUVERURcIiwgXCJGQUlMRURcIl0sXG4gICAgICAgIH0sXG4gICAgICB9LFxuICAgICAgdGFyZ2V0OiBuZXcgZXZlbnRzVGFyZ2V0cy5MYW1iZGFGdW5jdGlvbihyZXBvcnRGdW5jdGlvbiksXG4gICAgfSlcbiAgfVxufVxuIl19
55
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2xhY2stbm90aWZpY2F0aW9uLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2Nkay1waXBlbGluZXMvc2xhY2stbm90aWZpY2F0aW9uLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUFBLHlDQUF3QztBQUV4QyxnRUFBK0Q7QUFDL0QsMkNBQTBDO0FBQzFDLGlEQUFnRDtBQUNoRCxtQ0FBa0M7QUFDbEMsNkJBQTRCO0FBc0Q1Qjs7O0dBR0c7QUFDSCxNQUFhLGlCQUFrQixTQUFRLFVBQVUsQ0FBQyxTQUFTO0lBQ3pELFlBQ0UsS0FBMkIsRUFDM0IsRUFBVSxFQUNWLEtBQTZCOztRQUU3QixLQUFLLENBQUMsS0FBSyxFQUFFLEVBQUUsQ0FBQyxDQUFBO1FBRWhCLE1BQU0sV0FBVyxHQUEyQjtZQUMxQyxTQUFTLEVBQUUsS0FBSyxDQUFDLGVBQWU7WUFDaEMsYUFBYSxFQUFFLEtBQUssQ0FBQyxZQUFZO1lBQ2pDLGtCQUFrQixFQUFFLE1BQUEsS0FBSyxDQUFDLGlCQUFpQixtQ0FBSSxNQUFNO1NBQ3RELENBQUE7UUFFRCxJQUFJLEtBQUssQ0FBQyxtQkFBbUIsSUFBSSxJQUFJLEVBQUU7WUFDckMsV0FBVyxDQUFDLHFCQUFxQixHQUFHLEtBQUssQ0FBQyxtQkFBbUIsQ0FBQTtTQUM5RDtRQUVELE1BQU0sY0FBYyxHQUFHLElBQUksTUFBTSxDQUFDLGlCQUFpQixDQUFDLElBQUksRUFBRSxVQUFVLEVBQUU7WUFDcEUsSUFBSSxFQUFFLE1BQUEsS0FBSyxDQUFDLG1CQUFtQixtQ0FBSSxzQ0FBc0M7WUFDekUsSUFBSSxFQUFFLE1BQU0sQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUN6QixJQUFJLENBQUMsSUFBSSxDQUFDLFNBQVMsRUFBRSxpREFBaUQsQ0FBQyxDQUN4RTtZQUNELE9BQU8sRUFBRSxlQUFlO1lBQ3hCLE9BQU8sRUFBRSxNQUFNLENBQUMsT0FBTyxDQUFDLFdBQVc7WUFDbkMsT0FBTyxFQUFFLEdBQUcsQ0FBQyxRQUFRLENBQUMsT0FBTyxDQUFDLEVBQUUsQ0FBQztZQUNqQyxXQUFXO1lBQ1gsV0FBVyxFQUNULCtEQUErRDtTQUNsRSxDQUFDLENBQUE7UUFFRixjQUFjLENBQUMsY0FBYyxDQUFDLG9CQUFvQixDQUNoRCxJQUFJLEdBQUcsQ0FBQyxlQUFlLENBQUM7WUFDdEIsT0FBTyxFQUFFO2dCQUNQLG1DQUFtQztnQkFDbkMscUNBQXFDO2FBQ3RDO1lBQ0QsU0FBUyxFQUFFLENBQUMsS0FBSyxDQUFDLFFBQVEsQ0FBQyxXQUFXLENBQUM7U0FDeEMsQ0FBQyxDQUNILENBQUE7UUFFRCxLQUFLLENBQUMsZUFBZSxDQUFDLFNBQVMsQ0FBQyxjQUFjLEVBQUUsS0FBSyxDQUFDLGdCQUFnQixDQUFDLENBQUE7UUFFdkUsS0FBSyxDQUFDLFFBQVEsQ0FBQyxhQUFhLENBQUMsT0FBTyxHQUFHLENBQUMsTUFBQSxLQUFLLENBQUMsbUJBQW1CLG1DQUFJLEVBQUUsQ0FBQyxFQUFFO1lBQ3hFLFlBQVksRUFBRTtnQkFDWixNQUFNLEVBQUU7b0JBQ04sMEhBQTBIO29CQUMxSCxLQUFLLEVBQUUsQ0FBQyxXQUFXLEVBQUUsUUFBUSxFQUFFLFNBQVMsRUFBRSxZQUFZLENBQUM7aUJBQ3hEO2FBQ0Y7WUFDRCxNQUFNLEVBQUUsSUFBSSxhQUFhLENBQUMsY0FBYyxDQUFDLGNBQWMsQ0FBQztTQUN6RCxDQUFDLENBQUE7SUFDSixDQUFDO0NBQ0Y7QUFyREQsOENBcURDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0ICogYXMgY29uc3RydWN0cyBmcm9tIFwiY29uc3RydWN0c1wiXG5pbXBvcnQgKiBhcyBjb2RlcGlwZWxpbmUgZnJvbSBcImF3cy1jZGstbGliL2F3cy1jb2RlcGlwZWxpbmVcIlxuaW1wb3J0ICogYXMgZXZlbnRzVGFyZ2V0cyBmcm9tIFwiYXdzLWNkay1saWIvYXdzLWV2ZW50cy10YXJnZXRzXCJcbmltcG9ydCAqIGFzIGlhbSBmcm9tIFwiYXdzLWNkay1saWIvYXdzLWlhbVwiXG5pbXBvcnQgKiBhcyBsYW1iZGEgZnJvbSBcImF3cy1jZGstbGliL2F3cy1sYW1iZGFcIlxuaW1wb3J0ICogYXMgY2RrIGZyb20gXCJhd3MtY2RrLWxpYlwiXG5pbXBvcnQgKiBhcyBwYXRoIGZyb20gXCJwYXRoXCJcbmltcG9ydCAqIGFzIHMzIGZyb20gXCJhd3MtY2RrLWxpYi9hd3MtczNcIlxuXG5leHBvcnQgaW50ZXJmYWNlIFNsYWNrTm90aWZpY2F0aW9uUHJvcHMge1xuICAvKipcbiAgICogQ29kZVBpcGVsaW5lIHRvIG1vbml0b3IuXG4gICAqL1xuICBwaXBlbGluZTogY29kZXBpcGVsaW5lLklQaXBlbGluZVxuICAvKipcbiAgICogQXJ0aWZhY3RzIGJ1Y2tldCB1c2VkIGJ5IHBpcGVsaW5lXG4gICAqL1xuICBhcnRpZmFjdHNCdWNrZXQ6IHMzLklCdWNrZXRcbiAgLyoqXG4gICAqIFNsYWNrIHdlYmhvb2sgVVJMLlxuICAgKi9cbiAgc2xhY2tXZWJob29rVXJsOiBzdHJpbmdcbiAgLyoqXG4gICAqIENoYW5uZWwgbmFtZSBpbmNsdWRpbmcgbGVhZGluZyAjLlxuICAgKi9cbiAgc2xhY2tDaGFubmVsOiBzdHJpbmdcbiAgLyoqXG4gICAqIEFuIG9wdGlvbmFsIGZyaWVuZGx5IG5hbWUgdGhhdCB3aWxsIGJlIHVzZWQgaW4gdGhlIFNsYWNrIG5vdGlmaWNhdGlvbnMgaW5zdGVhZCBvZiB0aGUgQVdTIGFjY291bnQgSURcbiAgICovXG4gIGFjY291bnRGcmllbmRseU5hbWU/OiBzdHJpbmdcbiAgLyoqXG4gICAqIENvbnRyb2wgdGhlIGFtb3VudCBhbmQgdHlwZXMgb2Ygbm90aWZpY2F0aW9ucyBiZWluZyBzZW50IHRvIFNsYWNrLlxuICAgKiBcIldBUk5cIiBpcyB0aGUgbGVhc3QgdmVyYm9zZSwgd2hpbGUgXCJERUJVR1wiIGlzIHRoZSBtb3N0IHZlcmJvc2UuXG4gICAqXG4gICAqIFwiV0FSTlwiIC0gSW5jbHVkZXMgbm90aWZpY2F0aW9ucyByZWxhdGVkIHRvIHRoZSBmYWlsdXJlIG9mIGEgcGlwZWxpbmUgZXhlY3V0aW9uLlxuICAgKiBcIklORk9cIiAtIEFkZHMgbm90aWZpY2F0aW9ucyBmb3IgdGhlIHN1Y2Nlc3Mgb2YgYSBwaXBlbGluZSBleGVjdXRpb24uXG4gICAqIFwiREVCVUdcIiAtIEFkZHMgbm90aWZpY2F0aW9ucyBmb3IgdGhlIHN0YXJ0IGFuZCBzdXBlcnNlZGluZyBvZiBhIHBpcGVsaW5lIGV4ZWN1dGlvbi5cbiAgICpcbiAgICogQGRlZmF1bHQgXCJXQVJOXCJcbiAgICovXG4gIG5vdGlmaWNhdGlvbkxldmVsPzogXCJXQVJOXCIgfCBcIklORk9cIiB8IFwiREVCVUdcIlxuICAvKipcbiAgICogVGhlIGtleSBvZiB0aGUgb2JqZWN0IChlLmcuLCBgbXktcHJlZml4L215LWZpbGUuanNvbmApIHRoYXQgdHJpZ2dlcnMgdGhlIFMzIFNvdXJjZSBBY3Rpb24gYXNzb2NpYXRlZCB3aXRoIHRoZSBwaXBlbGluZS5cbiAgICogQnkgY29uZmlndXJpbmcgdGhpcyBwYXJhbWV0ZXIgeW91IGNhbiBzcGVjaWZ5IHdoaWNoIG9iamVjdHMgdGhlIExhbWJkYSBmdW5jdGlvbiB0aGF0IHNlbmRzIG1lc3NhZ2VzIHRvIFNsYWNrIGNhbiBhY2Nlc3MgaW4gdGhlIGFydGlmYWN0cyBidWNrZXQuXG4gICAqXG4gICAqIEBkZWZhdWx0IC0gdGhlIExhbWJkYSBmdW5jdGlvbiBjYW4gcmVhZCBhbGwgb2JqZWN0cyBpbiB0aGUgYXJ0aWZhY3RzIGJ1Y2tldC5cbiAgICovXG4gIHRyaWdnZXJPYmplY3RLZXk/OiBzdHJpbmdcbiAgLyoqXG4gICAqIFVzZWQgdG8gY29udHJvbCB0aGF0IG9ubHkgb25lIGxhbWJkYSBpcyBjcmVhdGVkIGluIHRoZSBhY2NvdW50LlxuICAgKlxuICAgKiBTcGVjaWZ5IGEgdmFsdWUgaGVyZSB3aGVuIHlvdSBtYW51YWxseSBjcmVhdGUgYSB7QGxpbmsgU2xhY2tOb3RpZmljYXRpb259IGZvciBhIHBpcGVsaW5lXG4gICAqIHdpdGhvdXQgdXNpbmcge0BsaW5rIExpZmxpZ0Nka1BpcGVsaW5lLmFkZFNsYWNrTm90aWZpY2F0aW9ufS5cbiAgICogRm9yIGV4YW1wbGU6IGBcIjY1ZjdhOWUwLWQwYTQtNGJhNy1hZDFmLTZkZWM4NTNiYmRiOFwiYC5cbiAgICpcbiAgICogQGRlZmF1bHQgXCI1NTk1NGZjOC0xODJlLTQ5N2UtYmQ2MC03YWYxNDk2ZGMyMjJcIlxuICAgKi9cbiAgc2luZ2xldG9uTGFtYmRhVXVpZD86IHN0cmluZ1xufVxuXG4vKipcbiAqIE1vbml0b3IgYSBDb2RlUGlwZWxpbmUgYW5kIHNlbmQgbWVzc2FnZSB0byBTbGFjayBvbiBmYWlsdXJlXG4gKiBhbmQgc29tZSBzdWNjZWVkZWQgZXZlbnRzLlxuICovXG5leHBvcnQgY2xhc3MgU2xhY2tOb3RpZmljYXRpb24gZXh0ZW5kcyBjb25zdHJ1Y3RzLkNvbnN0cnVjdCB7XG4gIGNvbnN0cnVjdG9yKFxuICAgIHNjb3BlOiBjb25zdHJ1Y3RzLkNvbnN0cnVjdCxcbiAgICBpZDogc3RyaW5nLFxuICAgIHByb3BzOiBTbGFja05vdGlmaWNhdGlvblByb3BzLFxuICApIHtcbiAgICBzdXBlcihzY29wZSwgaWQpXG5cbiAgICBjb25zdCBlbnZpcm9ubWVudDogUmVjb3JkPHN0cmluZywgc3RyaW5nPiA9IHtcbiAgICAgIFNMQUNLX1VSTDogcHJvcHMuc2xhY2tXZWJob29rVXJsLFxuICAgICAgU0xBQ0tfQ0hBTk5FTDogcHJvcHMuc2xhY2tDaGFubmVsLFxuICAgICAgTk9USUZJQ0FUSU9OX0xFVkVMOiBwcm9wcy5ub3RpZmljYXRpb25MZXZlbCA/PyBcIldBUk5cIixcbiAgICB9XG5cbiAgICBpZiAocHJvcHMuYWNjb3VudEZyaWVuZGx5TmFtZSAhPSBudWxsKSB7XG4gICAgICBlbnZpcm9ubWVudC5BQ0NPVU5UX0ZSSUVORExZX05BTUUgPSBwcm9wcy5hY2NvdW50RnJpZW5kbHlOYW1lXG4gICAgfVxuXG4gICAgY29uc3QgcmVwb3J0RnVuY3Rpb24gPSBuZXcgbGFtYmRhLlNpbmdsZXRvbkZ1bmN0aW9uKHRoaXMsIFwiRnVuY3Rpb25cIiwge1xuICAgICAgdXVpZDogcHJvcHMuc2luZ2xldG9uTGFtYmRhVXVpZCA/PyBcIjU1OTU0ZmM4LTE4MmUtNDk3ZS1iZDYwLTdhZjE0OTZkYzIyMlwiLFxuICAgICAgY29kZTogbGFtYmRhLkNvZGUuZnJvbUFzc2V0KFxuICAgICAgICBwYXRoLmpvaW4oX19kaXJuYW1lLCBcIi4uLy4uL2Fzc2V0cy9waXBlbGluZS1zbGFjay1ub3RpZmljYXRpb24tbGFtYmRhXCIpLFxuICAgICAgKSxcbiAgICAgIGhhbmRsZXI6IFwiaW5kZXguaGFuZGxlclwiLFxuICAgICAgcnVudGltZTogbGFtYmRhLlJ1bnRpbWUuUFlUSE9OXzNfMTEsXG4gICAgICB0aW1lb3V0OiBjZGsuRHVyYXRpb24uc2Vjb25kcygxMCksXG4gICAgICBlbnZpcm9ubWVudCxcbiAgICAgIGRlc2NyaXB0aW9uOlxuICAgICAgICBcIkhhbmRsZSBDb2RlUGlwZWxpbmUgcGlwZWxpbmUgc3RhdGUgY2hhbmdlIGFuZCByZXBvcnQgdG8gU2xhY2tcIixcbiAgICB9KVxuXG4gICAgcmVwb3J0RnVuY3Rpb24uZ3JhbnRQcmluY2lwYWwuYWRkVG9QcmluY2lwYWxQb2xpY3koXG4gICAgICBuZXcgaWFtLlBvbGljeVN0YXRlbWVudCh7XG4gICAgICAgIGFjdGlvbnM6IFtcbiAgICAgICAgICBcImNvZGVwaXBlbGluZTpMaXN0QWN0aW9uRXhlY3V0aW9uc1wiLFxuICAgICAgICAgIFwiY29kZXBpcGVsaW5lOkxpc3RQaXBlbGluZUV4ZWN1dGlvbnNcIixcbiAgICAgICAgXSxcbiAgICAgICAgcmVzb3VyY2VzOiBbcHJvcHMucGlwZWxpbmUucGlwZWxpbmVBcm5dLFxuICAgICAgfSksXG4gICAgKVxuXG4gICAgcHJvcHMuYXJ0aWZhY3RzQnVja2V0LmdyYW50UmVhZChyZXBvcnRGdW5jdGlvbiwgcHJvcHMudHJpZ2dlck9iamVjdEtleSlcblxuICAgIHByb3BzLnBpcGVsaW5lLm9uU3RhdGVDaGFuZ2UoXCJFdmVudFwiICsgKHByb3BzLnNpbmdsZXRvbkxhbWJkYVV1aWQgPz8gXCJcIiksIHtcbiAgICAgIGV2ZW50UGF0dGVybjoge1xuICAgICAgICBkZXRhaWw6IHtcbiAgICAgICAgICAvLyBBdmFpbGFibGUgc3RhdGVzOiBodHRwczovL2RvY3MuYXdzLmFtYXpvbi5jb20vY29kZXBpcGVsaW5lL2xhdGVzdC91c2VyZ3VpZGUvZGV0ZWN0LXN0YXRlLWNoYW5nZXMtY2xvdWR3YXRjaC1ldmVudHMuaHRtbFxuICAgICAgICAgIHN0YXRlOiBbXCJTVUNDRUVERURcIiwgXCJGQUlMRURcIiwgXCJTVEFSVEVEXCIsIFwiU1VQRVJTRURFRFwiXSxcbiAgICAgICAgfSxcbiAgICAgIH0sXG4gICAgICB0YXJnZXQ6IG5ldyBldmVudHNUYXJnZXRzLkxhbWJkYUZ1bmN0aW9uKHJlcG9ydEZ1bmN0aW9uKSxcbiAgICB9KVxuICB9XG59XG4iXX0=
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@liflig/cdk",
3
- "version": "2.14.4",
3
+ "version": "2.15.0",
4
4
  "description": "CDK library for Liflig",
5
5
  "repository": {
6
6
  "type": "git",
@@ -37,29 +37,29 @@
37
37
  "@aws-cdk/assert": "2.68.0",
38
38
  "@commitlint/cli": "17.7.1",
39
39
  "@commitlint/config-conventional": "17.7.0",
40
- "@types/aws-lambda": "8.10.119",
41
- "@types/jest": "29.5.4",
42
- "@types/node": "18.17.11",
40
+ "@types/aws-lambda": "8.10.121",
41
+ "@types/jest": "29.5.5",
42
+ "@types/node": "18.17.18",
43
43
  "@typescript-eslint/eslint-plugin": "5.62.0",
44
44
  "@typescript-eslint/parser": "5.62.0",
45
- "aws-cdk": "2.86.0",
46
- "aws-cdk-lib": "2.86.0",
47
- "constructs": "10.2.69",
48
- "eslint": "8.48.0",
49
- "eslint-config-prettier": "8.10.0",
50
- "eslint-plugin-prettier": "4.2.1",
45
+ "aws-cdk": "2.94.0",
46
+ "aws-cdk-lib": "2.94.0",
47
+ "constructs": "10.2.70",
48
+ "eslint": "8.49.0",
49
+ "eslint-config-prettier": "9.0.0",
50
+ "eslint-plugin-prettier": "5.0.0",
51
51
  "husky": "8.0.3",
52
- "jest": "29.6.4",
52
+ "jest": "29.7.0",
53
53
  "jest-cdk-snapshot": "2.0.1",
54
- "prettier": "2.8.8",
55
- "semantic-release": "21.1.1",
54
+ "prettier": "3.0.3",
55
+ "semantic-release": "21.1.2",
56
56
  "ts-jest": "29.1.1",
57
57
  "ts-node": "10.9.1",
58
58
  "typescript": "5.2.2"
59
59
  },
60
60
  "dependencies": {
61
61
  "@capraconsulting/webapp-deploy-lambda": "2.1.2",
62
- "aws-sdk": "2.1430.0",
62
+ "aws-sdk": "2.1446.0",
63
63
  "cpy": "8.1.2",
64
64
  "del": "6.1.1",
65
65
  "execa": "5.1.1",