eoapi-cdk 10.3.0 → 10.4.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.
Files changed (46) hide show
  1. package/.jsii +134 -4
  2. package/lib/bastion-host/index.js +1 -1
  3. package/lib/database/index.js +1 -1
  4. package/lib/ingestor-api/index.js +1 -1
  5. package/lib/lambda-api-gateway/index.js +1 -1
  6. package/lib/lambda-api-gateway-private/index.js +1 -1
  7. package/lib/stac-api/index.js +4 -4
  8. package/lib/stac-api/runtime/Dockerfile +25 -6
  9. package/lib/stac-api/runtime/pyproject.toml +14 -0
  10. package/lib/stac-api/runtime/uv.lock +855 -0
  11. package/lib/stac-auth-proxy/index.js +4 -4
  12. package/lib/stac-auth-proxy/runtime/Dockerfile +13 -6
  13. package/lib/stac-auth-proxy/runtime/pyproject.toml +13 -0
  14. package/lib/stac-auth-proxy/runtime/uv.lock +820 -0
  15. package/lib/stac-browser/index.js +1 -1
  16. package/lib/stac-loader/index.js +2 -2
  17. package/lib/stac-loader/runtime/Dockerfile +15 -7
  18. package/lib/stac-loader/runtime/pyproject.toml +6 -2
  19. package/lib/stac-loader/runtime/uv.lock +593 -0
  20. package/lib/stactools-item-generator/index.js +1 -1
  21. package/lib/stactools-item-generator/runtime/Dockerfile +10 -8
  22. package/lib/stactools-item-generator/runtime/pyproject.toml +1 -1
  23. package/lib/stactools-item-generator/runtime/uv.lock +177 -0
  24. package/lib/tipg-api/index.js +4 -4
  25. package/lib/tipg-api/runtime/Dockerfile +24 -13
  26. package/lib/tipg-api/runtime/pyproject.toml +13 -0
  27. package/lib/tipg-api/runtime/uv.lock +810 -0
  28. package/lib/titiler-pgstac-api/index.js +4 -4
  29. package/lib/titiler-pgstac-api/runtime/Dockerfile +27 -19
  30. package/lib/titiler-pgstac-api/runtime/pyproject.toml +19 -0
  31. package/lib/titiler-pgstac-api/runtime/uv.lock +1073 -0
  32. package/package.json +9 -9
  33. package/pyproject.toml +13 -6
  34. package/uv.lock +1373 -185
  35. package/lib/stac-api/runtime/requirements.txt +0 -2
  36. package/lib/tipg-api/runtime/requirements.txt +0 -1
  37. package/lib/titiler-pgstac-api/runtime/dev_requirements.txt +0 -2
  38. package/lib/titiler-pgstac-api/runtime/requirements.txt +0 -4
  39. /package/lib/stac-api/runtime/src/{__init__.py → stac_api/__init__.py} +0 -0
  40. /package/lib/stac-api/runtime/src/{handler.py → stac_api/handler.py} +0 -0
  41. /package/lib/stac-auth-proxy/runtime/src/{__init__.py → stac_auth_proxy_api/__init__.py} +0 -0
  42. /package/lib/stac-auth-proxy/runtime/src/{handler.py → stac_auth_proxy_api/handler.py} +0 -0
  43. /package/lib/tipg-api/runtime/src/{__init__.py → tipg_api/__init__.py} +0 -0
  44. /package/lib/tipg-api/runtime/src/{handler.py → tipg_api/handler.py} +0 -0
  45. /package/lib/titiler-pgstac-api/runtime/src/{__init__.py → titiler_pgstac_api/__init__.py} +0 -0
  46. /package/lib/titiler-pgstac-api/runtime/src/{handler.py → titiler_pgstac_api/handler.py} +0 -0
package/.jsii CHANGED
@@ -7,7 +7,7 @@
7
7
  ]
8
8
  },
9
9
  "dependencies": {
10
- "aws-cdk-lib": "^2.190.0",
10
+ "aws-cdk-lib": "^2.220.0",
11
11
  "constructs": "^10.4.2"
12
12
  },
13
13
  "dependencyClosure": {
@@ -160,6 +160,19 @@
160
160
  }
161
161
  }
162
162
  },
163
+ "aws-cdk-lib.aws_aiops": {
164
+ "targets": {
165
+ "dotnet": {
166
+ "package": "Amazon.CDK.AWS.AIOps"
167
+ },
168
+ "java": {
169
+ "package": "software.amazon.awscdk.services.aiops"
170
+ },
171
+ "python": {
172
+ "module": "aws_cdk.aws_aiops"
173
+ }
174
+ }
175
+ },
163
176
  "aws-cdk-lib.aws_amazonmq": {
164
177
  "targets": {
165
178
  "dotnet": {
@@ -407,6 +420,19 @@
407
420
  }
408
421
  }
409
422
  },
423
+ "aws-cdk-lib.aws_arcregionswitch": {
424
+ "targets": {
425
+ "dotnet": {
426
+ "package": "Amazon.CDK.AWS.ARCRegionSwitch"
427
+ },
428
+ "java": {
429
+ "package": "software.amazon.awscdk.services.arcregionswitch"
430
+ },
431
+ "python": {
432
+ "module": "aws_cdk.aws_arcregionswitch"
433
+ }
434
+ }
435
+ },
410
436
  "aws-cdk-lib.aws_arczonalshift": {
411
437
  "targets": {
412
438
  "dotnet": {
@@ -576,6 +602,19 @@
576
602
  }
577
603
  }
578
604
  },
605
+ "aws-cdk-lib.aws_bedrockagentcore": {
606
+ "targets": {
607
+ "dotnet": {
608
+ "package": "Amazon.CDK.AWS.BedrockAgentCore"
609
+ },
610
+ "java": {
611
+ "package": "software.amazon.awscdk.services.bedrockagentcore"
612
+ },
613
+ "python": {
614
+ "module": "aws_cdk.aws_bedrockagentcore"
615
+ }
616
+ }
617
+ },
579
618
  "aws-cdk-lib.aws_billingconductor": {
580
619
  "targets": {
581
620
  "dotnet": {
@@ -1552,6 +1591,19 @@
1552
1591
  }
1553
1592
  }
1554
1593
  },
1594
+ "aws-cdk-lib.aws_evs": {
1595
+ "targets": {
1596
+ "dotnet": {
1597
+ "package": "Amazon.CDK.AWS.EVS"
1598
+ },
1599
+ "java": {
1600
+ "package": "software.amazon.awscdk.services.evs"
1601
+ },
1602
+ "python": {
1603
+ "module": "aws_cdk.aws_evs"
1604
+ }
1605
+ }
1606
+ },
1555
1607
  "aws-cdk-lib.aws_finspace": {
1556
1608
  "targets": {
1557
1609
  "dotnet": {
@@ -2488,6 +2540,19 @@
2488
2540
  }
2489
2541
  }
2490
2542
  },
2543
+ "aws-cdk-lib.aws_mpa": {
2544
+ "targets": {
2545
+ "dotnet": {
2546
+ "package": "Amazon.CDK.AWS.MPA"
2547
+ },
2548
+ "java": {
2549
+ "package": "software.amazon.awscdk.services.mpa"
2550
+ },
2551
+ "python": {
2552
+ "module": "aws_cdk.aws_mpa"
2553
+ }
2554
+ }
2555
+ },
2491
2556
  "aws-cdk-lib.aws_msk": {
2492
2557
  "targets": {
2493
2558
  "dotnet": {
@@ -2618,6 +2683,32 @@
2618
2683
  }
2619
2684
  }
2620
2685
  },
2686
+ "aws-cdk-lib.aws_observabilityadmin": {
2687
+ "targets": {
2688
+ "dotnet": {
2689
+ "package": "Amazon.CDK.AWS.ObservabilityAdmin"
2690
+ },
2691
+ "java": {
2692
+ "package": "software.amazon.awscdk.services.observabilityadmin"
2693
+ },
2694
+ "python": {
2695
+ "module": "aws_cdk.aws_observabilityadmin"
2696
+ }
2697
+ }
2698
+ },
2699
+ "aws-cdk-lib.aws_odb": {
2700
+ "targets": {
2701
+ "dotnet": {
2702
+ "package": "Amazon.CDK.AWS.ODB"
2703
+ },
2704
+ "java": {
2705
+ "package": "software.amazon.awscdk.services.odb"
2706
+ },
2707
+ "python": {
2708
+ "module": "aws_cdk.aws_odb"
2709
+ }
2710
+ }
2711
+ },
2621
2712
  "aws-cdk-lib.aws_omics": {
2622
2713
  "targets": {
2623
2714
  "dotnet": {
@@ -3450,6 +3541,19 @@
3450
3541
  }
3451
3542
  }
3452
3543
  },
3544
+ "aws-cdk-lib.aws_smsvoice": {
3545
+ "targets": {
3546
+ "dotnet": {
3547
+ "package": "Amazon.CDK.AWS.SMSVOICE"
3548
+ },
3549
+ "java": {
3550
+ "package": "software.amazon.awscdk.services.smsvoice"
3551
+ },
3552
+ "python": {
3553
+ "module": "aws_cdk.aws_smsvoice"
3554
+ }
3555
+ }
3556
+ },
3453
3557
  "aws-cdk-lib.aws_sns": {
3454
3558
  "targets": {
3455
3559
  "dotnet": {
@@ -3515,6 +3619,19 @@
3515
3619
  }
3516
3620
  }
3517
3621
  },
3622
+ "aws-cdk-lib.aws_ssmguiconnect": {
3623
+ "targets": {
3624
+ "dotnet": {
3625
+ "package": "Amazon.CDK.AWS.SSMGuiConnect"
3626
+ },
3627
+ "java": {
3628
+ "package": "software.amazon.awscdk.services.ssmguiconnect"
3629
+ },
3630
+ "python": {
3631
+ "module": "aws_cdk.aws_ssmguiconnect"
3632
+ }
3633
+ }
3634
+ },
3518
3635
  "aws-cdk-lib.aws_ssmincidents": {
3519
3636
  "targets": {
3520
3637
  "dotnet": {
@@ -3749,6 +3866,19 @@
3749
3866
  }
3750
3867
  }
3751
3868
  },
3869
+ "aws-cdk-lib.aws_workspacesinstances": {
3870
+ "targets": {
3871
+ "dotnet": {
3872
+ "package": "Amazon.CDK.AWS.WorkspacesInstances"
3873
+ },
3874
+ "java": {
3875
+ "package": "software.amazon.awscdk.services.workspacesinstances"
3876
+ },
3877
+ "python": {
3878
+ "module": "aws_cdk.aws_workspacesinstances"
3879
+ }
3880
+ }
3881
+ },
3752
3882
  "aws-cdk-lib.aws_workspacesthinclient": {
3753
3883
  "targets": {
3754
3884
  "dotnet": {
@@ -3951,7 +4081,7 @@
3951
4081
  "stability": "experimental"
3952
4082
  },
3953
4083
  "homepage": "https://github.com/developmentseed/eoapi-cdk.git",
3954
- "jsiiVersion": "5.9.6 (build 4ea0b82)",
4084
+ "jsiiVersion": "5.9.10 (build 0e9cf43)",
3955
4085
  "keywords": [],
3956
4086
  "license": "ISC",
3957
4087
  "metadata": {
@@ -7780,6 +7910,6 @@
7780
7910
  "symbolId": "lib/titiler-pgstac-api/index:TitilerPgstacApiLambdaRuntimeProps"
7781
7911
  }
7782
7912
  },
7783
- "version": "10.3.0",
7784
- "fingerprint": "63xTPELHrRAOhjd1+YrjEGc7Sw2sdoryB8QFyf/zJ2U="
7913
+ "version": "10.4.0",
7914
+ "fingerprint": "s+tE9EQj7QrkWelZmO5T1/GRx3GCEMWv5NaqzffI+OE="
7785
7915
  }
