@orcabus/platform-cdk-constructs 0.0.6 → 0.0.7-alpha.4
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/.jsii +858 -6
- package/api-gateway/api-gateway.js +22 -4
- package/deployment-stack-pipeline/pipeline.js +6 -4
- package/dynamodb/config.d.ts +3 -0
- package/dynamodb/config.js +7 -0
- package/dynamodb/index.d.ts +27 -0
- package/dynamodb/index.js +106 -0
- package/ecs/config.d.ts +4 -0
- package/ecs/config.js +8 -0
- package/ecs/index.d.ts +46 -0
- package/ecs/index.js +103 -0
- package/index.d.ts +5 -2
- package/index.js +47 -5
- package/index.ts +16 -2
- package/lambda/build_python/Dockerfile +60 -0
- package/lambda/config.d.ts +11 -0
- package/lambda/config.js +19 -0
- package/lambda/index.d.ts +59 -0
- package/lambda/index.js +242 -0
- package/lambda/layers/mart_tools/poetry.lock +303 -0
- package/lambda/layers/mart_tools/pyproject.toml +27 -0
- package/lambda/layers/mart_tools/src/mart_tools/__init__.py +0 -0
- package/lambda/layers/mart_tools/src/mart_tools/mart/__init__.py +8 -0
- package/lambda/layers/mart_tools/src/mart_tools/mart/aws_helpers.py +79 -0
- package/lambda/layers/mart_tools/src/mart_tools/mart/dataframe_helpers.py +29 -0
- package/lambda/layers/mart_tools/src/mart_tools/mart/globals.py +5 -0
- package/lambda/layers/mart_tools/src/mart_tools/mart/models.py +71 -0
- package/lambda/layers/orcabus_api_tools/poetry.lock +273 -0
- package/lambda/layers/orcabus_api_tools/pyproject.toml +27 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/__init__.py +0 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/fastq/__init__.py +172 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/fastq/create_helpers.py +47 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/fastq/globals.py +13 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/fastq/job_helpers.py +53 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/fastq/models.py +253 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/fastq/query_helpers.py +248 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/fastq/update_helpers.py +221 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/fastq/workflow_helpers.py +25 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/fastq_unarchiving/__init__.py +92 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/fastq_unarchiving/create_helpers.py +27 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/fastq_unarchiving/globals.py +21 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/fastq_unarchiving/models.py +51 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/fastq_unarchiving/query_helpers.py +52 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/fastq_unarchiving/update_helpers.py +45 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/filemanager/__init__.py +98 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/filemanager/errors.py +45 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/filemanager/file_helpers.py +341 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/filemanager/globals.py +70 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/filemanager/models.py +59 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/mart/__init__.py +8 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/mart/aws_helpers.py +79 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/mart/dataframe_helpers.py +29 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/mart/globals.py +5 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/mart/models.py +71 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/metadata/__init__.py +250 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/metadata/contact_helpers.py +109 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/metadata/errors.py +104 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/metadata/globals.py +16 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/metadata/individual_helpers.py +139 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/metadata/library_helpers.py +196 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/metadata/lims_helpers.py +36 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/metadata/models.py +112 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/metadata/project_helpers.py +129 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/metadata/sample_helpers.py +132 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/metadata/subject_helpers.py +151 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/sequence/__init__.py +15 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/sequence/globals.py +2 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/sequence/models.py +44 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/sequence/sequence_helpers.py +62 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/utils/__init__.py +0 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/utils/aws_helpers.py +123 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/utils/miscell.py +17 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/utils/requests_helpers.py +163 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/workflow/__init__.py +0 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/workflow/errors.py +37 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/workflow/metadata_helpers.py +28 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/workflow/models.py +85 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/workflow/payload_helpers.py +64 -0
- package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/workflow/workflow_run_helpers.py +80 -0
- package/package.json +13 -7
- package/typedoc.json +3 -0
- package/utils/index.d.ts +3 -0
- package/utils/index.js +50 -1
package/index.ts
CHANGED
|
@@ -1,3 +1,17 @@
|
|
|
1
|
-
|
|
2
|
-
export * as deploymentPipeline from "./deployment-stack-pipeline";
|
|
1
|
+
// Get stage name
|
|
3
2
|
export * as utils from "./utils";
|
|
3
|
+
|
|
4
|
+
// Deployment pipeline constructs
|
|
5
|
+
export * as deploymentPipeline from "./deployment-stack-pipeline";
|
|
6
|
+
|
|
7
|
+
// Api Gateway constructs
|
|
8
|
+
export * as apigateway from "./api-gateway";
|
|
9
|
+
|
|
10
|
+
// DynamoDB Constructs
|
|
11
|
+
export * as dynamodb from "./dynamodb";
|
|
12
|
+
|
|
13
|
+
// ECS / Fargate Constructs
|
|
14
|
+
export * as ecs from "./ecs";
|
|
15
|
+
|
|
16
|
+
// Lambda Constructs
|
|
17
|
+
export * as lambda from "./lambda";
|
|
@@ -0,0 +1,60 @@
|
|
|
1
|
+
# NOTES -
|
|
2
|
+
# THIS IS FROM https://github.com/aws/aws-cdk/blob/main/packages/%40aws-cdk/aws-lambda-python-alpha/lib/Dockerfile
|
|
3
|
+
# We have made the following changes:
|
|
4
|
+
# 1. Added the installation of 'uv'
|
|
5
|
+
# 2. Addded in the UV_CACHE_DIR environment variable
|
|
6
|
+
# 3. Replaced 'pip' with 'uv pip' in the main code chunk
|
|
7
|
+
# 4. Replaced /usr/app/venv/pip with uv pip shell script wrapper
|
|
8
|
+
# The correct AWS SAM build image based on the runtime of the function will be
|
|
9
|
+
# passed as build arg. The default allows to do `docker build .` when testing.
|
|
10
|
+
ARG IMAGE=public.ecr.aws/sam/build-python3.12:latest
|
|
11
|
+
FROM $IMAGE
|
|
12
|
+
|
|
13
|
+
ARG TARGETPLATFORM
|
|
14
|
+
ARG PIP_INDEX_URL
|
|
15
|
+
ARG PIP_EXTRA_INDEX_URL
|
|
16
|
+
ARG HTTPS_PROXY
|
|
17
|
+
ARG POETRY_VERSION=1.5.1
|
|
18
|
+
|
|
19
|
+
# ADDITION: Install uv
|
|
20
|
+
RUN curl -LsSf https://astral.sh/uv/install.sh | XDG_CONFIG_HOME=/tmp UV_INSTALL_DIR=/usr/bin sh
|
|
21
|
+
|
|
22
|
+
# Add virtualenv path
|
|
23
|
+
ENV PATH="/usr/app/venv/bin:$PATH"
|
|
24
|
+
|
|
25
|
+
# set the pip cache location
|
|
26
|
+
ENV PIP_CACHE_DIR=/tmp/pip-cache
|
|
27
|
+
|
|
28
|
+
# set the poetry cache
|
|
29
|
+
ENV POETRY_CACHE_DIR=/tmp/poetry-cache
|
|
30
|
+
|
|
31
|
+
# ADDITION: set the uv cache
|
|
32
|
+
ENV UV_CACHE_DIR=/tmp/uv-cache
|
|
33
|
+
|
|
34
|
+
# ADDITION: Replace 'pip' with uv pip in the following chunk
|
|
35
|
+
RUN \
|
|
36
|
+
# create a new virtualenv for python to use
|
|
37
|
+
# so that it isn't using root
|
|
38
|
+
python -m venv /usr/app/venv && \
|
|
39
|
+
# Create a new location for the pip cache
|
|
40
|
+
mkdir /tmp/pip-cache && \
|
|
41
|
+
# Ensure all users can write to pip cache
|
|
42
|
+
chmod -R 777 /tmp/pip-cache && \
|
|
43
|
+
# Upgrade pip (required by cryptography v3.4 and above, which is a dependency of poetry)
|
|
44
|
+
uv pip install --upgrade pip && \
|
|
45
|
+
# Create a new location for the poetry cache
|
|
46
|
+
mkdir /tmp/poetry-cache && \
|
|
47
|
+
# Ensure all users can write to poetry cache
|
|
48
|
+
chmod -R 777 /tmp/poetry-cache && \
|
|
49
|
+
# Ensure all users can write to uv cache
|
|
50
|
+
chmod -R 777 /tmp/uv-cache && \
|
|
51
|
+
# Install pipenv and poetry
|
|
52
|
+
uv pip install poetry==$POETRY_VERSION && \
|
|
53
|
+
# Ensure no temporary files remain in the caches
|
|
54
|
+
rm -rf /tmp/pip-cache/* /tmp/poetry-cache/* /tmp/uv-cache/*
|
|
55
|
+
|
|
56
|
+
# ADDITION: Replace /usr/app/venv/pip with uv pip
|
|
57
|
+
RUN printf '#!/usr/bin/env bash \nuv pip $@\n' > /usr/app/venv/bin/pip && \
|
|
58
|
+
printf '#!/usr/bin/env bash \nuv pip $@\n' > /usr/app/venv/bin/pip3
|
|
59
|
+
|
|
60
|
+
CMD [ "python" ]
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
import { StageName } from "../utils";
|
|
2
|
+
export declare const DEFAULT_ORCABUS_TOKEN_SECRET_ID = "orcabus/token-service-jwt";
|
|
3
|
+
export declare const DEFAULT_HOSTNAME_SSM_PARAMETER = "/hosted_zone/umccr/name";
|
|
4
|
+
export declare const MART_ENV_VARS: {
|
|
5
|
+
athenaWorkgroupName: string;
|
|
6
|
+
athenaDatasourceName: string;
|
|
7
|
+
athenaDatabaseName: string;
|
|
8
|
+
};
|
|
9
|
+
export declare const MART_S3_BUCKET: Record<StageName, string>;
|
|
10
|
+
export declare const MART_S3_PREFIX = "athena-query-results/";
|
|
11
|
+
export declare const MART_LAMBDA_FUNCTION_NAME = "orcavault";
|
package/lambda/config.js
ADDED
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.MART_LAMBDA_FUNCTION_NAME = exports.MART_S3_PREFIX = exports.MART_S3_BUCKET = exports.MART_ENV_VARS = exports.DEFAULT_HOSTNAME_SSM_PARAMETER = exports.DEFAULT_ORCABUS_TOKEN_SECRET_ID = void 0;
|
|
4
|
+
const utils_1 = require("../utils");
|
|
5
|
+
exports.DEFAULT_ORCABUS_TOKEN_SECRET_ID = 'orcabus/token-service-jwt';
|
|
6
|
+
exports.DEFAULT_HOSTNAME_SSM_PARAMETER = '/hosted_zone/umccr/name';
|
|
7
|
+
exports.MART_ENV_VARS = {
|
|
8
|
+
athenaWorkgroupName: 'orcahouse',
|
|
9
|
+
athenaDatasourceName: 'orcavault',
|
|
10
|
+
athenaDatabaseName: 'mart'
|
|
11
|
+
};
|
|
12
|
+
exports.MART_S3_BUCKET = {
|
|
13
|
+
BETA: `data-sharing-artifacts-${utils_1.accountIdAlias.BETA}-${utils_1.region}`,
|
|
14
|
+
GAMMA: `data-sharing-artifacts-${utils_1.accountIdAlias.GAMMA}-${utils_1.region}`,
|
|
15
|
+
PROD: `data-sharing-artifacts-${utils_1.accountIdAlias.PROD}-${utils_1.region}`
|
|
16
|
+
};
|
|
17
|
+
exports.MART_S3_PREFIX = 'athena-query-results/';
|
|
18
|
+
exports.MART_LAMBDA_FUNCTION_NAME = 'orcavault';
|
|
19
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uZmlnLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiY29uZmlnLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUFBLG9DQUEyRDtBQUU5QyxRQUFBLCtCQUErQixHQUFJLDJCQUEyQixDQUFBO0FBQzlELFFBQUEsOEJBQThCLEdBQUcseUJBQXlCLENBQUE7QUFFMUQsUUFBQSxhQUFhLEdBQUc7SUFDekIsbUJBQW1CLEVBQUUsV0FBVztJQUNoQyxvQkFBb0IsRUFBRSxXQUFXO0lBQ2pDLGtCQUFrQixFQUFFLE1BQU07Q0FDN0IsQ0FBQTtBQUVZLFFBQUEsY0FBYyxHQUE4QjtJQUNyRCxJQUFJLEVBQUUsMEJBQTBCLHNCQUFjLENBQUMsSUFBSSxJQUFJLGNBQU0sRUFBRTtJQUMvRCxLQUFLLEVBQUUsMEJBQTBCLHNCQUFjLENBQUMsS0FBSyxJQUFJLGNBQU0sRUFBRTtJQUNqRSxJQUFJLEVBQUUsMEJBQTBCLHNCQUFjLENBQUMsSUFBSSxJQUFJLGNBQU0sRUFBRTtDQUNsRSxDQUFBO0FBQ1ksUUFBQSxjQUFjLEdBQUcsdUJBQXVCLENBQUE7QUFDeEMsUUFBQSx5QkFBeUIsR0FBRyxXQUFXLENBQUEiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge2FjY291bnRJZEFsaWFzLCByZWdpb24sIFN0YWdlTmFtZX0gZnJvbSBcIi4uL3V0aWxzXCI7XG5cbmV4cG9ydCBjb25zdCBERUZBVUxUX09SQ0FCVVNfVE9LRU5fU0VDUkVUX0lEID0gICdvcmNhYnVzL3Rva2VuLXNlcnZpY2Utand0J1xuZXhwb3J0IGNvbnN0IERFRkFVTFRfSE9TVE5BTUVfU1NNX1BBUkFNRVRFUiA9ICcvaG9zdGVkX3pvbmUvdW1jY3IvbmFtZSdcblxuZXhwb3J0IGNvbnN0IE1BUlRfRU5WX1ZBUlMgPSB7XG4gICAgYXRoZW5hV29ya2dyb3VwTmFtZTogJ29yY2Fob3VzZScsXG4gICAgYXRoZW5hRGF0YXNvdXJjZU5hbWU6ICdvcmNhdmF1bHQnLFxuICAgIGF0aGVuYURhdGFiYXNlTmFtZTogJ21hcnQnXG59XG5cbmV4cG9ydCBjb25zdCBNQVJUX1MzX0JVQ0tFVDogUmVjb3JkPFN0YWdlTmFtZSwgc3RyaW5nPiA9IHtcbiAgICBCRVRBOiBgZGF0YS1zaGFyaW5nLWFydGlmYWN0cy0ke2FjY291bnRJZEFsaWFzLkJFVEF9LSR7cmVnaW9ufWAsXG4gICAgR0FNTUE6IGBkYXRhLXNoYXJpbmctYXJ0aWZhY3RzLSR7YWNjb3VudElkQWxpYXMuR0FNTUF9LSR7cmVnaW9ufWAsXG4gICAgUFJPRDogYGRhdGEtc2hhcmluZy1hcnRpZmFjdHMtJHthY2NvdW50SWRBbGlhcy5QUk9EfS0ke3JlZ2lvbn1gXG59XG5leHBvcnQgY29uc3QgTUFSVF9TM19QUkVGSVggPSAnYXRoZW5hLXF1ZXJ5LXJlc3VsdHMvJ1xuZXhwb3J0IGNvbnN0IE1BUlRfTEFNQkRBX0ZVTkNUSU9OX05BTUUgPSAnb3JjYXZhdWx0J1xuIl19
|
|
@@ -0,0 +1,59 @@
|
|
|
1
|
+
import { PythonFunction, PythonFunctionProps } from "@aws-cdk/aws-lambda-python-alpha";
|
|
2
|
+
import { Construct } from "constructs";
|
|
3
|
+
export interface OrcabusResourcesProps {
|
|
4
|
+
/**
|
|
5
|
+
* Provide the orcabusTokenSecretId, otherwise it will default to @DEFAULT_ORCABUS_TOKEN_SECRET_ID
|
|
6
|
+
*/
|
|
7
|
+
readonly orcabusTokenSecretId?: string;
|
|
8
|
+
/**
|
|
9
|
+
* Provide the hostnameSsmParameterName, otherwise it will default to @DEFAULT_HOSTNAME_SSM_PARAMETER
|
|
10
|
+
*/
|
|
11
|
+
readonly hostnameSsmParameterName?: string;
|
|
12
|
+
}
|
|
13
|
+
export interface MartEnvironmentVariables {
|
|
14
|
+
/**
|
|
15
|
+
* Provide the athenaWorkgroupName, otherwise it will default to @MART_ENV_VARS.ATHENA_WORKGROUP_NAME
|
|
16
|
+
*/
|
|
17
|
+
readonly athenaWorkgroupName?: string;
|
|
18
|
+
/**
|
|
19
|
+
* Provide the athenaDatasourceName, otherwise it will default to @MART_ENV_VARS.athenaDatasourceName
|
|
20
|
+
*/
|
|
21
|
+
readonly athenaDatasourceName?: string;
|
|
22
|
+
/**
|
|
23
|
+
* Provide the athenaDatabaseName, otherwise it will default to @MART_ENV_VARS.ATHENA_DATABASE_NAME
|
|
24
|
+
*/
|
|
25
|
+
readonly athenaDatabaseName?: string;
|
|
26
|
+
}
|
|
27
|
+
export interface PythonUvFunctionProps extends PythonFunctionProps {
|
|
28
|
+
/**
|
|
29
|
+
* Whether or not to include the orcabus api tools layer in the lambda function build
|
|
30
|
+
*/
|
|
31
|
+
readonly includeOrcabusApiToolsLayer?: boolean;
|
|
32
|
+
/**
|
|
33
|
+
* Whether or not to include the mart layer in the lambda function build
|
|
34
|
+
* Note that the mart layer is a little heavier than the orcabus api tools layer
|
|
35
|
+
* Since we require pandas to be installed
|
|
36
|
+
*/
|
|
37
|
+
readonly includeMartLayer?: boolean;
|
|
38
|
+
/**
|
|
39
|
+
* Provide the orcabusTokenResources, optional, otherwise it will default to
|
|
40
|
+
* @DEFAULT_ORCABUS_TOKEN_SECRET_ID and @DEFAULT_HOSTNAME_SSM_PARAMETER
|
|
41
|
+
* for the secret and SSM parameter respectively
|
|
42
|
+
*/
|
|
43
|
+
readonly orcabusTokenResources?: OrcabusResourcesProps;
|
|
44
|
+
/**
|
|
45
|
+
* Provide the martEnvironmentVariables, optional, otherwise it will default to
|
|
46
|
+
* @MART_ENV_VARS.ATHENA_WORKGROUP_NAME, @MART_ENV_VARS.ATHENA_DATASOURCE_NAME
|
|
47
|
+
* and @MART_ENV_VARS.ATHENA_DATABASE_NAME for the athena workgroup, datasource and database respectively
|
|
48
|
+
*/
|
|
49
|
+
readonly martEnvironmentVariables?: MartEnvironmentVariables;
|
|
50
|
+
}
|
|
51
|
+
export declare class PythonUvFunction extends PythonFunction {
|
|
52
|
+
private static orcabusApiToolsLayer;
|
|
53
|
+
private static martLayer;
|
|
54
|
+
constructor(scope: Construct, id: string, props: PythonUvFunctionProps);
|
|
55
|
+
private buildOrcabusApiToolsLayer;
|
|
56
|
+
private buildMartToolsLayer;
|
|
57
|
+
private setOrcabusResources;
|
|
58
|
+
private setAthenaResources;
|
|
59
|
+
}
|
package/lambda/index.js
ADDED
|
@@ -0,0 +1,242 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
14
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
15
|
+
}) : function(o, v) {
|
|
16
|
+
o["default"] = v;
|
|
17
|
+
});
|
|
18
|
+
var __importStar = (this && this.__importStar) || (function () {
|
|
19
|
+
var ownKeys = function(o) {
|
|
20
|
+
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
21
|
+
var ar = [];
|
|
22
|
+
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
23
|
+
return ar;
|
|
24
|
+
};
|
|
25
|
+
return ownKeys(o);
|
|
26
|
+
};
|
|
27
|
+
return function (mod) {
|
|
28
|
+
if (mod && mod.__esModule) return mod;
|
|
29
|
+
var result = {};
|
|
30
|
+
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
31
|
+
__setModuleDefault(result, mod);
|
|
32
|
+
return result;
|
|
33
|
+
};
|
|
34
|
+
})();
|
|
35
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
36
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
37
|
+
};
|
|
38
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
39
|
+
exports.PythonUvFunction = void 0;
|
|
40
|
+
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
|
41
|
+
const aws_lambda_python_alpha_1 = require("@aws-cdk/aws-lambda-python-alpha");
|
|
42
|
+
const lambda = __importStar(require("aws-cdk-lib/aws-lambda"));
|
|
43
|
+
const secretsManager = __importStar(require("aws-cdk-lib/aws-secretsmanager"));
|
|
44
|
+
const ssm = __importStar(require("aws-cdk-lib/aws-ssm"));
|
|
45
|
+
const iam = __importStar(require("aws-cdk-lib/aws-iam"));
|
|
46
|
+
const s3 = __importStar(require("aws-cdk-lib/aws-s3"));
|
|
47
|
+
const aws_cdk_lib_1 = require("aws-cdk-lib");
|
|
48
|
+
const path_1 = __importDefault(require("path"));
|
|
49
|
+
const aws_lambda_1 = require("aws-cdk-lib/aws-lambda");
|
|
50
|
+
const config_1 = require("./config");
|
|
51
|
+
const utils_1 = require("../utils");
|
|
52
|
+
function getPythonUvDockerImage() {
|
|
53
|
+
return aws_cdk_lib_1.DockerImage.fromBuild(path_1.default.join(__dirname, 'build_python'));
|
|
54
|
+
}
|
|
55
|
+
class PythonUvFunction extends aws_lambda_python_alpha_1.PythonFunction {
|
|
56
|
+
static [JSII_RTTI_SYMBOL_1] = { fqn: "@orcabus/platform-cdk-constructs.lambda.PythonUvFunction", version: "0.0.7-alpha.4" };
|
|
57
|
+
// Class constructs, to be used for caching the layers
|
|
58
|
+
// This means that if there are multiple lambdas throughout the stack
|
|
59
|
+
// They will all use the same layer
|
|
60
|
+
static orcabusApiToolsLayer;
|
|
61
|
+
static martLayer;
|
|
62
|
+
constructor(scope, id, props) {
|
|
63
|
+
const uvProps = {
|
|
64
|
+
...props,
|
|
65
|
+
bundling: {
|
|
66
|
+
...props.bundling,
|
|
67
|
+
buildArgs: {
|
|
68
|
+
...props.bundling?.buildArgs,
|
|
69
|
+
// Add TARGETPLATFORM to build args if it's not already set
|
|
70
|
+
TARGETPLATFORM: props.bundling?.buildArgs?.TARGETPLATFORM ?? lambda.Architecture.ARM_64.dockerPlatform,
|
|
71
|
+
},
|
|
72
|
+
image: getPythonUvDockerImage(),
|
|
73
|
+
commandHooks: {
|
|
74
|
+
// @ts-ignore
|
|
75
|
+
beforeBundling(inputDir, outputDir) {
|
|
76
|
+
return [];
|
|
77
|
+
},
|
|
78
|
+
// @ts-ignore
|
|
79
|
+
afterBundling(inputDir, outputDir) {
|
|
80
|
+
return [`rm -rf ${outputDir}/pandas/tests`];
|
|
81
|
+
},
|
|
82
|
+
},
|
|
83
|
+
},
|
|
84
|
+
paramsAndSecrets: props.paramsAndSecrets ??
|
|
85
|
+
aws_lambda_1.ParamsAndSecretsLayerVersion.fromVersion(aws_lambda_1.ParamsAndSecretsVersions.V1_0_103, {
|
|
86
|
+
cacheEnabled: true,
|
|
87
|
+
cacheSize: 300,
|
|
88
|
+
logLevel: aws_lambda_1.ParamsAndSecretsLogLevel.DEBUG,
|
|
89
|
+
}),
|
|
90
|
+
};
|
|
91
|
+
super(scope, id, uvProps);
|
|
92
|
+
if (props.includeOrcabusApiToolsLayer) {
|
|
93
|
+
/* Set the environment variables for the Orcabus resources */
|
|
94
|
+
this.setOrcabusResources(props.orcabusTokenResources ?? {});
|
|
95
|
+
/* Build the orcabus Api tools layer */
|
|
96
|
+
this.buildOrcabusApiToolsLayer();
|
|
97
|
+
this.addLayers(PythonUvFunction.orcabusApiToolsLayer);
|
|
98
|
+
}
|
|
99
|
+
if (props.includeMartLayer) {
|
|
100
|
+
/* Set the environment variables for the mart resources */
|
|
101
|
+
this.setAthenaResources(props.martEnvironmentVariables ?? {});
|
|
102
|
+
/* Build the mart layer */
|
|
103
|
+
this.buildMartToolsLayer();
|
|
104
|
+
this.addLayers(PythonUvFunction.martLayer);
|
|
105
|
+
}
|
|
106
|
+
}
|
|
107
|
+
buildOrcabusApiToolsLayer() {
|
|
108
|
+
// Only build orcabus api layer if it doesn't exist
|
|
109
|
+
if (!PythonUvFunction.orcabusApiToolsLayer) {
|
|
110
|
+
PythonUvFunction.orcabusApiToolsLayer = new aws_lambda_python_alpha_1.PythonLayerVersion(this, 'orcabusApiToolsLayer', {
|
|
111
|
+
layerVersionName: 'orcabusApiToolsLayer',
|
|
112
|
+
entry: path_1.default.join(__dirname, 'layers/orcabus_api_tools'),
|
|
113
|
+
compatibleRuntimes: [lambda.Runtime.PYTHON_3_12],
|
|
114
|
+
compatibleArchitectures: [lambda.Architecture.ARM_64],
|
|
115
|
+
license: 'GPL3',
|
|
116
|
+
description: 'orcabusApiToolsLayer',
|
|
117
|
+
bundling: {
|
|
118
|
+
image: getPythonUvDockerImage(),
|
|
119
|
+
commandHooks: {
|
|
120
|
+
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
121
|
+
beforeBundling(inputDir, outputDir) {
|
|
122
|
+
return [];
|
|
123
|
+
},
|
|
124
|
+
afterBundling(inputDir, outputDir) {
|
|
125
|
+
return [
|
|
126
|
+
`pip install ${inputDir} --target ${outputDir}`,
|
|
127
|
+
`find ${outputDir} -name 'pandas' -exec rm -rf {}/tests/ \\;`,
|
|
128
|
+
];
|
|
129
|
+
},
|
|
130
|
+
},
|
|
131
|
+
},
|
|
132
|
+
});
|
|
133
|
+
}
|
|
134
|
+
}
|
|
135
|
+
buildMartToolsLayer() {
|
|
136
|
+
// Only build the layer if it doesn't exist
|
|
137
|
+
if (!PythonUvFunction.martLayer) {
|
|
138
|
+
PythonUvFunction.martLayer = new aws_lambda_python_alpha_1.PythonLayerVersion(this, 'martToolsLayer', {
|
|
139
|
+
layerVersionName: 'martToolsLayer',
|
|
140
|
+
entry: path_1.default.join(__dirname, 'layers/mart_tools'),
|
|
141
|
+
compatibleRuntimes: [lambda.Runtime.PYTHON_3_12],
|
|
142
|
+
compatibleArchitectures: [lambda.Architecture.ARM_64],
|
|
143
|
+
license: 'GPL3',
|
|
144
|
+
description: 'martToolsLayer',
|
|
145
|
+
bundling: {
|
|
146
|
+
image: getPythonUvDockerImage(),
|
|
147
|
+
commandHooks: {
|
|
148
|
+
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
149
|
+
beforeBundling(inputDir, outputDir) {
|
|
150
|
+
return [];
|
|
151
|
+
},
|
|
152
|
+
afterBundling(inputDir, outputDir) {
|
|
153
|
+
return [
|
|
154
|
+
`pip install ${inputDir} --target ${outputDir}`,
|
|
155
|
+
`find ${outputDir} -name 'pandas' -exec rm -rf {}/tests/ \\;`,
|
|
156
|
+
];
|
|
157
|
+
},
|
|
158
|
+
},
|
|
159
|
+
},
|
|
160
|
+
});
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
setOrcabusResources(props) {
|
|
164
|
+
// Set secret object
|
|
165
|
+
const orcabusTokenSecretId = secretsManager.Secret.fromSecretNameV2(this, 'orcabusTokenSecretId', props.orcabusTokenSecretId ?? config_1.DEFAULT_ORCABUS_TOKEN_SECRET_ID);
|
|
166
|
+
const hostnameSsmParameterName = ssm.StringParameter.fromStringParameterName(this, 'hostnameSsmParameterName', props.hostnameSsmParameterName ?? config_1.DEFAULT_HOSTNAME_SSM_PARAMETER);
|
|
167
|
+
// Add permissions for the secret and SSM parameter
|
|
168
|
+
// To the current version
|
|
169
|
+
orcabusTokenSecretId.grantRead(this.currentVersion);
|
|
170
|
+
hostnameSsmParameterName.grantRead(this.currentVersion);
|
|
171
|
+
// Add environment variables
|
|
172
|
+
this.addEnvironment('ORCABUS_TOKEN_SECRET_ID', orcabusTokenSecretId.secretName);
|
|
173
|
+
this.addEnvironment('HOSTNAME_SSM_PARAMETER_NAME', hostnameSsmParameterName.parameterName);
|
|
174
|
+
}
|
|
175
|
+
setAthenaResources(props) {
|
|
176
|
+
// Resolve the stage name by performing a reverse lookup using cdk.Aws.ACCOUNT_ID on accountIdAlias
|
|
177
|
+
const stageName = (0, utils_1.resolveStageName)();
|
|
178
|
+
const athenaS3Bucket = s3.Bucket.fromBucketName(this, 'athenaS3Bucket', config_1.MART_S3_BUCKET[stageName]);
|
|
179
|
+
const athenaFunctionName = lambda.Function.fromFunctionName(this, 'lambdaFunctionName', config_1.MART_LAMBDA_FUNCTION_NAME);
|
|
180
|
+
// Add env vars
|
|
181
|
+
// Iterate over kv pairs of MART_ENV_VARS
|
|
182
|
+
for (const [key, value] of Object.entries(config_1.MART_ENV_VARS)) {
|
|
183
|
+
// After
|
|
184
|
+
if (props[key] !== undefined) {
|
|
185
|
+
this.addEnvironment(key, props[key]);
|
|
186
|
+
}
|
|
187
|
+
else {
|
|
188
|
+
this.addEnvironment(key, value);
|
|
189
|
+
}
|
|
190
|
+
}
|
|
191
|
+
// Permissions
|
|
192
|
+
this.currentVersion.addToRolePolicy(
|
|
193
|
+
// From https://docs.aws.amazon.com/athena/latest/ug/example-policies-workgroup.html
|
|
194
|
+
new iam.PolicyStatement({
|
|
195
|
+
actions: [
|
|
196
|
+
// Workgroup lists
|
|
197
|
+
'athena:ListEngineVersions',
|
|
198
|
+
'athena:ListWorkGroups',
|
|
199
|
+
'athena:ListDataCatalogs',
|
|
200
|
+
'athena:ListDatabases',
|
|
201
|
+
'athena:GetDatabase',
|
|
202
|
+
'athena:ListTableMetadata',
|
|
203
|
+
'athena:GetTableMetadata',
|
|
204
|
+
'athena:GetDataCatalog',
|
|
205
|
+
],
|
|
206
|
+
resources: [`arn:aws:athena:${utils_1.region}:${utils_1.accountIdAlias[stageName]}:*`],
|
|
207
|
+
}));
|
|
208
|
+
this.currentVersion.addToRolePolicy(
|
|
209
|
+
// From https://docs.aws.amazon.com/athena/latest/ug/example-policies-workgroup.html
|
|
210
|
+
new iam.PolicyStatement({
|
|
211
|
+
actions: [
|
|
212
|
+
// Workgroup executions
|
|
213
|
+
'athena:BatchGetQueryExecution',
|
|
214
|
+
'athena:GetQueryExecution',
|
|
215
|
+
'athena:ListQueryExecutions',
|
|
216
|
+
'athena:StartQueryExecution',
|
|
217
|
+
'athena:StopQueryExecution',
|
|
218
|
+
'athena:GetQueryResults',
|
|
219
|
+
'athena:GetQueryResultsStream',
|
|
220
|
+
'athena:CreateNamedQuery',
|
|
221
|
+
'athena:GetNamedQuery',
|
|
222
|
+
'athena:BatchGetNamedQuery',
|
|
223
|
+
'athena:ListNamedQueries',
|
|
224
|
+
'athena:DeleteNamedQuery',
|
|
225
|
+
'athena:CreatePreparedStatement',
|
|
226
|
+
'athena:GetPreparedStatement',
|
|
227
|
+
'athena:ListPreparedStatements',
|
|
228
|
+
'athena:UpdatePreparedStatement',
|
|
229
|
+
'athena:DeletePreparedStatement',
|
|
230
|
+
],
|
|
231
|
+
resources: [
|
|
232
|
+
`arn:aws:athena:${utils_1.region}:${utils_1.accountIdAlias[stageName]}:workgroup/${config_1.MART_ENV_VARS.athenaWorkgroupName}`,
|
|
233
|
+
],
|
|
234
|
+
}));
|
|
235
|
+
// Add read access to the S3 bucket
|
|
236
|
+
athenaS3Bucket.grantReadWrite(this.currentVersion, `${config_1.MART_S3_PREFIX}*`);
|
|
237
|
+
// Add invoke access to the lambda function
|
|
238
|
+
athenaFunctionName.grantInvoke(this.currentVersion);
|
|
239
|
+
}
|
|
240
|
+
}
|
|
241
|
+
exports.PythonUvFunction = PythonUvFunction;
|
|
242
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["index.ts"],"names":[],"mappings":"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","sourcesContent":["import {PythonFunction, PythonFunctionProps, PythonLayerVersion} from \"@aws-cdk/aws-lambda-python-alpha\";\nimport {Construct} from \"constructs\";\nimport * as lambda from \"aws-cdk-lib/aws-lambda\";\nimport * as secretsManager from \"aws-cdk-lib/aws-secretsmanager\";\nimport * as ssm from \"aws-cdk-lib/aws-ssm\";\nimport * as iam from \"aws-cdk-lib/aws-iam\";\nimport * as s3 from \"aws-cdk-lib/aws-s3\";\nimport {DockerImage} from 'aws-cdk-lib';\n\nimport path from \"path\";\n\nimport {\n    ParamsAndSecretsLayerVersion,\n    ParamsAndSecretsLogLevel,\n    ParamsAndSecretsVersions,\n} from 'aws-cdk-lib/aws-lambda';\nimport {\n    DEFAULT_HOSTNAME_SSM_PARAMETER,\n    DEFAULT_ORCABUS_TOKEN_SECRET_ID,\n    MART_ENV_VARS, MART_LAMBDA_FUNCTION_NAME,\n    MART_S3_BUCKET,\n    MART_S3_PREFIX\n} from \"./config\";\nimport {accountIdAlias, region, resolveStageName} from \"../utils\";\n\n\nfunction getPythonUvDockerImage(): DockerImage {\n    return DockerImage.fromBuild(path.join(__dirname, 'build_python'));\n}\n\nexport interface OrcabusResourcesProps {\n    /**\n     * Provide the orcabusTokenSecretId, otherwise it will default to @DEFAULT_ORCABUS_TOKEN_SECRET_ID\n     */\n    readonly orcabusTokenSecretId?: string\n\n    /**\n     * Provide the hostnameSsmParameterName, otherwise it will default to @DEFAULT_HOSTNAME_SSM_PARAMETER\n     */\n    readonly hostnameSsmParameterName?: string\n}\n\nexport interface MartEnvironmentVariables {\n    /**\n     * Provide the athenaWorkgroupName, otherwise it will default to @MART_ENV_VARS.ATHENA_WORKGROUP_NAME\n     */\n    readonly athenaWorkgroupName?: string\n\n    /**\n     * Provide the athenaDatasourceName, otherwise it will default to @MART_ENV_VARS.athenaDatasourceName\n     */\n    readonly athenaDatasourceName?: string\n\n    /**\n     * Provide the athenaDatabaseName, otherwise it will default to @MART_ENV_VARS.ATHENA_DATABASE_NAME\n     */\n    readonly athenaDatabaseName?: string\n}\n\nexport interface PythonUvFunctionProps extends PythonFunctionProps {\n    /**\n     * Whether or not to include the orcabus api tools layer in the lambda function build\n     */\n    readonly includeOrcabusApiToolsLayer?: boolean\n\n    /**\n     * Whether or not to include the mart layer in the lambda function build\n     * Note that the mart layer is a little heavier than the orcabus api tools layer\n     * Since we require pandas to be installed\n     */\n    readonly includeMartLayer?: boolean\n\n    /**\n     * Provide the orcabusTokenResources, optional, otherwise it will default to\n     * @DEFAULT_ORCABUS_TOKEN_SECRET_ID and @DEFAULT_HOSTNAME_SSM_PARAMETER\n     * for the secret and SSM parameter respectively\n     */\n    readonly orcabusTokenResources?: OrcabusResourcesProps\n\n    /**\n     * Provide the martEnvironmentVariables, optional, otherwise it will default to\n     * @MART_ENV_VARS.ATHENA_WORKGROUP_NAME, @MART_ENV_VARS.ATHENA_DATASOURCE_NAME\n     * and @MART_ENV_VARS.ATHENA_DATABASE_NAME for the athena workgroup, datasource and database respectively\n     */\n    readonly martEnvironmentVariables?: MartEnvironmentVariables\n}\n\n\nexport class PythonUvFunction extends PythonFunction {\n    // Class constructs, to be used for caching the layers\n    // This means that if there are multiple lambdas throughout the stack\n    // They will all use the same layer\n    private static orcabusApiToolsLayer: lambda.ILayerVersion;\n    private static martLayer: lambda.ILayerVersion;\n\n    constructor(scope: Construct, id: string, props: PythonUvFunctionProps) {\n        const uvProps = {\n            ...props,\n            bundling: {\n                ...props.bundling,\n                buildArgs: {\n                    ...props.bundling?.buildArgs,\n                    // Add TARGETPLATFORM to build args if it's not already set\n                    TARGETPLATFORM:\n                        props.bundling?.buildArgs?.TARGETPLATFORM ?? lambda.Architecture.ARM_64.dockerPlatform,\n                },\n                image: getPythonUvDockerImage(),\n                commandHooks: {\n                    // @ts-ignore\n                    beforeBundling(inputDir: string, outputDir: string): string[] {\n                        return [];\n                    },\n                    // @ts-ignore\n                    afterBundling(inputDir: string, outputDir: string): string[] {\n                        return [`rm -rf ${outputDir}/pandas/tests`];\n                    },\n                },\n            },\n            paramsAndSecrets:\n                props.paramsAndSecrets ??\n                ParamsAndSecretsLayerVersion.fromVersion(ParamsAndSecretsVersions.V1_0_103, {\n                    cacheEnabled: true,\n                    cacheSize: 300,\n                    logLevel: ParamsAndSecretsLogLevel.DEBUG,\n                }),\n        };\n        super(scope, id, uvProps);\n\n        if (props.includeOrcabusApiToolsLayer) {\n            /* Set the environment variables for the Orcabus resources */\n            this.setOrcabusResources(props.orcabusTokenResources ?? {})\n\n            /* Build the orcabus Api tools layer */\n            this.buildOrcabusApiToolsLayer()\n            this.addLayers(PythonUvFunction.orcabusApiToolsLayer)\n        }\n\n        if (props.includeMartLayer) {\n            /* Set the environment variables for the mart resources */\n            this.setAthenaResources(props.martEnvironmentVariables ?? {})\n\n            /* Build the mart layer */\n            this.buildMartToolsLayer()\n            this.addLayers(PythonUvFunction.martLayer)\n        }\n    }\n\n    private buildOrcabusApiToolsLayer() {\n        // Only build orcabus api layer if it doesn't exist\n        if (!PythonUvFunction.orcabusApiToolsLayer) {\n            PythonUvFunction.orcabusApiToolsLayer = new PythonLayerVersion(this, 'orcabusApiToolsLayer', {\n                layerVersionName: 'orcabusApiToolsLayer',\n                entry: path.join(__dirname, 'layers/orcabus_api_tools'),\n                compatibleRuntimes: [lambda.Runtime.PYTHON_3_12],\n                compatibleArchitectures: [lambda.Architecture.ARM_64],\n                license: 'GPL3',\n                description: 'orcabusApiToolsLayer',\n                bundling: {\n                    image: getPythonUvDockerImage(),\n                    commandHooks: {\n                        // eslint-disable-next-line @typescript-eslint/no-unused-vars\n                        beforeBundling(inputDir: string, outputDir: string): string[] {\n                            return [];\n                        },\n                        afterBundling(inputDir: string, outputDir: string): string[] {\n                            return [\n                                `pip install ${inputDir} --target ${outputDir}`,\n                                `find ${outputDir} -name 'pandas' -exec rm -rf {}/tests/ \\\\;`,\n                            ];\n                        },\n                    },\n                },\n            });\n        }\n    }\n\n    private buildMartToolsLayer() {\n        // Only build the layer if it doesn't exist\n        if (!PythonUvFunction.martLayer) {\n            PythonUvFunction.martLayer = new PythonLayerVersion(this, 'martToolsLayer', {\n                layerVersionName: 'martToolsLayer',\n                entry: path.join(__dirname, 'layers/mart_tools'),\n                compatibleRuntimes: [lambda.Runtime.PYTHON_3_12],\n                compatibleArchitectures: [lambda.Architecture.ARM_64],\n                license: 'GPL3',\n                description: 'martToolsLayer',\n                bundling: {\n                    image: getPythonUvDockerImage(),\n                    commandHooks: {\n                        // eslint-disable-next-line @typescript-eslint/no-unused-vars\n                        beforeBundling(inputDir: string, outputDir: string): string[] {\n                            return [];\n                        },\n                        afterBundling(inputDir: string, outputDir: string): string[] {\n                            return [\n                                `pip install ${inputDir} --target ${outputDir}`,\n                                `find ${outputDir} -name 'pandas' -exec rm -rf {}/tests/ \\\\;`,\n                            ];\n                        },\n                    },\n                },\n            });\n        }\n\n    }\n\n    private setOrcabusResources(\n        props: OrcabusResourcesProps\n    ) {\n        // Set secret object\n        const orcabusTokenSecretId = secretsManager.Secret.fromSecretNameV2(\n            this, 'orcabusTokenSecretId',\n            props.orcabusTokenSecretId ?? DEFAULT_ORCABUS_TOKEN_SECRET_ID\n        );\n        const hostnameSsmParameterName = ssm.StringParameter.fromStringParameterName(\n            this, 'hostnameSsmParameterName',\n            props.hostnameSsmParameterName ?? DEFAULT_HOSTNAME_SSM_PARAMETER,\n        )\n\n        // Add permissions for the secret and SSM parameter\n        // To the current version\n        orcabusTokenSecretId.grantRead(this.currentVersion);\n        hostnameSsmParameterName.grantRead(this.currentVersion)\n\n        // Add environment variables\n        this.addEnvironment(\n            'ORCABUS_TOKEN_SECRET_ID', orcabusTokenSecretId.secretName,\n        )\n        this.addEnvironment(\n            'HOSTNAME_SSM_PARAMETER_NAME', hostnameSsmParameterName.parameterName,\n        )\n    }\n\n\n    private setAthenaResources(\n        props: MartEnvironmentVariables\n    ) {\n        // Resolve the stage name by performing a reverse lookup using cdk.Aws.ACCOUNT_ID on accountIdAlias\n        const stageName = resolveStageName()\n\n        const athenaS3Bucket = s3.Bucket.fromBucketName(\n            this, 'athenaS3Bucket',\n            MART_S3_BUCKET[stageName]\n        )\n\n        const athenaFunctionName = lambda.Function.fromFunctionName(\n            this, 'lambdaFunctionName', MART_LAMBDA_FUNCTION_NAME\n        )\n\n        // Add env vars\n        // Iterate over kv pairs of MART_ENV_VARS\n        for (const [key, value] of Object.entries(MART_ENV_VARS)) {\n            // After\n            if (props[key as keyof MartEnvironmentVariables] !== undefined) {\n                this.addEnvironment(key, props[key as keyof MartEnvironmentVariables]!);\n            } else {\n                this.addEnvironment(key, value);\n            }\n        }\n\n        // Permissions\n        this.currentVersion.addToRolePolicy(\n            // From https://docs.aws.amazon.com/athena/latest/ug/example-policies-workgroup.html\n            new iam.PolicyStatement({\n                actions: [\n                    // Workgroup lists\n                    'athena:ListEngineVersions',\n                    'athena:ListWorkGroups',\n                    'athena:ListDataCatalogs',\n                    'athena:ListDatabases',\n                    'athena:GetDatabase',\n                    'athena:ListTableMetadata',\n                    'athena:GetTableMetadata',\n                    'athena:GetDataCatalog',\n                ],\n                resources: [`arn:aws:athena:${region}:${accountIdAlias[stageName]}:*`],\n            })\n        );\n\n        this.currentVersion.addToRolePolicy(\n            // From https://docs.aws.amazon.com/athena/latest/ug/example-policies-workgroup.html\n            new iam.PolicyStatement({\n                actions: [\n                    // Workgroup executions\n                    'athena:BatchGetQueryExecution',\n                    'athena:GetQueryExecution',\n                    'athena:ListQueryExecutions',\n                    'athena:StartQueryExecution',\n                    'athena:StopQueryExecution',\n                    'athena:GetQueryResults',\n                    'athena:GetQueryResultsStream',\n                    'athena:CreateNamedQuery',\n                    'athena:GetNamedQuery',\n                    'athena:BatchGetNamedQuery',\n                    'athena:ListNamedQueries',\n                    'athena:DeleteNamedQuery',\n                    'athena:CreatePreparedStatement',\n                    'athena:GetPreparedStatement',\n                    'athena:ListPreparedStatements',\n                    'athena:UpdatePreparedStatement',\n                    'athena:DeletePreparedStatement',\n                ],\n                resources: [\n                    `arn:aws:athena:${region}:${accountIdAlias[stageName]}:workgroup/${MART_ENV_VARS.athenaWorkgroupName}`,\n                ],\n            })\n        );\n\n        // Add read access to the S3 bucket\n        athenaS3Bucket.grantReadWrite(\n            this.currentVersion,\n            `${MART_S3_PREFIX}*`\n        )\n\n        // Add invoke access to the lambda function\n        athenaFunctionName.grantInvoke(\n            this.currentVersion\n        )\n    }\n}\n\n\n\n\n"]}
|