eoapi-cdk 8.2.3 → 8.3.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/.jsii CHANGED
@@ -5459,21 +5459,85 @@
5459
5459
  "symbolId": "lib/ingestor-api/index:StacIngestorProps"
5460
5460
  },
5461
5461
  "eoapi-cdk.StacItemLoader": {
5462
+ "assembly": "eoapi-cdk",
5463
+ "base": "eoapi-cdk.StacLoader",
5464
+ "docs": {
5465
+ "deprecated": "Use StacLoader instead. StacItemLoader will be removed in a future version.",
5466
+ "stability": "deprecated"
5467
+ },
5468
+ "fqn": "eoapi-cdk.StacItemLoader",
5469
+ "initializer": {
5470
+ "docs": {
5471
+ "stability": "deprecated"
5472
+ },
5473
+ "locationInModule": {
5474
+ "filename": "lib/stac-loader/index.ts",
5475
+ "line": 484
5476
+ },
5477
+ "parameters": [
5478
+ {
5479
+ "name": "scope",
5480
+ "type": {
5481
+ "fqn": "constructs.Construct"
5482
+ }
5483
+ },
5484
+ {
5485
+ "name": "id",
5486
+ "type": {
5487
+ "primitive": "string"
5488
+ }
5489
+ },
5490
+ {
5491
+ "name": "props",
5492
+ "type": {
5493
+ "fqn": "eoapi-cdk.StacLoaderProps"
5494
+ }
5495
+ }
5496
+ ]
5497
+ },
5498
+ "kind": "class",
5499
+ "locationInModule": {
5500
+ "filename": "lib/stac-loader/index.ts",
5501
+ "line": 483
5502
+ },
5503
+ "name": "StacItemLoader",
5504
+ "symbolId": "lib/stac-loader/index:StacItemLoader"
5505
+ },
5506
+ "eoapi-cdk.StacItemLoaderProps": {
5507
+ "assembly": "eoapi-cdk",
5508
+ "datatype": true,
5509
+ "docs": {
5510
+ "deprecated": "Use StacLoaderProps instead. StacItemLoaderProps will be removed in a future version.",
5511
+ "stability": "deprecated"
5512
+ },
5513
+ "fqn": "eoapi-cdk.StacItemLoaderProps",
5514
+ "interfaces": [
5515
+ "eoapi-cdk.StacLoaderProps"
5516
+ ],
5517
+ "kind": "interface",
5518
+ "locationInModule": {
5519
+ "filename": "lib/stac-loader/index.ts",
5520
+ "line": 498
5521
+ },
5522
+ "name": "StacItemLoaderProps",
5523
+ "symbolId": "lib/stac-loader/index:StacItemLoaderProps"
5524
+ },
5525
+ "eoapi-cdk.StacLoader": {
5462
5526
  "assembly": "eoapi-cdk",
5463
5527
  "base": "constructs.Construct",
5464
5528
  "docs": {
5465
- "remarks": "The StacItemLoader creates a serverless, event-driven system for loading\nSTAC (SpatioTemporal Asset Catalog) items into a PostgreSQL database with\nthe pgstac extension. This construct supports multiple ingestion pathways\nfor flexible STAC item loading.\n\n## Architecture Overview\n\nThis construct creates the following AWS resources:\n- **SNS Topic**: Entry point for STAC items and S3 event notifications\n- **SQS Queue**: Buffers and batches messages before processing (60-second visibility timeout)\n- **Dead Letter Queue**: Captures failed loading attempts after 5 retries\n- **Lambda Function**: Python function that processes batches and inserts items into pgstac\n\n## Data Flow\n\nThe loader supports two primary data ingestion patterns:\n\n### Direct STAC Item Publishing\n1. STAC items (JSON) are published directly to the SNS topic in message bodies\n2. The SQS queue collects messages and batches them (up to {batchSize} items or 1 minute window)\n3. The Lambda function receives batches, validates items, and inserts into pgstac\n\n### S3 Event-Driven Loading\n1. An S3 bucket is configured to send notifications to the SNS topic when json files are created\n2. STAC items are uploaded to S3 buckets as JSON/GeoJSON files\n3. S3 event notifications are sent to the SNS topic when items are uploaded\n4. The Lambda function receives S3 events in the SQS message batch, fetches items from S3, and loads into pgstac\n\n## Batching Behavior\n\nThe SQS-to-Lambda integration uses intelligent batching to optimize performance:\n\n- **Batch Size**: Lambda waits to receive up to `batchSize` messages (default: 500)\n- **Batching Window**: If fewer than `batchSize` messages are available, Lambda\n triggers after `maxBatchingWindow` minutes (default: 1 minute)\n- **Trigger Condition**: Lambda executes when EITHER condition is met first\n- **Concurrency**: Limited to `maxConcurrency` concurrent executions to prevent database overload\n- **Partial Failures**: Uses `reportBatchItemFailures` to retry only failed items\n\nThis approach balances throughput (larger batches = fewer database connections)\nwith latency (time-based triggers prevent indefinite waiting).\n\n## Error Handling and Dead Letter Queue\n\nFailed messages are sent to the dead letter queue after 5 processing attempts.\n**Important**: This construct provides NO automated handling of dead letter queue\nmessages - monitoring, inspection, and reprocessing of failed items is the\nresponsibility of the implementing application.\n\nConsider implementing:\n- CloudWatch alarms on dead letter queue depth\n- Manual or automated reprocessing workflows\n- Logging and alerting for failed items\n- Regular cleanup of old dead letter messages (14-day retention)\n\n## Operational Characteristics\n\n- **Scalability**: Lambda scales automatically based on queue depth\n- **Reliability**: Dead letter queue captures failures for debugging\n- **Efficiency**: Batching optimizes database operations for high throughput\n- **Security**: Database credentials accessed via AWS Secrets Manager\n- **Observability**: CloudWatch logs retained for one week\n\n## Prerequisites\n\nBefore using this construct, ensure:\n- The pgstac database has collections loaded (items require existing collection IDs)\n- Database credentials are stored in AWS Secrets Manager\n- The pgstac extension is properly installed and configured\n\n## Usage Example\n\n```typescript\n// Create database first\nconst database = new PgStacDatabase(this, 'Database', {\n pgstacVersion: '0.9.5'\n});\n\n// Create item loader\nconst loader = new StacItemLoader(this, 'ItemLoader', {\n pgstacDb: database,\n batchSize: 1000, // Process up to 1000 items per batch\n maxBatchingWindowMinutes: 1, // Wait max 1 minute to fill batch\n lambdaTimeoutSeconds: 300 // Allow up to 300 seconds for database operations\n});\n\n// The topic ARN can be used by other services to publish items\nnew CfnOutput(this, 'LoaderTopicArn', {\n value: loader.topic.topicArn\n});\n```\n\n## Direct Item Publishing\n\nExternal services can publish STAC items directly to the topic:\n\n```bash\naws sns publish --topic-arn $ITEM_LOAD_TOPIC --message '{\n \"type\": \"Feature\",\n \"stac_version\": \"1.0.0\",\n \"id\": \"example-item\",\n \"properties\": {\"datetime\": \"2021-01-01T00:00:00Z\"},\n \"geometry\": {\"type\": \"Polygon\", \"coordinates\": [...]},\n \"collection\": \"example-collection\"\n}'\n```\n\n## S3 Event Configuration\n\nTo enable S3 event-driven loading, configure S3 bucket notifications to send\nevents to the SNS topic when STAC items (.json or .geojson files) are uploaded:\n\n```typescript\n// Configure S3 bucket to send notifications to the loader topic\nbucket.addEventNotification(\n s3.EventType.OBJECT_CREATED,\n new s3n.SnsDestination(loader.topic),\n { suffix: '.json' }\n);\n\nbucket.addEventNotification(\n s3.EventType.OBJECT_CREATED,\n new s3n.SnsDestination(loader.topic),\n { suffix: '.geojson' }\n);\n```\n\nWhen STAC items are uploaded to the configured S3 bucket, the loader will:\n1. Receive S3 event notifications via SNS\n2. Fetch the STAC item JSON from S3\n3. Validate and load the item into the pgstac database\n\n## Monitoring and Troubleshooting\n\n- Monitor Lambda logs: `/aws/lambda/{FunctionName}`\n- **Dead Letter Queue**: Check for failed items - **no automated handling provided**\n- Use batch item failure reporting for partial batch processing\n- CloudWatch metrics available for queue depth and Lambda performance\n\n### Dead Letter Queue Management\n\nApplications must implement their own dead letter queue monitoring:\n\n```typescript\n// Example: CloudWatch alarm for dead letter queue depth\nnew cloudwatch.Alarm(this, 'DeadLetterAlarm', {\n metric: loader.deadLetterQueue.metricApproximateNumberOfVisibleMessages(),\n threshold: 1,\n evaluationPeriods: 1\n});\n\n// Example: Lambda to reprocess dead letter messages\nconst reprocessFunction = new lambda.Function(this, 'Reprocess', {\n // Implementation to fetch and republish failed messages\n});\n```",
5529
+ "remarks": "The StacLoader creates a serverless, event-driven system for loading\nSTAC (SpatioTemporal Asset Catalog) objects into a PostgreSQL database with\nthe pgstac extension. This construct supports multiple ingestion pathways\nfor flexible STAC object loading.\n\n## Architecture Overview\n\nThis construct creates the following AWS resources:\n- **SNS Topic**: Entry point for STAC objects and S3 event notifications\n- **SQS Queue**: Buffers and batches messages before processing (60-second visibility timeout)\n- **Dead Letter Queue**: Captures failed loading attempts after 5 retries\n- **Lambda Function**: Python function that processes batches and inserts objects into pgstac\n\n## Data Flow\n\nThe loader supports two primary data ingestion patterns:\n\n### Direct STAC Object Publishing\n1. STAC objects (JSON) are published directly to the SNS topic in message bodies\n2. The SQS queue collects messages and batches them (up to {batchSize} objects or 1 minute window)\n3. The Lambda function receives batches, validates objects, and inserts into pgstac\n\n### S3 Event-Driven Loading\n1. An S3 bucket is configured to send notifications to the SNS topic when json files are created\n2. STAC objects are uploaded to S3 buckets as JSON/GeoJSON files\n3. S3 event notifications are sent to the SNS topic when objects are uploaded\n4. The Lambda function receives S3 events in the SQS message batch, fetches objects from S3, and loads into pgstac\n\n## Batching Behavior\n\nThe SQS-to-Lambda integration uses intelligent batching to optimize performance:\n\n- **Batch Size**: Lambda waits to receive up to `batchSize` messages (default: 500)\n- **Batching Window**: If fewer than `batchSize` messages are available, Lambda\n triggers after `maxBatchingWindow` minutes (default: 1 minute)\n- **Trigger Condition**: Lambda executes when EITHER condition is met first\n- **Concurrency**: Limited to `maxConcurrency` concurrent executions to prevent database overload\n- **Partial Failures**: Uses `reportBatchItemFailures` to retry only failed objects\n\nThis approach balances throughput (larger batches = fewer database connections)\nwith latency (time-based triggers prevent indefinite waiting).\n\n## Error Handling and Dead Letter Queue\n\nFailed messages are sent to the dead letter queue after 5 processing attempts.\n**Important**: This construct provides NO automated handling of dead letter queue\nmessages - monitoring, inspection, and reprocessing of failed objects is the\nresponsibility of the implementing application.\n\nConsider implementing:\n- CloudWatch alarms on dead letter queue depth\n- Manual or automated reprocessing workflows\n- Logging and alerting for failed objects\n- Regular cleanup of old dead letter messages (14-day retention)\n\n## Operational Characteristics\n\n- **Scalability**: Lambda scales automatically based on queue depth\n- **Reliability**: Dead letter queue captures failures for debugging\n- **Efficiency**: Batching optimizes database operations for high throughput\n- **Security**: Database credentials accessed via AWS Secrets Manager\n- **Observability**: CloudWatch logs retained for one week\n\n## Prerequisites\n\nBefore using this construct, ensure:\n- The pgstac database has collections loaded (objects require existing collection IDs)\n- Database credentials are stored in AWS Secrets Manager\n- The pgstac extension is properly installed and configured\n\n## Usage Example\n\n```typescript\n// Create database first\nconst database = new PgStacDatabase(this, 'Database', {\n pgstacVersion: '0.9.5'\n});\n\n// Create Object loader\nconst loader = new StacLoader(this, 'StacLoader', {\n pgstacDb: database,\n batchSize: 1000, // Process up to 1000 objects per batch\n maxBatchingWindowMinutes: 1, // Wait max 1 minute to fill batch\n lambdaTimeoutSeconds: 300 // Allow up to 300 seconds for database operations\n});\n\n// The topic ARN can be used by other services to publish objects\nnew CfnOutput(this, 'LoaderTopicArn', {\n value: loader.topic.topicArn\n});\n```\n\n## Direct Object Publishing\n\nExternal services can publish STAC objects directly to the topic:\n\n```bash\naws sns publish --topic-arn $STAC_LOAD_TOPIC --message '{\n \"id\": \"example-collection\",\n \"type\": \"Collection\",\n \"title\": \"Example Collection\",\n \"description\": \"An example collection\",\n \"license\": \"proprietary\",\n \"extent\": {\n \"spatial\": {\"bbox\": [[-180, -90, 180, 90]]},\n \"temporal\": {\"interval\": [[null, null]]},\n },\n \"stac_version\": \"1.1.0\",\n}'\n\naws sns publish --topic-arn $STAC_LOAD_TOPIC --message '{\n \"type\": \"Feature\",\n \"stac_version\": \"1.0.0\",\n \"id\": \"example-item\",\n \"properties\": {\"datetime\": \"2021-01-01T00:00:00Z\"},\n \"geometry\": {\"type\": \"Polygon\", \"coordinates\": [...]},\n \"collection\": \"example-collection\"\n}'\n\n\n```\n\n## S3 Event Configuration\n\nTo enable S3 event-driven loading, configure S3 bucket notifications to send\nevents to the SNS topic when STAC objects (.json or .geojson files) are uploaded:\n\n```typescript\n// Configure S3 bucket to send notifications to the loader topic\nbucket.addEventNotification(\n s3.EventType.OBJECT_CREATED,\n new s3n.SnsDestination(loader.topic),\n { suffix: '.json' }\n);\n\nbucket.addEventNotification(\n s3.EventType.OBJECT_CREATED,\n new s3n.SnsDestination(loader.topic),\n { suffix: '.geojson' }\n);\n```\n\nWhen STAC objects are uploaded to the configured S3 bucket, the loader will:\n1. Receive S3 event notifications via SNS\n2. Fetch the STAC JSON from S3\n3. Validate and load the objects into the pgstac database\n\n## Monitoring and Troubleshooting\n\n- Monitor Lambda logs: `/aws/lambda/{FunctionName}`\n- **Dead Letter Queue**: Check for failed objects - **no automated handling provided**\n- Use batch objects failure reporting for partial batch processing\n- CloudWatch metrics available for queue depth and Lambda performance\n\n### Dead Letter Queue Management\n\nApplications must implement their own dead letter queue monitoring:\n\n```typescript\n// Example: CloudWatch alarm for dead letter queue depth\nnew cloudwatch.Alarm(this, 'DeadLetterAlarm', {\n metric: loader.deadLetterQueue.metricApproximateNumberOfVisibleMessages(),\n threshold: 1,\n evaluationPeriods: 1\n});\n\n// Example: Lambda to reprocess dead letter messages\nconst reprocessFunction = new lambda.Function(this, 'Reprocess', {\n // Implementation to fetch and republish failed messages\n});\n```",
5466
5530
  "stability": "experimental",
5467
- "summary": "AWS CDK Construct for STAC Item Loading Infrastructure."
5531
+ "summary": "AWS CDK Construct for STAC Object Loading Infrastructure."
5468
5532
  },
5469
- "fqn": "eoapi-cdk.StacItemLoader",
5533
+ "fqn": "eoapi-cdk.StacLoader",
5470
5534
  "initializer": {
5471
5535
  "docs": {
5472
5536
  "stability": "experimental"
5473
5537
  },
5474
5538
  "locationInModule": {
5475
- "filename": "lib/stac-item-loader/index.ts",
5476
- "line": 365
5539
+ "filename": "lib/stac-loader/index.ts",
5540
+ "line": 380
5477
5541
  },
5478
5542
  "parameters": [
5479
5543
  {
@@ -5491,28 +5555,28 @@
5491
5555
  {
5492
5556
  "name": "props",
5493
5557
  "type": {
5494
- "fqn": "eoapi-cdk.StacItemLoaderProps"
5558
+ "fqn": "eoapi-cdk.StacLoaderProps"
5495
5559
  }
5496
5560
  }
5497
5561
  ]
5498
5562
  },
5499
5563
  "kind": "class",
5500
5564
  "locationInModule": {
5501
- "filename": "lib/stac-item-loader/index.ts",
5502
- "line": 313
5565
+ "filename": "lib/stac-loader/index.ts",
5566
+ "line": 328
5503
5567
  },
5504
- "name": "StacItemLoader",
5568
+ "name": "StacLoader",
5505
5569
  "properties": [
5506
5570
  {
5507
5571
  "docs": {
5508
5572
  "remarks": "Messages that fail processing after 5 attempts are sent here\nfor inspection and potential replay. Retains messages for 14 days\nto allow for debugging and manual intervention.\n\n**User Responsibility**: This construct provides NO automated monitoring,\nalerting, or reprocessing of dead letter queue messages. Applications\nusing this construct must implement their own:\n- Dead letter queue depth monitoring and alerting\n- Failed message inspection and debugging workflows\n- Manual or automated reprocessing mechanisms\n- Cleanup procedures for old failed messages",
5509
5573
  "stability": "experimental",
5510
- "summary": "Dead letter queue for failed item loading attempts."
5574
+ "summary": "Dead letter queue for failed objects loading attempts."
5511
5575
  },
5512
5576
  "immutable": true,
5513
5577
  "locationInModule": {
5514
- "filename": "lib/stac-item-loader/index.ts",
5515
- "line": 350
5578
+ "filename": "lib/stac-loader/index.ts",
5579
+ "line": 365
5516
5580
  },
5517
5581
  "name": "deadLetterQueue",
5518
5582
  "type": {
@@ -5521,14 +5585,14 @@
5521
5585
  },
5522
5586
  {
5523
5587
  "docs": {
5524
- "remarks": "This Python function receives batches of messages from SQS and processes\nthem based on their type:\n- Direct STAC items: Validates and loads directly into pgstac\n- S3 events: Fetches STAC items from S3, validates, and loads into pgstac\n\nThe function connects to PostgreSQL using credentials from Secrets Manager\nand uses pypgstac for efficient database operations.",
5588
+ "remarks": "This Python function receives batches of messages from SQS and processes\nthem based on their type:\n- Direct STAC objects: Validates and loads directly into pgstac\n- S3 events: Fetches STAC JSON from S3, validates, and loads into pgstac\n\nThe function connects to PostgreSQL using credentials from Secrets Manager\nand uses pypgstac for efficient database operations.",
5525
5589
  "stability": "experimental",
5526
- "summary": "The Lambda function that loads STAC items into the pgstac database."
5590
+ "summary": "The Lambda function that loads STAC objects into the pgstac database."
5527
5591
  },
5528
5592
  "immutable": true,
5529
5593
  "locationInModule": {
5530
- "filename": "lib/stac-item-loader/index.ts",
5531
- "line": 363
5594
+ "filename": "lib/stac-loader/index.ts",
5595
+ "line": 378
5532
5596
  },
5533
5597
  "name": "lambdaFunction",
5534
5598
  "type": {
@@ -5537,14 +5601,14 @@
5537
5601
  },
5538
5602
  {
5539
5603
  "docs": {
5540
- "remarks": "This queue collects both direct STAC items from SNS and S3 event\nnotifications, batching them for efficient database operations.\nConfigured with a visibility timeout that accommodates Lambda\nprocessing time plus buffer.",
5604
+ "remarks": "This queue collects both direct STAC objects from SNS and S3 event\nnotifications, batching them for efficient database operations.\nConfigured with a visibility timeout that accommodates Lambda\nprocessing time plus buffer.",
5541
5605
  "stability": "experimental",
5542
5606
  "summary": "The SQS queue that buffers messages before processing."
5543
5607
  },
5544
5608
  "immutable": true,
5545
5609
  "locationInModule": {
5546
- "filename": "lib/stac-item-loader/index.ts",
5547
- "line": 333
5610
+ "filename": "lib/stac-loader/index.ts",
5611
+ "line": 348
5548
5612
  },
5549
5613
  "name": "queue",
5550
5614
  "type": {
@@ -5553,14 +5617,14 @@
5553
5617
  },
5554
5618
  {
5555
5619
  "docs": {
5556
- "remarks": "This topic serves as the entry point for two types of events:\n1. Direct STAC item JSON documents published by external services\n2. S3 event notifications when STAC items are uploaded to configured buckets\n\nThe topic fans out to the SQS queue for batched processing.",
5620
+ "remarks": "This topic serves as the entry point for two types of events:\n1. Direct STAC JSON documents published by external services\n2. S3 event notifications when STAC objects are uploaded to configured buckets\n\nThe topic fans out to the SQS queue for batched processing.",
5557
5621
  "stability": "experimental",
5558
- "summary": "The SNS topic that receives STAC items and S3 event notifications for loading."
5622
+ "summary": "The SNS topic that receives STAC objects and S3 event notifications for loading."
5559
5623
  },
5560
5624
  "immutable": true,
5561
5625
  "locationInModule": {
5562
- "filename": "lib/stac-item-loader/index.ts",
5563
- "line": 323
5626
+ "filename": "lib/stac-loader/index.ts",
5627
+ "line": 338
5564
5628
  },
5565
5629
  "name": "topic",
5566
5630
  "type": {
@@ -5568,35 +5632,35 @@
5568
5632
  }
5569
5633
  }
5570
5634
  ],
5571
- "symbolId": "lib/stac-item-loader/index:StacItemLoader"
5635
+ "symbolId": "lib/stac-loader/index:StacLoader"
5572
5636
  },
5573
- "eoapi-cdk.StacItemLoaderProps": {
5637
+ "eoapi-cdk.StacLoaderProps": {
5574
5638
  "assembly": "eoapi-cdk",
5575
5639
  "datatype": true,
5576
5640
  "docs": {
5577
- "example": "const loader = new StacItemLoader(this, 'ItemLoader', {\n pgstacDb: database,\n batchSize: 1000,\n maxBatchingWindowMinutes: 1,\n lambdaTimeoutSeconds: 300\n});",
5578
- "remarks": "The StacItemLoader is part of a two-phase serverless STAC ingestion pipeline\nthat loads STAC items into a pgstac database. This construct creates\nthe infrastructure for receiving STAC items from multiple sources:\n1. SNS messages containing STAC metadata (direct ingestion)\n2. S3 event notifications for STAC items uploaded to S3 buckets\n\nItems from both sources are batched and inserted into PostgreSQL with the pgstac extension.",
5641
+ "example": "const loader = new StacLoader(this, 'StacLoader', {\n pgstacDb: database,\n batchSize: 1000,\n maxBatchingWindowMinutes: 1,\n lambdaTimeoutSeconds: 300\n});",
5642
+ "remarks": "The StacLoader is part of a two-phase serverless STAC ingestion pipeline\nthat loads STAC collections and items into a pgstac database. This construct creates\nthe infrastructure for receiving STAC objects from multiple sources:\n1. SNS messages containing STAC metadata (direct ingestion)\n2. S3 event notifications for STAC objects uploaded to S3 buckets\n\nObjects from both sources are batched and inserted into PostgreSQL with the pgstac extension.",
5579
5643
  "stability": "experimental",
5580
- "summary": "Configuration properties for the StacItemLoader construct."
5644
+ "summary": "Configuration properties for the StacLoader construct."
5581
5645
  },
5582
- "fqn": "eoapi-cdk.StacItemLoaderProps",
5646
+ "fqn": "eoapi-cdk.StacLoaderProps",
5583
5647
  "kind": "interface",
5584
5648
  "locationInModule": {
5585
- "filename": "lib/stac-item-loader/index.ts",
5649
+ "filename": "lib/stac-loader/index.ts",
5586
5650
  "line": 36
5587
5651
  },
5588
- "name": "StacItemLoaderProps",
5652
+ "name": "StacLoaderProps",
5589
5653
  "properties": [
5590
5654
  {
5591
5655
  "abstract": true,
5592
5656
  "docs": {
5593
- "remarks": "This database must have the pgstac extension installed and be properly\nconfigured with collections before items can be loaded. The loader will\nuse AWS Secrets Manager to securely access database credentials.",
5657
+ "remarks": "This database must have the pgstac extension installed and be properly\nconfigured with collections before objects can be loaded. The loader will\nuse AWS Secrets Manager to securely access database credentials.",
5594
5658
  "stability": "experimental",
5595
- "summary": "The PgSTAC database instance to load items into."
5659
+ "summary": "The PgSTAC database instance to load data into."
5596
5660
  },
5597
5661
  "immutable": true,
5598
5662
  "locationInModule": {
5599
- "filename": "lib/stac-item-loader/index.ts",
5663
+ "filename": "lib/stac-loader/index.ts",
5600
5664
  "line": 44
5601
5665
  },
5602
5666
  "name": "pgstacDb",
@@ -5608,13 +5672,13 @@
5608
5672
  "abstract": true,
5609
5673
  "docs": {
5610
5674
  "default": "500",
5611
- "remarks": "This determines the maximum number of STAC items that will be\nprocessed together in a single lambda invocation. Larger batch\nsizes improve database insertion efficiency but require more\nmemory and longer processing time.\n\n**Batching Behavior**: SQS will wait to accumulate up to this many\nmessages before triggering the Lambda, OR until the maxBatchingWindow\ntimeout is reached, whichever comes first. This creates an efficient\nbalance between throughput and latency.",
5675
+ "remarks": "This determines the maximum number of STAC objects that will be\nprocessed together in a single lambda invocation. Larger batch\nsizes improve database insertion efficiency but require more\nmemory and longer processing time.\n\n**Batching Behavior**: SQS will wait to accumulate up to this many\nmessages before triggering the Lambda, OR until the maxBatchingWindow\ntimeout is reached, whichever comes first. This creates an efficient\nbalance between throughput and latency.",
5612
5676
  "stability": "experimental",
5613
5677
  "summary": "SQS batch size for lambda event source."
5614
5678
  },
5615
5679
  "immutable": true,
5616
5680
  "locationInModule": {
5617
- "filename": "lib/stac-item-loader/index.ts",
5681
+ "filename": "lib/stac-loader/index.ts",
5618
5682
  "line": 104
5619
5683
  },
5620
5684
  "name": "batchSize",
@@ -5632,7 +5696,7 @@
5632
5696
  },
5633
5697
  "immutable": true,
5634
5698
  "locationInModule": {
5635
- "filename": "lib/stac-item-loader/index.ts",
5699
+ "filename": "lib/stac-loader/index.ts",
5636
5700
  "line": 142
5637
5701
  },
5638
5702
  "name": "environment",
@@ -5655,7 +5719,7 @@
5655
5719
  },
5656
5720
  "immutable": true,
5657
5721
  "locationInModule": {
5658
- "filename": "lib/stac-item-loader/index.ts",
5722
+ "filename": "lib/stac-loader/index.ts",
5659
5723
  "line": 149
5660
5724
  },
5661
5725
  "name": "lambdaFunctionOptions",
@@ -5674,7 +5738,7 @@
5674
5738
  },
5675
5739
  "immutable": true,
5676
5740
  "locationInModule": {
5677
- "filename": "lib/stac-item-loader/index.ts",
5741
+ "filename": "lib/stac-loader/index.ts",
5678
5742
  "line": 65
5679
5743
  },
5680
5744
  "name": "lambdaRuntime",
@@ -5687,13 +5751,13 @@
5687
5751
  "abstract": true,
5688
5752
  "docs": {
5689
5753
  "default": "300",
5690
- "remarks": "This should accommodate the time needed to process up to `batchSize`\nitems and perform database insertions. The SQS visibility timeout\nwill be set to this value plus 10 seconds.",
5754
+ "remarks": "This should accommodate the time needed to process up to `batchSize`\nobjects and perform database insertions. The SQS visibility timeout\nwill be set to this value plus 10 seconds.",
5691
5755
  "stability": "experimental",
5692
5756
  "summary": "The timeout for the item load lambda in seconds."
5693
5757
  },
5694
5758
  "immutable": true,
5695
5759
  "locationInModule": {
5696
- "filename": "lib/stac-item-loader/index.ts",
5760
+ "filename": "lib/stac-loader/index.ts",
5697
5761
  "line": 76
5698
5762
  },
5699
5763
  "name": "lambdaTimeoutSeconds",
@@ -5706,13 +5770,13 @@
5706
5770
  "abstract": true,
5707
5771
  "docs": {
5708
5772
  "default": "1",
5709
- "remarks": "Even if the batch size isn't reached, the lambda will be triggered\nafter this time period to ensure timely processing of items.\nThis prevents items from waiting indefinitely in low-volume scenarios.\n\n**Important**: This timeout works in conjunction with batchSize - SQS\nwill trigger the Lambda when EITHER the batch size is reached OR this\ntime window expires, ensuring items are processed in a timely manner\nregardless of volume.",
5773
+ "remarks": "Even if the batch size isn't reached, the lambda will be triggered\nafter this time period to ensure timely processing of objects.\nThis prevents objects from waiting indefinitely in low-volume scenarios.\n\n**Important**: This timeout works in conjunction with batchSize - SQS\nwill trigger the Lambda when EITHER the batch size is reached OR this\ntime window expires, ensuring objects are processed in a timely manner\nregardless of volume.",
5710
5774
  "stability": "experimental",
5711
5775
  "summary": "Maximum batching window in minutes."
5712
5776
  },
5713
5777
  "immutable": true,
5714
5778
  "locationInModule": {
5715
- "filename": "lib/stac-item-loader/index.ts",
5779
+ "filename": "lib/stac-loader/index.ts",
5716
5780
  "line": 120
5717
5781
  },
5718
5782
  "name": "maxBatchingWindowMinutes",
@@ -5727,11 +5791,11 @@
5727
5791
  "default": "2",
5728
5792
  "remarks": "This limit will be applied to the Lambda function and will control how\nmany concurrent batches will be released from the SQS queue.",
5729
5793
  "stability": "experimental",
5730
- "summary": "Maximum concurrent executions for the StacItemLoader Lambda function."
5794
+ "summary": "Maximum concurrent executions for the StacLoader Lambda function."
5731
5795
  },
5732
5796
  "immutable": true,
5733
5797
  "locationInModule": {
5734
- "filename": "lib/stac-item-loader/index.ts",
5798
+ "filename": "lib/stac-loader/index.ts",
5735
5799
  "line": 130
5736
5800
  },
5737
5801
  "name": "maxConcurrency",
@@ -5744,13 +5808,13 @@
5744
5808
  "abstract": true,
5745
5809
  "docs": {
5746
5810
  "default": "1024",
5747
- "remarks": "Higher memory allocation may improve performance when processing\nlarge batches of STAC items, especially for memory-intensive\ndatabase operations.",
5811
+ "remarks": "Higher memory allocation may improve performance when processing\nlarge batches of STAC objects, especially for memory-intensive\ndatabase operations.",
5748
5812
  "stability": "experimental",
5749
5813
  "summary": "Memory size for the lambda function in MB."
5750
5814
  },
5751
5815
  "immutable": true,
5752
5816
  "locationInModule": {
5753
- "filename": "lib/stac-item-loader/index.ts",
5817
+ "filename": "lib/stac-loader/index.ts",
5754
5818
  "line": 87
5755
5819
  },
5756
5820
  "name": "memorySize",
@@ -5767,7 +5831,7 @@
5767
5831
  },
5768
5832
  "immutable": true,
5769
5833
  "locationInModule": {
5770
- "filename": "lib/stac-item-loader/index.ts",
5834
+ "filename": "lib/stac-loader/index.ts",
5771
5835
  "line": 54
5772
5836
  },
5773
5837
  "name": "subnetSelection",
@@ -5784,7 +5848,7 @@
5784
5848
  },
5785
5849
  "immutable": true,
5786
5850
  "locationInModule": {
5787
- "filename": "lib/stac-item-loader/index.ts",
5851
+ "filename": "lib/stac-loader/index.ts",
5788
5852
  "line": 49
5789
5853
  },
5790
5854
  "name": "vpc",
@@ -5794,13 +5858,13 @@
5794
5858
  }
5795
5859
  }