@@ -158,5 +158,5 @@ class BastionHost extends constructs_1.Construct {
158
158
  }
159
159
  exports.BastionHost = BastionHost;
160
160
  _a = JSII_RTTI_SYMBOL_1;
161
- BastionHost[_a] = { fqn: "eoapi-cdk.BastionHost", version: "10.3.0" };
161
+ BastionHost[_a] = { fqn: "eoapi-cdk.BastionHost", version: "10.4.0" };
162
162
  //# 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_ec2 as ec2,\n  aws_iam as iam,\n  aws_rds as rds,\n  CfnOutput,\n} from \"aws-cdk-lib\";\nimport { Construct } from \"constructs\";\n\n/**\n * The database is located in an isolated subnet, meaning that it is not accessible from the public internet. As such, to interact with the database directly, a user must tunnel through a bastion host.\n *\n * ### Configuring\n *\n * This codebase controls _who_ is allowed to connect to the bastion host. This requires two steps:\n *\n * 1. Adding the IP address from which you are connecting to the `ipv4Allowlist` array\n * 1. Creating a bastion host system user by adding the user's configuration inform to `userdata.yaml`\n *\n * #### Adding an IP address to the `ipv4Allowlist` array\n *\n * The `BastionHost` construct takes in an `ipv4Allowlist` array as an argument. Find your IP address (eg `curl api.ipify.org`) and add that to the array along with the trailing CIDR block (likely `/32` to indicate that you are adding a single IP address).\n *\n * #### Creating a user via `userdata.yaml`\n *\n * Add an entry to the `users` array with a username (likely matching your local systems username, which you can get by running the `whoami` command in your terminal) and a public key (likely your default public key, which you can get by running `cat ~/.ssh/id_*.pub` in your terminal).\n *\n * #### Tips & Tricks when using the Bastion Host\n *\n * **Connecting to RDS Instance via SSM**\n *\n * ```sh\n * aws ssm start-session --target $INSTANCE_ID \\\n * --document-name AWS-StartPortForwardingSessionToRemoteHost \\\n * --parameters '{\n * \"host\": [\n * \"example-db.c5abcdefghij.us-west-2.rds.amazonaws.com\"\n * ],\n * \"portNumber\": [\n * \"5432\"\n * ],\n * \"localPortNumber\": [\n * \"9999\"\n * ]\n * }' \\\n * --profile $AWS_PROFILE\n * ```\n *\n * ```sh\n * psql -h localhost -p 9999 # continue adding username (-U) and db (-d) here...\n * ```\n *\n * Connect directly to Bastion Host:\n *\n * ```sh\n * aws ssm start-session --target $INSTANCE_ID --profile $AWS_PROFILE\n * ```\n *\n * **Setting up an SSH tunnel**\n *\n * In your `~/.ssh/config` file, add an entry like:\n *\n * ```\n * Host db-tunnel\n * Hostname {the-bastion-host-address}\n * LocalForward 9999 {the-db-hostname}:5432\n * ```\n *\n * Then a tunnel can be opened via:\n *\n * ```\n * ssh -N db-tunnel\n * ```\n *\n * And a connection to the DB can be made via:\n *\n * ```\n * psql -h 127.0.0.1 -p 9999 -U {username} -d {database}\n * ```\n *\n * **Handling `REMOTE HOST IDENTIFICATION HAS CHANGED!` error**\n *\n * If you've redeployed a bastion host that you've previously connected to, you may see an error like:\n *\n * ```\n * @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@\n * @    WARNING: REMOTE HOST IDENTIFICATION HAS CHANGED!     @\n * @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@\n * IT IS POSSIBLE THAT SOMEONE IS DOING SOMETHING NASTY!\n * Someone could be eavesdropping on you right now (man-in-the-middle attack)!\n * It is also possible that a host key has just been changed.\n * The fingerprint for the ECDSA key sent by the remote host is\n * SHA256:mPnxAOXTpb06PFgI1Qc8TMQ2e9b7goU8y2NdS5hzIr8.\n * Please contact your system administrator.\n * Add correct host key in /Users/username/.ssh/known_hosts to get rid of this message.\n * Offending ECDSA key in /Users/username/.ssh/known_hosts:28\n * ECDSA host key for ec2-12-34-56-789.us-west-2.compute.amazonaws.com has changed and you have requested strict checking.\n * Host key verification failed.\n * ```\n *\n * This is due to the server's fingerprint changing. We can scrub the fingerprint from our system with a command like:\n *\n * ```\n * ssh-keygen -R 12.34.56.789\n * ```\n *\n */\nexport class BastionHost extends Construct {\n  instance: ec2.Instance;\n\n  constructor(scope: Construct, id: string, props: BastionHostProps) {\n    super(scope, id);\n\n    const { stackName } = Stack.of(this);\n\n    // Build ec2 instance\n    this.instance = new ec2.Instance(this, \"bastion-host\", {\n      vpc: props.vpc,\n      vpcSubnets: { subnetType: ec2.SubnetType.PUBLIC },\n      instanceName: `${stackName} bastion host`,\n      instanceType: ec2.InstanceType.of(\n        ec2.InstanceClass.BURSTABLE4_GRAVITON,\n        ec2.InstanceSize.NANO\n      ),\n      machineImage: ec2.MachineImage.latestAmazonLinux({\n        generation: ec2.AmazonLinuxGeneration.AMAZON_LINUX_2,\n        cpuType: ec2.AmazonLinuxCpuType.ARM_64,\n      }),\n      userData: props.userData,\n      userDataCausesReplacement: true,\n    });\n\n    // Assign elastic IP\n    if (props.createElasticIp ?? true) {\n      new ec2.CfnEIP(this, \"IP\", {\n        instanceId: this.instance.instanceId,\n        tags: [{ key: \"Name\", value: stackName }],\n      });\n    }\n\n    // Allow bastion host to connect to db\n    this.instance.connections.allowTo(\n      props.db.connections.securityGroups[0],\n      ec2.Port.tcp(5432),\n      \"Allow connection from bastion host\"\n    );\n\n    // Allow IP access to bastion host\n    for (const ipv4 of props.ipv4Allowlist) {\n      this.instance.connections.allowFrom(\n        ec2.Peer.ipv4(ipv4),\n        ec2.Port.tcp(props.sshPort || 22),\n        \"SSH Access\"\n      );\n    }\n\n    // Integrate with SSM\n    this.instance.addToRolePolicy(\n      new iam.PolicyStatement({\n        actions: [\n          \"ssmmessages:*\",\n          \"ssm:UpdateInstanceInformation\",\n          \"ec2messages:*\",\n        ],\n        resources: [\"*\"],\n      })\n    );\n\n    new CfnOutput(this, \"instance-id-output\", {\n      value: this.instance.instanceId,\n      exportName: `${stackName}-instance-id`,\n    });\n    new CfnOutput(this, \"instance-public-ip-output\", {\n      value: this.instance.instancePublicIp,\n      exportName: `${stackName}-instance-public-ip`,\n    });\n    new CfnOutput(this, \"instance-public-dns-name-output\", {\n      value: this.instance.instancePublicDnsName,\n      exportName: `${stackName}-public-dns-name`,\n    });\n  }\n}\n\nexport interface BastionHostProps {\n  readonly vpc: ec2.IVpc;\n  readonly db: rds.IDatabaseInstance;\n  readonly userData: ec2.UserData;\n  readonly ipv4Allowlist: string[];\n  readonly sshPort?: number;\n\n  /**\n   * Whether or not an elastic IP should be created for the bastion host.\n   *\n   * @default false\n   */\n  readonly createElasticIp?: boolean;\n}\n"]}
@@ -221,5 +221,5 @@ class PgStacDatabase extends constructs_1.Construct {
221
221
  }
222
222
  exports.PgStacDatabase = PgStacDatabase;
223
223
  _a = JSII_RTTI_SYMBOL_1;
