eoapi-cdk 8.1.1 → 8.2.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 +594 -26
- package/lib/bastion-host/index.js +1 -1
- package/lib/database/index.d.ts +1 -0
- package/lib/database/index.js +5 -5
- package/lib/index.d.ts +2 -0
- package/lib/index.js +3 -1
- package/lib/ingestor-api/index.js +1 -1
- package/lib/stac-api/index.js +1 -1
- package/lib/stac-browser/index.js +1 -1
- package/lib/stac-item-loader/index.d.ts +322 -0
- package/lib/stac-item-loader/index.js +251 -0
- package/lib/stac-item-loader/runtime/Dockerfile +18 -0
- package/lib/stac-item-loader/runtime/pyproject.toml +17 -0
- package/lib/stac-item-loader/runtime/src/stac_item_loader/handler.py +241 -0
- package/lib/stactools-item-generator/index.d.ts +243 -0
- package/lib/stactools-item-generator/index.js +204 -0
- package/lib/stactools-item-generator/runtime/Dockerfile +20 -0
- package/lib/stactools-item-generator/runtime/pyproject.toml +16 -0
- package/lib/stactools-item-generator/runtime/src/stactools_item_generator/__init__.py +2 -0
- package/lib/stactools-item-generator/runtime/src/stactools_item_generator/handler.py +176 -0
- package/lib/stactools-item-generator/runtime/src/stactools_item_generator/item.py +77 -0
- package/lib/tipg-api/index.js +1 -1
- package/lib/titiler-pgstac-api/index.js +1 -1
- package/package.json +1 -1
- package/pyproject.toml +45 -0
- package/uv.lock +1065 -0
- package/.devcontainer/devcontainer.json +0 -4
- package/.github/pull_request_template.md +0 -4
- package/.github/workflows/build.yaml +0 -73
- package/.github/workflows/build_and_release.yaml +0 -13
- package/.github/workflows/conventional-pr.yaml +0 -26
- package/.github/workflows/deploy.yaml +0 -84
- package/.github/workflows/distribute.yaml +0 -46
- package/.github/workflows/docs.yaml +0 -26
- package/.github/workflows/lint.yaml +0 -26
- package/.github/workflows/tox.yaml +0 -26
- package/.nvmrc +0 -1
- package/.pre-commit-config.yaml +0 -23
- package/CHANGELOG.md +0 -471
- package/diagrams/bastion_diagram.excalidraw +0 -1416
- package/diagrams/bastion_diagram.png +0 -0
- package/diagrams/ingestor_diagram.excalidraw +0 -2274
- package/diagrams/ingestor_diagram.png +0 -0
- package/integration_tests/cdk/README.md +0 -55
- package/integration_tests/cdk/app.py +0 -186
- package/integration_tests/cdk/cdk.json +0 -32
- package/integration_tests/cdk/config.py +0 -52
- package/integration_tests/cdk/package-lock.json +0 -42
- package/integration_tests/cdk/package.json +0 -7
- package/integration_tests/cdk/requirements.txt +0 -7
- package/lib/database/lambda/package-lock.json +0 -1324
- package/lib/ingestor-api/runtime/tests/conftest.py +0 -270
- package/lib/ingestor-api/runtime/tests/test_collection.py +0 -87
- package/lib/ingestor-api/runtime/tests/test_collection_endpoint.py +0 -41
- package/lib/ingestor-api/runtime/tests/test_ingestor.py +0 -60
- package/lib/ingestor-api/runtime/tests/test_registration.py +0 -207
- package/lib/ingestor-api/runtime/tests/test_utils.py +0 -35
- package/lib/ingestor-api/runtime/tests/test_validators.py +0 -164
- package/ruff.toml +0 -23
- package/tox.ini +0 -16
- package/tsconfig.tsbuildinfo +0 -1
- /package/lib/{ingestor-api/runtime/tests → stac-item-loader/runtime/src/stac_item_loader}/__init__.py +0 -0
|
@@ -0,0 +1,204 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var _a;
|
|
3
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
4
|
+
exports.StactoolsItemGenerator = void 0;
|
|
5
|
+
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
|
6
|
+
const aws_cdk_lib_1 = require("aws-cdk-lib");
|
|
7
|
+
const constructs_1 = require("constructs");
|
|
8
|
+
const aws_ecr_assets_1 = require("aws-cdk-lib/aws-ecr-assets");
|
|
9
|
+
const path = require("path");
|
|
10
|
+
/**
|
|
11
|
+
* AWS CDK Construct for STAC Item Generation Infrastructure
|
|
12
|
+
*
|
|
13
|
+
* The StactoolsItemGenerator creates a serverless, event-driven system for generating
|
|
14
|
+
* STAC (SpatioTemporal Asset Catalog) items from source data. This construct
|
|
15
|
+
* implements the first phase of a two-stage ingestion pipeline that transforms
|
|
16
|
+
* raw geospatial data into standardized STAC metadata.
|
|
17
|
+
*
|
|
18
|
+
* ## Architecture Overview
|
|
19
|
+
*
|
|
20
|
+
* This construct creates the following AWS resources:
|
|
21
|
+
* - **SNS Topic**: Entry point for triggering item generation workflows
|
|
22
|
+
* - **SQS Queue**: Buffers generation requests (120-second visibility timeout)
|
|
23
|
+
* - **Dead Letter Queue**: Captures failed messages after 5 processing attempts
|
|
24
|
+
* - **Lambda Function**: Containerized function that generates STAC items using stactools
|
|
25
|
+
*
|
|
26
|
+
* ## Data Flow
|
|
27
|
+
*
|
|
28
|
+
* 1. External systems publish ItemRequest messages to the SNS topic with metadata about assets
|
|
29
|
+
* 2. The SQS queue buffers these messages and triggers the Lambda function
|
|
30
|
+
* 3. The Lambda function:
|
|
31
|
+
* - Uses `uvx` to install the required stactools package
|
|
32
|
+
* - Executes the `create-item` CLI command with provided arguments
|
|
33
|
+
* - Publishes generated STAC items to the ItemLoad topic
|
|
34
|
+
* 4. Failed processing attempts are sent to the dead letter queue
|
|
35
|
+
*
|
|
36
|
+
* ## Operational Characteristics
|
|
37
|
+
*
|
|
38
|
+
* - **Scalability**: Lambda scales automatically based on queue depth (up to maxConcurrency)
|
|
39
|
+
* - **Flexibility**: Supports any stactools package through dynamic installation
|
|
40
|
+
* - **Reliability**: Dead letter queue captures failed generation attempts
|
|
41
|
+
* - **Isolation**: Each generation task runs in a fresh container environment
|
|
42
|
+
* - **Observability**: CloudWatch logs retained for one week
|
|
43
|
+
*
|
|
44
|
+
* ## Message Schema
|
|
45
|
+
*
|
|
46
|
+
* The function expects messages matching the ItemRequest model:
|
|
47
|
+
*
|
|
48
|
+
* ```json
|
|
49
|
+
* {
|
|
50
|
+
* "package_name": "stactools-glad-global-forest-change",
|
|
51
|
+
* "group_name": "gladglobalforestchange",
|
|
52
|
+
* "create_item_args": [
|
|
53
|
+
* "https://example.com/data.tif"
|
|
54
|
+
* ],
|
|
55
|
+
* "collection_id": "glad-global-forest-change-1.11"
|
|
56
|
+
* }
|
|
57
|
+
* ```
|
|
58
|
+
*
|
|
59
|
+
* ## Usage Example
|
|
60
|
+
*
|
|
61
|
+
* ```typescript
|
|
62
|
+
* // Create item loader first (or get existing topic ARN)
|
|
63
|
+
* const loader = new StacItemLoader(this, 'ItemLoader', {
|
|
64
|
+
* pgstacDb: database
|
|
65
|
+
* });
|
|
66
|
+
*
|
|
67
|
+
* // Create item generator that feeds the loader
|
|
68
|
+
* const generator = new StactoolsItemGenerator(this, 'ItemGenerator', {
|
|
69
|
+
* itemLoadTopicArn: loader.topic.topicArn,
|
|
70
|
+
* lambdaTimeoutSeconds: 120, // Allow time for package installation
|
|
71
|
+
* maxConcurrency: 100, // Control parallel processing
|
|
72
|
+
* batchSize: 10 // Process 10 requests per invocation
|
|
73
|
+
* });
|
|
74
|
+
*
|
|
75
|
+
* // Grant permission to publish to the loader topic
|
|
76
|
+
* loader.topic.grantPublish(generator.lambdaFunction);
|
|
77
|
+
* ```
|
|
78
|
+
*
|
|
79
|
+
* ## Publishing Generation Requests
|
|
80
|
+
*
|
|
81
|
+
* Send messages to the generator topic to trigger item creation:
|
|
82
|
+
*
|
|
83
|
+
* ```bash
|
|
84
|
+
* aws sns publish --topic-arn $ITEM_GEN_TOPIC --message '{
|
|
85
|
+
* "package_name": "stactools-glad-global-forest-change",
|
|
86
|
+
* "group_name": "gladglobalforestchange",
|
|
87
|
+
* "create_item_args": [
|
|
88
|
+
* "https://storage.googleapis.com/earthenginepartners-hansen/GFC-2023-v1.11/Hansen_GFC-2023-v1.11_gain_40N_080W.tif"
|
|
89
|
+
* ],
|
|
90
|
+
* "collection_id": "glad-global-forest-change-1.11"
|
|
91
|
+
* }'
|
|
92
|
+
* ```
|
|
93
|
+
*
|
|
94
|
+
* ## Batch Processing Example
|
|
95
|
+
*
|
|
96
|
+
* For processing many assets, you can loop through URLs:
|
|
97
|
+
*
|
|
98
|
+
* ```bash
|
|
99
|
+
* while IFS= read -r url; do
|
|
100
|
+
* aws sns publish --topic-arn "$ITEM_GEN_TOPIC" --message "{
|
|
101
|
+
* \"package_name\": \"stactools-glad-glclu2020\",
|
|
102
|
+
* \"group_name\": \"gladglclu2020\",
|
|
103
|
+
* \"create_item_args\": [\"$url\"]
|
|
104
|
+
* }"
|
|
105
|
+
* done < urls.txt
|
|
106
|
+
* ```
|
|
107
|
+
*
|
|
108
|
+
* ## Monitoring and Troubleshooting
|
|
109
|
+
*
|
|
110
|
+
* - Monitor Lambda logs: `/aws/lambda/{FunctionName}`
|
|
111
|
+
* - Check dead letter queue for failed generation attempts
|
|
112
|
+
* - Use CloudWatch metrics to track processing rates and errors
|
|
113
|
+
* - Failed items can be replayed from the dead letter queue
|
|
114
|
+
*
|
|
115
|
+
* ## Supported Stactools Packages
|
|
116
|
+
*
|
|
117
|
+
* Any package available on PyPI that follows the stactools plugin pattern
|
|
118
|
+
* can be used. Examples include:
|
|
119
|
+
* - `stactools-glad-global-forest-change`
|
|
120
|
+
* - `stactools-glad-glclu2020`
|
|
121
|
+
* - `stactools-landsat`
|
|
122
|
+
* - `stactools-sentinel2`
|
|
123
|
+
*
|
|
124
|
+
* @see {@link https://github.com/stactools-packages} for available stactools packages
|
|
125
|
+
* @see {@link https://stactools.readthedocs.io/} for stactools documentation
|
|
126
|
+
*/
|
|
127
|
+
class StactoolsItemGenerator extends constructs_1.Construct {
|
|
128
|
+
constructor(scope, id, props) {
|
|
129
|
+
super(scope, id);
|
|
130
|
+
const timeoutSeconds = props.lambdaTimeoutSeconds ?? 120;
|
|
131
|
+
const lambdaRuntime = props.lambdaRuntime ?? aws_cdk_lib_1.aws_lambda.Runtime.PYTHON_3_11;
|
|
132
|
+
// Create dead letter queue
|
|
133
|
+
this.deadLetterQueue = new aws_cdk_lib_1.aws_sqs.Queue(this, "DeadLetterQueue", {
|
|
134
|
+
retentionPeriod: aws_cdk_lib_1.Duration.days(14),
|
|
135
|
+
});
|
|
136
|
+
// Create main queue
|
|
137
|
+
this.queue = new aws_cdk_lib_1.aws_sqs.Queue(this, "Queue", {
|
|
138
|
+
visibilityTimeout: aws_cdk_lib_1.Duration.seconds(timeoutSeconds + 10),
|
|
139
|
+
encryption: aws_cdk_lib_1.aws_sqs.QueueEncryption.SQS_MANAGED,
|
|
140
|
+
deadLetterQueue: {
|
|
141
|
+
maxReceiveCount: 5,
|
|
142
|
+
queue: this.deadLetterQueue,
|
|
143
|
+
},
|
|
144
|
+
});
|
|
145
|
+
// Create SNS topic
|
|
146
|
+
this.topic = new aws_cdk_lib_1.aws_sns.Topic(this, "Topic", {
|
|
147
|
+
displayName: `${id}-ItemGenTopic`,
|
|
148
|
+
});
|
|
149
|
+
// Subscribe the queue to the topic
|
|
150
|
+
this.topic.addSubscription(new aws_cdk_lib_1.aws_sns_subscriptions.SqsSubscription(this.queue));
|
|
151
|
+
// Create the lambda function
|
|
152
|
+
this.lambdaFunction = new aws_cdk_lib_1.aws_lambda.DockerImageFunction(this, "Function", {
|
|
153
|
+
code: aws_cdk_lib_1.aws_lambda.DockerImageCode.fromImageAsset(path.join(__dirname, ".."), {
|
|
154
|
+
file: "stactools-item-generator/runtime/Dockerfile",
|
|
155
|
+
platform: aws_ecr_assets_1.Platform.LINUX_AMD64,
|
|
156
|
+
buildArgs: {
|
|
157
|
+
PYTHON_VERSION: lambdaRuntime.toString().replace("python", ""),
|
|
158
|
+
},
|
|
159
|
+
}),
|
|
160
|
+
memorySize: props.memorySize ?? 1024,
|
|
161
|
+
timeout: aws_cdk_lib_1.Duration.seconds(timeoutSeconds),
|
|
162
|
+
logRetention: aws_cdk_lib_1.aws_logs.RetentionDays.ONE_WEEK,
|
|
163
|
+
environment: {
|
|
164
|
+
ITEM_LOAD_TOPIC_ARN: props.itemLoadTopicArn,
|
|
165
|
+
LOG_LEVEL: "INFO",
|
|
166
|
+
...props.environment,
|
|
167
|
+
},
|
|
168
|
+
});
|
|
169
|
+
// Add SQS event source to the lambda
|
|
170
|
+
this.lambdaFunction.addEventSource(new aws_cdk_lib_1.aws_lambda_event_sources.SqsEventSource(this.queue, {
|
|
171
|
+
batchSize: props.batchSize ?? 10,
|
|
172
|
+
reportBatchItemFailures: true,
|
|
173
|
+
maxConcurrency: props.maxConcurrency ?? 100,
|
|
174
|
+
}));
|
|
175
|
+
// Grant permissions to publish to the item load topic
|
|
176
|
+
// Note: This will be granted externally since we only have the ARN
|
|
177
|
+
// The consuming construct should handle this permission
|
|
178
|
+
// Create outputs
|
|
179
|
+
new aws_cdk_lib_1.CfnOutput(this, "TopicArn", {
|
|
180
|
+
value: this.topic.topicArn,
|
|
181
|
+
description: "ARN of the StactoolsItemGenerator SNS Topic",
|
|
182
|
+
exportName: "stactools-item-generator-topic-arn",
|
|
183
|
+
});
|
|
184
|
+
new aws_cdk_lib_1.CfnOutput(this, "QueueUrl", {
|
|
185
|
+
value: this.queue.queueUrl,
|
|
186
|
+
description: "URL of the StactoolsItemGenerator SQS Queue",
|
|
187
|
+
exportName: "stactools-item-generator-queue-url",
|
|
188
|
+
});
|
|
189
|
+
new aws_cdk_lib_1.CfnOutput(this, "DeadLetterQueueUrl", {
|
|
190
|
+
value: this.deadLetterQueue.queueUrl,
|
|
191
|
+
description: "URL of the StactoolsItemGenerator Dead Letter Queue",
|
|
192
|
+
exportName: "stactools-item-generator-deadletter-queue-url",
|
|
193
|
+
});
|
|
194
|
+
new aws_cdk_lib_1.CfnOutput(this, "FunctionName", {
|
|
195
|
+
value: this.lambdaFunction.functionName,
|
|
196
|
+
description: "Name of the StactoolsItemGenerator Lambda Function",
|
|
197
|
+
exportName: "stactools-item-generator-function-name",
|
|
198
|
+
});
|
|
199
|
+
}
|
|
200
|
+
}
|
|
201
|
+
exports.StactoolsItemGenerator = StactoolsItemGenerator;
|
|
202
|
+
_a = JSII_RTTI_SYMBOL_1;
|
|
203
|
+
StactoolsItemGenerator[_a] = { fqn: "eoapi-cdk.StactoolsItemGenerator", version: "8.2.0" };
|
|
204
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["index.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  aws_lambda as lambda,\n  aws_sqs as sqs,\n  aws_sns as sns,\n  aws_sns_subscriptions as snsSubscriptions,\n  aws_lambda_event_sources as lambdaEventSources,\n  aws_logs as logs,\n  Duration,\n  CfnOutput,\n} from \"aws-cdk-lib\";\nimport { Construct } from \"constructs\";\nimport { Platform } from \"aws-cdk-lib/aws-ecr-assets\";\nimport * as path from \"path\";\n\n/**\n * Configuration properties for the StactoolsItemGenerator construct.\n *\n * The StactoolsItemGenerator is part of a two-phase serverless STAC ingestion pipeline\n * that generates STAC items from source data. This construct creates the\n * infrastructure for the first phase of the pipeline - processing metadata\n * about assets and transforming them into standardized STAC items.\n *\n * @example\n * const generator = new StactoolsItemGenerator(this, 'ItemGenerator', {\n *   itemLoadTopicArn: loader.topic.topicArn,\n *   lambdaTimeoutSeconds: 120,\n *   maxConcurrency: 100,\n *   batchSize: 10\n * });\n */\nexport interface StactoolsItemGeneratorProps {\n  /**\n   * The lambda runtime to use for the item generation function.\n   *\n   * The function is containerized using Docker and can accommodate various\n   * stactools packages. The runtime version should be compatible with the\n   * packages you plan to use for STAC item generation.\n   *\n   * @default lambda.Runtime.PYTHON_3_11\n   */\n  readonly lambdaRuntime?: lambda.Runtime;\n\n  /**\n   * The timeout for the item generation lambda in seconds.\n   *\n   * This should accommodate the time needed to:\n   * - Install stactools packages using uvx\n   * - Download and process source data\n   * - Generate STAC metadata\n   * - Publish results to SNS\n   *\n   * The SQS visibility timeout will be set to this value plus 10 seconds.\n   *\n   * @default 120\n   */\n  readonly lambdaTimeoutSeconds?: number;\n\n  /**\n   * Memory size for the lambda function in MB.\n   *\n   * Higher memory allocation may be needed for processing large geospatial\n   * datasets or when stactools packages have high memory requirements.\n   * More memory also provides proportionally more CPU power.\n   *\n   * @default 1024\n   */\n  readonly memorySize?: number;\n\n  /**\n   * Maximum number of concurrent executions.\n   *\n   * This controls how many item generation tasks can run simultaneously.\n   * Higher concurrency enables faster processing of large batches but\n   * may strain downstream systems or external data sources.\n   *\n   * @default 100\n   */\n  readonly maxConcurrency?: number;\n\n  /**\n   * SQS batch size for lambda event source.\n   *\n   * This determines how many generation requests are processed together\n   * in a single lambda invocation. Unlike the loader, generation typically\n   * processes items individually, so smaller batch sizes are common.\n   *\n   * @default 10\n   */\n  readonly batchSize?: number;\n\n  /**\n   * Additional environment variables for the lambda function.\n   *\n   * These will be merged with default environment variables including\n   * ITEM_LOAD_TOPIC_ARN and LOG_LEVEL. Use this for custom configuration\n   * or to pass credentials for external data sources.\n   */\n  readonly environment?: { [key: string]: string };\n\n  /**\n   * ARN of the SNS topic to publish generated items to.\n   *\n   * This is typically the topic from a StacItemLoader construct.\n   * Generated STAC items will be published here for downstream\n   * processing and database insertion.\n   */\n  readonly itemLoadTopicArn: string;\n}\n\n/**\n * AWS CDK Construct for STAC Item Generation Infrastructure\n *\n * The StactoolsItemGenerator creates a serverless, event-driven system for generating\n * STAC (SpatioTemporal Asset Catalog) items from source data. This construct\n * implements the first phase of a two-stage ingestion pipeline that transforms\n * raw geospatial data into standardized STAC metadata.\n *\n * ## Architecture Overview\n *\n * This 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 *\n * 1. External systems publish ItemRequest messages to the SNS topic with metadata about assets\n * 2. The SQS queue buffers these messages and triggers the Lambda function\n * 3. 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\n * 4. 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 *\n * The 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)\n * const loader = new StacItemLoader(this, 'ItemLoader', {\n *   pgstacDb: database\n * });\n *\n * // Create item generator that feeds the loader\n * const 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\n * loader.topic.grantPublish(generator.lambdaFunction);\n * ```\n *\n * ## Publishing Generation Requests\n *\n * Send messages to the generator topic to trigger item creation:\n *\n * ```bash\n * aws 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 *\n * For processing many assets, you can loop through URLs:\n *\n * ```bash\n * while 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 *   }\"\n * done < 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 *\n * Any package available on PyPI that follows the stactools plugin pattern\n * can be used. Examples include:\n * - `stactools-glad-global-forest-change`\n * - `stactools-glad-glclu2020`\n * - `stactools-landsat`\n * - `stactools-sentinel2`\n *\n * @see {@link https://github.com/stactools-packages} for available stactools packages\n * @see {@link https://stactools.readthedocs.io/} for stactools documentation\n */\nexport class StactoolsItemGenerator extends Construct {\n  /**\n   * The SQS queue that buffers item generation requests.\n   *\n   * This queue receives messages from the SNS topic containing ItemRequest\n   * payloads. It's configured with a visibility timeout that matches the\n   * Lambda timeout plus buffer time to prevent duplicate processing.\n   */\n  public readonly queue: sqs.Queue;\n\n  /**\n   * Dead letter queue for failed item generation attempts.\n   *\n   * Messages that fail processing after 5 attempts are sent here for\n   * inspection and potential replay. This helps with debugging stactools\n   * package issues, network failures, or malformed requests.\n   */\n  public readonly deadLetterQueue: sqs.Queue;\n\n  /**\n   * The SNS topic that receives item generation requests.\n   *\n   * External systems publish ItemRequest messages to this topic to trigger\n   * STAC item generation. The topic fans out to the SQS queue for processing.\n   */\n  public readonly topic: sns.Topic;\n\n  /**\n   * The containerized Lambda function that generates STAC items.\n   *\n   * This Docker-based function dynamically installs stactools packages\n   * using uvx, processes source data, and publishes generated STAC items\n   * to the configured ItemLoad SNS topic.\n   */\n  public readonly lambdaFunction: lambda.DockerImageFunction;\n\n  constructor(scope: Construct, id: string, props: StactoolsItemGeneratorProps) {\n    super(scope, id);\n\n    const timeoutSeconds = props.lambdaTimeoutSeconds ?? 120;\n    const lambdaRuntime = props.lambdaRuntime ?? lambda.Runtime.PYTHON_3_11;\n\n    // Create dead letter queue\n    this.deadLetterQueue = new sqs.Queue(this, \"DeadLetterQueue\", {\n      retentionPeriod: Duration.days(14),\n    });\n\n    // Create main queue\n    this.queue = new sqs.Queue(this, \"Queue\", {\n      visibilityTimeout: Duration.seconds(timeoutSeconds + 10),\n      encryption: sqs.QueueEncryption.SQS_MANAGED,\n      deadLetterQueue: {\n        maxReceiveCount: 5,\n        queue: this.deadLetterQueue,\n      },\n    });\n\n    // Create SNS topic\n    this.topic = new sns.Topic(this, \"Topic\", {\n      displayName: `${id}-ItemGenTopic`,\n    });\n\n    // Subscribe the queue to the topic\n    this.topic.addSubscription(\n      new snsSubscriptions.SqsSubscription(this.queue)\n    );\n\n    // Create the lambda function\n    this.lambdaFunction = new lambda.DockerImageFunction(this, \"Function\", {\n      code: lambda.DockerImageCode.fromImageAsset(path.join(__dirname, \"..\"), {\n        file: \"stactools-item-generator/runtime/Dockerfile\",\n        platform: Platform.LINUX_AMD64,\n        buildArgs: {\n          PYTHON_VERSION: lambdaRuntime.toString().replace(\"python\", \"\"),\n        },\n      }),\n      memorySize: props.memorySize ?? 1024,\n      timeout: Duration.seconds(timeoutSeconds),\n      logRetention: logs.RetentionDays.ONE_WEEK,\n      environment: {\n        ITEM_LOAD_TOPIC_ARN: props.itemLoadTopicArn,\n        LOG_LEVEL: \"INFO\",\n        ...props.environment,\n      },\n    });\n\n    // Add SQS event source to the lambda\n    this.lambdaFunction.addEventSource(\n      new lambdaEventSources.SqsEventSource(this.queue, {\n        batchSize: props.batchSize ?? 10,\n        reportBatchItemFailures: true,\n        maxConcurrency: props.maxConcurrency ?? 100,\n      })\n    );\n\n    // Grant permissions to publish to the item load topic\n    // Note: This will be granted externally since we only have the ARN\n    // The consuming construct should handle this permission\n\n    // Create outputs\n    new CfnOutput(this, \"TopicArn\", {\n      value: this.topic.topicArn,\n      description: \"ARN of the StactoolsItemGenerator SNS Topic\",\n      exportName: \"stactools-item-generator-topic-arn\",\n    });\n\n    new CfnOutput(this, \"QueueUrl\", {\n      value: this.queue.queueUrl,\n      description: \"URL of the StactoolsItemGenerator SQS Queue\",\n      exportName: \"stactools-item-generator-queue-url\",\n    });\n\n    new CfnOutput(this, \"DeadLetterQueueUrl\", {\n      value: this.deadLetterQueue.queueUrl,\n      description: \"URL of the StactoolsItemGenerator Dead Letter Queue\",\n      exportName: \"stactools-item-generator-deadletter-queue-url\",\n    });\n\n    new CfnOutput(this, \"FunctionName\", {\n      value: this.lambdaFunction.functionName,\n      description: \"Name of the StactoolsItemGenerator Lambda Function\",\n      exportName: \"stactools-item-generator-function-name\",\n    });\n  }\n}\n"]}
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
ARG PYTHON_VERSION=3.11
|
|
2
|
+
FROM public.ecr.aws/lambda/python:${PYTHON_VERSION}
|
|
3
|
+
COPY --from=ghcr.io/astral-sh/uv:0.7.8 /uv /uvx /bin/
|
|
4
|
+
|
|
5
|
+
ENV UV_CACHE_DIR=/tmp/uv-cache/
|
|
6
|
+
ENV UV_COMPILE_BYTECODE=1
|
|
7
|
+
ENV PYTHONUNBUFFERED=1
|
|
8
|
+
ENV HOME=/tmp
|
|
9
|
+
ENV PATH=/tmp/.local/bin:$PATH
|
|
10
|
+
|
|
11
|
+
WORKDIR ${LAMBDA_TASK_ROOT}
|
|
12
|
+
|
|
13
|
+
COPY stactools-item-generator/runtime/pyproject.toml pyproject.toml
|
|
14
|
+
COPY stactools-item-generator/runtime/src/stactools_item_generator/ ${LAMBDA_TASK_ROOT}/stactools_item_generator/
|
|
15
|
+
|
|
16
|
+
RUN uv export --no-dev --no-editable -o requirements.txt && \
|
|
17
|
+
uv pip install --target ${LAMBDA_TASK_ROOT} -r requirements.txt && \
|
|
18
|
+
uv tool install --with requests stactools;
|
|
19
|
+
|
|
20
|
+
CMD ["stactools_item_generator.handler.handler"]
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
[project]
|
|
2
|
+
name = "stactools-item-generator"
|
|
3
|
+
version = "0.1.0"
|
|
4
|
+
description = "An application for generating STAC metadata with any stactools package"
|
|
5
|
+
authors = [
|
|
6
|
+
{ name = "hrodmn", email = "henry@developmentseed.org" }
|
|
7
|
+
]
|
|
8
|
+
requires-python = ">=3.11"
|
|
9
|
+
dependencies = [
|
|
10
|
+
"pydantic>=2.11.0",
|
|
11
|
+
"stac-pydantic>=3.2.0",
|
|
12
|
+
]
|
|
13
|
+
|
|
14
|
+
[build-system]
|
|
15
|
+
requires = ["hatchling"]
|
|
16
|
+
build-backend = "hatchling.build"
|
|
@@ -0,0 +1,176 @@
|
|
|
1
|
+
"""AWS Lambda handler for STAC Item Generation."""
|
|
2
|
+
|
|
3
|
+
import json
|
|
4
|
+
import logging
|
|
5
|
+
import os
|
|
6
|
+
import subprocess
|
|
7
|
+
import traceback
|
|
8
|
+
from typing import TYPE_CHECKING, Annotated, Any, Dict, List, Optional, TypedDict
|
|
9
|
+
|
|
10
|
+
import boto3
|
|
11
|
+
from pydantic import ValidationError
|
|
12
|
+
|
|
13
|
+
if TYPE_CHECKING:
|
|
14
|
+
from aws_lambda_typing.context import Context
|
|
15
|
+
else:
|
|
16
|
+
Context = Annotated[object, "Context object"]
|
|
17
|
+
|
|
18
|
+
from stactools_item_generator.item import ItemRequest, create_stac_item
|
|
19
|
+
|
|
20
|
+
logger = logging.getLogger()
|
|
21
|
+
if logger.hasHandlers():
|
|
22
|
+
logger.handlers.clear()
|
|
23
|
+
|
|
24
|
+
log_handler = logging.StreamHandler() # <--- Renamed handler variable
|
|
25
|
+
|
|
26
|
+
log_level_name = os.environ.get("LOG_LEVEL", "INFO").upper()
|
|
27
|
+
log_level = logging._nameToLevel.get(log_level_name, logging.INFO)
|
|
28
|
+
logger.setLevel(log_level)
|
|
29
|
+
|
|
30
|
+
formatter = logging.Formatter("%(asctime)s - %(name)s - %(levelname)s - %(message)s")
|
|
31
|
+
log_handler.setFormatter(formatter)
|
|
32
|
+
logger.addHandler(log_handler)
|
|
33
|
+
|
|
34
|
+
|
|
35
|
+
def get_topic_arn() -> str:
|
|
36
|
+
item_load_topic_arn = os.environ.get("ITEM_LOAD_TOPIC_ARN")
|
|
37
|
+
if not item_load_topic_arn:
|
|
38
|
+
logger.error("Environment variable ITEM_LOAD_TOPIC_ARN is not set.")
|
|
39
|
+
raise EnvironmentError("ITEM_LOAD_TOPIC_ARN must be set")
|
|
40
|
+
|
|
41
|
+
return item_load_topic_arn
|
|
42
|
+
|
|
43
|
+
|
|
44
|
+
def process_record(record: Dict[str, Any], sns_client) -> None:
|
|
45
|
+
"""
|
|
46
|
+
Processes a single SQS record (within a batch).
|
|
47
|
+
Extracts the request, calls create_stac_item, and publishes the result.
|
|
48
|
+
Raises exceptions on failure.
|
|
49
|
+
"""
|
|
50
|
+
message_id = record.get("messageId", "UNKNOWN_ID")
|
|
51
|
+
logger.info(f"Processing record: {message_id}")
|
|
52
|
+
message_str = None
|
|
53
|
+
try:
|
|
54
|
+
sqs_body_str = record["body"]
|
|
55
|
+
logger.debug(f"[{message_id}] SQS message body: {sqs_body_str}")
|
|
56
|
+
sns_notification = json.loads(sqs_body_str)
|
|
57
|
+
|
|
58
|
+
message_str = sns_notification["Message"]
|
|
59
|
+
logger.debug(f"[{message_id}] SNS Message content: {message_str}")
|
|
60
|
+
|
|
61
|
+
message_data = json.loads(message_str)
|
|
62
|
+
item_request = ItemRequest(**message_data)
|
|
63
|
+
logger.info(
|
|
64
|
+
f"[{message_id}] Parsed ItemRequest for package: {item_request.package_name}"
|
|
65
|
+
)
|
|
66
|
+
logger.debug(f"[{message_id}] Full ItemRequest: {item_request.model_dump_json()}")
|
|
67
|
+
|
|
68
|
+
stac_item = create_stac_item(item_request)
|
|
69
|
+
logger.info(f"[{message_id}] Successfully created STAC item: {stac_item.id}")
|
|
70
|
+
logger.debug(
|
|
71
|
+
f"[{message_id}] Generated STAC Item JSON (sample): "
|
|
72
|
+
f"{ {k: v for k, v in stac_item.model_dump().items() if k in ['id', 'collection', 'properties']} }"
|
|
73
|
+
)
|
|
74
|
+
|
|
75
|
+
stac_item_json = stac_item.model_dump_json()
|
|
76
|
+
|
|
77
|
+
item_load_topic_arn = get_topic_arn()
|
|
78
|
+
logger.info(
|
|
79
|
+
f"[{message_id}] Publishing STAC item {stac_item.id} to {item_load_topic_arn}"
|
|
80
|
+
)
|
|
81
|
+
response = sns_client.publish(
|
|
82
|
+
TopicArn=item_load_topic_arn,
|
|
83
|
+
Message=stac_item_json,
|
|
84
|
+
)
|
|
85
|
+
logger.info(
|
|
86
|
+
f"[{message_id}] SNS publish response MessageId: {response.get('MessageId')}"
|
|
87
|
+
)
|
|
88
|
+
|
|
89
|
+
except json.JSONDecodeError as e:
|
|
90
|
+
logger.error(f"[{message_id}] Failed to decode JSON: {e}")
|
|
91
|
+
logger.error(f"[{message_id}] Problematic data (SQS Body): {record.get('body')}")
|
|
92
|
+
raise
|
|
93
|
+
except ValidationError as e:
|
|
94
|
+
logger.error(f"[{message_id}] Failed to validate ItemRequest: {e}")
|
|
95
|
+
logger.error(f"[{message_id}] Validation errors:\n{e.errors()}")
|
|
96
|
+
problem_data = message_str if message_str is not None else record.get("body")
|
|
97
|
+
logger.error(
|
|
98
|
+
f"[{message_id}] Problematic data (SNS Message or SQS Body): {problem_data}"
|
|
99
|
+
)
|
|
100
|
+
raise
|
|
101
|
+
except (
|
|
102
|
+
subprocess.CalledProcessError
|
|
103
|
+
) as e: # <--- Catching the imported exception type
|
|
104
|
+
logger.error(f"[{message_id}] Subprocess command failed:")
|
|
105
|
+
logger.error(f"[{message_id}] Command: {' '.join(e.cmd)}")
|
|
106
|
+
logger.error(f"[{message_id}] Return code: {e.returncode}")
|
|
107
|
+
logger.error(f"[{message_id}] Stdout: {e.stdout}")
|
|
108
|
+
logger.error(f"[{message_id}] Stderr: {e.stderr}")
|
|
109
|
+
raise
|
|
110
|
+
except Exception as e:
|
|
111
|
+
logger.error(
|
|
112
|
+
f"[{message_id}] An unexpected error occurred processing record: {e}"
|
|
113
|
+
)
|
|
114
|
+
logger.error(traceback.format_exc())
|
|
115
|
+
raise
|
|
116
|
+
|
|
117
|
+
|
|
118
|
+
class BatchItemFailure(TypedDict):
|
|
119
|
+
itemIdentifier: str
|
|
120
|
+
|
|
121
|
+
|
|
122
|
+
class PartialBatchFailureResponse(TypedDict):
|
|
123
|
+
batchItemFailures: List[BatchItemFailure]
|
|
124
|
+
|
|
125
|
+
|
|
126
|
+
def handler(
|
|
127
|
+
event: Dict[str, Any], context: Context
|
|
128
|
+
) -> Optional[PartialBatchFailureResponse]:
|
|
129
|
+
"""
|
|
130
|
+
AWS Lambda handler function triggered by SQS with batching enabled.
|
|
131
|
+
|
|
132
|
+
Processes messages in batches, attempts to generate STAC items, publishes
|
|
133
|
+
successful results to SNS, and reports partial batch failures to SQS.
|
|
134
|
+
"""
|
|
135
|
+
try:
|
|
136
|
+
sns_client = boto3.client("sns", region_name=os.getenv("AWS_DEFAULT_REGION"))
|
|
137
|
+
except Exception as e:
|
|
138
|
+
logging.error(f"Error: {str(e)}")
|
|
139
|
+
raise EnvironmentError("AWS_DEFAULT_REGION must be set") from e
|
|
140
|
+
|
|
141
|
+
records = event.get("Records", [])
|
|
142
|
+
aws_request_id = getattr(context, "aws_request_id", "N/A")
|
|
143
|
+
remaining_time = getattr(context, "get_remaining_time_in_millis", lambda: "N/A")()
|
|
144
|
+
|
|
145
|
+
logger.info(f"Received batch with {len(records)} records.")
|
|
146
|
+
logger.debug(
|
|
147
|
+
f"Lambda Context: RequestId={aws_request_id}, RemainingTime={remaining_time}ms"
|
|
148
|
+
)
|
|
149
|
+
|
|
150
|
+
batch_item_failures: List[BatchItemFailure] = []
|
|
151
|
+
|
|
152
|
+
for record in records:
|
|
153
|
+
message_id = record.get("messageId")
|
|
154
|
+
if not message_id:
|
|
155
|
+
logger.warning("Record missing messageId, cannot report failure for it.")
|
|
156
|
+
continue
|
|
157
|
+
|
|
158
|
+
try:
|
|
159
|
+
process_record(record, sns_client)
|
|
160
|
+
logger.info(f"[{message_id}] Successfully processed.")
|
|
161
|
+
|
|
162
|
+
except Exception:
|
|
163
|
+
logger.error(f"[{message_id}] Marked as failed.")
|
|
164
|
+
batch_item_failures.append({"itemIdentifier": message_id})
|
|
165
|
+
|
|
166
|
+
if batch_item_failures:
|
|
167
|
+
logger.warning(
|
|
168
|
+
f"Finished processing batch. {len(batch_item_failures)} failure(s) reported."
|
|
169
|
+
)
|
|
170
|
+
logger.info(
|
|
171
|
+
f"Returning failed item identifiers: {[f['itemIdentifier'] for f in batch_item_failures]}"
|
|
172
|
+
)
|
|
173
|
+
return {"batchItemFailures": batch_item_failures}
|
|
174
|
+
else:
|
|
175
|
+
logger.info("Finished processing batch. All records successful.")
|
|
176
|
+
return None
|
|
@@ -0,0 +1,77 @@
|
|
|
1
|
+
import json
|
|
2
|
+
import logging
|
|
3
|
+
import subprocess
|
|
4
|
+
from tempfile import NamedTemporaryFile
|
|
5
|
+
from typing import Dict, List, Optional
|
|
6
|
+
|
|
7
|
+
from pydantic import BaseModel, ConfigDict, Field
|
|
8
|
+
from stac_pydantic.item import Item
|
|
9
|
+
|
|
10
|
+
logger = logging.getLogger()
|
|
11
|
+
logger.setLevel(logging.INFO)
|
|
12
|
+
|
|
13
|
+
|
|
14
|
+
class ItemRequest(BaseModel):
|
|
15
|
+
package_name: str = Field(..., description="Name of the stactools package")
|
|
16
|
+
group_name: str = Field(..., description="Group name for the STAC item")
|
|
17
|
+
create_item_args: List[str] = Field(
|
|
18
|
+
..., description="Arguments for create-item command"
|
|
19
|
+
)
|
|
20
|
+
create_item_options: Dict[str, str] = Field(
|
|
21
|
+
default_factory=dict, description="Options for create-item command"
|
|
22
|
+
)
|
|
23
|
+
collection_id: Optional[str] = Field(
|
|
24
|
+
None, description="value for the collection field of the item json"
|
|
25
|
+
)
|
|
26
|
+
|
|
27
|
+
model_config = ConfigDict(
|
|
28
|
+
json_schema_extra={
|
|
29
|
+
"example": {
|
|
30
|
+
"package_name": "stactools-glad-glclu2020",
|
|
31
|
+
"group_name": "gladglclu2020",
|
|
32
|
+
"create_item_args": [
|
|
33
|
+
"https://storage.googleapis.com/earthenginepartners-hansen/GLCLU2000-2020/v2/2000/50N_090W.tif"
|
|
34
|
+
],
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
)
|
|
38
|
+
|
|
39
|
+
|
|
40
|
+
def create_stac_item(request: ItemRequest) -> Item:
|
|
41
|
+
"""
|
|
42
|
+
Create a STAC item using a stactools package
|
|
43
|
+
"""
|
|
44
|
+
logger.info(f"Received request: {json.dumps(request.model_dump())}")
|
|
45
|
+
|
|
46
|
+
if not request.package_name:
|
|
47
|
+
raise ValueError("Missing required parameter: package_name")
|
|
48
|
+
|
|
49
|
+
command = [
|
|
50
|
+
"uvx",
|
|
51
|
+
"--with",
|
|
52
|
+
f"requests,{request.package_name}",
|
|
53
|
+
"--from",
|
|
54
|
+
"stactools",
|
|
55
|
+
"stac",
|
|
56
|
+
request.group_name,
|
|
57
|
+
"create-item",
|
|
58
|
+
*request.create_item_args,
|
|
59
|
+
]
|
|
60
|
+
|
|
61
|
+
for option, value in request.create_item_options.items():
|
|
62
|
+
command.extend([f"--{option}", value])
|
|
63
|
+
|
|
64
|
+
logger.info(f"Executing command: {' '.join(command)}")
|
|
65
|
+
|
|
66
|
+
with NamedTemporaryFile(suffix=".json") as output:
|
|
67
|
+
command.append(output.name)
|
|
68
|
+
result = subprocess.run(command, capture_output=True, text=True, check=True)
|
|
69
|
+
|
|
70
|
+
logger.info(f"Command output: {result.stdout}")
|
|
71
|
+
with open(output.name) as f:
|
|
72
|
+
item_dict = json.load(f)
|
|
73
|
+
|
|
74
|
+
if request.collection_id:
|
|
75
|
+
item_dict["collection"] = request.collection_id
|
|
76
|
+
|
|
77
|
+
return Item(**item_dict)
|
package/lib/tipg-api/index.js
CHANGED
|
@@ -57,5 +57,5 @@ class TiPgApiLambda extends constructs_1.Construct {
|
|
|
57
57
|
}
|
|
58
58
|
exports.TiPgApiLambda = TiPgApiLambda;
|
|
59
59
|
_a = JSII_RTTI_SYMBOL_1;
|
|
60
|
-
TiPgApiLambda[_a] = { fqn: "eoapi-cdk.TiPgApiLambda", version: "8.
|
|
60
|
+
TiPgApiLambda[_a] = { fqn: "eoapi-cdk.TiPgApiLambda", version: "8.2.0" };
|
|
61
61
|
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["index.ts"],"names":[],"mappings":";;;;;AAAA,6CAWqB;AACrB,2CAAuC;AAEvC,6BAA6B;AAE3B,MAAa,aAAc,SAAQ,sBAAS;IAI1C,YAAY,KAAgB,EAAE,EAAU,EAAE,KAAyB;QACjE,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAEjB,IAAI,CAAC,kBAAkB,GAAG,IAAI,wBAAM,CAAC,QAAQ,CAAC,IAAI,EAAE,QAAQ,EAAE;YAC5D,WAAW;YACX,OAAO,EAAE,wBAAM,CAAC,OAAO,CAAC,WAAW;YACnC,OAAO,EAAE,iBAAiB;YAC1B,UAAU,EAAE,IAAI;YAChB,YAAY,EAAE,sBAAI,CAAC,aAAa,CAAC,QAAQ;YACzC,OAAO,EAAE,sBAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;YAC7B,IAAI,EAAE,wBAAM,CAAC,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,IAAI,CAAC,EAAE;gBAC5D,IAAI,EAAE,6BAA6B;gBACnC,SAAS,EAAE,EAAE,cAAc,EAAE,MAAM,EAAE;aACtC,CAAC;YACF,GAAG,EAAE,KAAK,CAAC,GAAG;YACd,UAAU,EAAE,KAAK,CAAC,eAAe;YACjC,iBAAiB,EAAE,IAAI;YACvB,WAAW,EAAE;gBACX,iBAAiB,EAAE,KAAK,CAAC,QAAQ,CAAC,SAAS;gBAC3C,gBAAgB,EAAE,GAAG;gBACrB,gBAAgB,EAAE,GAAG;gBACrB,GAAG,KAAK,CAAC,MAAM;aAChB;YACD,iEAAiE;YACjE,GAAG,KAAK,CAAC,qBAAqB;SAC/B,CAAC,CAAC;QAEH,KAAK,CAAC,QAAQ,CAAC,SAAS,CAAC,IAAI,CAAC,kBAAkB,CAAC,CAAC;QAElD,IAAI,KAAK,CAAC,GAAG,EAAC,CAAC;YACb,IAAI,CAAC,kBAAkB,CAAC,WAAW,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE,EAAE,qBAAG,CAAC,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,6BAA6B,CAAC,CAAC;QAC3G,CAAC;QAED,MAAM,OAAO,GAAG,IAAI,8BAAY,CAAC,OAAO,CACtC,IAAI,EACJ,GAAG,mBAAK,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,SAAS,WAAW,EACtC;YACE,oBAAoB,EAAE,KAAK,CAAC,iBAAiB;gBAC3C,CAAC,CAAC;oBACE,UAAU,EAAE,KAAK,CAAC,iBAAiB;iBACpC;gBACH,CAAC,CAAC,SAAS;YACb,kBAAkB,EAChB,IAAI,2CAAyB,CAAC,qBAAqB,CACjD,aAAa,EACb,IAAI,CAAC,kBAAkB,EACvB,KAAK,CAAC,iBAAiB;gBACrB,CAAC,CAAC;oBACE,gBAAgB,EACd,IAAI,8BAAY,CAAC,gBAAgB,EAAE,CAAC,eAAe,CACjD,MAAM,EACN,8BAAY,CAAC,YAAY,CAAC,MAAM,CAC9B,KAAK,CAAC,iBAAiB,CAAC,IAAI,CAC7B,CACF;iBACJ;gBACH,CAAC,CAAC,SAAS,CACd;SACJ,CACF,CAAC;QAEF,IAAI,CAAC,GAAG,GAAG,OAAO,CAAC,GAAI,CAAC;QAExB,IAAI,uBAAS,CAAC,IAAI,EAAE,iBAAiB,EAAE;YACrC,UAAU,EAAE,GAAG,mBAAK,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,SAAS,UAAU;YACjD,KAAK,EAAE,IAAI,CAAC,GAAG;SAChB,CAAC,CAAC;IACL,CAAC;;AAvEH,sCAwEC","sourcesContent":["import {\n  Stack,\n  aws_apigatewayv2 as apigatewayv2,\n  aws_apigatewayv2_integrations as apigatewayv2_integrations,\n  aws_ec2 as ec2,\n  aws_lambda as lambda,\n  aws_logs as logs,\n  aws_rds as rds,\n  aws_secretsmanager as secretsmanager,\n  CfnOutput,\n  Duration,\n} from \"aws-cdk-lib\";\nimport { Construct } from \"constructs\";\nimport { CustomLambdaFunctionProps } from \"../utils\";\nimport * as path from 'path';\n\n  export class TiPgApiLambda extends Construct {\n    readonly url: string;\n    public tiPgLambdaFunction: lambda.Function;\n\n    constructor(scope: Construct, id: string, props: TiPgApiLambdaProps) {\n      super(scope, id);\n\n      this.tiPgLambdaFunction = new lambda.Function(this, \"lambda\", {\n        // defaults\n        runtime: lambda.Runtime.PYTHON_3_11,\n        handler: \"handler.handler\",\n        memorySize: 1024,\n        logRetention: logs.RetentionDays.ONE_WEEK,\n        timeout: Duration.seconds(30),\n        code: lambda.Code.fromDockerBuild(path.join(__dirname, '..'), {\n          file: \"tipg-api/runtime/Dockerfile\",\n          buildArgs: { PYTHON_VERSION: '3.11' },\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: \"1\",\n          DB_MAX_CONN_SIZE: \"1\",\n          ...props.apiEnv,\n        },\n        // overwrites defaults with user-provided configurable properties\n        ...props.lambdaFunctionOptions\n      });\n\n      props.dbSecret.grantRead(this.tiPgLambdaFunction);\n\n      if (props.vpc){\n        this.tiPgLambdaFunction.connections.allowTo(props.db, ec2.Port.tcp(5432), \"allow connections from tipg\");\n      }\n\n      const tipgApi = new apigatewayv2.HttpApi(\n        this,\n        `${Stack.of(this).stackName}-tipg-api`,\n        {\n          defaultDomainMapping: props.tipgApiDomainName\n            ? {\n                domainName: props.tipgApiDomainName,\n              }\n            : undefined,\n          defaultIntegration:\n            new apigatewayv2_integrations.HttpLambdaIntegration(\n              \"integration\",\n              this.tiPgLambdaFunction,\n              props.tipgApiDomainName\n                ? {\n                    parameterMapping:\n                      new apigatewayv2.ParameterMapping().overwriteHeader(\n                        \"host\",\n                        apigatewayv2.MappingValue.custom(\n                          props.tipgApiDomainName.name\n                        )\n                      ),\n                  }\n                : undefined\n            ),\n        }\n      );\n\n      this.url = tipgApi.url!;\n\n      new CfnOutput(this, \"tipg-api-output\", {\n        exportName: `${Stack.of(this).stackName}-tip-url`,\n        value: this.url,\n      });\n    }\n  }\n\n  export interface TiPgApiLambdaProps {\n    /**\n     * VPC into which the lambda should be deployed.\n     */\n    readonly vpc?: ec2.IVpc;\n\n    /**\n     * RDS Instance with installed pgSTAC or pgbouncer server.\n     */\n    readonly db: rds.IDatabaseInstance | ec2.IInstance;\n\n    /**\n     * Subnet into which the lambda should be deployed.\n     */\n    readonly subnetSelection?: ec2.SubnetSelection;\n\n    /**\n     * Secret containing connection information for pgSTAC database.\n     */\n    readonly dbSecret: secretsmanager.ISecret;\n\n    /**\n     * Customized environment variables to send to titiler-pgstac runtime.\n     */\n    readonly apiEnv?: Record<string, string>;\n\n    /**\n     * Custom Domain Name for tipg API. If defined, will create the\n     * domain name and integrate it with the tipg API.\n     *\n     * @default - undefined\n     */\n    readonly tipgApiDomainName?: apigatewayv2.IDomainName;\n\n    /**\n     * Can be used to override the default lambda function properties.\n     *\n     * @default - defined in the construct.\n     */\n    readonly lambdaFunctionOptions?: CustomLambdaFunctionProps;\n  }\n"]}
|
|
@@ -77,5 +77,5 @@ class TitilerPgstacApiLambda extends constructs_1.Construct {
|
|
|
77
77
|
}
|
|
78
78
|
exports.TitilerPgstacApiLambda = TitilerPgstacApiLambda;
|
|
79
79
|
_a = JSII_RTTI_SYMBOL_1;
|
|
80
|
-
TitilerPgstacApiLambda[_a] = { fqn: "eoapi-cdk.TitilerPgstacApiLambda", version: "8.
|
|
80
|
+
TitilerPgstacApiLambda[_a] = { fqn: "eoapi-cdk.TitilerPgstacApiLambda", version: "8.2.0" };
|
|
81
81
|
//# 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_apigatewayv2 as apigatewayv2,\n  aws_apigatewayv2_integrations as apigatewayv2_integrations,\n  aws_iam as iam,\n  aws_ec2 as ec2,\n  aws_rds as rds,\n  aws_lambda as lambda,\n  aws_secretsmanager as secretsmanager,\n  CfnOutput,\n  Duration,\n  aws_logs,\n} from \"aws-cdk-lib\";\nimport { Construct } from \"constructs\";\nimport { CustomLambdaFunctionProps } from \"../utils\";\nimport * as path from 'path';\n\n  // default settings that can be overridden by the user-provided environment.\n  let defaultTitilerPgstacEnv :{ [key: string]: any } = {\n    \"CPL_VSIL_CURL_ALLOWED_EXTENSIONS\": \".tif,.TIF,.tiff\",\n    \"GDAL_CACHEMAX\": \"200\",\n    \"GDAL_DISABLE_READDIR_ON_OPEN\": \"EMPTY_DIR\",\n    \"GDAL_INGESTED_BYTES_AT_OPEN\": \"32768\",\n    \"GDAL_HTTP_MERGE_CONSECUTIVE_RANGES\": \"YES\",\n    \"GDAL_HTTP_MULTIPLEX\": \"YES\",\n    \"GDAL_HTTP_VERSION\": \"2\",\n    \"PYTHONWARNINGS\": \"ignore\",\n    \"VSI_CACHE\": \"TRUE\",\n    \"VSI_CACHE_SIZE\": \"5000000\",\n    \"DB_MIN_CONN_SIZE\": \"1\",\n    \"DB_MAX_CONN_SIZE\": \"1\"\n  }\n\n  export class TitilerPgstacApiLambda extends Construct {\n    readonly url: string;\n    public titilerPgstacLambdaFunction: lambda.Function;\n\n    constructor(scope: Construct, id: string, props: TitilerPgStacApiLambdaProps) {\n      super(scope, id);\n\n      this.titilerPgstacLambdaFunction = new lambda.Function(this, \"lambda\", {\n        // defaults\n        runtime: lambda.Runtime.PYTHON_3_11,\n        handler: \"handler.handler\",\n        memorySize: 3008,\n        logRetention: aws_logs.RetentionDays.ONE_WEEK,\n        timeout: Duration.seconds(30),\n        code: lambda.Code.fromDockerBuild(path.join(__dirname, '..'), {\n          file: \"titiler-pgstac-api/runtime/Dockerfile\",\n          buildArgs: { PYTHON_VERSION: '3.11' }\n        }),\n        vpc: props.vpc,\n        vpcSubnets: props.subnetSelection,\n        allowPublicSubnet: true,\n        // if user provided environment variables, merge them with the defaults.\n        environment: props.apiEnv ? { ...defaultTitilerPgstacEnv, ...props.apiEnv, \"PGSTAC_SECRET_ARN\": props.dbSecret.secretArn } : defaultTitilerPgstacEnv,\n        // overwrites defaults with user-provided configurable properties\n        ...props.lambdaFunctionOptions,\n      });\n\n      // grant access to buckets using addToRolePolicy\n      if (props.buckets) {\n        props.buckets.forEach(bucket => {\n          this.titilerPgstacLambdaFunction.addToRolePolicy(new iam.PolicyStatement({\n            actions: [\"s3:GetObject\"],\n            resources: [`arn:aws:s3:::${bucket}/*`],\n          }));\n        });\n      }\n\n      props.dbSecret.grantRead(this.titilerPgstacLambdaFunction);\n\n      if (props.vpc) {\n        this.titilerPgstacLambdaFunction.connections.allowTo(props.db, ec2.Port.tcp(5432), \"allow connections from titiler\");\n      }\n\n      const stacApi = new apigatewayv2.HttpApi(\n        this,\n        `${Stack.of(this).stackName}-titiler-pgstac-api`,\n        {\n          defaultDomainMapping: props.titilerPgstacApiDomainName\n            ? {\n                domainName: props.titilerPgstacApiDomainName,\n              }\n            : undefined,\n          defaultIntegration:\n            new apigatewayv2_integrations.HttpLambdaIntegration(\n              \"integration\",\n              this.titilerPgstacLambdaFunction,\n              props.titilerPgstacApiDomainName\n                ? {\n                    parameterMapping:\n                      new apigatewayv2.ParameterMapping().overwriteHeader(\n                        \"host\",\n                        apigatewayv2.MappingValue.custom(\n                          props.titilerPgstacApiDomainName.name\n                        )\n                      ),\n                  }\n                : undefined\n            ),\n        }\n      );\n\n      this.url = stacApi.url!;\n\n      new CfnOutput(this, \"titiler-pgstac-api-output\", {\n        exportName: `${Stack.of(this).stackName}-titiler-pgstac-url`,\n        value: this.url,\n      });\n    }\n  }\n\n  export interface TitilerPgStacApiLambdaProps {\n    /**\n     * VPC into which the lambda should be deployed.\n     */\n    readonly vpc?: ec2.IVpc;\n\n    /**\n     * RDS Instance with installed pgSTAC or pgbouncer server.\n     */\n    readonly db: rds.IDatabaseInstance | ec2.IInstance;\n\n    /**\n     * Subnet into which the lambda should be deployed.\n     */\n    readonly subnetSelection?: ec2.SubnetSelection;\n\n    /**\n     * Secret containing connection information for pgSTAC database.\n     */\n    readonly dbSecret: secretsmanager.ISecret;\n\n    /**\n     * Customized environment variables to send to titiler-pgstac runtime. These will be merged with `defaultTitilerPgstacEnv`.\n     * The database secret arn is automatically added to the environment variables at deployment.\n    /*/\n    readonly apiEnv?: Record<string, string>;\n\n    /**\n     * list of buckets the lambda will be granted access to.\n     */\n    readonly buckets?: string[];\n\n    /**\n     * Custom Domain Name Options for Titiler Pgstac API,\n     *\n     * @default - undefined.\n     */\n    readonly titilerPgstacApiDomainName?: apigatewayv2.IDomainName;\n\n    /**\n     * Can be used to override the default lambda function properties.\n     *\n     * @default - defined in the construct.\n     */\n    readonly lambdaFunctionOptions?: CustomLambdaFunctionProps;\n  }\n"]}
|
package/package.json
CHANGED
package/pyproject.toml
ADDED
|
@@ -0,0 +1,45 @@
|
|
|
1
|
+
[project]
|
|
2
|
+
name = "eoapi-cdk"
|
|
3
|
+
version = "0.0"
|
|
4
|
+
requires-python = ">=3.11"
|
|
5
|
+
dependencies = [
|
|
6
|
+
"stactools-item-generator",
|
|
7
|
+
"stac-item-loader",
|
|
8
|
+
]
|
|
9
|
+
|
|
10
|
+
[tool.uv.sources]
|
|
11
|
+
stactools-item-generator = { workspace = true }
|
|
12
|
+
stac-item-loader = { workspace = true }
|
|
13
|
+
|
|
14
|
+
[tool.uv.workspace]
|
|
15
|
+
members = ["lib/stac-item-loader/runtime", "lib/stactools-item-generator/runtime"]
|
|
16
|
+
|
|
17
|
+
[dependency-groups]
|
|
18
|
+
deploy = [
|
|
19
|
+
"aws-cdk-lib==2.190.0",
|
|
20
|
+
"constructs==10.3.0",
|
|
21
|
+
"pydantic>=2.11.5",
|
|
22
|
+
"pydantic-settings[yaml]>=2.8.1",
|
|
23
|
+
"python-dotenv>=1.1.0",
|
|
24
|
+
"pyyaml>=6.0.2",
|
|
25
|
+
"types-pyyaml>=6.0.12.20250516",
|
|
26
|
+
]
|
|
27
|
+
dev = [
|
|
28
|
+
"aws-lambda-typing>=2.20.0",
|
|
29
|
+
"httpx>=0.28.1",
|
|
30
|
+
"pytest>=8.3.5",
|
|
31
|
+
"pytest-mock>=3.14.0",
|
|
32
|
+
"pytest-postgresql>=7.0.1",
|
|
33
|
+
]
|
|
34
|
+
|
|
35
|
+
[tool.pytest.ini_options]
|
|
36
|
+
addopts = "-vv --ignore=cdk.out --no-header --tb=native"
|
|
37
|
+
pythonpath = "."
|
|
38
|
+
testpaths = [
|
|
39
|
+
"lib/stactools-item-generator/runtime/tests",
|
|
40
|
+
"lib/stac-item-loader/runtime/tests",
|
|
41
|
+
]
|
|
42
|
+
|
|
43
|
+
[tool.ruff]
|
|
44
|
+
|
|
45
|
+
[tool.ruff.lint]
|