5796
5860
  ],
5797
- "symbolId": "lib/stac-item-loader/index:StacItemLoaderProps"
5861
+ "symbolId": "lib/stac-loader/index:StacLoaderProps"
5798
5862
  },
5799
5863
  "eoapi-cdk.StactoolsItemGenerator": {
5800
5864
  "assembly": "eoapi-cdk",
5801
5865
  "base": "constructs.Construct",
5802
5866
  "docs": {
5803
- "remarks": "The StactoolsItemGenerator creates a serverless, event-driven system for generating\nSTAC (SpatioTemporal Asset Catalog) items from source data. This construct\nimplements the first phase of a two-stage ingestion pipeline that transforms\nraw geospatial data into standardized STAC metadata.\n\n## Architecture Overview\n\nThis construct creates the following AWS resources:\n- **SNS Topic**: Entry point for triggering item generation workflows\n- **SQS Queue**: Buffers generation requests (120-second visibility timeout)\n- **Dead Letter Queue**: Captures failed messages after 5 processing attempts\n- **Lambda Function**: Containerized function that generates STAC items using stactools\n\n## Data Flow\n\n1. External systems publish ItemRequest messages to the SNS topic with metadata about assets\n2. The SQS queue buffers these messages and triggers the Lambda function\n3. The Lambda function:\n - Uses `uvx` to install the required stactools package\n - Executes the `create-item` CLI command with provided arguments\n - Publishes generated STAC items to the ItemLoad topic\n4. Failed processing attempts are sent to the dead letter queue\n\n## Operational Characteristics\n\n- **Scalability**: Lambda scales automatically based on queue depth (up to maxConcurrency)\n- **Flexibility**: Supports any stactools package through dynamic installation\n- **Reliability**: Dead letter queue captures failed generation attempts\n- **Isolation**: Each generation task runs in a fresh container environment\n- **Observability**: CloudWatch logs retained for one week\n\n## Message Schema\n\nThe function expects messages matching the ItemRequest model:\n\n```json\n{\n \"package_name\": \"stactools-glad-global-forest-change\",\n \"group_name\": \"gladglobalforestchange\",\n \"create_item_args\": [\n \"https://example.com/data.tif\"\n ],\n \"collection_id\": \"glad-global-forest-change-1.11\"\n}\n```\n\n## Usage Example\n\n```typescript\n// Create item loader first (or get existing topic ARN)\nconst loader = new StacItemLoader(this, 'ItemLoader', {\n pgstacDb: database\n});\n\n// Create item generator that feeds the loader\nconst generator = new StactoolsItemGenerator(this, 'ItemGenerator', {\n itemLoadTopicArn: loader.topic.topicArn,\n lambdaTimeoutSeconds: 120, // Allow time for package installation\n maxConcurrency: 100, // Control parallel processing\n batchSize: 10 // Process 10 requests per invocation\n});\n\n// Grant permission to publish to the loader topic\nloader.topic.grantPublish(generator.lambdaFunction);\n```\n\n## Publishing Generation Requests\n\nSend messages to the generator topic to trigger item creation:\n\n```bash\naws sns publish --topic-arn $ITEM_GEN_TOPIC --message '{\n \"package_name\": \"stactools-glad-global-forest-change\",\n \"group_name\": \"gladglobalforestchange\",\n \"create_item_args\": [\n \"https://storage.googleapis.com/earthenginepartners-hansen/GFC-2023-v1.11/Hansen_GFC-2023-v1.11_gain_40N_080W.tif\"\n ],\n \"collection_id\": \"glad-global-forest-change-1.11\"\n}'\n```\n\n## Batch Processing Example\n\nFor processing many assets, you can loop through URLs:\n\n```bash\nwhile IFS= read -r url; do\n aws sns publish --topic-arn \"$ITEM_GEN_TOPIC\" --message \"{\n \\\"package_name\\\": \\\"stactools-glad-glclu2020\\\",\n \\\"group_name\\\": \\\"gladglclu2020\\\",\n \\\"create_item_args\\\": [\\\"$url\\\"]\n }\"\ndone < urls.txt\n```\n\n## Monitoring and Troubleshooting\n\n- Monitor Lambda logs: `/aws/lambda/{FunctionName}`\n- Check dead letter queue for failed generation attempts\n- Use CloudWatch metrics to track processing rates and errors\n- Failed items can be replayed from the dead letter queue\n\n## Supported Stactools Packages\n\nAny package available on PyPI that follows the stactools plugin pattern\ncan be used. Examples include:\n- `stactools-glad-global-forest-change`\n- `stactools-glad-glclu2020`\n- `stactools-landsat`\n- `stactools-sentinel2`",
5867
+ "remarks": "The StactoolsItemGenerator creates a serverless, event-driven system for generating\nSTAC (SpatioTemporal Asset Catalog) items from source data. This construct\nimplements the first phase of a two-stage ingestion pipeline that transforms\nraw geospatial data into standardized STAC metadata.\n\n## Architecture Overview\n\nThis construct creates the following AWS resources:\n- **SNS Topic**: Entry point for triggering item generation workflows\n- **SQS Queue**: Buffers generation requests (120-second visibility timeout)\n- **Dead Letter Queue**: Captures failed messages after 5 processing attempts\n- **Lambda Function**: Containerized function that generates STAC items using stactools\n\n## Data Flow\n\n1. External systems publish ItemRequest messages to the SNS topic with metadata about assets\n2. The SQS queue buffers these messages and triggers the Lambda function\n3. The Lambda function:\n - Uses `uvx` to install the required stactools package\n - Executes the `create-item` CLI command with provided arguments\n - Publishes generated STAC items to the ItemLoad topic\n4. Failed processing attempts are sent to the dead letter queue\n\n## Operational Characteristics\n\n- **Scalability**: Lambda scales automatically based on queue depth (up to maxConcurrency)\n- **Flexibility**: Supports any stactools package through dynamic installation\n- **Reliability**: Dead letter queue captures failed generation attempts\n- **Isolation**: Each generation task runs in a fresh container environment\n- **Observability**: CloudWatch logs retained for one week\n\n## Message Schema\n\nThe function expects messages matching the ItemRequest model:\n\n```json\n{\n \"package_name\": \"stactools-glad-global-forest-change\",\n \"group_name\": \"gladglobalforestchange\",\n \"create_item_args\": [\n \"https://example.com/data.tif\"\n ],\n \"collection_id\": \"glad-global-forest-change-1.11\"\n}\n```\n\n## Usage Example\n\n```typescript\n// Create item loader first (or get existing topic ARN)\nconst loader = new StacLoader(this, 'ItemLoader', {\n pgstacDb: database\n});\n\n// Create item generator that feeds the loader\nconst generator = new StactoolsItemGenerator(this, 'ItemGenerator', {\n itemLoadTopicArn: loader.topic.topicArn,\n lambdaTimeoutSeconds: 120, // Allow time for package installation\n maxConcurrency: 100, // Control parallel processing\n batchSize: 10 // Process 10 requests per invocation\n});\n\n// Grant permission to publish to the loader topic\nloader.topic.grantPublish(generator.lambdaFunction);\n```\n\n## Publishing Generation Requests\n\nSend messages to the generator topic to trigger item creation:\n\n```bash\naws sns publish --topic-arn $ITEM_GEN_TOPIC --message '{\n \"package_name\": \"stactools-glad-global-forest-change\",\n \"group_name\": \"gladglobalforestchange\",\n \"create_item_args\": [\n \"https://storage.googleapis.com/earthenginepartners-hansen/GFC-2023-v1.11/Hansen_GFC-2023-v1.11_gain_40N_080W.tif\"\n ],\n \"collection_id\": \"glad-global-forest-change-1.11\"\n}'\n```\n\n## Batch Processing Example\n\nFor processing many assets, you can loop through URLs:\n\n```bash\nwhile IFS= read -r url; do\n aws sns publish --topic-arn \"$ITEM_GEN_TOPIC\" --message \"{\n \\\"package_name\\\": \\\"stactools-glad-glclu2020\\\",\n \\\"group_name\\\": \\\"gladglclu2020\\\",\n \\\"create_item_args\\\": [\\\"$url\\\"]\n }\"\ndone < urls.txt\n```\n\n## Monitoring and Troubleshooting\n\n- Monitor Lambda logs: `/aws/lambda/{FunctionName}`\n- Check dead letter queue for failed generation attempts\n- Use CloudWatch metrics to track processing rates and errors\n- Failed items can be replayed from the dead letter queue\n\n## Supported Stactools Packages\n\nAny package available on PyPI that follows the stactools plugin pattern\ncan be used. Examples include:\n- `stactools-glad-global-forest-change`\n- `stactools-glad-glclu2020`\n- `stactools-landsat`\n- `stactools-sentinel2`",
5804
5868
  "see": "{@link https://stactools.readthedocs.io/} for stactools documentation",
5805
5869
  "stability": "experimental",
5806
5870
  "summary": "AWS CDK Construct for STAC Item Generation Infrastructure."
@@ -5929,7 +5993,7 @@
5929
5993
  {
5930
5994
  "abstract": true,
5931
5995
  "docs": {
5932
- "remarks": "This is typically the topic from a StacItemLoader construct.\nGenerated STAC items will be published here for downstream\nprocessing and database insertion.",
5996
+ "remarks": "This is typically the topic from a StacLoader construct.\nGenerated STAC items will be published here for downstream\nprocessing and database insertion.",
5933
5997
  "stability": "experimental",
5934
5998
  "summary": "ARN of the SNS topic to publish generated items to."
5935
5999
  },
@@ -6587,6 +6651,6 @@
6587
6651
  "symbolId": "lib/titiler-pgstac-api/index:TitilerPgstacApiLambda"
6588
6652
  }
6589
6653
  },
6590
- "version": "8.2.3",
6591
- "fingerprint": "UnlsJrm4PSPeUv2V83ku+dQutHDfYeudJ06A/NuA0D4="
6654
+ "version": "8.3.0",
6655
+ "fingerprint": "+LZo2EalPFytHurMdz0n6AvEvjkWqv+ajo2WPpKTY6s="
6592
6656
  }