224
- PgStacDatabase[_a] = { fqn: "eoapi-cdk.PgStacDatabase", version: "10.3.0" };
224
+ PgStacDatabase[_a] = { fqn: "eoapi-cdk.PgStacDatabase", version: "10.4.0" };
225
225
  //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["index.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  CustomResource,\n  Duration,\n  RemovalPolicy,\n  Stack,\n  aws_lambda,\n  aws_logs,\n  aws_ec2 as ec2,\n  aws_rds as rds,\n  aws_secretsmanager as secretsmanager,\n} from \"aws-cdk-lib\";\nimport { Construct } from \"constructs\";\nimport {\n  CustomLambdaFunctionProps,\n  DEFAULT_PGSTAC_VERSION,\n  resolveLambdaCode,\n} from \"../utils\";\nimport { PgBouncer } from \"./PgBouncer\";\n\nconst instanceSizes: Record<string, number> = require(\"./instance-memory.json\");\n\nlet defaultPgSTACCustomOptions: { [key: string]: any } = {\n  context: \"FALSE\",\n  mosaic_index: \"TRUE\",\n};\n\nfunction hasVpc(\n  instance: rds.DatabaseInstance | rds.IDatabaseInstance\n): instance is rds.DatabaseInstance {\n  return (instance as rds.DatabaseInstance).vpc !== undefined;\n}\n\n/**\n * An RDS instance with pgSTAC installed and PgBouncer connection pooling.\n *\n * This construct creates an optimized pgSTAC database setup that includes:\n * - RDS PostgreSQL instance with pgSTAC extension\n * - PgBouncer connection pooler (enabled by default)\n * - Automated health monitoring system\n * - Optimized database parameters for the selected instance type\n *\n * ## Connection Pooling with PgBouncer\n *\n * By default, this construct deploys PgBouncer as a connection pooler running on\n * a dedicated EC2 instance. PgBouncer provides several benefits:\n *\n * - **Connection Management**: Pools and reuses database connections to reduce overhead\n * - **Performance**: Optimizes connection handling for high-traffic applications\n * - **Scalability**: Allows more concurrent connections than the RDS instance alone\n * - **Health Monitoring**: Includes comprehensive health checks to ensure availability\n *\n * ### PgBouncer Configuration\n * - Pool mode: Transaction-level pooling (default)\n * - Maximum client connections: 1000\n * - Default pool size: 20 connections per database/user combination\n * - Instance type: t3.micro EC2 instance\n *\n * ### Health Check System\n * The construct includes an automated health check system that validates:\n * - PgBouncer service is running and listening on port 5432\n * - Connection tests to ensure accessibility\n * - Cloud-init setup completion before validation\n * - Detailed diagnostics for troubleshooting\n *\n * ### Connection Details\n * When PgBouncer is enabled, applications connect through the PgBouncer instance\n * rather than directly to RDS. The `pgstacSecret` contains connection information\n * pointing to PgBouncer, and the `connectionTarget` property refers to the\n * PgBouncer EC2 instance.\n *\n * To disable PgBouncer and connect directly to RDS, set `addPgbouncer: false`.\n *\n * This is a wrapper around the `rds.DatabaseInstance` higher-level construct\n * making use of the BootstrapPgStac construct.\n */\nexport class PgStacDatabase extends Construct {\n  db: rds.DatabaseInstance;\n  pgstacSecret: secretsmanager.ISecret;\n  private _pgBouncerServer?: PgBouncer;\n\n  public readonly pgstacVersion: string;\n  public readonly connectionTarget: rds.IDatabaseInstance | ec2.Instance;\n  public readonly securityGroup?: ec2.SecurityGroup;\n  public readonly secretBootstrapper?: CustomResource;\n  public readonly pgbouncerHealthCheck?: CustomResource;\n\n  constructor(scope: Construct, id: string, props: PgStacDatabaseProps) {\n    super(scope, id);\n\n    const defaultParameters = this.getParameters(\n      props.instanceType?.toString() || \"m5.large\",\n      props.parameters\n    );\n    const parameterGroup = new rds.ParameterGroup(this, \"parameterGroup\", {\n      engine: props.engine,\n      parameters: {\n        max_connections: defaultParameters.maxConnections,\n        shared_buffers: defaultParameters.sharedBuffers,\n        effective_cache_size: defaultParameters.effectiveCacheSize,\n        work_mem: defaultParameters.workMem,\n        maintenance_work_mem: defaultParameters.maintenanceWorkMem,\n        max_locks_per_transaction: defaultParameters.maxLocksPerTransaction,\n        temp_buffers: defaultParameters.tempBuffers,\n        seq_page_cost: defaultParameters.seqPageCost,\n        random_page_cost: defaultParameters.randomPageCost,\n        ...props.parameters,\n      },\n    });\n\n    this.db = new rds.DatabaseInstance(this, \"db\", {\n      instanceIdentifier: Stack.of(this).stackName,\n      parameterGroup,\n      ...props,\n    });\n\n    this.pgstacVersion = props.pgstacVersion || DEFAULT_PGSTAC_VERSION;\n\n    const { code: userCode, ...otherLambdaOptions } =\n      props.bootstrapperLambdaFunctionOptions || {};\n\n    const handler = new aws_lambda.Function(this, \"lambda\", {\n      // defaults\n      runtime: aws_lambda.Runtime.PYTHON_3_12,\n      handler: \"handler.handler\",\n      memorySize: 128,\n      logRetention: aws_logs.RetentionDays.ONE_WEEK,\n      timeout: Duration.minutes(2),\n      code: resolveLambdaCode(userCode, __dirname, {\n        file: \"bootstrapper_runtime/Dockerfile\",\n        buildArgs: {\n          PYTHON_VERSION: \"3.12\",\n          PGSTAC_VERSION: this.pgstacVersion,\n        },\n      }),\n      vpc: hasVpc(this.db) ? this.db.vpc : props.vpc,\n      allowPublicSubnet: true,\n      // overwrites defaults with user-provided configurable properties,\n      ...otherLambdaOptions,\n    });\n\n    this.pgstacSecret = new secretsmanager.Secret(this, \"bootstrappersecret\", {\n      secretName: [\n        props.secretsPrefix || \"pgstac\",\n        id,\n        this.node.addr.slice(-8),\n      ].join(\"/\"),\n      generateSecretString: {\n        secretStringTemplate: JSON.stringify({\n          dbname: props.pgstacDbName || \"pgstac\",\n          engine: \"postgres\",\n          port: 5432,\n          host: this.db.instanceEndpoint.hostname,\n          username: props.pgstacUsername || \"pgstac_user\",\n        }),\n        generateStringKey: \"password\",\n        excludePunctuation: true,\n      },\n      description: `PgSTAC database bootstrapped by ${\n        Stack.of(this).stackName\n      }`,\n    });\n\n    // Allow lambda to...\n    // read new user secret\n    this.pgstacSecret.grantRead(handler);\n    // read database secret\n    this.db.secret!.grantRead(handler);\n    // connect to database\n    this.db.connections.allowFrom(handler, ec2.Port.tcp(5432));\n\n    let customResourceProperties: { [key: string]: any } =\n      props.customResourceProperties\n        ? { ...defaultPgSTACCustomOptions, ...props.customResourceProperties }\n        : defaultPgSTACCustomOptions;\n\n    // update properties\n    customResourceProperties[\"conn_secret_arn\"] = this.db.secret!.secretArn;\n    customResourceProperties[\"new_user_secret_arn\"] =\n      this.pgstacSecret.secretArn;\n\n    // if props.lambdaFunctionOptions doesn't have 'code' defined, update pgstac_version (needed for default runtime)\n    if (!userCode) {\n      customResourceProperties[\"pgstac_version\"] = this.pgstacVersion;\n    }\n\n    // add timestamp to properties to ensure the Lambda gets re-executed on each deploy\n    customResourceProperties[\"timestamp\"] = new Date().toISOString();\n\n    const bootstrapper = new CustomResource(this, \"bootstrapper\", {\n      serviceToken: handler.functionArn,\n      properties: customResourceProperties,\n      removalPolicy: RemovalPolicy.RETAIN, // This retains the custom resource (which doesn't really exist), not the database\n    });\n\n    // PgBouncer: connection poolercustomresource trigger on redeploy\n    const defaultPgbouncerInstanceProps: Partial<ec2.InstanceProps> = {\n      instanceName: `${Stack.of(this).stackName}-pgbouncer`,\n      instanceType: ec2.InstanceType.of(\n        ec2.InstanceClass.T3,\n        ec2.InstanceSize.MICRO\n      ),\n    };\n    const addPgbouncer = props.addPgbouncer ?? true;\n    if (addPgbouncer) {\n      this._pgBouncerServer = new PgBouncer(this, \"pgbouncer\", {\n        instanceProps: {\n          ...defaultPgbouncerInstanceProps,\n          ...props.pgbouncerInstanceProps,\n        },\n        vpc: props.vpc,\n        database: {\n          connections: this.db.connections,\n          secret: this.pgstacSecret,\n        },\n        dbMaxConnections: parseInt(defaultParameters.maxConnections),\n        usePublicSubnet:\n          !props.vpcSubnets ||\n          props.vpcSubnets.subnetType === ec2.SubnetType.PUBLIC,\n        pgBouncerConfig: {\n          poolMode: \"transaction\",\n          maxClientConn: 1000,\n          defaultPoolSize: 20,\n          minPoolSize: 10,\n          reservePoolSize: 5,\n          reservePoolTimeout: 5,\n        },\n      });\n\n      this._pgBouncerServer.node.addDependency(bootstrapper);\n\n      this.pgstacSecret = this._pgBouncerServer.pgbouncerSecret;\n      this.connectionTarget = this._pgBouncerServer.instance;\n      this.securityGroup = this._pgBouncerServer.securityGroup;\n      this.secretBootstrapper = this._pgBouncerServer.secretUpdateComplete;\n      this.pgbouncerHealthCheck = this._pgBouncerServer.healthCheck;\n    } else {\n      this.connectionTarget = this.db;\n    }\n  }\n\n  public getParameters(\n    instanceType: string,\n    parameters: PgStacDatabaseProps[\"parameters\"]\n  ): DatabaseParameters {\n    // https://github.com/aws/aws-cli/issues/1279#issuecomment-909318236\n    const memory_in_kb = instanceSizes[instanceType] * 1024;\n\n    // It's only necessary to consider passed in parameters for any value that used to\n    // derive subsequent values. Values that don't have dependencies will be overriden\n    // when we unpack the passed-in user parameters\n    const maxConnections = parameters?.maxConnections\n      ? Number.parseInt(parameters.maxConnections)\n      : // https://docs.aws.amazon.com/AmazonRDS/latest/UserGuide/CHAP_Limits.html#RDS_Limits.MaxConnections\n        Math.min(Math.round((memory_in_kb * 1024) / 9531392), 5000);\n    const sharedBuffers = parameters?.sharedBufers\n      ? Number.parseInt(parameters.sharedBufers)\n      : Math.round(0.25 * memory_in_kb);\n\n    const effectiveCacheSize = Math.round(0.75 * memory_in_kb);\n    const workMem = Math.floor(sharedBuffers / maxConnections);\n    const maintenanceWorkMem = Math.round(0.25 * sharedBuffers);\n\n    const tempBuffers = 128 * 1024;\n    const seqPageCost = 1;\n    const randomPageCost = 1.1;\n\n    return {\n      maxConnections: `${maxConnections}`,\n      sharedBuffers: `${sharedBuffers / 8}`, // Represented in 8kb blocks\n      effectiveCacheSize: `${effectiveCacheSize / 8}`, // Represented in 8kb blocks\n      workMem: `${workMem}`,\n      maintenanceWorkMem: `${maintenanceWorkMem}`,\n      maxLocksPerTransaction: \"1024\",\n      tempBuffers: `${tempBuffers / 8}`, // Represented in 8kb blocks\n      seqPageCost: `${seqPageCost}`,\n      randomPageCost: `${randomPageCost}`,\n    };\n  }\n}\n\nexport interface PgStacDatabaseProps extends rds.DatabaseInstanceProps {\n  /**\n   * Name of database that is to be created and onto which pgSTAC will be installed.\n   *\n   * @default pgstac\n   */\n  readonly pgstacDbName?: string;\n\n  /**\n   * Version of pgstac to install on the database\n   *\n   * @default 0.8.5\n   */\n  readonly pgstacVersion?: string;\n\n  /**\n   * Prefix to assign to the generated `secrets_manager.Secret`\n   *\n   * @default pgstac\n   */\n  readonly secretsPrefix?: string;\n\n  /**\n   * Name of user that will be generated for connecting to the pgSTAC database.\n   *\n   * @default pgstac_user\n   */\n  readonly pgstacUsername?: string;\n\n  /**\n   * Add pgbouncer instance for managing traffic to the pgSTAC database\n   *\n   * @default true\n   */\n  readonly addPgbouncer?: boolean;\n\n  /**\n   * Properties for the pgbouncer ec2 instance\n   *\n   * @default - defined in the construct\n   */\n  readonly pgbouncerInstanceProps?: ec2.InstanceProps | any;\n\n  /**\n   * Lambda function Custom Resource properties. A custom resource property is going to be created\n   * to trigger the boostrapping lambda function. This parameter allows the user to specify additional properties\n   * on top of the defaults ones.\n   *\n   */\n  readonly customResourceProperties?: {\n    [key: string]: any;\n  };\n\n  /**\n   * Can be used to override the default lambda function properties.\n   *\n   * @default - defined in the construct.\n   */\n  readonly bootstrapperLambdaFunctionOptions?: CustomLambdaFunctionProps;\n}\n\nexport interface DatabaseParameters {\n  /**\n   * @default - LEAST({DBInstanceClassMemory/9531392}, 5000)\n   */\n  readonly maxConnections: string;\n\n  /**\n   * Note: This value is measured in 8KB blocks.\n   *\n   * @default '{DBInstanceClassMemory/32768}' 25% of instance memory, ie `{(DBInstanceClassMemory/(1024*8)) * 0.25}`\n   */\n  readonly sharedBuffers: string;\n\n  /**\n   * @default - 75% of instance memory\n   */\n  readonly effectiveCacheSize: string;\n\n  /**\n   * @default - shared buffers divided by max connections\n   */\n  readonly workMem: string;\n\n  /**\n   * @default - 25% of shared buffers\n   */\n  readonly maintenanceWorkMem: string;\n\n  /**\n   * @default 1024\n   */\n  readonly maxLocksPerTransaction: string;\n\n  /**\n   * @default 131172 (128 * 1024)\n   */\n  readonly tempBuffers: string;\n\n  /**\n   * @default 1\n   */\n  readonly seqPageCost: string;\n\n  /**\n   * @default 1.1\n   */\n  readonly randomPageCost: string;\n}\n"]}
