@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.
Files changed (83) hide show
  1. package/.jsii +858 -6
  2. package/api-gateway/api-gateway.js +22 -4
  3. package/deployment-stack-pipeline/pipeline.js +6 -4
  4. package/dynamodb/config.d.ts +3 -0
  5. package/dynamodb/config.js +7 -0
  6. package/dynamodb/index.d.ts +27 -0
  7. package/dynamodb/index.js +106 -0
  8. package/ecs/config.d.ts +4 -0
  9. package/ecs/config.js +8 -0
  10. package/ecs/index.d.ts +46 -0
  11. package/ecs/index.js +103 -0
  12. package/index.d.ts +5 -2
  13. package/index.js +47 -5
  14. package/index.ts +16 -2
  15. package/lambda/build_python/Dockerfile +60 -0
  16. package/lambda/config.d.ts +11 -0
  17. package/lambda/config.js +19 -0
  18. package/lambda/index.d.ts +59 -0
  19. package/lambda/index.js +242 -0
  20. package/lambda/layers/mart_tools/poetry.lock +303 -0
  21. package/lambda/layers/mart_tools/pyproject.toml +27 -0
  22. package/lambda/layers/mart_tools/src/mart_tools/__init__.py +0 -0
  23. package/lambda/layers/mart_tools/src/mart_tools/mart/__init__.py +8 -0
  24. package/lambda/layers/mart_tools/src/mart_tools/mart/aws_helpers.py +79 -0
  25. package/lambda/layers/mart_tools/src/mart_tools/mart/dataframe_helpers.py +29 -0
  26. package/lambda/layers/mart_tools/src/mart_tools/mart/globals.py +5 -0
  27. package/lambda/layers/mart_tools/src/mart_tools/mart/models.py +71 -0
  28. package/lambda/layers/orcabus_api_tools/poetry.lock +273 -0
  29. package/lambda/layers/orcabus_api_tools/pyproject.toml +27 -0
  30. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/__init__.py +0 -0
  31. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/fastq/__init__.py +172 -0
  32. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/fastq/create_helpers.py +47 -0
  33. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/fastq/globals.py +13 -0
  34. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/fastq/job_helpers.py +53 -0
  35. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/fastq/models.py +253 -0
  36. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/fastq/query_helpers.py +248 -0
  37. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/fastq/update_helpers.py +221 -0
  38. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/fastq/workflow_helpers.py +25 -0
  39. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/fastq_unarchiving/__init__.py +92 -0
  40. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/fastq_unarchiving/create_helpers.py +27 -0
  41. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/fastq_unarchiving/globals.py +21 -0
  42. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/fastq_unarchiving/models.py +51 -0
  43. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/fastq_unarchiving/query_helpers.py +52 -0
  44. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/fastq_unarchiving/update_helpers.py +45 -0
  45. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/filemanager/__init__.py +98 -0
  46. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/filemanager/errors.py +45 -0
  47. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/filemanager/file_helpers.py +341 -0
  48. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/filemanager/globals.py +70 -0
  49. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/filemanager/models.py +59 -0
  50. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/mart/__init__.py +8 -0
  51. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/mart/aws_helpers.py +79 -0
  52. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/mart/dataframe_helpers.py +29 -0
  53. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/mart/globals.py +5 -0
  54. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/mart/models.py +71 -0
  55. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/metadata/__init__.py +250 -0
  56. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/metadata/contact_helpers.py +109 -0
  57. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/metadata/errors.py +104 -0
  58. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/metadata/globals.py +16 -0
  59. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/metadata/individual_helpers.py +139 -0
  60. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/metadata/library_helpers.py +196 -0
  61. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/metadata/lims_helpers.py +36 -0
  62. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/metadata/models.py +112 -0
  63. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/metadata/project_helpers.py +129 -0
  64. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/metadata/sample_helpers.py +132 -0
  65. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/metadata/subject_helpers.py +151 -0
  66. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/sequence/__init__.py +15 -0
  67. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/sequence/globals.py +2 -0
  68. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/sequence/models.py +44 -0
  69. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/sequence/sequence_helpers.py +62 -0
  70. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/utils/__init__.py +0 -0
  71. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/utils/aws_helpers.py +123 -0
  72. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/utils/miscell.py +17 -0
  73. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/utils/requests_helpers.py +163 -0
  74. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/workflow/__init__.py +0 -0
  75. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/workflow/errors.py +37 -0
  76. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/workflow/metadata_helpers.py +28 -0
  77. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/workflow/models.py +85 -0
  78. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/workflow/payload_helpers.py +64 -0
  79. package/lambda/layers/orcabus_api_tools/src/orcabus_api_tools/workflow/workflow_run_helpers.py +80 -0
  80. package/package.json +13 -7
  81. package/typedoc.json +3 -0
  82. package/utils/index.d.ts +3 -0
  83. package/utils/index.js +50 -1
package/index.ts CHANGED
@@ -1,3 +1,17 @@
1
- export * as apigateway from "./api-gateway";
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";
@@ -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
+ }
@@ -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"]}