@@ -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: "8.2.3" };
161
+ BastionHost[_a] = { fqn: "eoapi-cdk.BastionHost", version: "8.3.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"]}
@@ -181,5 +181,5 @@ class PgStacDatabase extends constructs_1.Construct {
181
181
  }
182
182
  exports.PgStacDatabase = PgStacDatabase;
183
183
  _a = JSII_RTTI_SYMBOL_1;
184
- PgStacDatabase[_a] = { fqn: "eoapi-cdk.PgStacDatabase", version: "8.2.3" };
184
+ PgStacDatabase[_a] = { fqn: "eoapi-cdk.PgStacDatabase", version: "8.3.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  Stack,\n  aws_rds as rds,\n  aws_ec2 as ec2,\n  aws_secretsmanager as secretsmanager,\n  aws_lambda,\n  CustomResource,\n  RemovalPolicy,\n  Duration,\n  aws_logs,\n} from \"aws-cdk-lib\";\nimport { Construct } from \"constructs\";\nimport { CustomLambdaFunctionProps, DEFAULT_PGSTAC_VERSION } 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. This is a wrapper around the\n * `rds.DatabaseInstance` higher-level construct making use\n * 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\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 handler = new aws_lambda.Function(this, \"lambda\", {\n      // defaults\n      runtime: aws_lambda.Runtime.PYTHON_3_11,\n      handler: \"handler.handler\",\n      memorySize: 128,\n      logRetention: aws_logs.RetentionDays.ONE_WEEK,\n      timeout: Duration.minutes(2),\n      code: aws_lambda.Code.fromDockerBuild(__dirname, {\n        file: \"bootstrapper_runtime/Dockerfile\",\n        buildArgs: {\n          PYTHON_VERSION: \"3.11\",\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      ...props.bootstrapperLambdaFunctionOptions,\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 (!props.bootstrapperLambdaFunctionOptions?.code) {\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    } 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"]}
package/lib/index.d.ts CHANGED
@@ -2,7 +2,7 @@ export * from "./bastion-host";
2
2
  export * from "./database";