@@ -181,5 +181,5 @@ class StacIngestor extends constructs_1.Construct {
181
181
  }
182
182
  exports.StacIngestor = StacIngestor;
183
183
  _a = JSII_RTTI_SYMBOL_1;
184
- StacIngestor[_a] = { fqn: "eoapi-cdk.StacIngestor", version: "10.3.0" };
184
+ StacIngestor[_a] = { fqn: "eoapi-cdk.StacIngestor", version: "10.4.0" };
185
185
  //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["index.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  aws_apigateway as apigateway,\n  aws_logs,\n  Duration,\n  aws_dynamodb as dynamodb,\n  aws_ec2 as ec2,\n  aws_lambda_event_sources as events,\n  aws_iam as iam,\n  aws_lambda as lambda,\n  RemovalPolicy,\n  aws_secretsmanager as secretsmanager,\n  aws_ssm as ssm,\n  Stack,\n} from \"aws-cdk-lib\";\nimport { Construct } from \"constructs\";\nimport {\n  CustomLambdaFunctionProps,\n  DEFAULT_PGSTAC_VERSION,\n  resolveLambdaCode,\n} from \"../utils\";\n\nexport class StacIngestor extends Construct {\n  table: dynamodb.Table;\n  public handlerRole: iam.Role;\n\n  constructor(scope: Construct, id: string, props: StacIngestorProps) {\n    super(scope, id);\n\n    this.table = this.buildTable();\n\n    const env: Record<string, string> = {\n      DYNAMODB_TABLE: this.table.tableName,\n      ROOT_PATH: `/${props.stage}`,\n      NO_PYDANTIC_SSM_SETTINGS: \"1\",\n      STAC_URL: props.stacUrl,\n      DATA_ACCESS_ROLE: props.dataAccessRole.roleArn,\n      ...props.apiEnv,\n    };\n\n    this.handlerRole = new iam.Role(this, \"execution-role\", {\n      description:\n        \"Role used by STAC Ingestor. Manually defined so that we can choose a name that is supported by the data access roles trust policy\",\n      assumedBy: new iam.ServicePrincipal(\"lambda.amazonaws.com\"),\n      managedPolicies: [\n        iam.ManagedPolicy.fromAwsManagedPolicyName(\n          \"service-role/AWSLambdaBasicExecutionRole\"\n        ),\n        iam.ManagedPolicy.fromAwsManagedPolicyName(\n          \"service-role/AWSLambdaVPCAccessExecutionRole\"\n        ),\n      ],\n    });\n\n    const handler = this.buildApiLambda({\n      table: this.table,\n      env,\n      dataAccessRole: props.dataAccessRole,\n      stage: props.stage,\n      dbSecret: props.stacDbSecret,\n      dbVpc: props.vpc,\n      dbSecurityGroup: props.stacDbSecurityGroup,\n      subnetSelection: props.subnetSelection,\n      lambdaFunctionOptions: props.apiLambdaFunctionOptions,\n      pgstacVersion: props.pgstacVersion,\n    });\n\n    this.buildApiEndpoint({\n      handler,\n      stage: props.stage,\n      endpointConfiguration: props.apiEndpointConfiguration,\n      policy: props.apiPolicy,\n      ingestorDomainNameOptions: props.ingestorDomainNameOptions,\n    });\n\n    this.buildIngestor({\n      table: this.table,\n      env: env,\n      dbSecret: props.stacDbSecret,\n      dbVpc: props.vpc,\n      dbSecurityGroup: props.stacDbSecurityGroup,\n      subnetSelection: props.subnetSelection,\n      lambdaFunctionOptions: props.ingestorLambdaFunctionOptions,\n      pgstacVersion: props.pgstacVersion,\n    });\n\n    this.registerSsmParameter({\n      name: \"dynamodb_table\",\n      value: this.table.tableName,\n      description: \"Name of table used to store ingestions\",\n    });\n  }\n\n  private buildTable(): dynamodb.Table {\n    const table = new dynamodb.Table(this, \"ingestions-table\", {\n      partitionKey: { name: \"created_by\", type: dynamodb.AttributeType.STRING },\n      sortKey: { name: \"id\", type: dynamodb.AttributeType.STRING },\n      billingMode: dynamodb.BillingMode.PAY_PER_REQUEST,\n      removalPolicy: RemovalPolicy.DESTROY,\n      stream: dynamodb.StreamViewType.NEW_IMAGE,\n    });\n\n    table.addGlobalSecondaryIndex({\n      indexName: \"status\",\n      partitionKey: { name: \"status\", type: dynamodb.AttributeType.STRING },\n      sortKey: { name: \"created_at\", type: dynamodb.AttributeType.STRING },\n    });\n\n    return table;\n  }\n\n  private buildApiLambda(props: {\n    table: dynamodb.ITable;\n    env: Record<string, string>;\n    dataAccessRole: iam.IRole;\n    stage: string;\n    dbSecret: secretsmanager.ISecret;\n    dbVpc: undefined | ec2.IVpc;\n    dbSecurityGroup: ec2.ISecurityGroup;\n    subnetSelection: undefined | ec2.SubnetSelection;\n    lambdaFunctionOptions?: CustomLambdaFunctionProps;\n    pgstacVersion?: string;\n  }): lambda.Function {\n    const { code: userCode, ...otherLambdaOptions } =\n      props.lambdaFunctionOptions || {};\n\n    const handler = new lambda.Function(this, \"api-handler\", {\n      // defaults\n      runtime: lambda.Runtime.PYTHON_3_12,\n      handler: \"src.handler.handler\",\n      memorySize: 2048,\n      logRetention: aws_logs.RetentionDays.ONE_WEEK,\n      timeout: Duration.seconds(30),\n      code: resolveLambdaCode(userCode, __dirname, {\n        file: \"runtime/Dockerfile\",\n        buildArgs: {\n          PYTHON_VERSION: \"3.12\",\n          PGSTAC_VERSION: props.pgstacVersion || DEFAULT_PGSTAC_VERSION,\n        },\n      }),\n      allowPublicSubnet: true,\n      vpc: props.dbVpc,\n      vpcSubnets: props.subnetSelection,\n      environment: { DB_SECRET_ARN: props.dbSecret.secretArn, ...props.env },\n      role: this.handlerRole,\n      // overwrites defaults with user-provided configurable properties\n      ...otherLambdaOptions,\n    });\n\n    // Allow handler to read DB secret\n    props.dbSecret.grantRead(handler);\n\n    // Allow handler to connect to DB\n\n    if (props.dbVpc) {\n      props.dbSecurityGroup.addIngressRule(\n        handler.connections.securityGroups[0],\n        ec2.Port.tcp(5432),\n        \"Allow connections from STAC Ingestor\"\n      );\n    }\n\n    props.table.grantReadWriteData(handler);\n\n    return handler;\n  }\n\n  private buildIngestor(props: {\n    table: dynamodb.ITable;\n    env: Record<string, string>;\n    dbSecret: secretsmanager.ISecret;\n    dbVpc: undefined | ec2.IVpc;\n    dbSecurityGroup: ec2.ISecurityGroup;\n    subnetSelection: undefined | ec2.SubnetSelection;\n    lambdaFunctionOptions?: CustomLambdaFunctionProps;\n    pgstacVersion?: string;\n  }): lambda.Function {\n    const { code: userCode, ...otherLambdaOptions } =\n      props.lambdaFunctionOptions || {};\n    const handler = new lambda.Function(this, \"stac-ingestor\", {\n      // defaults\n      runtime: lambda.Runtime.PYTHON_3_12,\n      handler: \"src.ingestor.handler\",\n      memorySize: 2048,\n      logRetention: aws_logs.RetentionDays.ONE_WEEK,\n      timeout: Duration.seconds(180),\n      code: resolveLambdaCode(userCode, __dirname, {\n        file: \"runtime/Dockerfile\",\n        buildArgs: {\n          PYTHON_VERSION: \"3.12\",\n          PGSTAC_VERSION: props.pgstacVersion || DEFAULT_PGSTAC_VERSION,\n        },\n      }),\n      vpc: props.dbVpc,\n      vpcSubnets: props.subnetSelection,\n      allowPublicSubnet: true,\n      environment: { DB_SECRET_ARN: props.dbSecret.secretArn, ...props.env },\n      role: this.handlerRole,\n      // overwrites defaults with user-provided configurable properties\n      ...otherLambdaOptions,\n    });\n\n    // Allow handler to read DB secret\n    props.dbSecret.grantRead(handler);\n\n    // Allow handler to connect to DB\n    if (props.dbVpc) {\n      props.dbSecurityGroup.addIngressRule(\n        handler.connections.securityGroups[0],\n        ec2.Port.tcp(5432),\n        \"Allow connections from STAC Ingestor\"\n      );\n    }\n\n    // Allow handler to write results back to DB\n    props.table.grantWriteData(handler);\n\n    // Trigger handler from writes to DynamoDB table\n    handler.addEventSource(\n      new events.DynamoEventSource(props.table, {\n        // Read when batches reach size...\n        batchSize: 1000,\n        // ... or when window is reached.\n        maxBatchingWindow: Duration.seconds(10),\n        // Read oldest data first.\n        startingPosition: lambda.StartingPosition.TRIM_HORIZON,\n        retryAttempts: 1,\n      })\n    );\n\n    return handler;\n  }\n\n  private buildApiEndpoint(props: {\n    handler: lambda.IFunction;\n    stage: string;\n    policy?: iam.PolicyDocument;\n    endpointConfiguration?: apigateway.EndpointConfiguration;\n    ingestorDomainNameOptions?: apigateway.DomainNameOptions;\n  }): apigateway.LambdaRestApi {\n    return new apigateway.LambdaRestApi(\n      this,\n      `${Stack.of(this).stackName}-ingestor-api`,\n      {\n        handler: props.handler,\n        proxy: true,\n\n        cloudWatchRole: true,\n        deployOptions: { stageName: props.stage },\n        endpointExportName: `${Stack.of(this)}-ingestor-api`,\n\n        endpointConfiguration: props.endpointConfiguration,\n        policy: props.policy,\n\n        domainName: props.ingestorDomainNameOptions\n          ? {\n              domainName: props.ingestorDomainNameOptions.domainName,\n              certificate: props.ingestorDomainNameOptions.certificate,\n            }\n          : undefined,\n      }\n    );\n  }\n\n  private registerSsmParameter(props: {\n    name: string;\n    value: string;\n    description: string;\n  }): ssm.IStringParameter {\n    const parameterNamespace = Stack.of(this).stackName;\n    return new ssm.StringParameter(\n      this,\n      `${props.name.replace(\"_\", \"-\")}-parameter`,\n      {\n        description: props.description,\n        parameterName: `/${parameterNamespace}/${props.name}`,\n        stringValue: props.value,\n      }\n    );\n  }\n}\n\nexport interface StacIngestorProps {\n  /**\n   * ARN of AWS Role used to validate access to S3 data\n   */\n  readonly dataAccessRole: iam.IRole;\n\n  /**\n   * URL of STAC API\n   */\n  readonly stacUrl: string;\n\n  /**\n   * Stage of deployment (e.g. `dev`, `prod`)\n   */\n  readonly stage: string;\n\n  /**\n   * Secret containing pgSTAC DB connection information\n   */\n  readonly stacDbSecret: secretsmanager.ISecret;\n\n  /**\n   * VPC running pgSTAC DB\n   */\n  readonly vpc?: ec2.IVpc;\n\n  /**\n   * Security Group used by pgSTAC DB\n   */\n  readonly stacDbSecurityGroup: ec2.ISecurityGroup;\n\n  /**\n   * Subnet into which the lambda should be deployed if using a VPC\n   */\n  readonly subnetSelection?: ec2.SubnetSelection;\n\n  /**\n   * Environment variables to be sent to Lambda.\n   */\n  readonly apiEnv?: Record<string, string>;\n\n  /**\n   * API Endpoint Configuration, useful for creating private APIs.\n   */\n  readonly apiEndpointConfiguration?: apigateway.EndpointConfiguration;\n\n  /**\n   * API Policy Document, useful for creating private APIs.\n   */\n  readonly apiPolicy?: iam.PolicyDocument;\n\n  /**\n   * Custom Domain Name Options for Ingestor API\n   */\n  readonly ingestorDomainNameOptions?: apigateway.DomainNameOptions;\n\n  /**\n   * Can be used to override the default lambda function properties.\n   *\n   * @default - default settings are defined in the construct.\n   */\n  readonly apiLambdaFunctionOptions?: CustomLambdaFunctionProps;\n\n  /**\n   * Can be used to override the default lambda function properties.\n   *\n   * @default - default settings are defined in the construct.\n   */\n  readonly ingestorLambdaFunctionOptions?: CustomLambdaFunctionProps;\n\n  /**\n   * pgstac version - must match the version installed on the pgstac database\n   *\n   * @default - default settings are defined in the construct\n   */\n  readonly pgstacVersion?: string;\n}\n"]}
