@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.
- package/assets/pipeline-slack-notification-lambda/index.py +156 -46
- package/lib/cdk-pipelines/liflig-cdk-pipeline.d.ts +3 -1
- package/lib/cdk-pipelines/liflig-cdk-pipeline.js +11 -8
- package/lib/cdk-pipelines/slack-notification.d.ts +21 -6
- package/lib/cdk-pipelines/slack-notification.js +7 -9
- package/package.json +14 -14
|
@@ -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
|
-
|
|
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
|
-
|
|
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
|
|
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
|
-
|
|
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
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
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
|
-
|
|
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
|
|
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
|
-
|
|
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
|
-
|
|
230
|
+
ci_metadata = get_metadata_from_trigger(pipeline_name, execution_id)
|
|
231
|
+
|
|
232
|
+
footer_text = get_footer_text(ci_metadata)
|
|
143
233
|
|
|
144
|
-
|
|
145
|
-
|
|
146
|
-
|
|
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
|
-
|
|
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
|
-
"
|
|
151
|
-
"
|
|
152
|
-
|
|
153
|
-
|
|
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
|
-
"
|
|
162
|
-
"username": "
|
|
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
|
-
|
|
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:
|
|
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: [
|
|
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
|
-
*
|
|
22
|
+
* An optional friendly name that will be used in the Slack notifications instead of the AWS account ID
|
|
18
23
|
*/
|
|
19
|
-
|
|
24
|
+
accountFriendlyName?: string;
|
|
20
25
|
/**
|
|
21
|
-
*
|
|
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
|
-
|
|
35
|
+
notificationLevel?: "WARN" | "INFO" | "DEBUG";
|
|
24
36
|
/**
|
|
25
|
-
*
|
|
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
|
-
|
|
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
|
-
|
|
21
|
+
NOTIFICATION_LEVEL: (_a = props.notificationLevel) !== null && _a !== void 0 ? _a : "WARN",
|
|
22
22
|
};
|
|
23
|
-
if (props.
|
|
24
|
-
environment.
|
|
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.
|
|
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,
|
|
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.
|
|
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.
|
|
41
|
-
"@types/jest": "29.5.
|
|
42
|
-
"@types/node": "18.17.
|
|
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.
|
|
46
|
-
"aws-cdk-lib": "2.
|
|
47
|
-
"constructs": "10.2.
|
|
48
|
-
"eslint": "8.
|
|
49
|
-
"eslint-config-prettier": "
|
|
50
|
-
"eslint-plugin-prettier": "
|
|
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.
|
|
52
|
+
"jest": "29.7.0",
|
|
53
53
|
"jest-cdk-snapshot": "2.0.1",
|
|
54
|
-
"prettier": "
|
|
55
|
-
"semantic-release": "21.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.
|
|
62
|
+
"aws-sdk": "2.1446.0",
|
|
63
63
|
"cpy": "8.1.2",
|
|
64
64
|
"del": "6.1.1",
|
|
65
65
|
"execa": "5.1.1",
|