3
3
  export * from "./ingestor-api";
4
4
  export * from "./stactools-item-generator";
5
- export * from "./stac-item-loader";
5
+ export * from "./stac-loader";
6
6
  export * from "./stac-api";
7
7
  export * from "./titiler-pgstac-api";
8
8
  export * from "./stac-browser";
package/lib/index.js CHANGED
@@ -18,10 +18,10 @@ __exportStar(require("./bastion-host"), exports);
18
18
  __exportStar(require("./database"), exports);
19
19
  __exportStar(require("./ingestor-api"), exports);
20
20
  __exportStar(require("./stactools-item-generator"), exports);
21
- __exportStar(require("./stac-item-loader"), exports);
21
+ __exportStar(require("./stac-loader"), exports);
22
22
  __exportStar(require("./stac-api"), exports);
23
23
  __exportStar(require("./titiler-pgstac-api"), exports);
24
24
  __exportStar(require("./stac-browser"), exports);
25
25
  __exportStar(require("./tipg-api"), exports);
26
26
  __exportStar(require("./utils"), exports);
27
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJpbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7O0FBQUEsaURBQStCO0FBQy9CLDZDQUEyQjtBQUMzQixpREFBK0I7QUFDL0IsNkRBQTJDO0FBQzNDLHFEQUFtQztBQUNuQyw2Q0FBMkI7QUFDM0IsdURBQXFDO0FBQ3JDLGlEQUErQjtBQUMvQiw2Q0FBMkI7QUFDM0IsMENBQXdCIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSBcIi4vYmFzdGlvbi1ob3N0XCI7XG5leHBvcnQgKiBmcm9tIFwiLi9kYXRhYmFzZVwiO1xuZXhwb3J0ICogZnJvbSBcIi4vaW5nZXN0b3ItYXBpXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9zdGFjdG9vbHMtaXRlbS1nZW5lcmF0b3JcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3N0YWMtaXRlbS1sb2FkZXJcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3N0YWMtYXBpXCI7XG5leHBvcnQgKiBmcm9tIFwiLi90aXRpbGVyLXBnc3RhYy1hcGlcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3N0YWMtYnJvd3NlclwiO1xuZXhwb3J0ICogZnJvbSBcIi4vdGlwZy1hcGlcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3V0aWxzXCI7XG4iXX0=
27
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJpbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7O0FBQUEsaURBQStCO0FBQy9CLDZDQUEyQjtBQUMzQixpREFBK0I7QUFDL0IsNkRBQTJDO0FBQzNDLGdEQUE4QjtBQUM5Qiw2Q0FBMkI7QUFDM0IsdURBQXFDO0FBQ3JDLGlEQUErQjtBQUMvQiw2Q0FBMkI7QUFDM0IsMENBQXdCIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSBcIi4vYmFzdGlvbi1ob3N0XCI7XG5leHBvcnQgKiBmcm9tIFwiLi9kYXRhYmFzZVwiO1xuZXhwb3J0ICogZnJvbSBcIi4vaW5nZXN0b3ItYXBpXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9zdGFjdG9vbHMtaXRlbS1nZW5lcmF0b3JcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3N0YWMtbG9hZGVyXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9zdGFjLWFwaVwiO1xuZXhwb3J0ICogZnJvbSBcIi4vdGl0aWxlci1wZ3N0YWMtYXBpXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9zdGFjLWJyb3dzZXJcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3RpcGctYXBpXCI7XG5leHBvcnQgKiBmcm9tIFwiLi91dGlsc1wiO1xuIl19
@@ -179,5 +179,5 @@ class StacIngestor extends constructs_1.Construct {
179
179
  }