@@ -24,5 +24,5 @@ class LambdaApiGateway extends constructs_1.Construct {
24
24
  }
25
25
  exports.LambdaApiGateway = LambdaApiGateway;
26
26
  _a = JSII_RTTI_SYMBOL_1;
27
- LambdaApiGateway[_a] = { fqn: "eoapi-cdk.LambdaApiGateway", version: "10.3.0" };
27
+ LambdaApiGateway[_a] = { fqn: "eoapi-cdk.LambdaApiGateway", version: "10.4.0" };
28
28
  //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJpbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7OztBQUFBLDZDQUtxQjtBQUNyQiwyQ0FBdUM7QUFzQnZDLE1BQWEsZ0JBQWlCLFNBQVEsc0JBQVM7SUFHN0MsWUFBWSxLQUFnQixFQUFFLEVBQVUsRUFBRSxLQUE0QjtRQUNwRSxLQUFLLENBQUMsS0FBSyxFQUFFLEVBQUUsQ0FBQyxDQUFDO1FBRWpCLE1BQU0sRUFDSixPQUFPLEdBQUcsR0FBRyxtQkFBSyxDQUFDLEVBQUUsQ0FBQyxJQUFJLENBQUMsQ0FBQyxTQUFTLElBQUksRUFBRSxFQUFFLEVBQzdDLFVBQVUsRUFDVixjQUFjLEdBQ2YsR0FBRyxLQUFLLENBQUM7UUFFVixNQUFNLG9CQUFvQixHQUFHLFVBQVUsQ0FBQyxDQUFDLENBQUMsRUFBRSxVQUFVLEVBQUUsQ0FBQyxDQUFDLENBQUMsU0FBUyxDQUFDO1FBRXJFLE1BQU0sa0JBQWtCLEdBQ3RCLElBQUksMkNBQXlCLENBQUMscUJBQXFCLENBQ2pELGFBQWEsRUFDYixjQUFjLEVBQ2QsVUFBVTtZQUNSLENBQUMsQ0FBQztnQkFDRSxnQkFBZ0IsRUFDZCxJQUFJLDhCQUFZLENBQUMsZ0JBQWdCLEVBQUUsQ0FBQyxlQUFlLENBQ2pELE1BQU0sRUFDTiw4QkFBWSxDQUFDLFlBQVksQ0FBQyxNQUFNLENBQUMsVUFBVSxDQUFDLElBQUksQ0FBQyxDQUNsRDthQUNKO1lBQ0gsQ0FBQyxDQUFDLFNBQVMsQ0FDZCxDQUFDO1FBRUosSUFBSSxDQUFDLEdBQUcsR0FBRyxJQUFJLDhCQUFZLENBQUMsT0FBTyxDQUFDLElBQUksRUFBRSxLQUFLLEVBQUU7WUFDL0MsT0FBTztZQUNQLG9CQUFvQjtZQUNwQixrQkFBa0IsRUFBRSxrQkFBa0I7U0FDdkMsQ0FBQyxDQUFDO0lBQ0wsQ0FBQzs7QUFsQ0gsNENBbUNDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtcbiAgU3RhY2ssXG4gIGF3c19hcGlnYXRld2F5djIgYXMgYXBpZ2F0ZXdheXYyLFxuICBhd3NfYXBpZ2F0ZXdheXYyX2ludGVncmF0aW9ucyBhcyBhcGlnYXRld2F5djJfaW50ZWdyYXRpb25zLFxuICBhd3NfbGFtYmRhIGFzIGxhbWJkYSxcbn0gZnJvbSBcImF3cy1jZGstbGliXCI7XG5pbXBvcnQgeyBDb25zdHJ1Y3QgfSBmcm9tIFwiY29uc3RydWN0c1wiO1xuXG5leHBvcnQgaW50ZXJmYWNlIExhbWJkYUFwaUdhdGV3YXlQcm9wcyB7XG4gIC8qKlxuICAgKiBMYW1iZGEgZnVuY3Rpb24gdG8gaW50ZWdyYXRlIHdpdGggdGhlIEFQSSBHYXRld2F5LlxuICAgKi9cbiAgcmVhZG9ubHkgbGFtYmRhRnVuY3Rpb246IGxhbWJkYS5GdW5jdGlvbiB8IGxhbWJkYS5WZXJzaW9uO1xuXG4gIC8qKlxuICAgKiBDdXN0b20gRG9tYWluIE5hbWUgZm9yIHRoZSBBUEkuIElmIGRlZmluZWQsIHdpbGwgY3JlYXRlIHRoZVxuICAgKiBkb21haW4gbmFtZSBhbmQgaW50ZWdyYXRlIGl0IHdpdGggdGhlIEFQSS5cbiAgICpcbiAgICogQGRlZmF1bHQgLSB1bmRlZmluZWRcbiAgICovXG4gIHJlYWRvbmx5IGRvbWFpbk5hbWU/OiBhcGlnYXRld2F5djIuSURvbWFpbk5hbWU7XG5cbiAgLyoqXG4gICAqIE5hbWUgb2YgdGhlIEFQSSBHYXRld2F5LlxuICAgKi9cbiAgcmVhZG9ubHkgYXBpTmFtZT86IHN0cmluZztcbn1cblxuZXhwb3J0IGNsYXNzIExhbWJkYUFwaUdhdGV3YXkgZXh0ZW5kcyBDb25zdHJ1Y3Qge1xuICByZWFkb25seSBhcGk6IGFwaWdhdGV3YXl2Mi5IdHRwQXBpO1xuXG4gIGNvbnN0cnVjdG9yKHNjb3BlOiBDb25zdHJ1Y3QsIGlkOiBzdHJpbmcsIHByb3BzOiBMYW1iZGFBcGlHYXRld2F5UHJvcHMpIHtcbiAgICBzdXBlcihzY29wZSwgaWQpO1xuXG4gICAgY29uc3Qge1xuICAgICAgYXBpTmFtZSA9IGAke1N0YWNrLm9mKHRoaXMpLnN0YWNrTmFtZX0tJHtpZH1gLFxuICAgICAgZG9tYWluTmFtZSxcbiAgICAgIGxhbWJkYUZ1bmN0aW9uLFxuICAgIH0gPSBwcm9wcztcblxuICAgIGNvbnN0IGRlZmF1bHREb21haW5NYXBwaW5nID0gZG9tYWluTmFtZSA/IHsgZG9tYWluTmFtZSB9IDogdW5kZWZpbmVkO1xuXG4gICAgY29uc3QgZGVmYXVsdEludGVncmF0aW9uID1cbiAgICAgIG5ldyBhcGlnYXRld2F5djJfaW50ZWdyYXRpb25zLkh0dHBMYW1iZGFJbnRlZ3JhdGlvbihcbiAgICAgICAgXCJpbnRlZ3JhdGlvblwiLFxuICAgICAgICBsYW1iZGFGdW5jdGlvbixcbiAgICAgICAgZG9tYWluTmFtZVxuICAgICAgICAgID8ge1xuICAgICAgICAgICAgICBwYXJhbWV0ZXJNYXBwaW5nOlxuICAgICAgICAgICAgICAgIG5ldyBhcGlnYXRld2F5djIuUGFyYW1ldGVyTWFwcGluZygpLm92ZXJ3cml0ZUhlYWRlcihcbiAgICAgICAgICAgICAgICAgIFwiaG9zdFwiLFxuICAgICAgICAgICAgICAgICAgYXBpZ2F0ZXdheXYyLk1hcHBpbmdWYWx1ZS5jdXN0b20oZG9tYWluTmFtZS5uYW1lKVxuICAgICAgICAgICAgICAgICksXG4gICAgICAgICAgICB9XG4gICAgICAgICAgOiB1bmRlZmluZWRcbiAgICAgICk7XG5cbiAgICB0aGlzLmFwaSA9IG5ldyBhcGlnYXRld2F5djIuSHR0cEFwaSh0aGlzLCBcImFwaVwiLCB7XG4gICAgICBhcGlOYW1lLFxuICAgICAgZGVmYXVsdERvbWFpbk1hcHBpbmcsXG4gICAgICBkZWZhdWx0SW50ZWdyYXRpb246IGRlZmF1bHRJbnRlZ3JhdGlvbixcbiAgICB9KTtcbiAgfVxufVxuIl19
@@ -52,5 +52,5 @@ class PrivateLambdaApiGateway extends constructs_1.Construct {
52
52
  }
53
53
  exports.PrivateLambdaApiGateway = PrivateLambdaApiGateway;
54
54
  _a = JSII_RTTI_SYMBOL_1;
