eoapi-cdk 11.3.1 → 11.4.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.jsii +40 -39
- package/lib/bastion-host/index.js +1 -1
- package/lib/database/bootstrapper_runtime/handler.py +157 -13
- package/lib/database/index.d.ts +36 -4
- package/lib/database/index.js +4 -4
- package/lib/ingestor-api/index.js +1 -1
- package/lib/lambda-api-gateway/index.js +1 -1
- package/lib/lambda-api-gateway-private/index.js +1 -1
- package/lib/stac-api/index.js +2 -2
- package/lib/stac-auth-proxy/index.js +2 -2
- package/lib/stac-browser/index.js +1 -1
- package/lib/stac-loader/index.js +2 -2
- package/lib/stactools-item-generator/index.js +1 -1
- package/lib/tipg-api/index.js +2 -2
- package/lib/titiler-pgstac-api/index.js +2 -2
- package/package.json +5 -5
package/.jsii
CHANGED
|
@@ -4081,7 +4081,7 @@
|
|
|
4081
4081
|
"stability": "experimental"
|
|
4082
4082
|
},
|
|
4083
4083
|
"homepage": "https://github.com/developmentseed/eoapi-cdk.git",
|
|
4084
|
-
"jsiiVersion": "5.9.
|
|
4084
|
+
"jsiiVersion": "5.9.34 (build 8773a22)",
|
|
4085
4085
|
"keywords": [],
|
|
4086
4086
|
"license": "ISC",
|
|
4087
4087
|
"metadata": {
|
|
@@ -4300,7 +4300,7 @@
|
|
|
4300
4300
|
"kind": "interface",
|
|
4301
4301
|
"locationInModule": {
|
|
4302
4302
|
"filename": "lib/database/index.ts",
|
|
4303
|
-
"line":
|
|
4303
|
+
"line": 495
|
|
4304
4304
|
},
|
|
4305
4305
|
"name": "DatabaseParameters",
|
|
4306
4306
|
"properties": [
|
|
@@ -4313,7 +4313,7 @@
|
|
|
4313
4313
|
"immutable": true,
|
|
4314
4314
|
"locationInModule": {
|
|
4315
4315
|
"filename": "lib/database/index.ts",
|
|
4316
|
-
"line":
|
|
4316
|
+
"line": 511
|
|
4317
4317
|
},
|
|
4318
4318
|
"name": "effectiveCacheSize",
|
|
4319
4319
|
"type": {
|
|
@@ -4329,7 +4329,7 @@
|
|
|
4329
4329
|
"immutable": true,
|
|
4330
4330
|
"locationInModule": {
|
|
4331
4331
|
"filename": "lib/database/index.ts",
|
|
4332
|
-
"line":
|
|
4332
|
+
"line": 521
|
|
4333
4333
|
},
|
|
4334
4334
|
"name": "maintenanceWorkMem",
|
|
4335
4335
|
"type": {
|
|
@@ -4345,7 +4345,7 @@
|
|
|
4345
4345
|
"immutable": true,
|
|
4346
4346
|
"locationInModule": {
|
|
4347
4347
|
"filename": "lib/database/index.ts",
|
|
4348
|
-
"line":
|
|
4348
|
+
"line": 499
|
|
4349
4349
|
},
|
|
4350
4350
|
"name": "maxConnections",
|
|
4351
4351
|
"type": {
|
|
@@ -4361,7 +4361,7 @@
|
|
|
4361
4361
|
"immutable": true,
|
|
4362
4362
|
"locationInModule": {
|
|
4363
4363
|
"filename": "lib/database/index.ts",
|
|
4364
|
-
"line":
|
|
4364
|
+
"line": 526
|
|
4365
4365
|
},
|
|
4366
4366
|
"name": "maxLocksPerTransaction",
|
|
4367
4367
|
"type": {
|
|
@@ -4377,7 +4377,7 @@
|
|
|
4377
4377
|
"immutable": true,
|
|
4378
4378
|
"locationInModule": {
|
|
4379
4379
|
"filename": "lib/database/index.ts",
|
|
4380
|
-
"line":
|
|
4380
|
+
"line": 541
|
|
4381
4381
|
},
|
|
4382
4382
|
"name": "randomPageCost",
|
|
4383
4383
|
"type": {
|
|
@@ -4393,7 +4393,7 @@
|
|
|
4393
4393
|
"immutable": true,
|
|
4394
4394
|
"locationInModule": {
|
|
4395
4395
|
"filename": "lib/database/index.ts",
|
|
4396
|
-
"line":
|
|
4396
|
+
"line": 536
|
|
4397
4397
|
},
|
|
4398
4398
|
"name": "seqPageCost",
|
|
4399
4399
|
"type": {
|
|
@@ -4410,7 +4410,7 @@
|
|
|
4410
4410
|
"immutable": true,
|
|
4411
4411
|
"locationInModule": {
|
|
4412
4412
|
"filename": "lib/database/index.ts",
|
|
4413
|
-
"line":
|
|
4413
|
+
"line": 506
|
|
4414
4414
|
},
|
|
4415
4415
|
"name": "sharedBuffers",
|
|
4416
4416
|
"type": {
|
|
@@ -4426,7 +4426,7 @@
|
|
|
4426
4426
|
"immutable": true,
|
|
4427
4427
|
"locationInModule": {
|
|
4428
4428
|
"filename": "lib/database/index.ts",
|
|
4429
|
-
"line":
|
|
4429
|
+
"line": 531
|
|
4430
4430
|
},
|
|
4431
4431
|
"name": "tempBuffers",
|
|
4432
4432
|
"type": {
|
|
@@ -4442,7 +4442,7 @@
|
|
|
4442
4442
|
"immutable": true,
|
|
4443
4443
|
"locationInModule": {
|
|
4444
4444
|
"filename": "lib/database/index.ts",
|
|
4445
|
-
"line":
|
|
4445
|
+
"line": 516
|
|
4446
4446
|
},
|
|
4447
4447
|
"name": "workMem",
|
|
4448
4448
|
"type": {
|
|
@@ -4988,7 +4988,7 @@
|
|
|
4988
4988
|
},
|
|
4989
4989
|
"locationInModule": {
|
|
4990
4990
|
"filename": "lib/database/index.ts",
|
|
4991
|
-
"line":
|
|
4991
|
+
"line": 130
|
|
4992
4992
|
},
|
|
4993
4993
|
"parameters": [
|
|
4994
4994
|
{
|
|
@@ -5014,7 +5014,7 @@
|
|
|
5014
5014
|
"kind": "class",
|
|
5015
5015
|
"locationInModule": {
|
|
5016
5016
|
"filename": "lib/database/index.ts",
|
|
5017
|
-
"line":
|
|
5017
|
+
"line": 118
|
|
5018
5018
|
},
|
|
5019
5019
|
"methods": [
|
|
5020
5020
|
{
|
|
@@ -5023,7 +5023,7 @@
|
|
|
5023
5023
|
},
|
|
5024
5024
|
"locationInModule": {
|
|
5025
5025
|
"filename": "lib/database/index.ts",
|
|
5026
|
-
"line":
|
|
5026
|
+
"line": 308
|
|
5027
5027
|
},
|
|
5028
5028
|
"name": "getParameters",
|
|
5029
5029
|
"parameters": [
|
|
@@ -5062,7 +5062,7 @@
|
|
|
5062
5062
|
"immutable": true,
|
|
5063
5063
|
"locationInModule": {
|
|
5064
5064
|
"filename": "lib/database/index.ts",
|
|
5065
|
-
"line":
|
|
5065
|
+
"line": 124
|
|
5066
5066
|
},
|
|
5067
5067
|
"name": "connectionTarget",
|
|
5068
5068
|
"type": {
|
|
@@ -5085,7 +5085,7 @@
|
|
|
5085
5085
|
"immutable": true,
|
|
5086
5086
|
"locationInModule": {
|
|
5087
5087
|
"filename": "lib/database/index.ts",
|
|
5088
|
-
"line":
|
|
5088
|
+
"line": 123
|
|
5089
5089
|
},
|
|
5090
5090
|
"name": "pgstacVersion",
|
|
5091
5091
|
"type": {
|
|
@@ -5099,7 +5099,7 @@
|
|
|
5099
5099
|
"immutable": true,
|
|
5100
5100
|
"locationInModule": {
|
|
5101
5101
|
"filename": "lib/database/index.ts",
|
|
5102
|
-
"line":
|
|
5102
|
+
"line": 127
|
|
5103
5103
|
},
|
|
5104
5104
|
"name": "pgbouncerHealthCheck",
|
|
5105
5105
|
"optional": true,
|
|
@@ -5114,7 +5114,7 @@
|
|
|
5114
5114
|
"immutable": true,
|
|
5115
5115
|
"locationInModule": {
|
|
5116
5116
|
"filename": "lib/database/index.ts",
|
|
5117
|
-
"line":
|
|
5117
|
+
"line": 128
|
|
5118
5118
|
},
|
|
5119
5119
|
"name": "pgbouncerInstanceId",
|
|
5120
5120
|
"optional": true,
|
|
@@ -5129,7 +5129,7 @@
|
|
|
5129
5129
|
"immutable": true,
|
|
5130
5130
|
"locationInModule": {
|
|
5131
5131
|
"filename": "lib/database/index.ts",
|
|
5132
|
-
"line":
|
|
5132
|
+
"line": 126
|
|
5133
5133
|
},
|
|
5134
5134
|
"name": "secretBootstrapper",
|
|
5135
5135
|
"optional": true,
|
|
@@ -5144,7 +5144,7 @@
|
|
|
5144
5144
|
"immutable": true,
|
|
5145
5145
|
"locationInModule": {
|
|
5146
5146
|
"filename": "lib/database/index.ts",
|
|
5147
|
-
"line":
|
|
5147
|
+
"line": 125
|
|
5148
5148
|
},
|
|
5149
5149
|
"name": "securityGroup",
|
|
5150
5150
|
"optional": true,
|
|
@@ -5158,7 +5158,7 @@
|
|
|
5158
5158
|
},
|
|
5159
5159
|
"locationInModule": {
|
|
5160
5160
|
"filename": "lib/database/index.ts",
|
|
5161
|
-
"line":
|
|
5161
|
+
"line": 119
|
|
5162
5162
|
},
|
|
5163
5163
|
"name": "db",
|
|
5164
5164
|
"type": {
|
|
@@ -5171,7 +5171,7 @@
|
|
|
5171
5171
|
},
|
|
5172
5172
|
"locationInModule": {
|
|
5173
5173
|
"filename": "lib/database/index.ts",
|
|
5174
|
-
"line":
|
|
5174
|
+
"line": 120
|
|
5175
5175
|
},
|
|
5176
5176
|
"name": "pgstacSecret",
|
|
5177
5177
|
"type": {
|
|
@@ -5194,7 +5194,7 @@
|
|
|
5194
5194
|
"kind": "interface",
|
|
5195
5195
|
"locationInModule": {
|
|
5196
5196
|
"filename": "lib/database/index.ts",
|
|
5197
|
-
"line":
|
|
5197
|
+
"line": 348
|
|
5198
5198
|
},
|
|
5199
5199
|
"name": "PgStacDatabaseProps",
|
|
5200
5200
|
"properties": [
|
|
@@ -5208,7 +5208,7 @@
|
|
|
5208
5208
|
"immutable": true,
|
|
5209
5209
|
"locationInModule": {
|
|
5210
5210
|
"filename": "lib/database/index.ts",
|
|
5211
|
-
"line":
|
|
5211
|
+
"line": 417
|
|
5212
5212
|
},
|
|
5213
5213
|
"name": "addPatchManager",
|
|
5214
5214
|
"optional": true,
|
|
@@ -5226,7 +5226,7 @@
|
|
|
5226
5226
|
"immutable": true,
|
|
5227
5227
|
"locationInModule": {
|
|
5228
5228
|
"filename": "lib/database/index.ts",
|
|
5229
|
-
"line":
|
|
5229
|
+
"line": 382
|
|
5230
5230
|
},
|
|
5231
5231
|
"name": "addPgbouncer",
|
|
5232
5232
|
"optional": true,
|
|
@@ -5244,7 +5244,7 @@
|
|
|
5244
5244
|
"immutable": true,
|
|
5245
5245
|
"locationInModule": {
|
|
5246
5246
|
"filename": "lib/database/index.ts",
|
|
5247
|
-
"line":
|
|
5247
|
+
"line": 473
|
|
5248
5248
|
},
|
|
5249
5249
|
"name": "bootstrapperLambdaFunctionOptions",
|
|
5250
5250
|
"optional": true,
|
|
@@ -5255,14 +5255,15 @@
|
|
|
5255
5255
|
{
|
|
5256
5256
|
"abstract": true,
|
|
5257
5257
|
"docs": {
|
|
5258
|
-
"
|
|
5258
|
+
"example": "customResourceProperties: {\n update_collection_extent: \"TRUE\",\n use_queue: \"TRUE\",\n pg_cron_schedule: \"*\\/10 * * * *\",\n}",
|
|
5259
|
+
"remarks": "These are merged with the defaults and forwarded\nto the handler as `event[\"ResourceProperties\"]`.\n\n## Supported pgSTAC settings\n\nEach setting follows honest boolean semantics: `\"TRUE\"` enables, `\"FALSE\"`\ndisables (reverts to pgSTAC's built-in default), and omitting the key\nentirely is a no-op — the database is left as-is. This means upgrading\nwithout changing config never silently alters a manually-managed setting.\n\n| Property | Default | Description |\n|----------------------------|----------|-------------|\n| `context` | (unset) | Enable `CONTEXT=ON` in pgSTAC (item count on search). |\n| `mosaic_index` | `\"TRUE\"` | Create a partial index on searches of type `mosaic`. Set `\"FALSE\"` to drop it. |\n| `update_collection_extent` | (unset) | Automatically update collection spatial/temporal extents on item ingest. Combine with `use_queue` to reduce per-transaction overhead. |\n| `use_queue` | (unset) | Process extent updates asynchronously via an internal queue. `\"TRUE\"` installs pg_cron and schedules the drain job; `\"FALSE\"` removes the job. Requires `pg_cron` (see below). |\n| `pg_cron_schedule` | `\"*\\/10 * * * *\"` | Cron schedule for `CALL run_queued_queries()`. Only used when `use_queue` is `\"TRUE\"`. |\n\n## pg_cron requirement\n\nWhen `use_queue` is `\"TRUE\"`, the bootstrapper installs the `pg_cron`\nextension and schedules a job to periodically drain the queue.\n`pg_cron` must be present in `shared_preload_libraries` **before**\ndeployment or the bootstrap will fail. Add it via `props.parameters`:\n\n```\nparameters: { shared_preload_libraries: \"pg_cron\" }\n```",
|
|
5259
5260
|
"stability": "experimental",
|
|
5260
|
-
"summary": "Lambda
|
|
5261
|
+
"summary": "Additional properties passed to the bootstrapper Lambda as CloudFormation Custom Resource properties."
|
|
5261
5262
|
},
|
|
5262
5263
|
"immutable": true,
|
|
5263
5264
|
"locationInModule": {
|
|
5264
5265
|
"filename": "lib/database/index.ts",
|
|
5265
|
-
"line":
|
|
5266
|
+
"line": 464
|
|
5266
5267
|
},
|
|
5267
5268
|
"name": "customResourceProperties",
|
|
5268
5269
|
"optional": true,
|
|
@@ -5286,7 +5287,7 @@
|
|
|
5286
5287
|
"immutable": true,
|
|
5287
5288
|
"locationInModule": {
|
|
5288
5289
|
"filename": "lib/database/index.ts",
|
|
5289
|
-
"line":
|
|
5290
|
+
"line": 492
|
|
5290
5291
|
},
|
|
5291
5292
|
"name": "forceBootstrap",
|
|
5292
5293
|
"optional": true,
|
|
@@ -5304,7 +5305,7 @@
|
|
|
5304
5305
|
"immutable": true,
|
|
5305
5306
|
"locationInModule": {
|
|
5306
5307
|
"filename": "lib/database/index.ts",
|
|
5307
|
-
"line":
|
|
5308
|
+
"line": 424
|
|
5308
5309
|
},
|
|
5309
5310
|
"name": "maintenanceWindow",
|
|
5310
5311
|
"optional": true,
|
|
@@ -5323,7 +5324,7 @@
|
|
|
5323
5324
|
"immutable": true,
|
|
5324
5325
|
"locationInModule": {
|
|
5325
5326
|
"filename": "lib/database/index.ts",
|
|
5326
|
-
"line":
|
|
5327
|
+
"line": 409
|
|
5327
5328
|
},
|
|
5328
5329
|
"name": "pgbouncerAmiSsmParameter",
|
|
5329
5330
|
"optional": true,
|
|
@@ -5341,7 +5342,7 @@
|
|
|
5341
5342
|
"immutable": true,
|
|
5342
5343
|
"locationInModule": {
|
|
5343
5344
|
"filename": "lib/database/index.ts",
|
|
5344
|
-
"line":
|
|
5345
|
+
"line": 389
|
|
5345
5346
|
},
|
|
5346
5347
|
"name": "pgbouncerInstanceProps",
|
|
5347
5348
|
"optional": true,
|
|
@@ -5359,7 +5360,7 @@
|
|
|
5359
5360
|
"immutable": true,
|
|
5360
5361
|
"locationInModule": {
|
|
5361
5362
|
"filename": "lib/database/index.ts",
|
|
5362
|
-
"line":
|
|
5363
|
+
"line": 354
|
|
5363
5364
|
},
|
|
5364
5365
|
"name": "pgstacDbName",
|
|
5365
5366
|
"optional": true,
|
|
@@ -5377,7 +5378,7 @@
|
|
|
5377
5378
|
"immutable": true,
|
|
5378
5379
|
"locationInModule": {
|
|
5379
5380
|
"filename": "lib/database/index.ts",
|
|
5380
|
-
"line":
|
|
5381
|
+
"line": 375
|
|
5381
5382
|
},
|
|
5382
5383
|
"name": "pgstacUsername",
|
|
5383
5384
|
"optional": true,
|
|
@@ -5395,7 +5396,7 @@
|
|
|
5395
5396
|
"immutable": true,
|
|
5396
5397
|
"locationInModule": {
|
|
5397
5398
|
"filename": "lib/database/index.ts",
|
|
5398
|
-
"line":
|
|
5399
|
+
"line": 361
|
|
5399
5400
|
},
|
|
5400
5401
|
"name": "pgstacVersion",
|
|
5401
5402
|
"optional": true,
|
|
@@ -5413,7 +5414,7 @@
|
|
|
5413
5414
|
"immutable": true,
|
|
5414
5415
|
"locationInModule": {
|
|
5415
5416
|
"filename": "lib/database/index.ts",
|
|
5416
|
-
"line":
|
|
5417
|
+
"line": 368
|
|
5417
5418
|
},
|
|
5418
5419
|
"name": "secretsPrefix",
|
|
5419
5420
|
"optional": true,
|
|
@@ -7999,6 +8000,6 @@
|
|
|
7999
8000
|
"symbolId": "lib/titiler-pgstac-api/index:TitilerPgstacApiLambdaRuntimeProps"
|
|
8000
8001
|
}
|
|
8001
8002
|
},
|
|
8002
|
-
"version": "11.
|
|
8003
|
-
"fingerprint": "
|
|
8003
|
+
"version": "11.4.0",
|
|
8004
|
+
"fingerprint": "XAgc9qDybKGNxcDQS+Srt8N+/5AJbD+nSQHWHr6Q12o="
|
|
8004
8005
|
}
|
|
@@ -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: "11.
|
|
161
|
+
BastionHost[_a] = { fqn: "eoapi-cdk.BastionHost", version: "11.4.0" };
|
|
162
162
|
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["index.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Stack,\n  aws_ec2 as ec2,\n  aws_iam as iam,\n  aws_rds as rds,\n  CfnOutput,\n} from \"aws-cdk-lib\";\nimport { Construct } from \"constructs\";\n\n/**\n * The database is located in an isolated subnet, meaning that it is not accessible from the public internet. As such, to interact with the database directly, a user must tunnel through a bastion host.\n *\n * ### Configuring\n *\n * This codebase controls _who_ is allowed to connect to the bastion host. This requires two steps:\n *\n * 1. Adding the IP address from which you are connecting to the `ipv4Allowlist` array\n * 1. Creating a bastion host system user by adding the user's configuration inform to `userdata.yaml`\n *\n * #### Adding an IP address to the `ipv4Allowlist` array\n *\n * The `BastionHost` construct takes in an `ipv4Allowlist` array as an argument. Find your IP address (eg `curl api.ipify.org`) and add that to the array along with the trailing CIDR block (likely `/32` to indicate that you are adding a single IP address).\n *\n * #### Creating a user via `userdata.yaml`\n *\n * Add an entry to the `users` array with a username (likely matching your local systems username, which you can get by running the `whoami` command in your terminal) and a public key (likely your default public key, which you can get by running `cat ~/.ssh/id_*.pub` in your terminal).\n *\n * #### Tips & Tricks when using the Bastion Host\n *\n * **Connecting to RDS Instance via SSM**\n *\n * ```sh\n * aws ssm start-session --target $INSTANCE_ID \\\n * --document-name AWS-StartPortForwardingSessionToRemoteHost \\\n * --parameters '{\n * \"host\": [\n * \"example-db.c5abcdefghij.us-west-2.rds.amazonaws.com\"\n * ],\n * \"portNumber\": [\n * \"5432\"\n * ],\n * \"localPortNumber\": [\n * \"9999\"\n * ]\n * }' \\\n * --profile $AWS_PROFILE\n * ```\n *\n * ```sh\n * psql -h localhost -p 9999 # continue adding username (-U) and db (-d) here...\n * ```\n *\n * Connect directly to Bastion Host:\n *\n * ```sh\n * aws ssm start-session --target $INSTANCE_ID --profile $AWS_PROFILE\n * ```\n *\n * **Setting up an SSH tunnel**\n *\n * In your `~/.ssh/config` file, add an entry like:\n *\n * ```\n * Host db-tunnel\n * Hostname {the-bastion-host-address}\n * LocalForward 9999 {the-db-hostname}:5432\n * ```\n *\n * Then a tunnel can be opened via:\n *\n * ```\n * ssh -N db-tunnel\n * ```\n *\n * And a connection to the DB can be made via:\n *\n * ```\n * psql -h 127.0.0.1 -p 9999 -U {username} -d {database}\n * ```\n *\n * **Handling `REMOTE HOST IDENTIFICATION HAS CHANGED!` error**\n *\n * If you've redeployed a bastion host that you've previously connected to, you may see an error like:\n *\n * ```\n * @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@\n * @    WARNING: REMOTE HOST IDENTIFICATION HAS CHANGED!     @\n * @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@\n * IT IS POSSIBLE THAT SOMEONE IS DOING SOMETHING NASTY!\n * Someone could be eavesdropping on you right now (man-in-the-middle attack)!\n * It is also possible that a host key has just been changed.\n * The fingerprint for the ECDSA key sent by the remote host is\n * SHA256:mPnxAOXTpb06PFgI1Qc8TMQ2e9b7goU8y2NdS5hzIr8.\n * Please contact your system administrator.\n * Add correct host key in /Users/username/.ssh/known_hosts to get rid of this message.\n * Offending ECDSA key in /Users/username/.ssh/known_hosts:28\n * ECDSA host key for ec2-12-34-56-789.us-west-2.compute.amazonaws.com has changed and you have requested strict checking.\n * Host key verification failed.\n * ```\n *\n * This is due to the server's fingerprint changing. We can scrub the fingerprint from our system with a command like:\n *\n * ```\n * ssh-keygen -R 12.34.56.789\n * ```\n *\n */\nexport class BastionHost extends Construct {\n  instance: ec2.Instance;\n\n  constructor(scope: Construct, id: string, props: BastionHostProps) {\n    super(scope, id);\n\n    const { stackName } = Stack.of(this);\n\n    // Build ec2 instance\n    this.instance = new ec2.Instance(this, \"bastion-host\", {\n      vpc: props.vpc,\n      vpcSubnets: { subnetType: ec2.SubnetType.PUBLIC },\n      instanceName: `${stackName} bastion host`,\n      instanceType: ec2.InstanceType.of(\n        ec2.InstanceClass.BURSTABLE4_GRAVITON,\n        ec2.InstanceSize.NANO\n      ),\n      machineImage: ec2.MachineImage.latestAmazonLinux({\n        generation: ec2.AmazonLinuxGeneration.AMAZON_LINUX_2,\n        cpuType: ec2.AmazonLinuxCpuType.ARM_64,\n      }),\n      userData: props.userData,\n      userDataCausesReplacement: true,\n    });\n\n    // Assign elastic IP\n    if (props.createElasticIp ?? true) {\n      new ec2.CfnEIP(this, \"IP\", {\n        instanceId: this.instance.instanceId,\n        tags: [{ key: \"Name\", value: stackName }],\n      });\n    }\n\n    // Allow bastion host to connect to db\n    this.instance.connections.allowTo(\n      props.db.connections.securityGroups[0],\n      ec2.Port.tcp(5432),\n      \"Allow connection from bastion host\"\n    );\n\n    // Allow IP access to bastion host\n    for (const ipv4 of props.ipv4Allowlist) {\n      this.instance.connections.allowFrom(\n        ec2.Peer.ipv4(ipv4),\n        ec2.Port.tcp(props.sshPort || 22),\n        \"SSH Access\"\n      );\n    }\n\n    // Integrate with SSM\n    this.instance.addToRolePolicy(\n      new iam.PolicyStatement({\n        actions: [\n          \"ssmmessages:*\",\n          \"ssm:UpdateInstanceInformation\",\n          \"ec2messages:*\",\n        ],\n        resources: [\"*\"],\n      })\n    );\n\n    new CfnOutput(this, \"instance-id-output\", {\n      value: this.instance.instanceId,\n      exportName: `${stackName}-instance-id`,\n    });\n    new CfnOutput(this, \"instance-public-ip-output\", {\n      value: this.instance.instancePublicIp,\n      exportName: `${stackName}-instance-public-ip`,\n    });\n    new CfnOutput(this, \"instance-public-dns-name-output\", {\n      value: this.instance.instancePublicDnsName,\n      exportName: `${stackName}-public-dns-name`,\n    });\n  }\n}\n\nexport interface BastionHostProps {\n  readonly vpc: ec2.IVpc;\n  readonly db: rds.IDatabaseInstance;\n  readonly userData: ec2.UserData;\n  readonly ipv4Allowlist: string[];\n  readonly sshPort?: number;\n\n  /**\n   * Whether or not an elastic IP should be created for the bastion host.\n   *\n   * @default false\n   */\n  readonly createElasticIp?: boolean;\n}\n"]}
|
|
@@ -17,6 +17,9 @@ from pypgstac.migrate import Migrate
|
|
|
17
17
|
logger = logging.getLogger("eoapi-bootstrap")
|
|
18
18
|
|
|
19
19
|
|
|
20
|
+
DEFAULT_PG_CRON_SCHEDULE = "*/10 * * * *" # every 10 minutes
|
|
21
|
+
|
|
22
|
+
|
|
20
23
|
def send(
|
|
21
24
|
event,
|
|
22
25
|
context,
|
|
@@ -143,30 +146,151 @@ def register_extensions(cursor) -> None:
|
|
|
143
146
|
###############################################################################
|
|
144
147
|
# PgSTAC Customization
|
|
145
148
|
###############################################################################
|
|
146
|
-
def
|
|
149
|
+
def _is_enabled(params: dict, key: str) -> bool:
|
|
150
|
+
"""Parse a string boolean CloudFormation custom resource property.
|
|
151
|
+
|
|
152
|
+
CloudFormation custom resource properties are always strings, so boolean
|
|
153
|
+
flags are passed as ``"TRUE"`` or ``"FALSE"``. Callers must check that the
|
|
154
|
+
key is present before calling this function; absent keys are a no-op and
|
|
155
|
+
should not reach this function.
|
|
156
|
+
|
|
157
|
+
Args:
|
|
158
|
+
params: ResourceProperties dict from the CloudFormation event.
|
|
159
|
+
key: Property name to look up.
|
|
160
|
+
|
|
161
|
+
Returns:
|
|
162
|
+
True if the property value is ``"TRUE"`` (case-insensitive).
|
|
147
163
|
"""
|
|
148
|
-
|
|
164
|
+
return str(params.get(key, "FALSE")).upper() == "TRUE"
|
|
165
|
+
|
|
166
|
+
|
|
167
|
+
def _apply_pgstac_setting(cursor, name: str, enabled: bool) -> None:
|
|
168
|
+
"""Upsert or delete a row in the pgstac_settings table.
|
|
169
|
+
|
|
170
|
+
When ``enabled`` is True, upserts ``'on'``. When False, deletes the row so
|
|
171
|
+
pgstac reverts to its built-in default. When the setting is absent from
|
|
172
|
+
params entirely, this function should not be called — absent means no-op.
|
|
173
|
+
|
|
174
|
+
Args:
|
|
175
|
+
cursor: Database cursor with ``search_path`` including the pgstac schema.
|
|
176
|
+
name: Setting name (primary key in pgstac_settings).
|
|
177
|
+
enabled: Whether to enable or disable the setting.
|
|
178
|
+
"""
|
|
179
|
+
if enabled:
|
|
180
|
+
cursor.execute(
|
|
181
|
+
sql.SQL(
|
|
182
|
+
"INSERT INTO pgstac_settings (name, value) VALUES ({name}, 'on') "
|
|
183
|
+
"ON CONFLICT ON CONSTRAINT pgstac_settings_pkey DO UPDATE SET value = excluded.value;"
|
|
184
|
+
).format(name=sql.Literal(name))
|
|
185
|
+
)
|
|
186
|
+
else:
|
|
187
|
+
cursor.execute(
|
|
188
|
+
sql.SQL("DELETE FROM pgstac_settings WHERE name = {name};").format(
|
|
189
|
+
name=sql.Literal(name)
|
|
190
|
+
)
|
|
191
|
+
)
|
|
192
|
+
|
|
193
|
+
|
|
194
|
+
def customization(cursor, params) -> None:
|
|
195
|
+
"""Apply pgSTAC database customizations based on the provided params.
|
|
196
|
+
|
|
197
|
+
Each setting follows honest boolean semantics:
|
|
198
|
+
|
|
199
|
+
- ``"TRUE"`` → enable (upsert ``'on'`` in pgstac_settings)
|
|
200
|
+
- ``"FALSE"`` → disable (delete from pgstac_settings, reverting to pgstac default)
|
|
201
|
+
- absent → no-op (the database is left exactly as-is)
|
|
202
|
+
|
|
203
|
+
This means the flags are safe to use as true booleans, and upgrading without
|
|
204
|
+
changing config never silently alters a manually-managed setting.
|
|
149
205
|
|
|
150
206
|
ref: https://github.com/stac-utils/pgstac/blob/main/docs/src/pgstac.md
|
|
207
|
+
"""
|
|
208
|
+
if "context" in params:
|
|
209
|
+
_apply_pgstac_setting(cursor, "context", _is_enabled(params, "context"))
|
|
210
|
+
|
|
211
|
+
if "mosaic_index" in params:
|
|
212
|
+
if _is_enabled(params, "mosaic_index"):
|
|
213
|
+
cursor.execute(
|
|
214
|
+
sql.SQL(
|
|
215
|
+
"CREATE INDEX IF NOT EXISTS searches_mosaic ON searches ((true)) "
|
|
216
|
+
"WHERE metadata->>'type'='mosaic';"
|
|
217
|
+
)
|
|
218
|
+
)
|
|
219
|
+
else:
|
|
220
|
+
cursor.execute(sql.SQL("DROP INDEX IF EXISTS searches_mosaic;"))
|
|
221
|
+
|
|
222
|
+
# Enable automatic spatial/temporal extent updates on item ingest.
|
|
223
|
+
# For large ingests, combine with use_queue=TRUE to reduce transaction overhead.
|
|
224
|
+
if "update_collection_extent" in params:
|
|
225
|
+
_apply_pgstac_setting(
|
|
226
|
+
cursor,
|
|
227
|
+
"update_collection_extent",
|
|
228
|
+
_is_enabled(params, "update_collection_extent"),
|
|
229
|
+
)
|
|
230
|
+
|
|
231
|
+
# Enable asynchronous queue processing for extent updates.
|
|
232
|
+
# Requires pg_cron to periodically invoke CALL pgstac.run_queued_queries().
|
|
233
|
+
if "use_queue" in params:
|
|
234
|
+
_apply_pgstac_setting(cursor, "use_queue", _is_enabled(params, "use_queue"))
|
|
235
|
+
|
|
236
|
+
|
|
237
|
+
def unregister_pg_cron(cursor) -> None:
|
|
238
|
+
"""Remove the run_queued_queries pg_cron job if it exists.
|
|
239
|
+
|
|
240
|
+
Safe to call even if pg_cron is not installed or the job does not exist.
|
|
151
241
|
|
|
242
|
+
Args:
|
|
243
|
+
cursor: Database cursor connected to the ``postgres`` database as a superuser.
|
|
152
244
|
"""
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
pgstac_settings = """
|
|
156
|
-
INSERT INTO pgstac_settings (name, value)
|
|
157
|
-
VALUES ('context', 'on')
|
|
158
|
-
ON CONFLICT ON CONSTRAINT pgstac_settings_pkey DO UPDATE SET value = excluded.value;"""
|
|
159
|
-
cursor.execute(sql.SQL(pgstac_settings))
|
|
160
|
-
|
|
161
|
-
if str(params.get("mosaic_index", "TRUE")).upper() == "TRUE":
|
|
162
|
-
# Create index of searches with `mosaic`` type
|
|
245
|
+
cursor.execute(sql.SQL("SELECT 1 FROM pg_extension WHERE extname = 'pg_cron';"))
|
|
246
|
+
if cursor.fetchone():
|
|
163
247
|
cursor.execute(
|
|
164
248
|
sql.SQL(
|
|
165
|
-
"
|
|
249
|
+
"SELECT cron.unschedule(jobid) FROM cron.job "
|
|
250
|
+
"WHERE jobname = 'pgstac-run-queued-queries';"
|
|
166
251
|
)
|
|
167
252
|
)
|
|
168
253
|
|
|
169
254
|
|
|
255
|
+
def register_pg_cron(cursor, db_name: str, schedule: str) -> None:
|
|
256
|
+
"""Install the pg_cron extension and schedule run_queued_queries.
|
|
257
|
+
|
|
258
|
+
pg_cron can only be installed in the ``postgres`` database, so the cursor
|
|
259
|
+
must be connected there as a superuser. ``cron.schedule_in_database`` is
|
|
260
|
+
used to run the job against the pgSTAC database. The job is upserted by
|
|
261
|
+
name so repeated bootstrap runs are safe.
|
|
262
|
+
|
|
263
|
+
pg_cron must be listed in ``shared_preload_libraries`` in the RDS parameter
|
|
264
|
+
group before this will succeed.
|
|
265
|
+
|
|
266
|
+
Side effect: sets ``search_path = pgstac, public`` as the default for all
|
|
267
|
+
connections to ``db_name`` via ``ALTER DATABASE``. This is required because
|
|
268
|
+
pg_cron background worker sessions start with no ``search_path``, and
|
|
269
|
+
prepending ``SET search_path`` to the cron command would open a transaction
|
|
270
|
+
that prevents ``run_queued_queries()`` from issuing its own ``COMMIT``.
|
|
271
|
+
|
|
272
|
+
Args:
|
|
273
|
+
cursor: Database cursor connected to the ``postgres`` database as a superuser.
|
|
274
|
+
db_name: Name of the pgSTAC database where run_queued_queries will run.
|
|
275
|
+
schedule: Cron schedule expression (e.g. ``"*/5 * * * *"``).
|
|
276
|
+
"""
|
|
277
|
+
cursor.execute(sql.SQL("CREATE EXTENSION IF NOT EXISTS pg_cron;"))
|
|
278
|
+
cursor.execute(
|
|
279
|
+
sql.SQL("ALTER DATABASE {db_name} SET search_path TO pgstac, public;").format(
|
|
280
|
+
db_name=sql.Identifier(db_name),
|
|
281
|
+
)
|
|
282
|
+
)
|
|
283
|
+
cursor.execute(
|
|
284
|
+
sql.SQL(
|
|
285
|
+
"SELECT cron.schedule_in_database({job_name}, {schedule}, 'CALL pgstac.run_queued_queries();', {db_name});"
|
|
286
|
+
).format(
|
|
287
|
+
job_name=sql.Literal("pgstac-run-queued-queries"),
|
|
288
|
+
schedule=sql.Literal(schedule),
|
|
289
|
+
db_name=sql.Literal(db_name),
|
|
290
|
+
)
|
|
291
|
+
)
|
|
292
|
+
|
|
293
|
+
|
|
170
294
|
def handler(event, context):
|
|
171
295
|
"""Lambda Handler."""
|
|
172
296
|
print(f"Handling {event}")
|
|
@@ -213,6 +337,26 @@ def handler(event, context):
|
|
|
213
337
|
password=eoapi_params["password"],
|
|
214
338
|
)
|
|
215
339
|
|
|
340
|
+
if "use_queue" in params:
|
|
341
|
+
if _is_enabled(params, "use_queue"):
|
|
342
|
+
schedule = params.get(
|
|
343
|
+
"pg_cron_schedule", DEFAULT_PG_CRON_SCHEDULE
|
|
344
|
+
)
|
|
345
|
+
print(
|
|
346
|
+
f"Scheduling pg_cron job 'pgstac-run-queued-queries' "
|
|
347
|
+
f"with schedule '{schedule}'..."
|
|
348
|
+
)
|
|
349
|
+
register_pg_cron(
|
|
350
|
+
cursor=cur,
|
|
351
|
+
db_name=eoapi_params["dbname"],
|
|
352
|
+
schedule=schedule,
|
|
353
|
+
)
|
|
354
|
+
else:
|
|
355
|
+
print(
|
|
356
|
+
"Removing pg_cron job 'pgstac-run-queued-queries' if present..."
|
|
357
|
+
)
|
|
358
|
+
unregister_pg_cron(cursor=cur)
|
|
359
|
+
|
|
216
360
|
# Install postgis and pgstac on the eoapi database with
|
|
217
361
|
# superuser permissions
|
|
218
362
|
print(f"Connecting to eoAPI '{eoapi_params['dbname']}' database...")
|
package/lib/database/index.d.ts
CHANGED
|
@@ -127,10 +127,42 @@ export interface PgStacDatabaseProps extends rds.DatabaseInstanceProps {
|
|
|
127
127
|
*/
|
|
128
128
|
readonly maintenanceWindow?: ssm.CfnMaintenanceWindow;
|
|
129
129
|
/**
|
|
130
|
-
*
|
|
131
|
-
*
|
|
132
|
-
*
|
|
133
|
-
*
|
|
130
|
+
* Additional properties passed to the bootstrapper Lambda as CloudFormation
|
|
131
|
+
* Custom Resource properties. These are merged with the defaults and forwarded
|
|
132
|
+
* to the handler as `event["ResourceProperties"]`.
|
|
133
|
+
*
|
|
134
|
+
* ## Supported pgSTAC settings
|
|
135
|
+
*
|
|
136
|
+
* Each setting follows honest boolean semantics: `"TRUE"` enables, `"FALSE"`
|
|
137
|
+
* disables (reverts to pgSTAC's built-in default), and omitting the key
|
|
138
|
+
* entirely is a no-op — the database is left as-is. This means upgrading
|
|
139
|
+
* without changing config never silently alters a manually-managed setting.
|
|
140
|
+
*
|
|
141
|
+
* | Property | Default | Description |
|
|
142
|
+
* |----------------------------|----------|-------------|
|
|
143
|
+
* | `context` | (unset) | Enable `CONTEXT=ON` in pgSTAC (item count on search). |
|
|
144
|
+
* | `mosaic_index` | `"TRUE"` | Create a partial index on searches of type `mosaic`. Set `"FALSE"` to drop it. |
|
|
145
|
+
* | `update_collection_extent` | (unset) | Automatically update collection spatial/temporal extents on item ingest. Combine with `use_queue` to reduce per-transaction overhead. |
|
|
146
|
+
* | `use_queue` | (unset) | Process extent updates asynchronously via an internal queue. `"TRUE"` installs pg_cron and schedules the drain job; `"FALSE"` removes the job. Requires `pg_cron` (see below). |
|
|
147
|
+
* | `pg_cron_schedule` | `"*\/10 * * * *"` | Cron schedule for `CALL run_queued_queries()`. Only used when `use_queue` is `"TRUE"`. |
|
|
148
|
+
*
|
|
149
|
+
* ## pg_cron requirement
|
|
150
|
+
*
|
|
151
|
+
* When `use_queue` is `"TRUE"`, the bootstrapper installs the `pg_cron`
|
|
152
|
+
* extension and schedules a job to periodically drain the queue.
|
|
153
|
+
* `pg_cron` must be present in `shared_preload_libraries` **before**
|
|
154
|
+
* deployment or the bootstrap will fail. Add it via `props.parameters`:
|
|
155
|
+
*
|
|
156
|
+
* ```
|
|
157
|
+
* parameters: { shared_preload_libraries: "pg_cron" }
|
|
158
|
+
* ```
|
|
159
|
+
*
|
|
160
|
+
* @example
|
|
161
|
+
* customResourceProperties: {
|
|
162
|
+
* update_collection_extent: "TRUE",
|
|
163
|
+
* use_queue: "TRUE",
|
|
164
|
+
* pg_cron_schedule: "*\/10 * * * *",
|
|
165
|
+
* }
|
|
134
166
|
*/
|
|
135
167
|
readonly customResourceProperties?: {
|
|
136
168
|
[key: string]: any;
|