eoapi-cdk 7.6.2 → 8.0.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.
@@ -6,8 +6,37 @@ import asyncio
6
6
  import os
7
7
 
8
8
  from mangum import Mangum
9
+ from stac_fastapi.pgstac.app import app
10
+ from stac_fastapi.pgstac.config import PostgresSettings
11
+ from stac_fastapi.pgstac.db import close_db_connection, connect_to_db
12
+ from utils import get_secret_dict
13
+
14
+ secret = get_secret_dict(secret_arn_env_var="PGSTAC_SECRET_ARN")
15
+ postgres_settings = PostgresSettings(
16
+ postgres_host_reader=secret["host"],
17
+ postgres_host_writer=secret["host"],
18
+ postgres_dbname=secret["dbname"],
19
+ postgres_user=secret["username"],
20
+ postgres_pass=secret["password"],
21
+ postgres_port=int(secret["port"]),
22
+ )
23
+
24
+
25
+ @app.on_event("startup")
26
+ async def startup_event():
27
+ """Connect to database on startup."""
28
+ print("Setting up DB connection...")
29
+ await connect_to_db(app, postgres_settings=postgres_settings)
30
+ print("DB connection setup.")
31
+
32
+
33
+ @app.on_event("shutdown")
34
+ async def shutdown_event():
35
+ """Close database connection."""
36
+ print("Closing up DB connection...")
37
+ await close_db_connection(app)
38
+ print("DB connection closed.")
9
39
 
10
- from .app import app
11
40
 
12
41
  handler = Mangum(app, lifespan="off")
13
42
 
@@ -91,5 +91,5 @@ class StacBrowser extends constructs_1.Construct {
91
91
  }
92
92
  exports.StacBrowser = StacBrowser;
93
93
  _a = JSII_RTTI_SYMBOL_1;
94
- StacBrowser[_a] = { fqn: "eoapi-cdk.StacBrowser", version: "7.6.2" };
94
+ StacBrowser[_a] = { fqn: "eoapi-cdk.StacBrowser", version: "8.0.0" };
95
95
  //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["index.ts"],"names":[],"mappings":"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","sourcesContent":["import { Stack, aws_s3 as s3, aws_s3_deployment as s3_deployment} from \"aws-cdk-lib\";\nimport { RemovalPolicy, CfnOutput } from \"aws-cdk-lib\";\nimport { PolicyStatement, ServicePrincipal, Effect } from \"aws-cdk-lib/aws-iam\";\n\nimport { Construct } from \"constructs\";\nimport { execSync } from \"child_process\";\nimport * as fs from 'fs';\n\nconst DEFAULT_CLONE_DIRECTORY = './stac-browser';\n\nexport class StacBrowser extends Construct {\n\n    public bucket: s3.IBucket;\n    public bucketDeployment: s3_deployment.BucketDeployment;\n\n    constructor(scope: Construct, id: string, props: StacBrowserProps) {\n        super(scope, id);\n\n        const buildPath = this.buildApp(props, props.cloneDirectory || DEFAULT_CLONE_DIRECTORY);\n\n        // import a bucket from props.bucketArn if defined, otherwise create a new bucket\n        if (props.bucketArn) {\n            this.bucket = s3.Bucket.fromBucketArn(this, 'Bucket', props.bucketArn);\n        } else {\n            this.bucket = new s3.Bucket(this, 'Bucket', {\n                accessControl: s3.BucketAccessControl.PRIVATE,\n                removalPolicy: RemovalPolicy.DESTROY,\n                websiteIndexDocument: props.websiteIndexDocument\n            })\n        }\n\n        // if props.cloudFrontDistributionArn is defined and props.bucketArn is not defined, add a bucket policy to allow read access from the cloudfront distribution\n        if (props.cloudFrontDistributionArn && !props.bucketArn) {\n            this.bucket.addToResourcePolicy(new PolicyStatement({\n                        sid: 'AllowCloudFrontServicePrincipal',\n                        effect: Effect.ALLOW,\n                        actions: ['s3:GetObject'],\n                        principals: [new ServicePrincipal('cloudfront.amazonaws.com')],\n                        resources: [this.bucket.arnForObjects('*')],\n                        conditions: {\n                            'StringEquals': {\n                                'aws:SourceArn': props.cloudFrontDistributionArn\n                            }\n                        }\n                    }));\n        }\n\n        // add the compiled code to the bucket as a bucket deployment\n        this.bucketDeployment = new s3_deployment.BucketDeployment(this, 'BucketDeployment', {\n            destinationBucket: this.bucket,\n            sources: [s3_deployment.Source.asset(buildPath)]\n          });\n\n        new CfnOutput(this, \"bucket-name\", {\n        exportName: `${Stack.of(this).stackName}-bucket-name`,\n        value: this.bucket.bucketName,\n        });\n\n    }\n\n    private buildApp(props: StacBrowserProps, cloneDirectory: string): string {\n\n        // Define where to clone and build\n        const githubRepoUrl = 'https://github.com/radiantearth/stac-browser.git';\n\n\n        // Maybe the repo already exists in cloneDirectory. Try checking out the desired version and if it fails, delete and reclone.\n        try {\n            console.log(`Checking if a valid cloned repo exists with version ${props.githubRepoTag}...`)\n            execSync(`git checkout tags/${props.githubRepoTag}`, { cwd: cloneDirectory });\n        }\n        catch (error) {\n\n            // if directory exists, raise an error\n            if (fs.existsSync(cloneDirectory)) {\n                throw new Error(`Directory ${cloneDirectory} already exists and is not a valid clone of ${githubRepoUrl}. Please delete this directory or specify a different cloneDirectory.`);\n            }\n\n            // else, we clone and check out the version.\n\n            // Clone the repo\n            console.log(`Cloning ${githubRepoUrl} into ${cloneDirectory}...`)\n            execSync(`git clone ${githubRepoUrl} ${cloneDirectory}`);\n\n            // Check out the desired version\n            console.log(`Checking out version ${props.githubRepoTag}...`)\n            execSync(`git checkout tags/${props.githubRepoTag}`, { cwd: cloneDirectory });\n\n        }\n\n        // Install the dependencies and build the application\n        console.log(`Installing dependencies`)\n        execSync('npm install', { cwd: cloneDirectory });\n\n        // If a config file is provided, copy it to the stac-browser directory at \"config.js\", replaces the default config.js.\n        if (props.configFilePath) {\n            // check that the file exists at this location. if not, raise an error and print current working directory.\n            if (!fs.existsSync(props.configFilePath)) {\n                throw new Error(`Config file ${props.configFilePath} does not exist. Current working directory is ${process.cwd()}`);\n            }\n            console.log(`Copying config file ${props.configFilePath} to ${cloneDirectory}/config.js`)\n            fs.copyFileSync(props.configFilePath, `${cloneDirectory}/config.js`);\n        }\n\n        // Build the app with catalogUrl\n        console.log(`Building app with catalogUrl=${props.stacCatalogUrl} into ${cloneDirectory}`)\n        execSync(`npm run build -- --catalogUrl=${props.stacCatalogUrl}`, { cwd: cloneDirectory });\n\n        return './stac-browser/dist'\n\n    }\n\n\n}\n\nexport interface StacBrowserProps {\n\n    /**\n     * Bucket ARN. If specified, the identity used to deploy the stack must have the appropriate permissions to create a deployment for this bucket.\n     * In addition, if specified, `cloudFrontDistributionArn` is ignored since the policy of an imported resource can't be modified.\n     *\n     * @default - No bucket ARN. A new bucket will be created.\n     */\n\n    readonly bucketArn?: string;\n\n    /**\n     * STAC catalog URL. Overrides the catalog URL in the stac-browser configuration.\n     */\n    readonly stacCatalogUrl: string;\n\n    /**\n     * Path to config file for the STAC browser. If not provided, default configuration in the STAC browser\n     * repository is used.\n     */\n    readonly configFilePath?: string;\n\n    /**\n     * Tag of the radiant earth stac-browser repo to use to build the app.\n     */\n    readonly githubRepoTag: string;\n\n\n    /**\n     * The ARN of the cloudfront distribution that will be added to the bucket policy with read access.\n     * If `bucketArn` is specified, this parameter is ignored since the policy of an imported bucket can't be modified.\n     *\n     * @default - No cloudfront distribution ARN. The bucket policy will not be modified.\n     */\n    readonly cloudFrontDistributionArn?: string;\n\n    /**\n     * The name of the index document (e.g. \"index.html\") for the website. Enables static website\n     * hosting for this bucket.\n     *\n     * @default - No index document.\n     */\n    readonly websiteIndexDocument?: string;\n\n    /**\n     * Location in the filesystem where to compile the browser code.\n     *\n     * @default - DEFAULT_CLONE_DIRECTORY\n     */\n    readonly cloneDirectory?: string;\n\n}\n"]}