55
- PrivateLambdaApiGateway[_a] = { fqn: "eoapi-cdk.PrivateLambdaApiGateway", version: "10.3.0" };
55
+ PrivateLambdaApiGateway[_a] = { fqn: "eoapi-cdk.PrivateLambdaApiGateway", version: "10.4.0" };
56
56
  //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJpbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7OztBQUFBLDZDQUtxQjtBQUNyQiwyQ0FBdUM7QUEwRHZDLE1BQWEsdUJBQXdCLFNBQVEsc0JBQVM7SUFJcEQsWUFDRSxLQUFnQixFQUNoQixFQUFVLEVBQ1YsS0FBbUM7UUFFbkMsS0FBSyxDQUFDLEtBQUssRUFBRSxFQUFFLENBQUMsQ0FBQztRQUVqQixNQUFNLEVBQ0osV0FBVyxFQUNYLFdBQVcsRUFDWCxjQUFjLEVBQ2QsR0FBRyxFQUNILDBCQUEwQixFQUMxQixpQkFBaUIsR0FBRyxJQUFJLEVBQ3hCLGFBQWEsRUFDYixNQUFNLEVBQ04sd0JBQXdCLEdBQ3pCLEdBQUcsS0FBSyxDQUFDO1FBRVYsSUFBSSxpQkFBaUIsRUFBRSxDQUFDO1lBQ3RCLHNDQUFzQztZQUN0QyxJQUFJLENBQUMsV0FBVyxHQUFHLElBQUkscUJBQUcsQ0FBQyxvQkFBb0IsQ0FBQyxJQUFJLEVBQUUsY0FBYyxFQUFFO2dCQUNwRSxHQUFHO2dCQUNILE9BQU8sRUFBRSxxQkFBRyxDQUFDLDhCQUE4QixDQUFDLFVBQVU7Z0JBQ3RELE9BQU8sRUFBRSwwQkFBMEI7YUFDcEMsQ0FBQyxDQUFDO1FBQ0wsQ0FBQztRQUVELE1BQU0sa0JBQWtCLEdBQUcsSUFBSSw0QkFBVSxDQUFDLGlCQUFpQixDQUN6RCxjQUFjLEVBQ2Qsd0JBQXdCLENBQ3pCLENBQUM7UUFFRixrQ0FBa0M7UUFDbEMsSUFBSSxDQUFDLEdBQUcsR0FBRyxJQUFJLDRCQUFVLENBQUMsT0FBTyxDQUFDLElBQUksRUFBRSxVQUFVLEVBQUU7WUFDbEQsV0FBVyxFQUFFLFdBQVcsSUFBSSxHQUFHLEtBQUssQ0FBQyxJQUFJLENBQUMsRUFBRSxjQUFjO1lBQzFELFdBQVcsRUFBRSxXQUFXLElBQUksMEJBQTBCO1lBQ3RELGFBQWEsRUFBRSxDQUFDLDRCQUFVLENBQUMsWUFBWSxDQUFDLE9BQU8sQ0FBQztZQUNoRCxNQUFNLEVBQ0osTUFBTTtnQkFDTixJQUFJLHFCQUFHLENBQUMsY0FBYyxDQUFDO29CQUNyQixVQUFVLEVBQUU7d0JBQ1YsSUFBSSxxQkFBRyxDQUFDLGVBQWUsQ0FBQzs0QkFDdEIsTUFBTSxFQUFFLHFCQUFHLENBQUMsTUFBTSxDQUFDLEtBQUs7NEJBQ3hCLFVBQVUsRUFBRSxDQUFDLElBQUkscUJBQUcsQ0FBQyxZQUFZLEVBQUUsQ0FBQzs0QkFDcEMsT0FBTyxFQUFFLENBQUMsb0JBQW9CLENBQUM7NEJBQy9CLFNBQVMsRUFBRSxDQUFDLGdCQUFnQixDQUFDOzRCQUM3QixVQUFVLEVBQUU7Z0NBQ1YsWUFBWSxFQUFFLEVBQUUsZUFBZSxFQUFFLEdBQUcsQ0FBQyxLQUFLLEVBQUU7NkJBQzdDO3lCQUNGLENBQUM7cUJBQ0g7aUJBQ0YsQ0FBQztZQUNKLGFBQWEsRUFBRSxhQUFhLElBQUk7Z0JBQzlCLFlBQVksRUFBRSw0QkFBVSxDQUFDLGtCQUFrQixDQUFDLElBQUk7Z0JBQ2hELGdCQUFnQixFQUFFLElBQUk7YUFDdkI7WUFDRCxrQkFBa0I7WUFDbEIsb0JBQW9CLEVBQUU7Z0JBQ3BCLGlCQUFpQixFQUFFLDRCQUFVLENBQUMsaUJBQWlCLENBQUMsSUFBSTthQUNyRDtTQUNGLENBQUMsQ0FBQztRQUVILElBQUksQ0FBQyxHQUFHLENBQUMsSUFBSSxDQUFDLFNBQVMsQ0FBQyxLQUFLLENBQUMsQ0FBQztRQUMvQixJQUFJLENBQUMsR0FBRyxDQUFDLElBQUksQ0FBQyxXQUFXLENBQUMsVUFBVSxDQUFDLENBQUMsU0FBUyxDQUFDLEtBQUssQ0FBQyxDQUFDO0lBQ3pELENBQUM7O0FBckVILDBEQXNFQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7XG4gIGF3c19hcGlnYXRld2F5IGFzIGFwaWdhdGV3YXksXG4gIGF3c19lYzIgYXMgZWMyLFxuICBhd3NfaWFtIGFzIGlhbSxcbiAgYXdzX2xhbWJkYSBhcyBsYW1iZGEsXG59IGZyb20gXCJhd3MtY2RrLWxpYlwiO1xuaW1wb3J0IHsgQ29uc3RydWN0IH0gZnJvbSBcImNvbnN0cnVjdHNcIjtcblxuZXhwb3J0IGludGVyZmFjZSBQcml2YXRlTGFtYmRhQXBpR2F0ZXdheVByb3BzIHtcbiAgLyoqXG4gICAqIExhbWJkYSBmdW5jdGlvbiB0byBpbnRlZ3JhdGUgd2l0aCB0aGUgQVBJIEdhdGV3YXkuXG4gICAqL1xuICByZWFkb25seSBsYW1iZGFGdW5jdGlvbjogbGFtYmRhLklGdW5jdGlvbjtcblxuICAvKipcbiAgICogTGFtYmRhIGludGVncmF0aW9uIG9wdGlvbnMgZm9yIHRoZSBBUEkgR2F0ZXdheS5cbiAgICovXG4gIHJlYWRvbmx5IGxhbWJkYUludGVncmF0aW9uT3B0aW9ucz86IGFwaWdhdGV3YXkuTGFtYmRhSW50ZWdyYXRpb25PcHRpb25zO1xuXG4gIC8qKlxuICAgKiBWUEMgdG8gY3JlYXRlIHRoZSBBUEkgR2F0ZXdheSBpbi5cbiAgICovXG4gIHJlYWRvbmx5IHZwYzogZWMyLklWcGM7XG5cbiAgLyoqXG4gICAqIFdoZXRoZXIgdG8gY3JlYXRlIGEgVlBDIGVuZHBvaW50IGZvciB0aGUgQVBJIEdhdGV3YXkuXG4gICAqXG4gICAqIEBkZWZhdWx0IC0gdHJ1ZVxuICAgKi9cbiAgcmVhZG9ubHkgY3JlYXRlVnBjRW5kcG9pbnQ/OiBib29sZWFuO1xuXG4gIC8qKlxuICAgKiBUaGUgc3VibmV0cyBpbiB3aGljaCB0byBjcmVhdGUgYSBWUEMgZW5kcG9pbnQgbmV0d29yayBpbnRlcmZhY2UuIEF0IG1vc3Qgb25lIHBlciBhdmFpbGFiaWxpdHkgem9uZS5cblxuICAgKi9cbiAgcmVhZG9ubHkgdnBjRW5kcG9pbnRTdWJuZXRTZWxlY3Rpb24/OiBlYzIuU3VibmV0U2VsZWN0aW9uO1xuXG4gIC8qKlxuICAgKiBOYW1lIGZvciB0aGUgQVBJIEdhdGV3YXkuXG4gICAqXG4gICAqIEBkZWZhdWx0IC0gYCR7c2NvcGUubm9kZS5pZH0tcHJpdmF0ZS1hcGlgXG4gICAqL1xuICByZWFkb25seSByZXN0QXBpTmFtZT86IHN0cmluZztcblxuICAvKipcbiAgICogRGVzY3JpcHRpb24gZm9yIHRoZSBBUEkgR2F0ZXdheS5cbiAgICpcbiAgICogQGRlZmF1bHQgLSBcIlByaXZhdGUgUkVTVCBBUEkgR2F0ZXdheVwiXG4gICAqL1xuICByZWFkb25seSBkZXNjcmlwdGlvbj86IHN0cmluZztcblxuICAvKipcbiAgICogRGVwbG95IG9wdGlvbnMgZm9yIHRoZSBBUEkgR2F0ZXdheS5cbiAgICovXG4gIHJlYWRvbmx5IGRlcGxveU9wdGlvbnM/OiBhcGlnYXRld2F5LlN0YWdlT3B0aW9ucztcblxuICAvKipcbiAgICogUG9saWN5IGZvciB0aGUgQVBJIEdhdGV3YXkuXG4gICAqXG4gICAqIEBkZWZhdWx0IC0gUG9saWN5IHRoYXQgYWxsb3dzIGFueSBwcmluY2lwYWwgd2l0aCB0aGUgc2FtZSBWUEMgdG8gaW52b2tlIHRoZSBBUEkuXG4gICAqL1xuICByZWFkb25seSBwb2xpY3k/OiBpYW0uUG9saWN5RG9jdW1lbnQ7XG59XG5cbmV4cG9ydCBjbGFzcyBQcml2YXRlTGFtYmRhQXBpR2F0ZXdheSBleHRlbmRzIENvbnN0cnVjdCB7XG4gIHB1YmxpYyByZWFkb25seSBhcGk6IGFwaWdhdGV3YXkuUmVzdEFwaTtcbiAgcHVibGljIHJlYWRvbmx5IHZwY0VuZHBvaW50PzogZWMyLkludGVyZmFjZVZwY0VuZHBvaW50O1xuXG4gIGNvbnN0cnVjdG9yKFxuICAgIHNjb3BlOiBDb25zdHJ1Y3QsXG4gICAgaWQ6IHN0cmluZyxcbiAgICBwcm9wczogUHJpdmF0ZUxhbWJkYUFwaUdhdGV3YXlQcm9wc1xuICApIHtcbiAgICBzdXBlcihzY29wZSwgaWQpO1xuXG4gICAgY29uc3Qge1xuICAgICAgcmVzdEFwaU5hbWUsXG4gICAgICBkZXNjcmlwdGlvbixcbiAgICAgIGxhbWJkYUZ1bmN0aW9uLFxuICAgICAgdnBjLFxuICAgICAgdnBjRW5kcG9pbnRTdWJuZXRTZWxlY3Rpb24sXG4gICAgICBjcmVhdGVWcGNFbmRwb2ludCA9IHRydWUsXG4gICAgICBkZXBsb3lPcHRpb25zLFxuICAgICAgcG9saWN5LFxuICAgICAgbGFtYmRhSW50ZWdyYXRpb25PcHRpb25zLFxuICAgIH0gPSBwcm9wcztcblxuICAgIGlmIChjcmVhdGVWcGNFbmRwb2ludCkge1xuICAgICAgLy8gQ3JlYXRlIFZQQyBFbmRwb2ludCBmb3IgQVBJIEdhdGV3YXlcbiAgICAgIHRoaXMudnBjRW5kcG9pbnQgPSBuZXcgZWMyLkludGVyZmFjZVZwY0VuZHBvaW50KHRoaXMsIFwidnBjLWVuZHBvaW50XCIsIHtcbiAgICAgICAgdnBjLFxuICAgICAgICBzZXJ2aWNlOiBlYzIuSW50ZXJmYWNlVnBjRW5kcG9pbnRBd3NTZXJ2aWNlLkFQSUdBVEVXQVksXG4gICAgICAgIHN1Ym5ldHM6IHZwY0VuZHBvaW50U3VibmV0U2VsZWN0aW9uLFxuICAgICAgfSk7XG4gICAgfVxuXG4gICAgY29uc3QgZGVmYXVsdEludGVncmF0aW9uID0gbmV3IGFwaWdhdGV3YXkuTGFtYmRhSW50ZWdyYXRpb24oXG4gICAgICBsYW1iZGFGdW5jdGlvbixcbiAgICAgIGxhbWJkYUludGVncmF0aW9uT3B0aW9uc1xuICAgICk7XG5cbiAgICAvLyBDcmVhdGUgUHJpdmF0ZSBSRVNUIEFQSSBHYXRld2F5XG4gICAgdGhpcy5hcGkgPSBuZXcgYXBpZ2F0ZXdheS5SZXN0QXBpKHRoaXMsIFwicmVzdC1hcGlcIiwge1xuICAgICAgcmVzdEFwaU5hbWU6IHJlc3RBcGlOYW1lID8/IGAke3Njb3BlLm5vZGUuaWR9LXByaXZhdGUtYXBpYCxcbiAgICAgIGRlc2NyaXB0aW9uOiBkZXNjcmlwdGlvbiA/PyBcIlByaXZhdGUgUkVTVCBBUEkgR2F0ZXdheVwiLFxuICAgICAgZW5kcG9pbnRUeXBlczogW2FwaWdhdGV3YXkuRW5kcG9pbnRUeXBlLlBSSVZBVEVdLFxuICAgICAgcG9saWN5OlxuICAgICAgICBwb2xpY3kgPz9cbiAgICAgICAgbmV3IGlhbS5Qb2xpY3lEb2N1bWVudCh7XG4gICAgICAgICAgc3RhdGVtZW50czogW1xuICAgICAgICAgICAgbmV3IGlhbS5Qb2xpY3lTdGF0ZW1lbnQoe1xuICAgICAgICAgICAgICBlZmZlY3Q6IGlhbS5FZmZlY3QuQUxMT1csXG4gICAgICAgICAgICAgIHByaW5jaXBhbHM6IFtuZXcgaWFtLkFueVByaW5jaXBhbCgpXSxcbiAgICAgICAgICAgICAgYWN0aW9uczogW1wiZXhlY3V0ZS1hcGk6SW52b2tlXCJdLFxuICAgICAgICAgICAgICByZXNvdXJjZXM6IFtcImV4ZWN1dGUtYXBpOi8qXCJdLFxuICAgICAgICAgICAgICBjb25kaXRpb25zOiB7XG4gICAgICAgICAgICAgICAgU3RyaW5nRXF1YWxzOiB7IFwiYXdzOlNvdXJjZVZwY1wiOiB2cGMudnBjSWQgfSxcbiAgICAgICAgICAgICAgfSxcbiAgICAgICAgICAgIH0pLFxuICAgICAgICAgIF0sXG4gICAgICAgIH0pLFxuICAgICAgZGVwbG95T3B0aW9uczogZGVwbG95T3B0aW9ucyA/PyB7XG4gICAgICAgIGxvZ2dpbmdMZXZlbDogYXBpZ2F0ZXdheS5NZXRob2RMb2dnaW5nTGV2ZWwuSU5GTyxcbiAgICAgICAgZGF0YVRyYWNlRW5hYmxlZDogdHJ1ZSxcbiAgICAgIH0sXG4gICAgICBkZWZhdWx0SW50ZWdyYXRpb24sXG4gICAgICBkZWZhdWx0TWV0aG9kT3B0aW9uczoge1xuICAgICAgICBhdXRob3JpemF0aW9uVHlwZTogYXBpZ2F0ZXdheS5BdXRob3JpemF0aW9uVHlwZS5OT05FLFxuICAgICAgfSxcbiAgICB9KTtcblxuICAgIHRoaXMuYXBpLnJvb3QuYWRkTWV0aG9kKFwiQU5ZXCIpO1xuICAgIHRoaXMuYXBpLnJvb3QuYWRkUmVzb3VyY2UoXCJ7cHJveHkrfVwiKS5hZGRNZXRob2QoXCJBTllcIik7XG4gIH1cbn1cbiJdfQ==