180
180
  exports.StacIngestor = StacIngestor;
181
181
  _a = JSII_RTTI_SYMBOL_1;
182
- StacIngestor[_a] = { fqn: "eoapi-cdk.StacIngestor", version: "8.2.3" };
182
+ StacIngestor[_a] = { fqn: "eoapi-cdk.StacIngestor", version: "8.3.0" };
183
183
  //# 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_dynamodb as dynamodb,\n  aws_ec2 as ec2,\n  aws_iam as iam,\n  aws_lambda as lambda,\n  aws_logs,\n  aws_lambda_event_sources as events,\n  aws_secretsmanager as secretsmanager,\n  aws_ssm as ssm,\n  Duration,\n  RemovalPolicy,\n  Stack,\n} from \"aws-cdk-lib\";\nimport { Construct } from \"constructs\";\nimport { CustomLambdaFunctionProps, DEFAULT_PGSTAC_VERSION } 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 handler = new lambda.Function(this, \"api-handler\", {\n      // defaults\n      runtime: lambda.Runtime.PYTHON_3_11,\n      handler: \"src.handler.handler\",\n      memorySize: 2048,\n      logRetention: aws_logs.RetentionDays.ONE_WEEK,\n      timeout: Duration.seconds(30),\n      code: lambda.Code.fromDockerBuild(__dirname, {\n        file: \"runtime/Dockerfile\",\n        buildArgs: {\n          PYTHON_VERSION: \"3.11\",\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      ...props.lambdaFunctionOptions,\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 handler = new lambda.Function(this, \"stac-ingestor\", {\n      // defaults\n      runtime: lambda.Runtime.PYTHON_3_11,\n      handler: \"src.ingestor.handler\",\n      memorySize: 2048,\n      logRetention: aws_logs.RetentionDays.ONE_WEEK,\n      timeout: Duration.seconds(180),\n      code: lambda.Code.fromDockerBuild(__dirname, {\n        file: \"runtime/Dockerfile\",\n        buildArgs: {\n          PYTHON_VERSION: \"3.11\",\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      ...props.lambdaFunctionOptions,\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"]}