@@ -7,6 +7,7 @@ const aws_cdk_lib_1 = require("aws-cdk-lib");
7
7
  const aws_apigatewayv2_alpha_1 = require("@aws-cdk/aws-apigatewayv2-alpha");
8
8
  const aws_apigatewayv2_integrations_alpha_1 = require("@aws-cdk/aws-apigatewayv2-integrations-alpha");
9
9
  const constructs_1 = require("constructs");
10
+ const path = require("path");
10
11
  class TiPgApiLambda extends constructs_1.Construct {
11
12
  constructor(scope, id, props) {
12
13
  super(scope, id);
@@ -17,8 +18,8 @@ class TiPgApiLambda extends constructs_1.Construct {
17
18
  memorySize: 1024,
18
19
  logRetention: aws_cdk_lib_1.aws_logs.RetentionDays.ONE_WEEK,
19
20
  timeout: aws_cdk_lib_1.Duration.seconds(30),
20
- code: aws_cdk_lib_1.aws_lambda.Code.fromDockerBuild(__dirname, {
21
- file: "runtime/Dockerfile",
21
+ code: aws_cdk_lib_1.aws_lambda.Code.fromDockerBuild(path.join(__dirname, '..'), {
22
+ file: "tipg-api/runtime/Dockerfile",
22
23
  buildArgs: { PYTHON_VERSION: '3.11' },
23
24
  }),
24
25
  vpc: props.vpc,
@@ -54,5 +55,5 @@ class TiPgApiLambda extends constructs_1.Construct {
54
55
  }
55
56
  exports.TiPgApiLambda = TiPgApiLambda;
56
57
  _a = JSII_RTTI_SYMBOL_1;
57
- TiPgApiLambda[_a] = { fqn: "eoapi-cdk.TiPgApiLambda", version: "7.6.2" };
58
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJpbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7OztBQUFBLDZDQVN1QjtBQUNyQiw0RUFBc0c7QUFDdEcsc0dBQXFGO0FBQ3JGLDJDQUF1QztBQUd2QyxNQUFhLGFBQWMsU0FBUSxzQkFBUztJQUkxQyxZQUFZLEtBQWdCLEVBQUUsRUFBVSxFQUFFLEtBQXlCO1FBQ2pFLEtBQUssQ0FBQyxLQUFLLEVBQUUsRUFBRSxDQUFDLENBQUM7UUFFakIsSUFBSSxDQUFDLGtCQUFrQixHQUFHLElBQUksd0JBQU0sQ0FBQyxRQUFRLENBQUMsSUFBSSxFQUFFLFFBQVEsRUFBRTtZQUM1RCxXQUFXO1lBQ1gsT0FBTyxFQUFFLHdCQUFNLENBQUMsT0FBTyxDQUFDLFdBQVc7WUFDbkMsT0FBTyxFQUFFLGlCQUFpQjtZQUMxQixVQUFVLEVBQUUsSUFBSTtZQUNoQixZQUFZLEVBQUUsc0JBQUksQ0FBQyxhQUFhLENBQUMsUUFBUTtZQUN6QyxPQUFPLEVBQUUsc0JBQVEsQ0FBQyxPQUFPLENBQUMsRUFBRSxDQUFDO1lBQzdCLElBQUksRUFBRSx3QkFBTSxDQUFDLElBQUksQ0FBQyxlQUFlLENBQUMsU0FBUyxFQUFFO2dCQUM3QyxJQUFJLEVBQUUsb0JBQW9CO2dCQUMxQixTQUFTLEVBQUUsRUFBRSxjQUFjLEVBQUUsTUFBTSxFQUFFO2FBQ3BDLENBQUM7WUFDRixHQUFHLEVBQUUsS0FBSyxDQUFDLEdBQUc7WUFDZCxVQUFVLEVBQUUsS0FBSyxDQUFDLGVBQWU7WUFDakMsaUJBQWlCLEVBQUUsSUFBSTtZQUN2QixXQUFXLEVBQUU7Z0JBQ1gsaUJBQWlCLEVBQUUsS0FBSyxDQUFDLFFBQVEsQ0FBQyxTQUFTO2dCQUMzQyxnQkFBZ0IsRUFBRSxHQUFHO2dCQUNyQixnQkFBZ0IsRUFBRSxHQUFHO2dCQUNyQixHQUFHLEtBQUssQ0FBQyxNQUFNO2FBQ2hCO1lBQ0QsaUVBQWlFO1lBQ2pFLEdBQUcsS0FBSyxDQUFDLHFCQUFxQjtTQUMvQixDQUFDLENBQUM7UUFFSCxLQUFLLENBQUMsUUFBUSxDQUFDLFNBQVMsQ0FBQyxJQUFJLENBQUMsa0JBQWtCLENBQUMsQ0FBQztRQUVsRCxJQUFJLEtBQUssQ0FBQyxHQUFHLEVBQUMsQ0FBQztZQUNiLElBQUksQ0FBQyxrQkFBa0IsQ0FBQyxXQUFXLENBQUMsT0FBTyxDQUFDLEtBQUssQ0FBQyxFQUFFLEVBQUUscUJBQUcsQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFDLElBQUksQ0FBQyxFQUFFLDZCQUE2QixDQUFDLENBQUM7UUFDM0csQ0FBQztRQUVELE1BQU0sT0FBTyxHQUFHLElBQUksZ0NBQU8sQ0FBQyxJQUFJLEVBQUUsR0FBRyxtQkFBSyxDQUFDLEVBQUUsQ0FBQyxJQUFJLENBQUMsQ0FBQyxTQUFTLFdBQVcsRUFBRTtZQUN4RSxvQkFBb0IsRUFBRSxLQUFLLENBQUMsaUJBQWlCLENBQUMsQ0FBQyxDQUFDO2dCQUM5QyxVQUFVLEVBQUUsS0FBSyxDQUFDLGlCQUFpQjthQUNwQyxDQUFDLENBQUMsQ0FBQyxTQUFTO1lBQ2Isa0JBQWtCLEVBQUUsSUFBSSwyREFBcUIsQ0FDM0MsYUFBYSxFQUNiLElBQUksQ0FBQyxrQkFBa0IsRUFDdkIsS0FBSyxDQUFDLGlCQUFpQixDQUFDLENBQUMsQ0FBQztnQkFDdEIsZ0JBQWdCLEVBQUUsSUFBSSx5Q0FBZ0IsRUFBRSxDQUFDLGVBQWUsQ0FBQyxNQUFNLEVBQUUscUNBQVksQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFDLGlCQUFpQixDQUFDLElBQUksQ0FBQyxDQUFDO2FBQ3RILENBQUMsQ0FBQyxDQUFDLFNBQVMsQ0FDZDtTQUNGLENBQUMsQ0FBQztRQUVILElBQUksQ0FBQyxHQUFHLEdBQUcsT0FBTyxDQUFDLEdBQUksQ0FBQztRQUV4QixJQUFJLHVCQUFTLENBQUMsSUFBSSxFQUFFLGlCQUFpQixFQUFFO1lBQ3JDLFVBQVUsRUFBRSxHQUFHLG1CQUFLLENBQUMsRUFBRSxDQUFDLElBQUksQ0FBQyxDQUFDLFNBQVMsVUFBVTtZQUNqRCxLQUFLLEVBQUUsSUFBSSxDQUFDLEdBQUc7U0FDaEIsQ0FBQyxDQUFDO0lBQ0wsQ0FBQzs7QUF4REgsc0NBeURDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtcbiAgICBTdGFjayxcbiAgICBhd3NfZWMyIGFzIGVjMixcbiAgICBhd3NfbGFtYmRhIGFzIGxhbWJkYSxcbiAgICBhd3NfbG9ncyBhcyBsb2dzLFxuICAgIGF3c19yZHMgYXMgcmRzLFxuICAgIGF3c19zZWNyZXRzbWFuYWdlciBhcyBzZWNyZXRzbWFuYWdlcixcbiAgICBDZm5PdXRwdXQsXG4gICAgRHVyYXRpb24sXG4gIH0gZnJvbSBcImF3cy1jZGstbGliXCI7XG4gIGltcG9ydCB7IElEb21haW5OYW1lLCBIdHRwQXBpLCBQYXJhbWV0ZXJNYXBwaW5nLCBNYXBwaW5nVmFsdWV9IGZyb20gXCJAYXdzLWNkay9hd3MtYXBpZ2F0ZXdheXYyLWFscGhhXCI7XG4gIGltcG9ydCB7IEh0dHBMYW1iZGFJbnRlZ3JhdGlvbiB9IGZyb20gXCJAYXdzLWNkay9hd3MtYXBpZ2F0ZXdheXYyLWludGVncmF0aW9ucy1hbHBoYVwiO1xuICBpbXBvcnQgeyBDb25zdHJ1Y3QgfSBmcm9tIFwiY29uc3RydWN0c1wiO1xuICBpbXBvcnQgeyBDdXN0b21MYW1iZGFGdW5jdGlvblByb3BzIH0gZnJvbSBcIi4uL3V0aWxzXCI7XG5cbiAgZXhwb3J0IGNsYXNzIFRpUGdBcGlMYW1iZGEgZXh0ZW5kcyBDb25zdHJ1Y3Qge1xuICAgIHJlYWRvbmx5IHVybDogc3RyaW5nO1xuICAgIHB1YmxpYyB0aVBnTGFtYmRhRnVuY3Rpb246IGxhbWJkYS5GdW5jdGlvbjtcblxuICAgIGNvbnN0cnVjdG9yKHNjb3BlOiBDb25zdHJ1Y3QsIGlkOiBzdHJpbmcsIHByb3BzOiBUaVBnQXBpTGFtYmRhUHJvcHMpIHtcbiAgICAgIHN1cGVyKHNjb3BlLCBpZCk7XG5cbiAgICAgIHRoaXMudGlQZ0xhbWJkYUZ1bmN0aW9uID0gbmV3IGxhbWJkYS5GdW5jdGlvbih0aGlzLCBcImxhbWJkYVwiLCB7XG4gICAgICAgIC8vIGRlZmF1bHRzXG4gICAgICAgIHJ1bnRpbWU6IGxhbWJkYS5SdW50aW1lLlBZVEhPTl8zXzExLFxuICAgICAgICBoYW5kbGVyOiBcImhhbmRsZXIuaGFuZGxlclwiLFxuICAgICAgICBtZW1vcnlTaXplOiAxMDI0LFxuICAgICAgICBsb2dSZXRlbnRpb246IGxvZ3MuUmV0ZW50aW9uRGF5cy5PTkVfV0VFSyxcbiAgICAgICAgdGltZW91dDogRHVyYXRpb24uc2Vjb25kcygzMCksXG4gICAgICAgIGNvZGU6IGxhbWJkYS5Db2RlLmZyb21Eb2NrZXJCdWlsZChfX2Rpcm5hbWUsIHtcbiAgICAgICAgZmlsZTogXCJydW50aW1lL0RvY2tlcmZpbGVcIixcbiAgICAgICAgYnVpbGRBcmdzOiB7IFBZVEhPTl9WRVJTSU9OOiAnMy4xMScgfSxcbiAgICAgICAgfSksXG4gICAgICAgIHZwYzogcHJvcHMudnBjLFxuICAgICAgICB2cGNTdWJuZXRzOiBwcm9wcy5zdWJuZXRTZWxlY3Rpb24sXG4gICAgICAgIGFsbG93UHVibGljU3VibmV0OiB0cnVlLFxuICAgICAgICBlbnZpcm9ubWVudDoge1xuICAgICAgICAgIFBHU1RBQ19TRUNSRVRfQVJOOiBwcm9wcy5kYlNlY3JldC5zZWNyZXRBcm4sXG4gICAgICAgICAgREJfTUlOX0NPTk5fU0laRTogXCIxXCIsXG4gICAgICAgICAgREJfTUFYX0NPTk5fU0laRTogXCIxXCIsXG4gICAgICAgICAgLi4ucHJvcHMuYXBpRW52LFxuICAgICAgICB9LFxuICAgICAgICAvLyBvdmVyd3JpdGVzIGRlZmF1bHRzIHdpdGggdXNlci1wcm92aWRlZCBjb25maWd1cmFibGUgcHJvcGVydGllc1xuICAgICAgICAuLi5wcm9wcy5sYW1iZGFGdW5jdGlvbk9wdGlvbnNcbiAgICAgIH0pO1xuXG4gICAgICBwcm9wcy5kYlNlY3JldC5ncmFudFJlYWQodGhpcy50aVBnTGFtYmRhRnVuY3Rpb24pO1xuXG4gICAgICBpZiAocHJvcHMudnBjKXtcbiAgICAgICAgdGhpcy50aVBnTGFtYmRhRnVuY3Rpb24uY29ubmVjdGlvbnMuYWxsb3dUbyhwcm9wcy5kYiwgZWMyLlBvcnQudGNwKDU0MzIpLCBcImFsbG93IGNvbm5lY3Rpb25zIGZyb20gdGlwZ1wiKTtcbiAgICAgIH1cblxuICAgICAgY29uc3QgdGlwZ0FwaSA9IG5ldyBIdHRwQXBpKHRoaXMsIGAke1N0YWNrLm9mKHRoaXMpLnN0YWNrTmFtZX0tdGlwZy1hcGlgLCB7XG4gICAgICAgIGRlZmF1bHREb21haW5NYXBwaW5nOiBwcm9wcy50aXBnQXBpRG9tYWluTmFtZSA/IHtcbiAgICAgICAgICBkb21haW5OYW1lOiBwcm9wcy50aXBnQXBpRG9tYWluTmFtZVxuICAgICAgICB9IDogdW5kZWZpbmVkLFxuICAgICAgICBkZWZhdWx0SW50ZWdyYXRpb246IG5ldyBIdHRwTGFtYmRhSW50ZWdyYXRpb24oXG4gICAgICAgICAgXCJpbnRlZ3JhdGlvblwiLFxuICAgICAgICAgIHRoaXMudGlQZ0xhbWJkYUZ1bmN0aW9uLFxuICAgICAgICAgIHByb3BzLnRpcGdBcGlEb21haW5OYW1lID8ge1xuICAgICAgICAgICAgICBwYXJhbWV0ZXJNYXBwaW5nOiBuZXcgUGFyYW1ldGVyTWFwcGluZygpLm92ZXJ3cml0ZUhlYWRlcignaG9zdCcsIE1hcHBpbmdWYWx1ZS5jdXN0b20ocHJvcHMudGlwZ0FwaURvbWFpbk5hbWUubmFtZSkpXG4gICAgICAgICAgfSA6IHVuZGVmaW5lZFxuICAgICAgICApLFxuICAgICAgfSk7XG5cbiAgICAgIHRoaXMudXJsID0gdGlwZ0FwaS51cmwhO1xuXG4gICAgICBuZXcgQ2ZuT3V0cHV0KHRoaXMsIFwidGlwZy1hcGktb3V0cHV0XCIsIHtcbiAgICAgICAgZXhwb3J0TmFtZTogYCR7U3RhY2sub2YodGhpcykuc3RhY2tOYW1lfS10aXAtdXJsYCxcbiAgICAgICAgdmFsdWU6IHRoaXMudXJsLFxuICAgICAgfSk7XG4gICAgfVxuICB9XG5cbiAgZXhwb3J0IGludGVyZmFjZSBUaVBnQXBpTGFtYmRhUHJvcHMge1xuXG4gICAgLyoqXG4gICAgICogVlBDIGludG8gd2hpY2ggdGhlIGxhbWJkYSBzaG91bGQgYmUgZGVwbG95ZWQuXG4gICAgICovXG4gICAgcmVhZG9ubHkgdnBjPzogZWMyLklWcGM7XG5cbiAgICAvKipcbiAgICAgKiBSRFMgSW5zdGFuY2Ugd2l0aCBpbnN0YWxsZWQgcGdTVEFDIG9yIHBnYm91bmNlciBzZXJ2ZXIuXG4gICAgICovXG4gICAgcmVhZG9ubHkgZGI6IHJkcy5JRGF0YWJhc2VJbnN0YW5jZSB8IGVjMi5JSW5zdGFuY2U7XG5cbiAgICAvKipcbiAgICAgKiBTdWJuZXQgaW50byB3aGljaCB0aGUgbGFtYmRhIHNob3VsZCBiZSBkZXBsb3llZC5cbiAgICAgKi9cbiAgICByZWFkb25seSBzdWJuZXRTZWxlY3Rpb24/OiBlYzIuU3VibmV0U2VsZWN0aW9uO1xuXG4gICAgLyoqXG4gICAgICogU2VjcmV0IGNvbnRhaW5pbmcgY29ubmVjdGlvbiBpbmZvcm1hdGlvbiBmb3IgcGdTVEFDIGRhdGFiYXNlLlxuICAgICAqL1xuICAgIHJlYWRvbmx5IGRiU2VjcmV0OiBzZWNyZXRzbWFuYWdlci5JU2VjcmV0O1xuXG4gICAgLyoqXG4gICAgICogQ3VzdG9taXplZCBlbnZpcm9ubWVudCB2YXJpYWJsZXMgdG8gc2VuZCB0byB0aXRpbGVyLXBnc3RhYyBydW50aW1lLlxuICAgICAqL1xuICAgIHJlYWRvbmx5IGFwaUVudj86IFJlY29yZDxzdHJpbmcsIHN0cmluZz47XG5cbiAgICAvKipcbiAgICAgKiBDdXN0b20gRG9tYWluIE5hbWUgZm9yIHRpcGcgQVBJLiBJZiBkZWZpbmVkLCB3aWxsIGNyZWF0ZSB0aGVcbiAgICAgKiBkb21haW4gbmFtZSBhbmQgaW50ZWdyYXRlIGl0IHdpdGggdGhlIHRpcGcgQVBJLlxuICAgICAqXG4gICAgICogQGRlZmF1bHQgLSB1bmRlZmluZWRcbiAgICAgKi9cbiAgICByZWFkb25seSB0aXBnQXBpRG9tYWluTmFtZT86IElEb21haW5OYW1lO1xuXG5cbiAgICAvKipcbiAgICAgKiBDYW4gYmUgdXNlZCB0byBvdmVycmlkZSB0aGUgZGVmYXVsdCBsYW1iZGEgZnVuY3Rpb24gcHJvcGVydGllcy5cbiAgICAgKlxuICAgICAqIEBkZWZhdWx0IC0gZGVmaW5lZCBpbiB0aGUgY29uc3RydWN0LlxuICAgICAqL1xuICAgIHJlYWRvbmx5IGxhbWJkYUZ1bmN0aW9uT3B0aW9ucz86IEN1c3RvbUxhbWJkYUZ1bmN0aW9uUHJvcHM7XG4gIH1cbiJdfQ==
58
+ TiPgApiLambda[_a] = { fqn: "eoapi-cdk.TiPgApiLambda", version: "8.0.0" };
59
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJpbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7OztBQUFBLDZDQVNxQjtBQUNyQiw0RUFBc0c7QUFDdEcsc0dBQXFGO0FBQ3JGLDJDQUF1QztBQUV2Qyw2QkFBNkI7QUFFM0IsTUFBYSxhQUFjLFNBQVEsc0JBQVM7SUFJMUMsWUFBWSxLQUFnQixFQUFFLEVBQVUsRUFBRSxLQUF5QjtRQUNqRSxLQUFLLENBQUMsS0FBSyxFQUFFLEVBQUUsQ0FBQyxDQUFDO1FBRWpCLElBQUksQ0FBQyxrQkFBa0IsR0FBRyxJQUFJLHdCQUFNLENBQUMsUUFBUSxDQUFDLElBQUksRUFBRSxRQUFRLEVBQUU7WUFDNUQsV0FBVztZQUNYLE9BQU8sRUFBRSx3QkFBTSxDQUFDLE9BQU8sQ0FBQyxXQUFXO1lBQ25DLE9BQU8sRUFBRSxpQkFBaUI7WUFDMUIsVUFBVSxFQUFFLElBQUk7WUFDaEIsWUFBWSxFQUFFLHNCQUFJLENBQUMsYUFBYSxDQUFDLFFBQVE7WUFDekMsT0FBTyxFQUFFLHNCQUFRLENBQUMsT0FBTyxDQUFDLEVBQUUsQ0FBQztZQUM3QixJQUFJLEVBQUUsd0JBQU0sQ0FBQyxJQUFJLENBQUMsZUFBZSxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsU0FBUyxFQUFFLElBQUksQ0FBQyxFQUFFO2dCQUM1RCxJQUFJLEVBQUUsNkJBQTZCO2dCQUNuQyxTQUFTLEVBQUUsRUFBRSxjQUFjLEVBQUUsTUFBTSxFQUFFO2FBQ3RDLENBQUM7WUFDRixHQUFHLEVBQUUsS0FBSyxDQUFDLEdBQUc7WUFDZCxVQUFVLEVBQUUsS0FBSyxDQUFDLGVBQWU7WUFDakMsaUJBQWlCLEVBQUUsSUFBSTtZQUN2QixXQUFXLEVBQUU7Z0JBQ1gsaUJBQWlCLEVBQUUsS0FBSyxDQUFDLFFBQVEsQ0FBQyxTQUFTO2dCQUMzQyxnQkFBZ0IsRUFBRSxHQUFHO2dCQUNyQixnQkFBZ0IsRUFBRSxHQUFHO2dCQUNyQixHQUFHLEtBQUssQ0FBQyxNQUFNO2FBQ2hCO1lBQ0QsaUVBQWlFO1lBQ2pFLEdBQUcsS0FBSyxDQUFDLHFCQUFxQjtTQUMvQixDQUFDLENBQUM7UUFFSCxLQUFLLENBQUMsUUFBUSxDQUFDLFNBQVMsQ0FBQyxJQUFJLENBQUMsa0JBQWtCLENBQUMsQ0FBQztRQUVsRCxJQUFJLEtBQUssQ0FBQyxHQUFHLEVBQUMsQ0FBQztZQUNiLElBQUksQ0FBQyxrQkFBa0IsQ0FBQyxXQUFXLENBQUMsT0FBTyxDQUFDLEtBQUssQ0FBQyxFQUFFLEVBQUUscUJBQUcsQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFDLElBQUksQ0FBQyxFQUFFLDZCQUE2QixDQUFDLENBQUM7UUFDM0csQ0FBQztRQUVELE1BQU0sT0FBTyxHQUFHLElBQUksZ0NBQU8sQ0FBQyxJQUFJLEVBQUUsR0FBRyxtQkFBSyxDQUFDLEVBQUUsQ0FBQyxJQUFJLENBQUMsQ0FBQyxTQUFTLFdBQVcsRUFBRTtZQUN4RSxvQkFBb0IsRUFBRSxLQUFLLENBQUMsaUJBQWlCLENBQUMsQ0FBQyxDQUFDO2dCQUM5QyxVQUFVLEVBQUUsS0FBSyxDQUFDLGlCQUFpQjthQUNwQyxDQUFDLENBQUMsQ0FBQyxTQUFTO1lBQ2Isa0JBQWtCLEVBQUUsSUFBSSwyREFBcUIsQ0FDM0MsYUFBYSxFQUNiLElBQUksQ0FBQyxrQkFBa0IsRUFDdkIsS0FBSyxDQUFDLGlCQUFpQixDQUFDLENBQUMsQ0FBQztnQkFDdEIsZ0JBQWdCLEVBQUUsSUFBSSx5Q0FBZ0IsRUFBRSxDQUFDLGVBQWUsQ0FBQyxNQUFNLEVBQUUscUNBQVksQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFDLGlCQUFpQixDQUFDLElBQUksQ0FBQyxDQUFDO2FBQ3RILENBQUMsQ0FBQyxDQUFDLFNBQVMsQ0FDZDtTQUNGLENBQUMsQ0FBQztRQUVILElBQUksQ0FBQyxHQUFHLEdBQUcsT0FBTyxDQUFDLEdBQUksQ0FBQztRQUV4QixJQUFJLHVCQUFTLENBQUMsSUFBSSxFQUFFLGlCQUFpQixFQUFFO1lBQ3JDLFVBQVUsRUFBRSxHQUFHLG1CQUFLLENBQUMsRUFBRSxDQUFDLElBQUksQ0FBQyxDQUFDLFNBQVMsVUFBVTtZQUNqRCxLQUFLLEVBQUUsSUFBSSxDQUFDLEdBQUc7U0FDaEIsQ0FBQyxDQUFDO0lBQ0wsQ0FBQzs7QUF4REgsc0NBeURDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtcbiAgU3RhY2ssXG4gIGF3c19lYzIgYXMgZWMyLFxuICBhd3NfbGFtYmRhIGFzIGxhbWJkYSxcbiAgYXdzX2xvZ3MgYXMgbG9ncyxcbiAgYXdzX3JkcyBhcyByZHMsXG4gIGF3c19zZWNyZXRzbWFuYWdlciBhcyBzZWNyZXRzbWFuYWdlcixcbiAgQ2ZuT3V0cHV0LFxuICBEdXJhdGlvbixcbn0gZnJvbSBcImF3cy1jZGstbGliXCI7XG5pbXBvcnQgeyBJRG9tYWluTmFtZSwgSHR0cEFwaSwgUGFyYW1ldGVyTWFwcGluZywgTWFwcGluZ1ZhbHVlfSBmcm9tIFwiQGF3cy1jZGsvYXdzLWFwaWdhdGV3YXl2Mi1hbHBoYVwiO1xuaW1wb3J0IHsgSHR0cExhbWJkYUludGVncmF0aW9uIH0gZnJvbSBcIkBhd3MtY2RrL2F3cy1hcGlnYXRld2F5djItaW50ZWdyYXRpb25zLWFscGhhXCI7XG5pbXBvcnQgeyBDb25zdHJ1Y3QgfSBmcm9tIFwiY29uc3RydWN0c1wiO1xuaW1wb3J0IHsgQ3VzdG9tTGFtYmRhRnVuY3Rpb25Qcm9wcyB9IGZyb20gXCIuLi91dGlsc1wiO1xuaW1wb3J0ICogYXMgcGF0aCBmcm9tICdwYXRoJztcblxuICBleHBvcnQgY2xhc3MgVGlQZ0FwaUxhbWJkYSBleHRlbmRzIENvbnN0cnVjdCB7XG4gICAgcmVhZG9ubHkgdXJsOiBzdHJpbmc7XG4gICAgcHVibGljIHRpUGdMYW1iZGFGdW5jdGlvbjogbGFtYmRhLkZ1bmN0aW9uO1xuXG4gICAgY29uc3RydWN0b3Ioc2NvcGU6IENvbnN0cnVjdCwgaWQ6IHN0cmluZywgcHJvcHM6IFRpUGdBcGlMYW1iZGFQcm9wcykge1xuICAgICAgc3VwZXIoc2NvcGUsIGlkKTtcblxuICAgICAgdGhpcy50aVBnTGFtYmRhRnVuY3Rpb24gPSBuZXcgbGFtYmRhLkZ1bmN0aW9uKHRoaXMsIFwibGFtYmRhXCIsIHtcbiAgICAgICAgLy8gZGVmYXVsdHNcbiAgICAgICAgcnVudGltZTogbGFtYmRhLlJ1bnRpbWUuUFlUSE9OXzNfMTEsXG4gICAgICAgIGhhbmRsZXI6IFwiaGFuZGxlci5oYW5kbGVyXCIsXG4gICAgICAgIG1lbW9yeVNpemU6IDEwMjQsXG4gICAgICAgIGxvZ1JldGVudGlvbjogbG9ncy5SZXRlbnRpb25EYXlzLk9ORV9XRUVLLFxuICAgICAgICB0aW1lb3V0OiBEdXJhdGlvbi5zZWNvbmRzKDMwKSxcbiAgICAgICAgY29kZTogbGFtYmRhLkNvZGUuZnJvbURvY2tlckJ1aWxkKHBhdGguam9pbihfX2Rpcm5hbWUsICcuLicpLCB7XG4gICAgICAgICAgZmlsZTogXCJ0aXBnLWFwaS9ydW50aW1lL0RvY2tlcmZpbGVcIixcbiAgICAgICAgICBidWlsZEFyZ3M6IHsgUFlUSE9OX1ZFUlNJT046ICczLjExJyB9LFxuICAgICAgICB9KSxcbiAgICAgICAgdnBjOiBwcm9wcy52cGMsXG4gICAgICAgIHZwY1N1Ym5ldHM6IHByb3BzLnN1Ym5ldFNlbGVjdGlvbixcbiAgICAgICAgYWxsb3dQdWJsaWNTdWJuZXQ6IHRydWUsXG4gICAgICAgIGVudmlyb25tZW50OiB7XG4gICAgICAgICAgUEdTVEFDX1NFQ1JFVF9BUk46IHByb3BzLmRiU2VjcmV0LnNlY3JldEFybixcbiAgICAgICAgICBEQl9NSU5fQ09OTl9TSVpFOiBcIjFcIixcbiAgICAgICAgICBEQl9NQVhfQ09OTl9TSVpFOiBcIjFcIixcbiAgICAgICAgICAuLi5wcm9wcy5hcGlFbnYsXG4gICAgICAgIH0sXG4gICAgICAgIC8vIG92ZXJ3cml0ZXMgZGVmYXVsdHMgd2l0aCB1c2VyLXByb3ZpZGVkIGNvbmZpZ3VyYWJsZSBwcm9wZXJ0aWVzXG4gICAgICAgIC4uLnByb3BzLmxhbWJkYUZ1bmN0aW9uT3B0aW9uc1xuICAgICAgfSk7XG5cbiAgICAgIHByb3BzLmRiU2VjcmV0LmdyYW50UmVhZCh0aGlzLnRpUGdMYW1iZGFGdW5jdGlvbik7XG5cbiAgICAgIGlmIChwcm9wcy52cGMpe1xuICAgICAgICB0aGlzLnRpUGdMYW1iZGFGdW5jdGlvbi5jb25uZWN0aW9ucy5hbGxvd1RvKHByb3BzLmRiLCBlYzIuUG9ydC50Y3AoNTQzMiksIFwiYWxsb3cgY29ubmVjdGlvbnMgZnJvbSB0aXBnXCIpO1xuICAgICAgfVxuXG4gICAgICBjb25zdCB0aXBnQXBpID0gbmV3IEh0dHBBcGkodGhpcywgYCR7U3RhY2sub2YodGhpcykuc3RhY2tOYW1lfS10aXBnLWFwaWAsIHtcbiAgICAgICAgZGVmYXVsdERvbWFpbk1hcHBpbmc6IHByb3BzLnRpcGdBcGlEb21haW5OYW1lID8ge1xuICAgICAgICAgIGRvbWFpbk5hbWU6IHByb3BzLnRpcGdBcGlEb21haW5OYW1lXG4gICAgICAgIH0gOiB1bmRlZmluZWQsXG4gICAgICAgIGRlZmF1bHRJbnRlZ3JhdGlvbjogbmV3IEh0dHBMYW1iZGFJbnRlZ3JhdGlvbihcbiAgICAgICAgICBcImludGVncmF0aW9uXCIsXG4gICAgICAgICAgdGhpcy50aVBnTGFtYmRhRnVuY3Rpb24sXG4gICAgICAgICAgcHJvcHMudGlwZ0FwaURvbWFpbk5hbWUgPyB7XG4gICAgICAgICAgICAgIHBhcmFtZXRlck1hcHBpbmc6IG5ldyBQYXJhbWV0ZXJNYXBwaW5nKCkub3ZlcndyaXRlSGVhZGVyKCdob3N0JywgTWFwcGluZ1ZhbHVlLmN1c3RvbShwcm9wcy50aXBnQXBpRG9tYWluTmFtZS5uYW1lKSlcbiAgICAgICAgICB9IDogdW5kZWZpbmVkXG4gICAgICAgICksXG4gICAgICB9KTtcblxuICAgICAgdGhpcy51cmwgPSB0aXBnQXBpLnVybCE7XG5cbiAgICAgIG5ldyBDZm5PdXRwdXQodGhpcywgXCJ0aXBnLWFwaS1vdXRwdXRcIiwge1xuICAgICAgICBleHBvcnROYW1lOiBgJHtTdGFjay5vZih0aGlzKS5zdGFja05hbWV9LXRpcC11cmxgLFxuICAgICAgICB2YWx1ZTogdGhpcy51cmwsXG4gICAgICB9KTtcbiAgICB9XG4gIH1cblxuICBleHBvcnQgaW50ZXJmYWNlIFRpUGdBcGlMYW1iZGFQcm9wcyB7XG5cbiAgICAvKipcbiAgICAgKiBWUEMgaW50byB3aGljaCB0aGUgbGFtYmRhIHNob3VsZCBiZSBkZXBsb3llZC5cbiAgICAgKi9cbiAgICByZWFkb25seSB2cGM/OiBlYzIuSVZwYztcblxuICAgIC8qKlxuICAgICAqIFJEUyBJbnN0YW5jZSB3aXRoIGluc3RhbGxlZCBwZ1NUQUMgb3IgcGdib3VuY2VyIHNlcnZlci5cbiAgICAgKi9cbiAgICByZWFkb25seSBkYjogcmRzLklEYXRhYmFzZUluc3RhbmNlIHwgZWMyLklJbnN0YW5jZTtcblxuICAgIC8qKlxuICAgICAqIFN1Ym5ldCBpbnRvIHdoaWNoIHRoZSBsYW1iZGEgc2hvdWxkIGJlIGRlcGxveWVkLlxuICAgICAqL1xuICAgIHJlYWRvbmx5IHN1Ym5ldFNlbGVjdGlvbj86IGVjMi5TdWJuZXRTZWxlY3Rpb247XG5cbiAgICAvKipcbiAgICAgKiBTZWNyZXQgY29udGFpbmluZyBjb25uZWN0aW9uIGluZm9ybWF0aW9uIGZvciBwZ1NUQUMgZGF0YWJhc2UuXG4gICAgICovXG4gICAgcmVhZG9ubHkgZGJTZWNyZXQ6IHNlY3JldHNtYW5hZ2VyLklTZWNyZXQ7XG5cbiAgICAvKipcbiAgICAgKiBDdXN0b21pemVkIGVudmlyb25tZW50IHZhcmlhYmxlcyB0byBzZW5kIHRvIHRpdGlsZXItcGdzdGFjIHJ1bnRpbWUuXG4gICAgICovXG4gICAgcmVhZG9ubHkgYXBpRW52PzogUmVjb3JkPHN0cmluZywgc3RyaW5nPjtcblxuICAgIC8qKlxuICAgICAqIEN1c3RvbSBEb21haW4gTmFtZSBmb3IgdGlwZyBBUEkuIElmIGRlZmluZWQsIHdpbGwgY3JlYXRlIHRoZVxuICAgICAqIGRvbWFpbiBuYW1lIGFuZCBpbnRlZ3JhdGUgaXQgd2l0aCB0aGUgdGlwZyBBUEkuXG4gICAgICpcbiAgICAgKiBAZGVmYXVsdCAtIHVuZGVmaW5lZFxuICAgICAqL1xuICAgIHJlYWRvbmx5IHRpcGdBcGlEb21haW5OYW1lPzogSURvbWFpbk5hbWU7XG5cblxuICAgIC8qKlxuICAgICAqIENhbiBiZSB1c2VkIHRvIG92ZXJyaWRlIHRoZSBkZWZhdWx0IGxhbWJkYSBmdW5jdGlvbiBwcm9wZXJ0aWVzLlxuICAgICAqXG4gICAgICogQGRlZmF1bHQgLSBkZWZpbmVkIGluIHRoZSBjb25zdHJ1Y3QuXG4gICAgICovXG4gICAgcmVhZG9ubHkgbGFtYmRhRnVuY3Rpb25PcHRpb25zPzogQ3VzdG9tTGFtYmRhRnVuY3Rpb25Qcm9wcztcbiAgfVxuIl19
@@ -4,7 +4,7 @@ FROM --platform=linux/amd64 public.ecr.aws/lambda/python:${PYTHON_VERSION}
4
4
  WORKDIR /tmp
5
5
  RUN python -m pip install pip -U
6
6
 
7
- COPY runtime/requirements.txt requirements.txt
7
+ COPY tipg-api/runtime/requirements.txt requirements.txt
8
8
  RUN python -m pip install -r requirements.txt "mangum>=0.14,<0.15" -t /asset --no-binary pydantic
9
9
 
10
10
  # Reduce package size and remove useless files
@@ -13,6 +13,7 @@ RUN cd /asset && find . -type d -a -name '__pycache__' -print0 | xargs -0 rm -rf
13
13
  RUN cd /asset && find . -type f -a -name '*.py' -print0 | xargs -0 rm -f
14
14
  RUN find /asset -type d -a -name 'tests' -print0 | xargs -0 rm -rf
15
15
 
16
- COPY runtime/src/*.py /asset/
16
+ COPY tipg-api/runtime/src/*.py /asset/
17
+ COPY utils/utils.py /asset/
17
18
 
18
19
  CMD ["echo", "hello world"]
@@ -1 +1 @@
1
- tipg==0.6.3
1
+ tipg>=1.0,<1.1
@@ -6,9 +6,18 @@ import asyncio
6
6
  import os
7
7
 
8
8
  from mangum import Mangum
9
- from utils import load_pgstac_secret
9
+ from utils import get_secret_dict
10
10
 
11
- load_pgstac_secret(os.environ["PGSTAC_SECRET_ARN"]) # required for the below imports
11
+ secret = get_secret_dict(secret_arn_env_var="PGSTAC_SECRET_ARN")
12
+ os.environ.update(
13
+ {
14
+ "postgres_host": secret["host"],
15
+ "postgres_dbname": secret["dbname"],
16
+ "postgres_user": secret["username"],
17
+ "postgres_pass": secret["password"],
18
+ "postgres_port": str(secret["port"]),
19
+ }
20
+ )
12
21
 
13
22
  from tipg.collections import register_collection_catalog # noqa: E402
14
23
  from tipg.database import connect_to_db # noqa: E402
@@ -7,6 +7,7 @@ const aws_cdk_lib_1 = require("aws-cdk-lib");
7
7
  const aws_apigatewayv2_alpha_1 = require("@aws-cdk/aws-apigatewayv2-alpha");
8
8
  const aws_apigatewayv2_integrations_alpha_1 = require("@aws-cdk/aws-apigatewayv2-integrations-alpha");
9
9
  const constructs_1 = require("constructs");
10
+ const path = require("path");
10
11
  // default settings that can be overridden by the user-provided environment.
11
12
  let defaultTitilerPgstacEnv = {
12
13
  "CPL_VSIL_CURL_ALLOWED_EXTENSIONS": ".tif,.TIF,.tiff",
@@ -32,8 +33,8 @@ class TitilerPgstacApiLambda extends constructs_1.Construct {
32
33
  memorySize: 3008,
33
34
  logRetention: aws_cdk_lib_1.aws_logs.RetentionDays.ONE_WEEK,
34
35
  timeout: aws_cdk_lib_1.Duration.seconds(30),
35
- code: aws_cdk_lib_1.aws_lambda.Code.fromDockerBuild(__dirname, {
36
- file: "runtime/Dockerfile",
36
+ code: aws_cdk_lib_1.aws_lambda.Code.fromDockerBuild(path.join(__dirname, '..'), {
37
+ file: "titiler-pgstac-api/runtime/Dockerfile",
37
38
  buildArgs: { PYTHON_VERSION: '3.11' }
38
39
  }),
39
40
  vpc: props.vpc,
@@ -74,5 +75,5 @@ class TitilerPgstacApiLambda extends constructs_1.Construct {
74
75
  }
75
76
  exports.TitilerPgstacApiLambda = TitilerPgstacApiLambda;
76
77
  _a = JSII_RTTI_SYMBOL_1;
77
- TitilerPgstacApiLambda[_a] = { fqn: "eoapi-cdk.TitilerPgstacApiLambda", version: "7.6.2" };
78
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["index.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n    Stack,\n    aws_iam as iam,\n    aws_ec2 as ec2,\n    aws_rds as rds,\n    aws_lambda as lambda,\n    aws_secretsmanager as secretsmanager,\n    CfnOutput,\n    Duration,\n    aws_logs\n  } from \"aws-cdk-lib\";\n  import { IDomainName, HttpApi, ParameterMapping, MappingValue} from \"@aws-cdk/aws-apigatewayv2-alpha\";\n  import { HttpLambdaIntegration } from \"@aws-cdk/aws-apigatewayv2-integrations-alpha\";\n  import { Construct } from \"constructs\";\nimport { CustomLambdaFunctionProps } from \"../utils\";\n\n\n  // default settings that can be overridden by the user-provided environment.\n  let defaultTitilerPgstacEnv :{ [key: string]: any } = {\n    \"CPL_VSIL_CURL_ALLOWED_EXTENSIONS\": \".tif,.TIF,.tiff\",\n    \"GDAL_CACHEMAX\": \"200\",\n    \"GDAL_DISABLE_READDIR_ON_OPEN\": \"EMPTY_DIR\",\n    \"GDAL_INGESTED_BYTES_AT_OPEN\": \"32768\",\n    \"GDAL_HTTP_MERGE_CONSECUTIVE_RANGES\": \"YES\",\n    \"GDAL_HTTP_MULTIPLEX\": \"YES\",\n    \"GDAL_HTTP_VERSION\": \"2\",\n    \"PYTHONWARNINGS\": \"ignore\",\n    \"VSI_CACHE\": \"TRUE\",\n    \"VSI_CACHE_SIZE\": \"5000000\",\n    \"DB_MIN_CONN_SIZE\": \"1\",\n    \"DB_MAX_CONN_SIZE\": \"1\"\n  }\n\n  export class TitilerPgstacApiLambda extends Construct {\n    readonly url: string;\n    public titilerPgstacLambdaFunction: lambda.Function;\n\n    constructor(scope: Construct, id: string, props: TitilerPgStacApiLambdaProps) {\n      super(scope, id);\n\n      this.titilerPgstacLambdaFunction = new lambda.Function(this, \"lambda\", {\n        // defaults\n        runtime: lambda.Runtime.PYTHON_3_11,\n        handler: \"handler.handler\",\n        memorySize: 3008,\n        logRetention: aws_logs.RetentionDays.ONE_WEEK,\n        timeout: Duration.seconds(30),\n        code: lambda.Code.fromDockerBuild(__dirname, {\n          file: \"runtime/Dockerfile\",\n          buildArgs: { PYTHON_VERSION: '3.11' }\n        }),\n        vpc: props.vpc,\n        vpcSubnets: props.subnetSelection,\n        allowPublicSubnet: true,\n        // if user provided environment variables, merge them with the defaults.\n        environment: props.apiEnv ? { ...defaultTitilerPgstacEnv, ...props.apiEnv, \"PGSTAC_SECRET_ARN\": props.dbSecret.secretArn } : defaultTitilerPgstacEnv,\n        // overwrites defaults with user-provided configurable properties\n        ...props.lambdaFunctionOptions,\n      });\n\n      // grant access to buckets using addToRolePolicy\n      if (props.buckets) {\n        props.buckets.forEach(bucket => {\n          this.titilerPgstacLambdaFunction.addToRolePolicy(new iam.PolicyStatement({\n            actions: [\"s3:GetObject\"],\n            resources: [`arn:aws:s3:::${bucket}/*`],\n          }));\n        });\n      }\n\n      props.dbSecret.grantRead(this.titilerPgstacLambdaFunction);\n\n      if (props.vpc) {\n        this.titilerPgstacLambdaFunction.connections.allowTo(props.db, ec2.Port.tcp(5432), \"allow connections from titiler\");\n      }\n\n      const stacApi = new HttpApi(this, `${Stack.of(this).stackName}-titiler-pgstac-api`, {\n        defaultDomainMapping: props.titilerPgstacApiDomainName ? {\n          domainName: props.titilerPgstacApiDomainName\n        } : undefined,\n        defaultIntegration: new HttpLambdaIntegration(\n          \"integration\",\n          this.titilerPgstacLambdaFunction,\n          props.titilerPgstacApiDomainName ? {\n              parameterMapping: new ParameterMapping().overwriteHeader('host', MappingValue.custom(props.titilerPgstacApiDomainName.name))\n          } : undefined\n        ),\n      });\n\n      this.url = stacApi.url!;\n\n      new CfnOutput(this, \"titiler-pgstac-api-output\", {\n        exportName: `${Stack.of(this).stackName}-titiler-pgstac-url`,\n        value: this.url,\n      });\n    }\n  }\n\n  export interface TitilerPgStacApiLambdaProps {\n\n    /**\n     * VPC into which the lambda should be deployed.\n     */\n    readonly vpc?: ec2.IVpc;\n\n    /**\n     * RDS Instance with installed pgSTAC or pgbouncer server.\n     */\n    readonly db: rds.IDatabaseInstance | ec2.IInstance;\n\n    /**\n     * Subnet into which the lambda should be deployed.\n     */\n    readonly subnetSelection?: ec2.SubnetSelection;\n\n    /**\n     * Secret containing connection information for pgSTAC database.\n     */\n    readonly dbSecret: secretsmanager.ISecret;\n\n    /**\n     * Customized environment variables to send to titiler-pgstac runtime. These will be merged with `defaultTitilerPgstacEnv`.\n     * The database secret arn is automatically added to the environment variables at deployment.\n    /*/\n    readonly apiEnv?: Record<string, string>;\n\n    /**\n     * list of buckets the lambda will be granted access to.\n     */\n    readonly buckets?: string[];\n\n    /**\n     * Custom Domain Name Options for Titiler Pgstac API,\n     *\n     * @default - undefined.\n     */\n    readonly titilerPgstacApiDomainName?: IDomainName;\n\n    /**\n     * Can be used to override the default lambda function properties.\n     *\n     * @default - defined in the construct.\n     */\n    readonly lambdaFunctionOptions?: CustomLambdaFunctionProps;\n\n  }\n"]}
78
+ TitilerPgstacApiLambda[_a] = { fqn: "eoapi-cdk.TitilerPgstacApiLambda", version: "8.0.0" };
79
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["index.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Stack,\n  aws_iam as iam,\n  aws_ec2 as ec2,\n  aws_rds as rds,\n  aws_lambda as lambda,\n  aws_secretsmanager as secretsmanager,\n  CfnOutput,\n  Duration,\n  aws_logs\n} from \"aws-cdk-lib\";\nimport { IDomainName, HttpApi, ParameterMapping, MappingValue} from \"@aws-cdk/aws-apigatewayv2-alpha\";\nimport { HttpLambdaIntegration } from \"@aws-cdk/aws-apigatewayv2-integrations-alpha\";\nimport { Construct } from \"constructs\";\nimport { CustomLambdaFunctionProps } from \"../utils\";\nimport * as path from 'path';\n\n  // default settings that can be overridden by the user-provided environment.\n  let defaultTitilerPgstacEnv :{ [key: string]: any } = {\n    \"CPL_VSIL_CURL_ALLOWED_EXTENSIONS\": \".tif,.TIF,.tiff\",\n    \"GDAL_CACHEMAX\": \"200\",\n    \"GDAL_DISABLE_READDIR_ON_OPEN\": \"EMPTY_DIR\",\n    \"GDAL_INGESTED_BYTES_AT_OPEN\": \"32768\",\n    \"GDAL_HTTP_MERGE_CONSECUTIVE_RANGES\": \"YES\",\n    \"GDAL_HTTP_MULTIPLEX\": \"YES\",\n    \"GDAL_HTTP_VERSION\": \"2\",\n    \"PYTHONWARNINGS\": \"ignore\",\n    \"VSI_CACHE\": \"TRUE\",\n    \"VSI_CACHE_SIZE\": \"5000000\",\n    \"DB_MIN_CONN_SIZE\": \"1\",\n    \"DB_MAX_CONN_SIZE\": \"1\"\n  }\n\n  export class TitilerPgstacApiLambda extends Construct {\n    readonly url: string;\n    public titilerPgstacLambdaFunction: lambda.Function;\n\n    constructor(scope: Construct, id: string, props: TitilerPgStacApiLambdaProps) {\n      super(scope, id);\n\n      this.titilerPgstacLambdaFunction = new lambda.Function(this, \"lambda\", {\n        // defaults\n        runtime: lambda.Runtime.PYTHON_3_11,\n        handler: \"handler.handler\",\n        memorySize: 3008,\n        logRetention: aws_logs.RetentionDays.ONE_WEEK,\n        timeout: Duration.seconds(30),\n        code: lambda.Code.fromDockerBuild(path.join(__dirname, '..'), {\n          file: \"titiler-pgstac-api/runtime/Dockerfile\",\n          buildArgs: { PYTHON_VERSION: '3.11' }\n        }),\n        vpc: props.vpc,\n        vpcSubnets: props.subnetSelection,\n        allowPublicSubnet: true,\n        // if user provided environment variables, merge them with the defaults.\n        environment: props.apiEnv ? { ...defaultTitilerPgstacEnv, ...props.apiEnv, \"PGSTAC_SECRET_ARN\": props.dbSecret.secretArn } : defaultTitilerPgstacEnv,\n        // overwrites defaults with user-provided configurable properties\n        ...props.lambdaFunctionOptions,\n      });\n\n      // grant access to buckets using addToRolePolicy\n      if (props.buckets) {\n        props.buckets.forEach(bucket => {\n          this.titilerPgstacLambdaFunction.addToRolePolicy(new iam.PolicyStatement({\n            actions: [\"s3:GetObject\"],\n            resources: [`arn:aws:s3:::${bucket}/*`],\n          }));\n        });\n      }\n\n      props.dbSecret.grantRead(this.titilerPgstacLambdaFunction);\n\n      if (props.vpc) {\n        this.titilerPgstacLambdaFunction.connections.allowTo(props.db, ec2.Port.tcp(5432), \"allow connections from titiler\");\n      }\n\n      const stacApi = new HttpApi(this, `${Stack.of(this).stackName}-titiler-pgstac-api`, {\n        defaultDomainMapping: props.titilerPgstacApiDomainName ? {\n          domainName: props.titilerPgstacApiDomainName\n        } : undefined,\n        defaultIntegration: new HttpLambdaIntegration(\n          \"integration\",\n          this.titilerPgstacLambdaFunction,\n          props.titilerPgstacApiDomainName ? {\n              parameterMapping: new ParameterMapping().overwriteHeader('host', MappingValue.custom(props.titilerPgstacApiDomainName.name))\n          } : undefined\n        ),\n      });\n\n      this.url = stacApi.url!;\n\n      new CfnOutput(this, \"titiler-pgstac-api-output\", {\n        exportName: `${Stack.of(this).stackName}-titiler-pgstac-url`,\n        value: this.url,\n      });\n    }\n  }\n\n  export interface TitilerPgStacApiLambdaProps {\n\n    /**\n     * VPC into which the lambda should be deployed.\n     */\n    readonly vpc?: ec2.IVpc;\n\n    /**\n     * RDS Instance with installed pgSTAC or pgbouncer server.\n     */\n    readonly db: rds.IDatabaseInstance | ec2.IInstance;\n\n    /**\n     * Subnet into which the lambda should be deployed.\n     */\n    readonly subnetSelection?: ec2.SubnetSelection;\n\n    /**\n     * Secret containing connection information for pgSTAC database.\n     */\n    readonly dbSecret: secretsmanager.ISecret;\n\n    /**\n     * Customized environment variables to send to titiler-pgstac runtime. These will be merged with `defaultTitilerPgstacEnv`.\n     * The database secret arn is automatically added to the environment variables at deployment.\n    /*/\n    readonly apiEnv?: Record<string, string>;\n\n    /**\n     * list of buckets the lambda will be granted access to.\n     */\n    readonly buckets?: string[];\n\n    /**\n     * Custom Domain Name Options for Titiler Pgstac API,\n     *\n     * @default - undefined.\n     */\n    readonly titilerPgstacApiDomainName?: IDomainName;\n\n    /**\n     * Can be used to override the default lambda function properties.\n     *\n     * @default - defined in the construct.\n     */\n    readonly lambdaFunctionOptions?: CustomLambdaFunctionProps;\n\n  }\n"]}
@@ -7,7 +7,7 @@ RUN python -m pip install pip -U
7
7
  # Install system dependencies to compile (numexpr)
8
8
  RUN yum install -y gcc-c++
9
9
 
10
- COPY runtime/requirements.txt requirements.txt
10
+ COPY titiler-pgstac-api/runtime/requirements.txt requirements.txt
11
11
  RUN python -m pip install -r requirements.txt "mangum>=0.14,<0.15" -t /asset
12
12
 
13
13
  # Remove system dependencies
@@ -20,6 +20,13 @@ RUN find /asset -type f -name '*.py' -print0 | xargs -0 rm -f
20
20
  RUN find /asset -type d -name 'tests' -print0 | xargs -0 rm -rf
21
21
  RUN rm -rdf /asset/numpy/doc/ /asset/boto3* /asset/botocore* /asset/bin /asset/geos_license /asset/Misc
22
22
 
23
- COPY runtime/src/*.py /asset/
23
+ # Strip debug symbols from compiled C/C++ code (except for numpy.libs!)
24
+ RUN cd /asset && \
25
+ find . -type f -name '*.so*' \
26
+ -not -path "./numpy.libs/*" \
27
+ -exec strip --strip-unneeded {} \;
28
+
29
+ COPY titiler-pgstac-api/runtime/src/*.py /asset/
30
+ COPY utils/utils.py /asset/
24
31
 
25
32
  CMD ["echo", "hello world"]
@@ -1,2 +1,2 @@
1
- titiler.pgstac==1.2.2
1
+ titiler.pgstac>=1.7,<1.8
2
2
  psycopg[binary, pool]
@@ -8,9 +8,7 @@ import os
8
8
  from mangum import Mangum
9
9
  from utils import get_secret_dict
10
10
 
11
- pgstac_secret_arn = os.environ["PGSTAC_SECRET_ARN"]
12
-
13
- secret = get_secret_dict(pgstac_secret_arn)
11
+ secret = get_secret_dict(secret_arn_env_var="PGSTAC_SECRET_ARN")
14
12
  os.environ.update(
15
13
  {
16
14
  "postgres_host": secret["host"],
@@ -1,25 +1,28 @@
1
1
  import base64
2
2
  import json
3
+ import os
3
4
 
4
5
  import boto3
5
6
 
6
7
 
7
- def get_secret_dict(secret_name: str):
8
+ def get_secret_dict(secret_arn_env_var: str):
8
9
  """Retrieve secrets from AWS Secrets Manager
9
10
 
10
11
  Args:
11
- secret_name (str): name of aws secrets manager secret containing database connection secrets
12
- profile_name (str, optional): optional name of aws profile for use in debugger only
12
+ secret_arn_env_var (str): environment variable that contains the secret ARN
13
13
 
14
14
  Returns:
15
15
  secrets (dict): decrypted secrets in dict
16
16
  """
17
+ secret_arn = os.environ.get(secret_arn_env_var)
18
+ if not secret_arn:
19
+ raise ValueError(f"{secret_arn_env_var} is not set!")
17
20
 
18
21
  # Create a Secrets Manager client
19
22
  session = boto3.session.Session()
20
23
  client = session.client(service_name="secretsmanager")
21
24
 
22
- get_secret_value_response = client.get_secret_value(SecretId=secret_name)
25
+ get_secret_value_response = client.get_secret_value(SecretId=secret_arn)
23
26
 
24
27
  if "SecretString" in get_secret_value_response:
25
28
  return json.loads(get_secret_value_response["SecretString"])
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "eoapi-cdk",
3
- "version": "7.6.2",
3
+ "version": "8.0.0",
4
4
  "description": "A set of constructs deploying pgSTAC with CDK",
5
5
  "main": "lib/index.js",
6
6
  "types": "lib/index.d.ts",