@@ -49,7 +49,7 @@ class PgStacApiLambdaRuntime extends constructs_1.Construct {
49
49
  this.lambdaFunction = new aws_cdk_lib_1.aws_lambda.Function(this, "lambda", {
50
50
  // defaults
51
51
  runtime: aws_cdk_lib_1.aws_lambda.Runtime.PYTHON_3_12,
52
- handler: "handler.handler",
52
+ handler: "stac_api.handler.handler",
53
53
  memorySize: 8192,
54
54
  logRetention: aws_cdk_lib_1.aws_logs.RetentionDays.ONE_WEEK,
55
55
  timeout: aws_cdk_lib_1.Duration.seconds(30),
@@ -81,7 +81,7 @@ class PgStacApiLambdaRuntime extends constructs_1.Construct {
81
81
  }
82
82
  exports.PgStacApiLambdaRuntime = PgStacApiLambdaRuntime;
83
83
  _a = JSII_RTTI_SYMBOL_1;
84
- PgStacApiLambdaRuntime[_a] = { fqn: "eoapi-cdk.PgStacApiLambdaRuntime", version: "10.3.0" };
84
+ PgStacApiLambdaRuntime[_a] = { fqn: "eoapi-cdk.PgStacApiLambdaRuntime", version: "10.4.0" };
85
85
  class PgStacApiLambda extends constructs_1.Construct {
86
86
  constructor(scope, id, props) {
87
87
  super(scope, id);
@@ -111,5 +111,5 @@ class PgStacApiLambda extends constructs_1.Construct {
111
111
  }
112
112
  exports.PgStacApiLambda = PgStacApiLambda;
113
113
  _b = JSII_RTTI_SYMBOL_1;
114
- PgStacApiLambda[_b] = { fqn: "eoapi-cdk.PgStacApiLambda", version: "10.3.0" };
115
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["index.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  aws_apigatewayv2 as apigatewayv2,\n  aws_logs,\n  CfnOutput,\n  Duration,\n  aws_ec2 as ec2,\n  aws_lambda as lambda,\n  aws_rds as rds,\n  aws_secretsmanager as secretsmanager,\n  Stack,\n} from \"aws-cdk-lib\";\nimport { Construct } from \"constructs\";\nimport * as path from \"path\";\nimport { LambdaApiGateway } from \"../lambda-api-gateway\";\nimport { CustomLambdaFunctionProps, resolveLambdaCode } from \"../utils\";\n\nexport const EXTENSIONS = {\n  QUERY: \"query\",\n  SORT: \"sort\",\n  FIELDS: \"fields\",\n  FILTER: \"filter\",\n  FREE_TEXT: \"free_text\",\n  PAGINATION: \"pagination\",\n  COLLECTION_SEARCH: \"collection_search\",\n  TRANSACTION: \"transaction\",\n  BULK_TRANSACTION: \"bulk_transactions\",\n} as const;\n\ntype ExtensionType = (typeof EXTENSIONS)[keyof typeof EXTENSIONS];\n\n/**\n * Validates if a given string is a valid STAC extension\n */\nfunction isValidExtension(value: string): value is ExtensionType {\n  return Object.values(EXTENSIONS).includes(value as any);\n}\n\nexport class PgStacApiLambdaRuntime extends Construct {\n  public readonly lambdaFunction: lambda.Function;\n\n  constructor(\n    scope: Construct,\n    id: string,\n    props: PgStacApiLambdaRuntimeProps\n  ) {\n    super(scope, id);\n\n    const defaultExtensions: ExtensionType[] = [\n      EXTENSIONS.QUERY,\n      EXTENSIONS.SORT,\n      EXTENSIONS.FIELDS,\n      EXTENSIONS.FILTER,\n      EXTENSIONS.FREE_TEXT,\n      EXTENSIONS.PAGINATION,\n      EXTENSIONS.COLLECTION_SEARCH,\n    ];\n\n    if (props.enabledExtensions) {\n      for (const ext of props.enabledExtensions) {\n        if (!isValidExtension(ext)) {\n          throw new Error(\n            `Invalid extension: \"${ext}\". Must be one of: ${Object.values(\n              EXTENSIONS\n            ).join(\", \")}`\n          );\n        }\n      }\n    }\n\n    const enabledExtensions = props.enabledExtensions || defaultExtensions;\n\n    const { code: userCode, ...otherLambdaOptions } =\n      props.lambdaFunctionOptions || {};\n\n    this.lambdaFunction = new lambda.Function(this, \"lambda\", {\n      // defaults\n      runtime: lambda.Runtime.PYTHON_3_12,\n      handler: \"handler.handler\",\n      memorySize: 8192,\n      logRetention: aws_logs.RetentionDays.ONE_WEEK,\n      timeout: Duration.seconds(30),\n      code: resolveLambdaCode(userCode, path.join(__dirname, \"..\"), {\n        file: \"stac-api/runtime/Dockerfile\",\n        buildArgs: { PYTHON_VERSION: \"3.12\" },\n      }),\n      vpc: props.vpc,\n      vpcSubnets: props.subnetSelection,\n      allowPublicSubnet: true,\n      environment: {\n        PGSTAC_SECRET_ARN: props.dbSecret.secretArn,\n        DB_MIN_CONN_SIZE: \"0\",\n        DB_MAX_CONN_SIZE: \"1\",\n        ENABLED_EXTENSIONS: enabledExtensions.join(\",\"),\n        ...props.apiEnv,\n      },\n      snapStart: props.enableSnapStart\n        ? lambda.SnapStartConf.ON_PUBLISHED_VERSIONS\n        : undefined,\n      // overwrites defaults with user-provided configurable properties (excluding code)\n      ...otherLambdaOptions,\n    });\n\n    props.dbSecret.grantRead(this.lambdaFunction);\n\n    if (props.vpc) {\n      this.lambdaFunction.connections.allowTo(\n        props.db,\n        ec2.Port.tcp(5432),\n        \"allow connections from stac-fastapi-pgstac\"\n      );\n    }\n  }\n}\n\nexport interface PgStacApiLambdaRuntimeProps {\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 fastapi-pgstac runtime.\n   */\n  readonly apiEnv?: Record<string, string>;\n\n  /**\n   * List of STAC API extensions to enable.\n   *\n   * @default - query, sort, fields, filter, free_text, pagination, collection_search\n   */\n  readonly enabledExtensions?: ExtensionType[];\n\n  /**\n   * Enable SnapStart to reduce cold start latency.\n   *\n   * SnapStart creates a snapshot of the initialized Lambda function, allowing new instances\n   * to start from this pre-initialized state instead of starting from scratch.\n   *\n   * Benefits:\n   * - Significantly reduces cold start times (typically 10x faster)\n   * - Improves API response time for infrequent requests\n   *\n   * Considerations:\n   * - Additional cost: charges for snapshot storage and restore operations\n   * - Requires Lambda versioning (automatically configured by this construct)\n   * - Database connections are recreated on restore using snapshot lifecycle hooks\n   *\n   * @see https://docs.aws.amazon.com/lambda/latest/dg/snapstart.html\n   * @default false\n   */\n  readonly enableSnapStart?: boolean;\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\nexport class PgStacApiLambda extends Construct {\n  /**\n   * URL for the STAC API.\n   */\n  readonly url: string;\n\n  /**\n   * Lambda function for the STAC API.\n   */\n  readonly lambdaFunction: lambda.Function;\n\n  /**\n   * @deprecated - use lambdaFunction instead\n   */\n  public stacApiLambdaFunction: lambda.Function;\n\n  constructor(scope: Construct, id: string, props: PgStacApiLambdaProps) {\n    super(scope, id);\n\n    const runtime = new PgStacApiLambdaRuntime(this, \"runtime\", {\n      vpc: props.vpc,\n      subnetSelection: props.subnetSelection,\n      db: props.db,\n      dbSecret: props.dbSecret,\n      enabledExtensions: props.enabledExtensions,\n      apiEnv: props.apiEnv,\n      enableSnapStart: props.enableSnapStart,\n      lambdaFunctionOptions: props.lambdaFunctionOptions,\n    });\n    this.stacApiLambdaFunction = this.lambdaFunction = runtime.lambdaFunction;\n\n    const { api } = new LambdaApiGateway(this, \"stac-api\", {\n      lambdaFunction: props.enableSnapStart!\n        ? runtime.lambdaFunction.currentVersion\n        : runtime.lambdaFunction,\n      domainName: props.domainName ?? props.stacApiDomainName,\n    });\n\n    this.url = api.url!;\n\n    new CfnOutput(this, \"stac-api-output\", {\n      exportName: `${Stack.of(this).stackName}-url`,\n      value: this.url,\n    });\n  }\n}\n\nexport interface PgStacApiLambdaProps extends PgStacApiLambdaRuntimeProps {\n  /**\n   * Domain Name for the STAC API. If defined, will create the domain name and integrate it with the STAC API.\n   *\n   * @default - undefined\n   */\n  readonly domainName?: apigatewayv2.IDomainName;\n\n  /**\n   * Custom Domain Name Options for STAC API.\n   *\n   * @deprecated Use 'domainName' instead.\n   * @default - undefined.\n   */\n  readonly stacApiDomainName?: apigatewayv2.IDomainName;\n}\n"]}
114
+ PgStacApiLambda[_b] = { fqn: "eoapi-cdk.PgStacApiLambda", version: "10.4.0" };
115
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["index.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  aws_apigatewayv2 as apigatewayv2,\n  aws_logs,\n  CfnOutput,\n  Duration,\n  aws_ec2 as ec2,\n  aws_lambda as lambda,\n  aws_rds as rds,\n  aws_secretsmanager as secretsmanager,\n  Stack,\n} from \"aws-cdk-lib\";\nimport { Construct } from \"constructs\";\nimport * as path from \"path\";\nimport { LambdaApiGateway } from \"../lambda-api-gateway\";\nimport { CustomLambdaFunctionProps, resolveLambdaCode } from \"../utils\";\n\nexport const EXTENSIONS = {\n  QUERY: \"query\",\n  SORT: \"sort\",\n  FIELDS: \"fields\",\n  FILTER: \"filter\",\n  FREE_TEXT: \"free_text\",\n  PAGINATION: \"pagination\",\n  COLLECTION_SEARCH: \"collection_search\",\n  TRANSACTION: \"transaction\",\n  BULK_TRANSACTION: \"bulk_transactions\",\n} as const;\n\ntype ExtensionType = (typeof EXTENSIONS)[keyof typeof EXTENSIONS];\n\n/**\n * Validates if a given string is a valid STAC extension\n */\nfunction isValidExtension(value: string): value is ExtensionType {\n  return Object.values(EXTENSIONS).includes(value as any);\n}\n\nexport class PgStacApiLambdaRuntime extends Construct {\n  public readonly lambdaFunction: lambda.Function;\n\n  constructor(\n    scope: Construct,\n    id: string,\n    props: PgStacApiLambdaRuntimeProps,\n  ) {\n    super(scope, id);\n\n    const defaultExtensions: ExtensionType[] = [\n      EXTENSIONS.QUERY,\n      EXTENSIONS.SORT,\n      EXTENSIONS.FIELDS,\n      EXTENSIONS.FILTER,\n      EXTENSIONS.FREE_TEXT,\n      EXTENSIONS.PAGINATION,\n      EXTENSIONS.COLLECTION_SEARCH,\n    ];\n\n    if (props.enabledExtensions) {\n      for (const ext of props.enabledExtensions) {\n        if (!isValidExtension(ext)) {\n          throw new Error(\n            `Invalid extension: \"${ext}\". Must be one of: ${Object.values(\n              EXTENSIONS,\n            ).join(\", \")}`,\n          );\n        }\n      }\n    }\n\n    const enabledExtensions = props.enabledExtensions || defaultExtensions;\n\n    const { code: userCode, ...otherLambdaOptions } =\n      props.lambdaFunctionOptions || {};\n\n    this.lambdaFunction = new lambda.Function(this, \"lambda\", {\n      // defaults\n      runtime: lambda.Runtime.PYTHON_3_12,\n      handler: \"stac_api.handler.handler\",\n      memorySize: 8192,\n      logRetention: aws_logs.RetentionDays.ONE_WEEK,\n      timeout: Duration.seconds(30),\n      code: resolveLambdaCode(userCode, path.join(__dirname, \"..\"), {\n        file: \"stac-api/runtime/Dockerfile\",\n        buildArgs: { PYTHON_VERSION: \"3.12\" },\n      }),\n      vpc: props.vpc,\n      vpcSubnets: props.subnetSelection,\n      allowPublicSubnet: true,\n      environment: {\n        PGSTAC_SECRET_ARN: props.dbSecret.secretArn,\n        DB_MIN_CONN_SIZE: \"0\",\n        DB_MAX_CONN_SIZE: \"1\",\n        ENABLED_EXTENSIONS: enabledExtensions.join(\",\"),\n        ...props.apiEnv,\n      },\n      snapStart: props.enableSnapStart\n        ? lambda.SnapStartConf.ON_PUBLISHED_VERSIONS\n        : undefined,\n      // overwrites defaults with user-provided configurable properties (excluding code)\n      ...otherLambdaOptions,\n    });\n\n    props.dbSecret.grantRead(this.lambdaFunction);\n\n    if (props.vpc) {\n      this.lambdaFunction.connections.allowTo(\n        props.db,\n        ec2.Port.tcp(5432),\n        \"allow connections from stac-fastapi-pgstac\",\n      );\n    }\n  }\n}\n\nexport interface PgStacApiLambdaRuntimeProps {\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 fastapi-pgstac runtime.\n   */\n  readonly apiEnv?: Record<string, string>;\n\n  /**\n   * List of STAC API extensions to enable.\n   *\n   * @default - query, sort, fields, filter, free_text, pagination, collection_search\n   */\n  readonly enabledExtensions?: ExtensionType[];\n\n  /**\n   * Enable SnapStart to reduce cold start latency.\n   *\n   * SnapStart creates a snapshot of the initialized Lambda function, allowing new instances\n   * to start from this pre-initialized state instead of starting from scratch.\n   *\n   * Benefits:\n   * - Significantly reduces cold start times (typically 10x faster)\n   * - Improves API response time for infrequent requests\n   *\n   * Considerations:\n   * - Additional cost: charges for snapshot storage and restore operations\n   * - Requires Lambda versioning (automatically configured by this construct)\n   * - Database connections are recreated on restore using snapshot lifecycle hooks\n   *\n   * @see https://docs.aws.amazon.com/lambda/latest/dg/snapstart.html\n   * @default false\n   */\n  readonly enableSnapStart?: boolean;\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\nexport class PgStacApiLambda extends Construct {\n  /**\n   * URL for the STAC API.\n   */\n  readonly url: string;\n\n  /**\n   * Lambda function for the STAC API.\n   */\n  readonly lambdaFunction: lambda.Function;\n\n  /**\n   * @deprecated - use lambdaFunction instead\n   */\n  public stacApiLambdaFunction: lambda.Function;\n\n  constructor(scope: Construct, id: string, props: PgStacApiLambdaProps) {\n    super(scope, id);\n\n    const runtime = new PgStacApiLambdaRuntime(this, \"runtime\", {\n      vpc: props.vpc,\n      subnetSelection: props.subnetSelection,\n      db: props.db,\n      dbSecret: props.dbSecret,\n      enabledExtensions: props.enabledExtensions,\n      apiEnv: props.apiEnv,\n      enableSnapStart: props.enableSnapStart,\n      lambdaFunctionOptions: props.lambdaFunctionOptions,\n    });\n    this.stacApiLambdaFunction = this.lambdaFunction = runtime.lambdaFunction;\n\n    const { api } = new LambdaApiGateway(this, \"stac-api\", {\n      lambdaFunction: props.enableSnapStart!\n        ? runtime.lambdaFunction.currentVersion\n        : runtime.lambdaFunction,\n      domainName: props.domainName ?? props.stacApiDomainName,\n    });\n\n    this.url = api.url!;\n\n    new CfnOutput(this, \"stac-api-output\", {\n      exportName: `${Stack.of(this).stackName}-url`,\n      value: this.url,\n    });\n  }\n}\n\nexport interface PgStacApiLambdaProps extends PgStacApiLambdaRuntimeProps {\n  /**\n   * Domain Name for the STAC API. If defined, will create the domain name and integrate it with the STAC API.\n   *\n   * @default - undefined\n   */\n  readonly domainName?: apigatewayv2.IDomainName;\n\n  /**\n   * Custom Domain Name Options for STAC API.\n   *\n   * @deprecated Use 'domainName' instead.\n   * @default - undefined.\n   */\n  readonly stacApiDomainName?: apigatewayv2.IDomainName;\n}\n"]}
@@ -1,14 +1,33 @@
1
1
  ARG PYTHON_VERSION
2
2
  FROM --platform=linux/amd64 public.ecr.aws/lambda/python:${PYTHON_VERSION}
3
+ COPY --from=ghcr.io/astral-sh/uv:latest /uv /uvx /bin/
3
4
 
4
5
  WORKDIR /tmp
5
- RUN python -m pip install pip -U
6
+ COPY stac-api/runtime/uv.lock stac-api/runtime/pyproject.toml ./
7
+ COPY stac-api/runtime/src/ ./src/
8
+ COPY utils/utils.py /asset/
6
9
 
7
- COPY stac-api/runtime/requirements.txt requirements.txt
8
- RUN python -m pip install -r requirements.txt -t /asset --no-binary pydantic
10
+ RUN <<EOF
11
+ uv export --locked --no-editable --no-dev --format requirements.txt -o requirements.txt
12
+ uv pip install \
13
+ --compile-bytecode \
14
+ --no-binary pydantic \
15
+ --target /asset \
16
+ --no-cache-dir \
17
+ --disable-pip-version-check \
18
+ -r requirements.txt
19
+ EOF
9
20
 
10
- RUN mkdir -p /asset/src
11
- COPY stac-api/runtime/src/*.py /asset/
12
- COPY utils/utils.py /asset/
21
+ # Reduce package size and remove useless files
22
+ WORKDIR /asset
23
+ RUN <<EOF
24
+ dnf install -y findutils && \
25
+ dnf clean all && \
26
+ rm -rf /var/cache/dnf
27
+ find . -type f -name '*.pyc' | while read f; do n=$(echo $f | sed 's/__pycache__\///' | sed 's/.cpython-[0-9]*//'); cp $f $n; done;
28
+ find . -type d -a -name '__pycache__' -print0 | xargs -0 rm -rf
29
+ find . -type f -a -name '*.py' -print0 | xargs -0 rm -f
30
+ find . -type d -a -name 'tests' -print0 | xargs -0 rm -rf
31
+ EOF
13
32
 
14
33
  CMD ["echo", "hello world"]
@@ -0,0 +1,14 @@
1
+ [project]
2
+ name = "stac-api"
3
+ version = "0.0.0"
4
+ description = "stac-api runtime"
5
+ authors = [{ name = "hrodmn", email = "henry@developmentseed.org" }]
6
+ requires-python = ">=3.12"
7
+ dependencies = [
8
+ "stac-fastapi-pgstac[awslambda]>=6.0,<6.1",
9
+ "starlette-cramjam>=0.4,<0.5",
10
+ ]
11
+
12
+ [build-system]
13
+ requires = ["hatchling"]
14
+ build-backend = "hatchling.build"