eoapi-cdk 5.3.0 → 5.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 CHANGED
@@ -3589,6 +3589,73 @@
3589
3589
  }
3590
3590
  },
3591
3591
  "types": {
3592
+ "eoapi-cdk.ApiCode": {
3593
+ "assembly": "eoapi-cdk",
3594
+ "datatype": true,
3595
+ "docs": {
3596
+ "stability": "experimental"
3597
+ },
3598
+ "fqn": "eoapi-cdk.ApiCode",
3599
+ "kind": "interface",
3600
+ "locationInModule": {
3601
+ "filename": "lib/ingestor-api/index.ts",
3602
+ "line": 326
3603
+ },
3604
+ "name": "ApiCode",
3605
+ "properties": [
3606
+ {
3607
+ "abstract": true,
3608
+ "docs": {
3609
+ "stability": "experimental",
3610
+ "summary": "Path to the source of the function or the location for dependencies, for the api lambda."
3611
+ },
3612
+ "immutable": true,
3613
+ "locationInModule": {
3614
+ "filename": "lib/ingestor-api/index.ts",
3615
+ "line": 331
3616
+ },
3617
+ "name": "entry",
3618
+ "type": {
3619
+ "primitive": "string"
3620
+ }
3621
+ },
3622
+ {
3623
+ "abstract": true,
3624
+ "docs": {
3625
+ "stability": "experimental",
3626
+ "summary": "The name of the exported handler in the `api_lambda_index` file."
3627
+ },
3628
+ "immutable": true,
3629
+ "locationInModule": {
3630
+ "filename": "lib/ingestor-api/index.ts",
3631
+ "line": 341
3632
+ },
3633
+ "name": "handler",
3634
+ "optional": true,
3635
+ "type": {
3636
+ "primitive": "string"
3637
+ }
3638
+ },
3639
+ {
3640
+ "abstract": true,
3641
+ "docs": {
3642
+ "stability": "experimental",
3643
+ "summary": "Path to the index file containing the exported handler, relative to `api_lambda_entry`."
3644
+ },
3645
+ "immutable": true,
3646
+ "locationInModule": {
3647
+ "filename": "lib/ingestor-api/index.ts",
3648
+ "line": 336
3649
+ },
3650
+ "name": "index",
3651
+ "optional": true,
3652
+ "type": {
3653
+ "primitive": "string"
3654
+ }
3655
+ }
3656
+ ],
3657
+ "symbolId": "lib/ingestor-api/index:ApiCode"
3658
+ },
3592
3659
  "eoapi-cdk.ApiEntrypoint": {
3593
3660
  "assembly": "eoapi-cdk",
3594
3661
  "datatype": true,
@@ -4208,6 +4275,73 @@
4208
4275
  ],
4209
4276
  "symbolId": "lib/database/index:DatabaseParameters"
4210
4277
  },
4278
+ "eoapi-cdk.IngestorCode": {
4279
+ "assembly": "eoapi-cdk",
4280
+ "datatype": true,
4281
+ "docs": {
4282
+ "stability": "experimental"
4283
+ },
4284
+ "fqn": "eoapi-cdk.IngestorCode",
4285
+ "kind": "interface",
4286
+ "locationInModule": {
4287
+ "filename": "lib/ingestor-api/index.ts",
4288
+ "line": 345
4289
+ },
4290
+ "name": "IngestorCode",
4291
+ "properties": [
4292
+ {
4293
+ "abstract": true,
4294
+ "docs": {
4295
+ "stability": "experimental",
4296
+ "summary": "Path to the source of the function or the location for dependencies, for the ingestor lambda."
4297
+ },
4298
+ "immutable": true,
4299
+ "locationInModule": {
4300
+ "filename": "lib/ingestor-api/index.ts",
4301
+ "line": 350
4302
+ },
4303
+ "name": "entry",
4304
+ "type": {
4305
+ "primitive": "string"
4306
+ }
4307
+ },
4308
+ {
4309
+ "abstract": true,
4310
+ "docs": {
4311
+ "stability": "experimental",
4312
+ "summary": "The name of the exported handler in the `ingestor_lambda_index` file."
4313
+ },
4314
+ "immutable": true,
4315
+ "locationInModule": {
4316
+ "filename": "lib/ingestor-api/index.ts",
4317
+ "line": 360
4318
+ },
4319
+ "name": "handler",
4320
+ "optional": true,
4321
+ "type": {
4322
+ "primitive": "string"
4323
+ }
4324
+ },
4325
+ {
4326
+ "abstract": true,
4327
+ "docs": {
4328
+ "stability": "experimental",
4329
+ "summary": "Path to the index file containing the exported handler, relative to `ingestor_lambda_entry`."
4330
+ },
4331
+ "immutable": true,
4332
+ "locationInModule": {
4333
+ "filename": "lib/ingestor-api/index.ts",
4334
+ "line": 355
4335
+ },
4336
+ "name": "index",
4337
+ "optional": true,
4338
+ "type": {
4339
+ "primitive": "string"
4340
+ }
4341
+ }
4342
+ ],
4343
+ "symbolId": "lib/ingestor-api/index:IngestorCode"
4344
+ },
4211
4345
  "eoapi-cdk.PgStacApiLambda": {
4212
4346
  "assembly": "eoapi-cdk",
4213
4347
  "base": "constructs.Construct",
@@ -4891,7 +5025,7 @@
4891
5025
  "kind": "interface",
4892
5026
  "locationInModule": {
4893
5027
  "filename": "lib/ingestor-api/index.ts",
4894
- "line": 238
5028
+ "line": 255
4895
5029
  },
4896
5030
  "name": "StacIngestorProps",
4897
5031
  "properties": [
@@ -4904,7 +5038,7 @@
4904
5038
  "immutable": true,
4905
5039
  "locationInModule": {
4906
5040
  "filename": "lib/ingestor-api/index.ts",
4907
- "line": 242
5041
+ "line": 259
4908
5042
  },
4909
5043
  "name": "dataAccessRole",
4910
5044
  "type": {
@@ -4920,7 +5054,7 @@
4920
5054
  "immutable": true,
4921
5055
  "locationInModule": {
4922
5056
  "filename": "lib/ingestor-api/index.ts",
4923
- "line": 257
5057
+ "line": 274
4924
5058
  },
4925
5059
  "name": "stacDbSecret",
4926
5060
  "type": {
@@ -4936,7 +5070,7 @@
4936
5070
  "immutable": true,
4937
5071
  "locationInModule": {
4938
5072
  "filename": "lib/ingestor-api/index.ts",
4939
- "line": 267
5073
+ "line": 284
4940
5074
  },
4941
5075
  "name": "stacDbSecurityGroup",
4942
5076
  "type": {
@@ -4952,7 +5086,7 @@
4952
5086
  "immutable": true,
4953
5087
  "locationInModule": {
4954
5088
  "filename": "lib/ingestor-api/index.ts",
4955
- "line": 247
5089
+ "line": 264
4956
5090
  },
4957
5091
  "name": "stacUrl",
4958
5092
  "type": {
@@ -4968,7 +5102,7 @@
4968
5102
  "immutable": true,
4969
5103
  "locationInModule": {
4970
5104
  "filename": "lib/ingestor-api/index.ts",
4971
- "line": 252
5105
+ "line": 269
4972
5106
  },
4973
5107
  "name": "stage",
4974
5108
  "type": {
@@ -4984,7 +5118,7 @@
4984
5118
  "immutable": true,
4985
5119
  "locationInModule": {
4986
5120
  "filename": "lib/ingestor-api/index.ts",
4987
- "line": 272
5121
+ "line": 289
4988
5122
  },
4989
5123
  "name": "subnetSelection",
4990
5124
  "type": {
@@ -5000,13 +5134,31 @@
5000
5134
  "immutable": true,
5001
5135
  "locationInModule": {
5002
5136
  "filename": "lib/ingestor-api/index.ts",
5003
- "line": 262
5137
+ "line": 279
5004
5138
  },
5005
5139
  "name": "vpc",
5006
5140
  "type": {
5007
5141
  "fqn": "aws-cdk-lib.aws_ec2.IVpc"
5008
5142
  }
5009
5143
  },
5144
+ {
5145
+ "abstract": true,
5146
+ "docs": {
5147
+ "default": "- default in the runtime folder.",
5148
+ "stability": "experimental",
5149
+ "summary": "Custom code for the ingestor api."
5150
+ },
5151
+ "immutable": true,
5152
+ "locationInModule": {
5153
+ "filename": "lib/ingestor-api/index.ts",
5154
+ "line": 316
5155
+ },
5156
+ "name": "apiCode",
5157
+ "optional": true,
5158
+ "type": {
5159
+ "fqn": "eoapi-cdk.ApiCode"
5160
+ }
5161
+ },
5010
5162
  {
5011
5163
  "abstract": true,
5012
5164
  "docs": {
@@ -5016,7 +5168,7 @@
5016
5168
  "immutable": true,
5017
5169
  "locationInModule": {
5018
5170
  "filename": "lib/ingestor-api/index.ts",
5019
- "line": 282
5171
+ "line": 299
5020
5172
  },
5021
5173
  "name": "apiEndpointConfiguration",
5022
5174
  "optional": true,
@@ -5033,7 +5185,7 @@
5033
5185
  "immutable": true,
5034
5186
  "locationInModule": {
5035
5187
  "filename": "lib/ingestor-api/index.ts",
5036
- "line": 277
5188
+ "line": 294
5037
5189
  },
5038
5190
  "name": "apiEnv",
5039
5191
  "optional": true,
@@ -5055,7 +5207,7 @@
5055
5207
  "immutable": true,
5056
5208
  "locationInModule": {
5057
5209
  "filename": "lib/ingestor-api/index.ts",
5058
- "line": 287
5210
+ "line": 304
5059
5211
  },
5060
5212
  "name": "apiPolicy",
5061
5213
  "optional": true,
@@ -5063,6 +5215,24 @@
5063
5215
  "fqn": "aws-cdk-lib.aws_iam.PolicyDocument"
5064
5216
  }
5065
5217
  },
5218
+ {
5219
+ "abstract": true,
5220
+ "docs": {
5221
+ "default": "- default in the runtime folder.",
5222
+ "stability": "experimental",
5223
+ "summary": "Custom code for the ingestor."
5224
+ },
5225
+ "immutable": true,
5226
+ "locationInModule": {
5227
+ "filename": "lib/ingestor-api/index.ts",
5228
+ "line": 323
5229
+ },
5230
+ "name": "ingestorCode",
5231
+ "optional": true,
5232
+ "type": {
5233
+ "fqn": "eoapi-cdk.IngestorCode"
5234
+ }
5235
+ },
5066
5236
  {
5067
5237
  "abstract": true,
5068
5238
  "docs": {
@@ -5072,7 +5242,7 @@
5072
5242
  "immutable": true,
5073
5243
  "locationInModule": {
5074
5244
  "filename": "lib/ingestor-api/index.ts",
5075
- "line": 292
5245
+ "line": 309
5076
5246
  },
5077
5247
  "name": "ingestorDomainNameOptions",
5078
5248
  "optional": true,
@@ -5304,9 +5474,9 @@
5304
5474
  {
5305
5475
  "abstract": true,
5306
5476
  "docs": {
5307
- "default": "- simplified version of fastapi-pgstac",
5477
+ "default": "- simplified version of tipg.",
5308
5478
  "stability": "experimental",
5309
- "summary": "Custom code to run for fastapi-pgstac."
5479
+ "summary": "Custom code to run for the application."
5310
5480
  },
5311
5481
  "immutable": true,
5312
5482
  "locationInModule": {
@@ -5373,7 +5543,7 @@
5373
5543
  "kind": "interface",
5374
5544
  "locationInModule": {
5375
5545
  "filename": "lib/titiler-pgstac-api/index.ts",
5376
- "line": 85
5546
+ "line": 95
5377
5547
  },
5378
5548
  "name": "TitilerPgStacApiLambdaProps",
5379
5549
  "properties": [
@@ -5386,7 +5556,7 @@
5386
5556
  "immutable": true,
5387
5557
  "locationInModule": {
5388
5558
  "filename": "lib/titiler-pgstac-api/index.ts",
5389
- "line": 95
5559
+ "line": 105
5390
5560
  },
5391
5561
  "name": "db",
5392
5562
  "type": {
@@ -5402,7 +5572,7 @@
5402
5572
  "immutable": true,
5403
5573
  "locationInModule": {
5404
5574
  "filename": "lib/titiler-pgstac-api/index.ts",
5405
- "line": 105
5575
+ "line": 115
5406
5576
  },
5407
5577
  "name": "dbSecret",
5408
5578
  "type": {
@@ -5418,7 +5588,7 @@
5418
5588
  "immutable": true,
5419
5589
  "locationInModule": {
5420
5590
  "filename": "lib/titiler-pgstac-api/index.ts",
5421
- "line": 100
5591
+ "line": 110
5422
5592
  },
5423
5593
  "name": "subnetSelection",
5424
5594
  "type": {
@@ -5434,7 +5604,7 @@
5434
5604
  "immutable": true,
5435
5605
  "locationInModule": {
5436
5606
  "filename": "lib/titiler-pgstac-api/index.ts",
5437
- "line": 90
5607
+ "line": 100
5438
5608
  },
5439
5609
  "name": "vpc",
5440
5610
  "type": {
@@ -5444,13 +5614,14 @@
5444
5614
  {
5445
5615
  "abstract": true,
5446
5616
  "docs": {
5617
+ "remarks": "These will be merged with `defaultTitilerPgstacEnv`.\nThe database secret arn is automatically added to the environment variables at deployment. \n/",
5447
5618
  "stability": "experimental",
5448
5619
  "summary": "Customized environment variables to send to titiler-pgstac runtime."
5449
5620
  },
5450
5621
  "immutable": true,
5451
5622
  "locationInModule": {
5452
5623
  "filename": "lib/titiler-pgstac-api/index.ts",
5453
- "line": 110
5624
+ "line": 121
5454
5625
  },
5455
5626
  "name": "apiEnv",
5456
5627
  "optional": true,
@@ -5472,7 +5643,7 @@
5472
5643
  "immutable": true,
5473
5644
  "locationInModule": {
5474
5645
  "filename": "lib/titiler-pgstac-api/index.ts",
5475
- "line": 115
5646
+ "line": 126
5476
5647
  },
5477
5648
  "name": "buckets",
5478
5649
  "optional": true,
@@ -5488,13 +5659,32 @@
5488
5659
  {
5489
5660
  "abstract": true,
5490
5661
  "docs": {
5662
+ "default": "- defined in the construct.",
5663
+ "stability": "experimental",
5664
+ "summary": "Optional settings for the titiler-pgstac python lambda function."
5665
+ },
5666
+ "immutable": true,
5667
+ "locationInModule": {
5668
+ "filename": "lib/titiler-pgstac-api/index.ts",
5669
+ "line": 140
5670
+ },
5671
+ "name": "pythonLambdaOptions",
5672
+ "optional": true,
5673
+ "type": {
5674
+ "fqn": "eoapi-cdk.TitilerPgstacPythonLambdaOptions"
5675
+ }
5676
+ },
5677
+ {
5678
+ "abstract": true,
5679
+ "docs": {
5680
+ "default": "- undefined.",
5491
5681
  "stability": "experimental",
5492
5682
  "summary": "Custom Domain Name Options for Titiler Pgstac API,."
5493
5683
  },
5494
5684
  "immutable": true,
5495
5685
  "locationInModule": {
5496
5686
  "filename": "lib/titiler-pgstac-api/index.ts",
5497
- "line": 120
5687
+ "line": 133
5498
5688
  },
5499
5689
  "name": "titilerPgstacApiDomainName",
5500
5690
  "optional": true,
@@ -5518,7 +5708,7 @@
5518
5708
  },
5519
5709
  "locationInModule": {
5520
5710
  "filename": "lib/titiler-pgstac-api/index.ts",
5521
- "line": 20
5711
+ "line": 40
5522
5712
  },
5523
5713
  "parameters": [
5524
5714
  {
@@ -5544,7 +5734,7 @@
5544
5734
  "kind": "class",
5545
5735
  "locationInModule": {
5546
5736
  "filename": "lib/titiler-pgstac-api/index.ts",
5547
- "line": 16
5737
+ "line": 36
5548
5738
  },
5549
5739
  "name": "TitilerPgstacApiLambda",
5550
5740
  "properties": [
@@ -5555,7 +5745,7 @@
5555
5745
  "immutable": true,
5556
5746
  "locationInModule": {
5557
5747
  "filename": "lib/titiler-pgstac-api/index.ts",
5558
- "line": 17
5748
+ "line": 37
5559
5749
  },
5560
5750
  "name": "url",
5561
5751
  "type": {
@@ -5568,7 +5758,7 @@
5568
5758
  },
5569
5759
  "locationInModule": {
5570
5760
  "filename": "lib/titiler-pgstac-api/index.ts",
5571
- "line": 18
5761
+ "line": 38
5572
5762
  },
5573
5763
  "name": "titilerPgstacLambdaFunction",
5574
5764
  "type": {
@@ -5577,8 +5767,140 @@
5577
5767
  }
5578
5768
  ],
5579
5769
  "symbolId": "lib/titiler-pgstac-api/index:TitilerPgstacApiLambda"
5770
+ },
5771
+ "eoapi-cdk.TitilerPgstacPythonLambdaOptions": {
5772
+ "assembly": "eoapi-cdk",
5773
+ "datatype": true,
5774
+ "docs": {
5775
+ "stability": "experimental"
5776
+ },
5777
+ "fqn": "eoapi-cdk.TitilerPgstacPythonLambdaOptions",
5778
+ "kind": "interface",
5779
+ "locationInModule": {
5780
+ "filename": "lib/titiler-pgstac-api/index.ts",
5781
+ "line": 145
5782
+ },
5783
+ "name": "TitilerPgstacPythonLambdaOptions",
5784
+ "properties": [
5785
+ {
5786
+ "abstract": true,
5787
+ "docs": {
5788
+ "stability": "experimental",
5789
+ "summary": "The system architectures compatible with this lambda function."
5790
+ },
5791
+ "immutable": true,
5792
+ "locationInModule": {
5793
+ "filename": "lib/titiler-pgstac-api/index.ts",
5794
+ "line": 181
5795
+ },
5796
+ "name": "architecture",
5797
+ "type": {
5798
+ "fqn": "aws-cdk-lib.aws_lambda.Architecture"
5799
+ }
5800
+ },
5801
+ {
5802
+ "abstract": true,
5803
+ "docs": {
5804
+ "stability": "experimental",
5805
+ "summary": "Path to the source of the function or the location for dependencies."
5806
+ },
5807
+ "immutable": true,
5808
+ "locationInModule": {
5809
+ "filename": "lib/titiler-pgstac-api/index.ts",
5810
+ "line": 150
5811
+ },
5812
+ "name": "entry",
5813
+ "type": {
5814
+ "primitive": "string"
5815
+ }
5816
+ },
5817
+ {
5818
+ "abstract": true,
5819
+ "docs": {
5820
+ "stability": "experimental",
5821
+ "summary": "The name of the exported handler in the index file."
5822
+ },
5823
+ "immutable": true,
5824
+ "locationInModule": {
5825
+ "filename": "lib/titiler-pgstac-api/index.ts",
5826
+ "line": 165
5827
+ },
5828
+ "name": "handler",
5829
+ "type": {
5830
+ "primitive": "string"
5831
+ }
5832
+ },
5833
+ {
5834
+ "abstract": true,
5835
+ "docs": {
5836
+ "stability": "experimental",
5837
+ "summary": "The path (relative to entry) to the index file containing the exported handler."
5838
+ },
5839
+ "immutable": true,
5840
+ "locationInModule": {
5841
+ "filename": "lib/titiler-pgstac-api/index.ts",
5842
+ "line": 161
5843
+ },
5844
+ "name": "index",
5845
+ "type": {
5846
+ "primitive": "string"
5847
+ }
5848
+ },
5849
+ {
5850
+ "abstract": true,
5851
+ "docs": {
5852
+ "stability": "experimental",
5853
+ "summary": "The amount of memory, in MB, that is allocated to your Lambda function."
5854
+ },
5855
+ "immutable": true,
5856
+ "locationInModule": {
5857
+ "filename": "lib/titiler-pgstac-api/index.ts",
5858
+ "line": 176
5859
+ },
5860
+ "name": "memorySize",
5861
+ "type": {
5862
+ "primitive": "number"
5863
+ }
5864
+ },
5865
+ {
5866
+ "abstract": true,
5867
+ "docs": {
5868
+ "remarks": "Only runtimes of the Python family are\nsupported.",
5869
+ "stability": "experimental",
5870
+ "summary": "The runtime environment."
5871
+ },
5872
+ "immutable": true,
5873
+ "locationInModule": {
5874
+ "filename": "lib/titiler-pgstac-api/index.ts",
5875
+ "line": 155
5876
+ },
5877
+ "name": "runtime",
5878
+ "type": {
5879
+ "fqn": "aws-cdk-lib.aws_lambda.Runtime"
5880
+ }
5881
+ },
5882
+ {
5883
+ "abstract": true,
5884
+ "docs": {
5885
+ "remarks": "Use this to specify custom bundling options like\nthe bundling Docker image, asset hash type, custom hash, architecture, etc.",
5886
+ "stability": "experimental",
5887
+ "summary": "Bundling options to use for this function."
5888
+ },
5889
+ "immutable": true,
5890
+ "locationInModule": {
5891
+ "filename": "lib/titiler-pgstac-api/index.ts",
5892
+ "line": 171
5893
+ },
5894
+ "name": "bundling",
5895
+ "optional": true,
5896
+ "type": {
5897
+ "fqn": "aws-cdk-lib.BundlingOptions"
5898
+ }
5899
+ }
5900
+ ],
5901
+ "symbolId": "lib/titiler-pgstac-api/index:TitilerPgstacPythonLambdaOptions"
5580
5902
  }
5581
5903
  },
5582
- "version": "5.3.0",
5583
- "fingerprint": "xaJ5JxX0D7rKagYaYvOlg9jlky8AnyKw+GXghxm0MzI="
5904
+ "version": "5.4.0",
5905
+ "fingerprint": "GRDDvKc58fv7CdF/bcMJAzC9U6to2+Gn2sVG26wwEtg="
5584
5906
  }
package/CHANGELOG.md CHANGED
@@ -1,3 +1,10 @@
1
+ # [5.4.0](https://github.com/developmentseed/eoapi-cdk/compare/v5.3.0...v5.4.0) (2023-09-05)
2
+
3
+
4
+ ### Features
5
+
6
+ * custom runtimes option for titiler and ingestor ([#66](https://github.com/developmentseed/eoapi-cdk/issues/66)) ([3aaedae](https://github.com/developmentseed/eoapi-cdk/commit/3aaedaef86da558eac163348771a545b789ed8b9))
7
+
1
8
  # [5.3.0](https://github.com/developmentseed/eoapi-cdk/compare/v5.2.0...v5.3.0) (2023-09-01)
2
9
 
3
10
 
@@ -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: "5.3.0" };
161
+ BastionHost[_a] = { fqn: "eoapi-cdk.BastionHost", version: "5.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"]}
@@ -69,5 +69,5 @@ class BootstrapPgStac extends constructs_1.Construct {
69
69
  }
70
70
  exports.BootstrapPgStac = BootstrapPgStac;
71
71
  _a = JSII_RTTI_SYMBOL_1;
72
- BootstrapPgStac[_a] = { fqn: "eoapi-cdk.BootstrapPgStac", version: "5.3.0" };
72
+ BootstrapPgStac[_a] = { fqn: "eoapi-cdk.BootstrapPgStac", version: "5.4.0" };
73
73
  //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["index.ts"],"names":[],"mappings":";;;;;AAAA,6CAUqB;AACrB,2CAAuC;AAEvC,SAAS,MAAM,CACb,QAA8D;IAE9D,OAAQ,QAAqC,CAAC,GAAG,KAAK,SAAS,CAAC;AAClE,CAAC;AAED,MAAM,sBAAsB,GAAG,QAAQ,CAAC;AAExC;;GAEG;AACH,MAAa,eAAgB,SAAQ,sBAAS;IAG5C,YAAY,KAAgB,EAAE,EAAU,EAAE,KAA2B;QACnE,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAEjB,MAAM,EAAE,aAAa,GAAG,sBAAsB,EAAE,GAAG,KAAK,CAAC;QACzD,MAAM,OAAO,GAAG,IAAI,wBAAU,CAAC,QAAQ,CAAC,IAAI,EAAE,QAAQ,EAAE;YACtD,OAAO,EAAE,iBAAiB;YAC1B,OAAO,EAAE,wBAAU,CAAC,OAAO,CAAC,UAAU;YACtC,IAAI,EAAE,wBAAU,CAAC,IAAI,CAAC,eAAe,CAAC,SAAS,EAAE;gBAC/C,IAAI,EAAE,oBAAoB;gBAC1B,SAAS,EAAE,EAAE,cAAc,EAAE,aAAa,EAAE;aAC7C,CAAC;YACF,OAAO,EAAE,sBAAQ,CAAC,OAAO,CAAC,CAAC,CAAC;YAC5B,GAAG,EAAE,MAAM,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC,KAAK,CAAC,GAAG;YAC5D,YAAY,EAAE,sBAAQ,CAAC,aAAa,CAAC,QAAQ;SAC9C,CAAC,CAAC;QAEH,IAAI,CAAC,MAAM,GAAG,IAAI,gCAAkB,CAAC,MAAM,CAAC,IAAI,EAAE,QAAQ,EAAE;YAC1D,UAAU,EAAE;gBACV,KAAK,CAAC,aAAa,IAAI,QAAQ;gBAC/B,EAAE;gBACF,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;aACzB,CAAC,IAAI,CAAC,GAAG,CAAC;YACX,oBAAoB,EAAE;gBACpB,oBAAoB,EAAE,IAAI,CAAC,SAAS,CAAC;oBACnC,MAAM,EAAE,KAAK,CAAC,YAAY,IAAI,QAAQ;oBACtC,MAAM,EAAE,UAAU;oBAClB,IAAI,EAAE,IAAI;oBACV,IAAI,EAAE,KAAK,CAAC,QAAQ,CAAC,gBAAgB,CAAC,QAAQ;oBAC9C,QAAQ,EAAE,KAAK,CAAC,cAAc,IAAI,aAAa;iBAChD,CAAC;gBACF,iBAAiB,EAAE,UAAU;gBAC7B,kBAAkB,EAAE,IAAI;aACzB;YACD,WAAW,EAAE,mCACX,mBAAK,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,SACjB,EAAE;SACH,CAAC,CAAC;QAEH,qBAAqB;QACrB,uBAAuB;QACvB,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC;QAC/B,uBAAuB;QACvB,KAAK,CAAC,QAAQ,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC;QAClC,sBAAsB;QACtB,KAAK,CAAC,QAAQ,CAAC,WAAW,CAAC,SAAS,CAAC,OAAO,EAAE,qBAAO,CAAC,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,CAAC;QAEtE,iDAAiD;QACjD,IAAI,4BAAc,CAAC,IAAI,EAAE,cAAc,EAAE;YACvC,YAAY,EAAE,OAAO,CAAC,WAAW;YACjC,UAAU,EAAE;gBACV,sDAAsD;gBACtD,gEAAgE;gBAChE,cAAc,EAAE,aAAa;gBAC7B,eAAe,EAAE,KAAK,CAAC,QAAQ,CAAC,SAAS;gBACzC,mBAAmB,EAAE,IAAI,CAAC,MAAM,CAAC,SAAS;aAC3C;YACD,aAAa,EAAE,2BAAa,CAAC,MAAM;SACpC,CAAC,CAAC;IACL,CAAC;;AA7DH,0CA8DC","sourcesContent":["import {\n  aws_ec2,\n  aws_rds,\n  aws_lambda,\n  aws_logs,\n  aws_secretsmanager,\n  CustomResource,\n  Duration,\n  Stack,\n  RemovalPolicy,\n} from \"aws-cdk-lib\";\nimport { Construct } from \"constructs\";\n\nfunction hasVpc(\n  instance: aws_rds.DatabaseInstance | aws_rds.IDatabaseInstance\n): instance is aws_rds.DatabaseInstance {\n  return (instance as aws_rds.DatabaseInstance).vpc !== undefined;\n}\n\nconst DEFAULT_PGSTAC_VERSION = \"0.6.13\";\n\n/**\n * Bootstraps a database instance, installing pgSTAC onto the database.\n */\nexport class BootstrapPgStac extends Construct {\n  secret: aws_secretsmanager.ISecret;\n\n  constructor(scope: Construct, id: string, props: BootstrapPgStacProps) {\n    super(scope, id);\n\n    const { pgstacVersion = DEFAULT_PGSTAC_VERSION } = props;\n    const handler = new aws_lambda.Function(this, \"lambda\", {\n      handler: \"handler.handler\",\n      runtime: aws_lambda.Runtime.PYTHON_3_8,\n      code: aws_lambda.Code.fromDockerBuild(__dirname, {\n        file: \"runtime/Dockerfile\",\n        buildArgs: { PGSTAC_VERSION: pgstacVersion },\n      }),\n      timeout: Duration.minutes(2),\n      vpc: hasVpc(props.database) ? props.database.vpc : props.vpc,\n      logRetention: aws_logs.RetentionDays.ONE_WEEK,\n    });\n\n    this.secret = new aws_secretsmanager.Secret(this, \"secret\", {\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: props.database.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.secret.grantRead(handler);\n    // read database secret\n    props.dbSecret.grantRead(handler);\n    // connect to database\n    props.database.connections.allowFrom(handler, aws_ec2.Port.tcp(5432));\n\n    // this.connections = props.database.connections;\n    new CustomResource(this, \"bootstrapper\", {\n      serviceToken: handler.functionArn,\n      properties: {\n        // By setting pgstac_version in the properties assures\n        // that Create/Update events will be passed to the service token\n        pgstac_version: pgstacVersion,\n        conn_secret_arn: props.dbSecret.secretArn,\n        new_user_secret_arn: this.secret.secretArn,\n      },\n      removalPolicy: RemovalPolicy.RETAIN, // This retains the custom resource (which doesn't really exist), not the database\n    });\n  }\n}\n\nexport interface BootstrapPgStacProps {\n  /**\n   * VPC in which the database resides.\n   *\n   * Note - Must be explicitely set if the `database` only conforms to the\n   * `aws_rds.IDatabaseInstace` interface (ie it is a reference to a database instance\n   * rather than a database instance.)\n   *\n   * @default - `vpc` property of the `database` instance provided.\n   */\n  readonly vpc?: aws_ec2.IVpc;\n\n  /**\n   * Database onto which pgSTAC should be installed.\n   */\n  readonly database: aws_rds.DatabaseInstance | aws_rds.IDatabaseInstance;\n\n  /**\n   * Secret containing valid connection details for the database instance. Secret must\n   * conform to the format of CDK's `DatabaseInstance` (i.e. a JSON object containing a\n   * `username`, `password`, `host`, `port`, and optionally a `dbname`). If a `dbname`\n   * property is not specified within the secret, the bootstrapper will attempt to\n   * connect to a database with the name of `\"postgres\"`.\n   */\n  readonly dbSecret: aws_secretsmanager.ISecret;\n\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   * 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   * pgSTAC version to be installed.\n   *\n   * @default 0.6.8\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"]}
@@ -80,5 +80,5 @@ class PgStacDatabase extends constructs_1.Construct {
80
80
  }
81
81
  exports.PgStacDatabase = PgStacDatabase;
82
82
  _a = JSII_RTTI_SYMBOL_1;
83
- PgStacDatabase[_a] = { fqn: "eoapi-cdk.PgStacDatabase", version: "5.3.0" };
83
+ PgStacDatabase[_a] = { fqn: "eoapi-cdk.PgStacDatabase", version: "5.4.0" };
84
84
  //# 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_secretsmanager as secretsmanager,\n} from \"aws-cdk-lib\";\nimport { Construct } from \"constructs\";\nimport { BootstrapPgStac, BootstrapPgStacProps } from \"../bootstrapper\";\n\nconst instanceSizes: Record<string, number> = require(\"./instance-memory.json\");\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\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        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    const bootstrap = new BootstrapPgStac(this, \"bootstrap-pgstac-instance\", {\n      vpc: props.vpc,\n      database: this.db,\n      dbSecret: this.db.secret!,\n      pgstacDbName: props.pgstacDbName,\n      pgstacVersion: props.pgstacVersion,\n      pgstacUsername: props.pgstacUsername,\n      secretsPrefix: props.secretsPrefix,\n    });\n\n    this.pgstacSecret = bootstrap.secret;\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}`,\n      workMem: `${workMem}`,\n      maintenanceWorkMem: `${maintenanceWorkMem}`,\n      maxLocksPerTransaction: \"1024\",\n      tempBuffers: `${tempBuffers}`,\n      seqPageCost: `${seqPageCost}`,\n      randomPageCost: `${randomPageCost}`,\n    };\n  }\n}\n\nexport interface PgStacDatabaseProps extends rds.DatabaseInstanceProps {\n  readonly pgstacDbName?: BootstrapPgStacProps[\"pgstacDbName\"];\n  readonly pgstacVersion?: BootstrapPgStacProps[\"pgstacVersion\"];\n  readonly pgstacUsername?: BootstrapPgStacProps[\"pgstacUsername\"];\n  readonly secretsPrefix?: BootstrapPgStacProps[\"secretsPrefix\"];\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"]}
@@ -1,4 +1,5 @@
1
1
  import { aws_apigateway as apigateway, aws_dynamodb as dynamodb, aws_ec2 as ec2, aws_iam as iam, aws_secretsmanager as secretsmanager } from "aws-cdk-lib";
2
+ import { PythonFunctionProps } from "@aws-cdk/aws-lambda-python-alpha";
2
3
  import { Construct } from "constructs";
3
4
  export declare class StacIngestor extends Construct {
4
5
  table: dynamodb.Table;
@@ -55,4 +56,44 @@ export interface StacIngestorProps {
55
56
  * Custom Domain Name Options for Ingestor API
56
57
  */
57
58
  readonly ingestorDomainNameOptions?: apigateway.DomainNameOptions;
59
+ /**
60
+ * Custom code for the ingestor api.
61
+ *
62
+ * @default - default in the runtime folder.
63
+ */
64
+ readonly apiCode?: ApiCode;
65
+ /**
66
+ * Custom code for the ingestor.
67
+ *
68
+ * @default - default in the runtime folder.
69
+ */
70
+ readonly ingestorCode?: IngestorCode;
71
+ }
72
+ export interface ApiCode {
73
+ /**
74
+ * Path to the source of the function or the location for dependencies, for the api lambda.
75
+ */
76
+ readonly entry: PythonFunctionProps["entry"];
77
+ /**
78
+ * Path to the index file containing the exported handler, relative to `api_lambda_entry`.
79
+ */
80
+ readonly index: PythonFunctionProps["index"];
81
+ /**
82
+ * The name of the exported handler in the `api_lambda_index` file.
83
+ */
84
+ readonly handler: PythonFunctionProps["handler"];
85
+ }
86
+ export interface IngestorCode {
87
+ /**
88
+ * Path to the source of the function or the location for dependencies, for the ingestor lambda.
89
+ */
90
+ readonly entry: PythonFunctionProps["entry"];
91
+ /**
92
+ * Path to the index file containing the exported handler, relative to `ingestor_lambda_entry`.
93
+ */
94
+ readonly index: PythonFunctionProps["index"];
95
+ /**
96
+ * The name of the exported handler in the `ingestor_lambda_index` file.
97
+ */
98
+ readonly handler: PythonFunctionProps["handler"];
58
99
  }
@@ -35,6 +35,7 @@ class StacIngestor extends constructs_1.Construct {
35
35
  dbVpc: props.vpc,
36
36
  dbSecurityGroup: props.stacDbSecurityGroup,
37
37
  subnetSelection: props.subnetSelection,
38
+ apiCode: props.apiCode,
38
39
  });
39
40
  this.buildApiEndpoint({
40
41
  handler,
@@ -50,6 +51,7 @@ class StacIngestor extends constructs_1.Construct {
50
51
  dbVpc: props.vpc,
51
52
  dbSecurityGroup: props.stacDbSecurityGroup,
52
53
  subnetSelection: props.subnetSelection,
54
+ ingestorCode: props.ingestorCode,
53
55
  });
54
56
  this.registerSsmParameter({
55
57
  name: "dynamodb_table",
@@ -73,9 +75,13 @@ class StacIngestor extends constructs_1.Construct {
73
75
  return table;
74
76
  }
75
77
  buildApiLambda(props) {
76
- const handler = new aws_lambda_python_alpha_1.PythonFunction(this, "api-handler", {
78
+ const apiCode = props.apiCode || {
77
79
  entry: `${__dirname}/runtime`,
78
80
  index: "src/handler.py",
81
+ handler: "handler",
82
+ };
83
+ const handler = new aws_lambda_python_alpha_1.PythonFunction(this, "api-handler", {
84
+ ...apiCode,
79
85
  runtime: aws_cdk_lib_1.aws_lambda.Runtime.PYTHON_3_9,
80
86
  timeout: aws_cdk_lib_1.Duration.seconds(30),
81
87
  environment: { DB_SECRET_ARN: props.dbSecret.secretArn, ...props.env },
@@ -93,9 +99,13 @@ class StacIngestor extends constructs_1.Construct {
93
99
  return handler;
94
100
  }
95
101
  buildIngestor(props) {
96
- const handler = new aws_lambda_python_alpha_1.PythonFunction(this, "stac-ingestor", {
102
+ const ingestorCode = props.ingestorCode || {
97
103
  entry: `${__dirname}/runtime`,
98
104
  index: "src/ingestor.py",
105
+ handler: "handler",
106
+ };
107
+ const handler = new aws_lambda_python_alpha_1.PythonFunction(this, "stac-ingestor", {
108
+ ...ingestorCode,
99
109
  runtime: aws_cdk_lib_1.aws_lambda.Runtime.PYTHON_3_9,
100
110
  timeout: aws_cdk_lib_1.Duration.seconds(180),
101
111
  environment: { DB_SECRET_ARN: props.dbSecret.secretArn, ...props.env },
@@ -148,5 +158,5 @@ class StacIngestor extends constructs_1.Construct {
148
158
  }
149
159
  exports.StacIngestor = StacIngestor;
150
160
  _a = JSII_RTTI_SYMBOL_1;
151
- StacIngestor[_a] = { fqn: "eoapi-cdk.StacIngestor", version: "5.3.0" };
152
- //# 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_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 { PythonFunction } from \"@aws-cdk/aws-lambda-python-alpha\";\nimport { Construct } from \"constructs\";\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    });\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    });\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: ec2.IVpc;\n    dbSecurityGroup: ec2.ISecurityGroup;\n    subnetSelection: ec2.SubnetSelection\n  }): PythonFunction {\n    \n    const handler = new PythonFunction(this, \"api-handler\", {\n      entry: `${__dirname}/runtime`,\n      index: \"src/handler.py\",\n      runtime: lambda.Runtime.PYTHON_3_9,\n      timeout: Duration.seconds(30),\n      environment: { DB_SECRET_ARN: props.dbSecret.secretArn, ...props.env },\n      vpc: props.dbVpc,\n      vpcSubnets: props.subnetSelection,\n      allowPublicSubnet: true,\n      role: this.handlerRole,\n      memorySize: 2048,\n    });\n\n    // Allow handler to read DB secret\n    props.dbSecret.grantRead(handler);\n\n    // Allow handler to connect to DB\n    props.dbSecurityGroup.addIngressRule(\n      handler.connections.securityGroups[0],\n      ec2.Port.tcp(5432),\n      \"Allow connections from STAC Ingestor\"\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: ec2.IVpc;\n    dbSecurityGroup: ec2.ISecurityGroup;\n    subnetSelection: ec2.SubnetSelection;\n  }): PythonFunction {\n    const handler = new PythonFunction(this, \"stac-ingestor\", {\n      entry: `${__dirname}/runtime`,\n      index: \"src/ingestor.py\",\n      runtime: lambda.Runtime.PYTHON_3_9,\n      timeout: Duration.seconds(180),\n      environment: { DB_SECRET_ARN: props.dbSecret.secretArn, ...props.env },\n      vpc: props.dbVpc,\n      vpcSubnets: props.subnetSelection,\n      allowPublicSubnet: true,\n      memorySize: 2048,\n    });\n\n    // Allow handler to read DB secret\n    props.dbSecret.grantRead(handler);\n\n    // Allow handler to connect to DB\n    props.dbSecurityGroup.addIngressRule(\n      handler.connections.securityGroups[0],\n      ec2.Port.tcp(5432),\n      \"Allow connections from STAC Ingestor\"\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\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          domainName: props.ingestorDomainNameOptions.domainName,\n          certificate: props.ingestorDomainNameOptions.certificate,\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   * Boolean indicating whether or not pgSTAC DB is in a public subnet\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"]}
161
+ StacIngestor[_a] = { fqn: "eoapi-cdk.StacIngestor", version: "5.4.0" };
162
+ //# 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_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 { PythonFunction, PythonFunctionProps } from \"@aws-cdk/aws-lambda-python-alpha\";\nimport { Construct } from \"constructs\";\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      apiCode: props.apiCode,\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      ingestorCode: props.ingestorCode,\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: ec2.IVpc;\n    dbSecurityGroup: ec2.ISecurityGroup;\n    subnetSelection: ec2.SubnetSelection\n    apiCode?: ApiCode;\n  }): PythonFunction {\n    \n    const apiCode = props.apiCode || {\n      entry: `${__dirname}/runtime`,\n      index: \"src/handler.py\",\n      handler: \"handler\",\n    };\n\n    const handler = new PythonFunction(this, \"api-handler\", {\n      ...apiCode,\n      runtime: lambda.Runtime.PYTHON_3_9,\n      timeout: Duration.seconds(30),\n      environment: { DB_SECRET_ARN: props.dbSecret.secretArn, ...props.env },\n      vpc: props.dbVpc,\n      vpcSubnets: props.subnetSelection,\n      allowPublicSubnet: true,\n      role: this.handlerRole,\n      memorySize: 2048,\n    });\n\n    // Allow handler to read DB secret\n    props.dbSecret.grantRead(handler);\n\n    // Allow handler to connect to DB\n    props.dbSecurityGroup.addIngressRule(\n      handler.connections.securityGroups[0],\n      ec2.Port.tcp(5432),\n      \"Allow connections from STAC Ingestor\"\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: ec2.IVpc;\n    dbSecurityGroup: ec2.ISecurityGroup;\n    subnetSelection: ec2.SubnetSelection;\n    ingestorCode?: IngestorCode;\n  }): PythonFunction {\n\n\n\n    const ingestorCode = props.ingestorCode || {\n      entry: `${__dirname}/runtime`,\n      index: \"src/ingestor.py\",\n      handler: \"handler\",\n    };\n\n    const handler = new PythonFunction(this, \"stac-ingestor\", {\n      ...ingestorCode,\n      runtime: lambda.Runtime.PYTHON_3_9,\n      timeout: Duration.seconds(180),\n      environment: { DB_SECRET_ARN: props.dbSecret.secretArn, ...props.env },\n      vpc: props.dbVpc,\n      vpcSubnets: props.subnetSelection,\n      allowPublicSubnet: true,\n      memorySize: 2048,\n    });\n\n    // Allow handler to read DB secret\n    props.dbSecret.grantRead(handler);\n\n    // Allow handler to connect to DB\n    props.dbSecurityGroup.addIngressRule(\n      handler.connections.securityGroups[0],\n      ec2.Port.tcp(5432),\n      \"Allow connections from STAC Ingestor\"\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\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          domainName: props.ingestorDomainNameOptions.domainName,\n          certificate: props.ingestorDomainNameOptions.certificate,\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   * Boolean indicating whether or not pgSTAC DB is in a public subnet\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   * Custom code for the ingestor api.\n   *\n   * @default - default in the runtime folder. \n   */\n   readonly apiCode?: ApiCode;\n\n   /**\n   * Custom code for the ingestor.\n   *\n   * @default - default in the runtime folder. \n   */\n   readonly ingestorCode?: IngestorCode;\n}\n\nexport interface ApiCode {\n\n  /**\n   * Path to the source of the function or the location for dependencies, for the api lambda. \n   */\n  readonly entry: PythonFunctionProps[\"entry\"];\n\n  /**\n   * Path to the index file containing the exported handler, relative to `api_lambda_entry`. \n   */\n  readonly index: PythonFunctionProps[\"index\"];\n\n  /**\n   * The name of the exported handler in the `api_lambda_index` file. \n   */\n  readonly handler: PythonFunctionProps[\"handler\"];\n\n}\n\nexport interface IngestorCode {\n\n  /**\n   * Path to the source of the function or the location for dependencies, for the ingestor lambda. \n   */\n  readonly entry: PythonFunctionProps[\"entry\"];\n\n  /**\n   * Path to the index file containing the exported handler, relative to `ingestor_lambda_entry`. \n   */\n  readonly index: PythonFunctionProps[\"index\"];\n\n  /**\n   * The name of the exported handler in the `ingestor_lambda_index` file.\n   */\n  readonly handler: PythonFunctionProps[\"handler\"];\n\n}"]}
@@ -59,5 +59,5 @@ class PgStacApiLambda extends constructs_1.Construct {
59
59
  }
60
60
  exports.PgStacApiLambda = PgStacApiLambda;
61
61
  _a = JSII_RTTI_SYMBOL_1;
62
- PgStacApiLambda[_a] = { fqn: "eoapi-cdk.PgStacApiLambda", version: "5.3.0" };
62
+ PgStacApiLambda[_a] = { fqn: "eoapi-cdk.PgStacApiLambda", version: "5.4.0" };
63
63
  //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJpbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7OztBQUFBLDZDQU9xQjtBQUNyQiw4RUFHMEM7QUFDMUMsNEVBQXVFO0FBQ3ZFLHNHQUFxRjtBQUNyRiwyQ0FBdUM7QUFFdkMsTUFBYSxlQUFnQixTQUFRLHNCQUFTO0lBSTVDLFlBQVksS0FBZ0IsRUFBRSxFQUFVLEVBQUUsS0FBMkI7UUFDbkUsS0FBSyxDQUFDLEtBQUssRUFBRSxFQUFFLENBQUMsQ0FBQztRQUVqQixNQUFNLE9BQU8sR0FBRyxLQUFLLENBQUMsT0FBTyxJQUFJO1lBQy9CLEtBQUssRUFBRSxHQUFHLFNBQVMsVUFBVTtZQUM3QixLQUFLLEVBQUUsZ0JBQWdCO1lBQ3ZCLE9BQU8sRUFBRSxTQUFTO1NBQ25CLENBQUM7UUFFRixJQUFJLENBQUMscUJBQXFCLEdBQUcsSUFBSSx3Q0FBYyxDQUFDLElBQUksRUFBRSxVQUFVLEVBQUU7WUFDaEUsR0FBRyxPQUFPO1lBQ1Y7Ozs7Ozs7Ozs7aUJBVUs7WUFDTCxPQUFPLEVBQUUsd0JBQU0sQ0FBQyxPQUFPLENBQUMsVUFBVTtZQUNsQyxZQUFZLEVBQUUsd0JBQU0sQ0FBQyxZQUFZLENBQUMsTUFBTTtZQUN4QyxXQUFXLEVBQUU7Z0JBQ1gsaUJBQWlCLEVBQUUsS0FBSyxDQUFDLFFBQVEsQ0FBQyxTQUFTO2dCQUMzQyxnQkFBZ0IsRUFBRSxHQUFHO2dCQUNyQixnQkFBZ0IsRUFBRSxHQUFHO2dCQUNyQixHQUFHLEtBQUssQ0FBQyxNQUFNO2FBQ2hCO1lBQ0QsR0FBRyxFQUFFLEtBQUssQ0FBQyxHQUFHO1lBQ2QsVUFBVSxFQUFFLEtBQUssQ0FBQyxlQUFlO1lBQ2pDLGlCQUFpQixFQUFFLElBQUk7WUFDdkIsVUFBVSxFQUFFLElBQUk7U0FDakIsQ0FBQyxDQUFDO1FBRUgsS0FBSyxDQUFDLFFBQVEsQ0FBQyxTQUFTLENBQUMsSUFBSSxDQUFDLHFCQUFxQixDQUFDLENBQUM7UUFDckQsSUFBSSxDQUFDLHFCQUFxQixDQUFDLFdBQVcsQ0FBQyxPQUFPLENBQUMsS0FBSyxDQUFDLEVBQUUsRUFBRSxxQkFBRyxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQztRQUU3RSxNQUFNLE9BQU8sR0FBRyxJQUFJLGdDQUFPLENBQUMsSUFBSSxFQUFFLEdBQUcsbUJBQUssQ0FBQyxFQUFFLENBQUMsSUFBSSxDQUFDLENBQUMsU0FBUyxXQUFXLEVBQUU7WUFDeEUsb0JBQW9CLEVBQUUsS0FBSyxDQUFDLGlCQUFpQixDQUFDLENBQUMsQ0FBQztnQkFDOUMsVUFBVSxFQUFFLEtBQUssQ0FBQyxpQkFBaUI7YUFDcEMsQ0FBQyxDQUFDLENBQUMsU0FBUztZQUNiLGtCQUFrQixFQUFFLElBQUksMkRBQXFCLENBQUMsYUFBYSxFQUFFLElBQUksQ0FBQyxxQkFBcUIsQ0FBQztTQUN6RixDQUFDLENBQUM7UUFFSCxJQUFJLENBQUMsR0FBRyxHQUFHLE9BQU8sQ0FBQyxHQUFJLENBQUM7UUFFeEIsSUFBSSx1QkFBUyxDQUFDLElBQUksRUFBRSxpQkFBaUIsRUFBRTtZQUNyQyxVQUFVLEVBQUUsR0FBRyxtQkFBSyxDQUFDLEVBQUUsQ0FBQyxJQUFJLENBQUMsQ0FBQyxTQUFTLE1BQU07WUFDN0MsS0FBSyxFQUFFLElBQUksQ0FBQyxHQUFHO1NBQ2hCLENBQUMsQ0FBQztJQUNMLENBQUM7O0FBeERILDBDQXlEQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7XG4gIFN0YWNrLFxuICBhd3NfZWMyIGFzIGVjMixcbiAgYXdzX3JkcyBhcyByZHMsXG4gIGF3c19sYW1iZGEgYXMgbGFtYmRhLFxuICBhd3Nfc2VjcmV0c21hbmFnZXIgYXMgc2VjcmV0c21hbmFnZXIsXG4gIENmbk91dHB1dCxcbn0gZnJvbSBcImF3cy1jZGstbGliXCI7XG5pbXBvcnQge1xuICBQeXRob25GdW5jdGlvbixcbiAgUHl0aG9uRnVuY3Rpb25Qcm9wcyxcbn0gZnJvbSBcIkBhd3MtY2RrL2F3cy1sYW1iZGEtcHl0aG9uLWFscGhhXCI7XG5pbXBvcnQgeyBJRG9tYWluTmFtZSwgSHR0cEFwaSB9IGZyb20gXCJAYXdzLWNkay9hd3MtYXBpZ2F0ZXdheXYyLWFscGhhXCI7XG5pbXBvcnQgeyBIdHRwTGFtYmRhSW50ZWdyYXRpb24gfSBmcm9tIFwiQGF3cy1jZGsvYXdzLWFwaWdhdGV3YXl2Mi1pbnRlZ3JhdGlvbnMtYWxwaGFcIjtcbmltcG9ydCB7IENvbnN0cnVjdCB9IGZyb20gXCJjb25zdHJ1Y3RzXCI7XG5cbmV4cG9ydCBjbGFzcyBQZ1N0YWNBcGlMYW1iZGEgZXh0ZW5kcyBDb25zdHJ1Y3Qge1xuICByZWFkb25seSB1cmw6IHN0cmluZztcbiAgcHVibGljIHN0YWNBcGlMYW1iZGFGdW5jdGlvbjogUHl0aG9uRnVuY3Rpb247XG5cbiAgY29uc3RydWN0b3Ioc2NvcGU6IENvbnN0cnVjdCwgaWQ6IHN0cmluZywgcHJvcHM6IFBnU3RhY0FwaUxhbWJkYVByb3BzKSB7XG4gICAgc3VwZXIoc2NvcGUsIGlkKTtcblxuICAgIGNvbnN0IGFwaUNvZGUgPSBwcm9wcy5hcGlDb2RlIHx8IHtcbiAgICAgIGVudHJ5OiBgJHtfX2Rpcm5hbWV9L3J1bnRpbWVgLFxuICAgICAgaW5kZXg6IFwic3JjL2hhbmRsZXIucHlcIixcbiAgICAgIGhhbmRsZXI6IFwiaGFuZGxlclwiLFxuICAgIH07XG5cbiAgICB0aGlzLnN0YWNBcGlMYW1iZGFGdW5jdGlvbiA9IG5ldyBQeXRob25GdW5jdGlvbih0aGlzLCBcInN0YWMtYXBpXCIsIHtcbiAgICAgIC4uLmFwaUNvZGUsXG4gICAgICAvKipcbiAgICAgICAqIE5PVEU6IFVuYWJsZSB0byB1c2UgUHkzLjksIGR1ZSB0byBpc3N1ZXMgd2l0aCBoYXNoZXM6XG4gICAgICAgKlxuICAgICAgICogICAgRVJST1I6IEhhc2hlcyBhcmUgcmVxdWlyZWQgaW4gLS1yZXF1aXJlLWhhc2hlcyBtb2RlLCBidXQgdGhleSBhcmUgbWlzc2luZ1xuICAgICAgICogICAgZnJvbSBzb21lIHJlcXVpcmVtZW50cy4gSGVyZSBpcyBhIGxpc3Qgb2YgdGhvc2UgcmVxdWlyZW1lbnRzIGFsb25nIHdpdGggdGhlXG4gICAgICAgKiAgICBoYXNoZXMgdGhlaXIgZG93bmxvYWRlZCBhcmNoaXZlcyBhY3R1YWxseSBoYWQuIEFkZCBsaW5lcyBsaWtlIHRoZXNlIHRvIHlvdXJcbiAgICAgICAqICAgIHJlcXVpcmVtZW50cyBmaWxlcyB0byBwcmV2ZW50IHRhbXBlcmluZy4gKElmIHlvdSBkaWQgbm90IGVuYWJsZVxuICAgICAgICogICAgLS1yZXF1aXJlLWhhc2hlcyBtYW51YWxseSwgbm90ZSB0aGF0IGl0IHR1cm5zIG9uIGF1dG9tYXRpY2FsbHkgd2hlbiBhbnlcbiAgICAgICAqICAgIHBhY2thZ2UgaGFzIGEgaGFzaC4pXG4gICAgICAgKiAgICAgICAgYW55aW89PTMuNi4xIC0taGFzaD1zaGEyNTY6Y2IyOWI5YzcwNjIwNTA2YTlhOGY4N2EzMDk1OTE3MTM0NDY5NTMzMDJkN2Q5OTUzNDRkMGQ3YzZjMGM5YTdiZVxuICAgICAgICogKi9cbiAgICAgIHJ1bnRpbWU6IGxhbWJkYS5SdW50aW1lLlBZVEhPTl8zXzgsXG4gICAgICBhcmNoaXRlY3R1cmU6IGxhbWJkYS5BcmNoaXRlY3R1cmUuWDg2XzY0LFxuICAgICAgZW52aXJvbm1lbnQ6IHtcbiAgICAgICAgUEdTVEFDX1NFQ1JFVF9BUk46IHByb3BzLmRiU2VjcmV0LnNlY3JldEFybixcbiAgICAgICAgREJfTUlOX0NPTk5fU0laRTogXCIwXCIsXG4gICAgICAgIERCX01BWF9DT05OX1NJWkU6IFwiMVwiLFxuICAgICAgICAuLi5wcm9wcy5hcGlFbnYsXG4gICAgICB9LFxuICAgICAgdnBjOiBwcm9wcy52cGMsXG4gICAgICB2cGNTdWJuZXRzOiBwcm9wcy5zdWJuZXRTZWxlY3Rpb24sXG4gICAgICBhbGxvd1B1YmxpY1N1Ym5ldDogdHJ1ZSxcbiAgICAgIG1lbW9yeVNpemU6IDgxOTIsXG4gICAgfSk7XG5cbiAgICBwcm9wcy5kYlNlY3JldC5ncmFudFJlYWQodGhpcy5zdGFjQXBpTGFtYmRhRnVuY3Rpb24pO1xuICAgIHRoaXMuc3RhY0FwaUxhbWJkYUZ1bmN0aW9uLmNvbm5lY3Rpb25zLmFsbG93VG8ocHJvcHMuZGIsIGVjMi5Qb3J0LnRjcCg1NDMyKSk7XG5cbiAgICBjb25zdCBzdGFjQXBpID0gbmV3IEh0dHBBcGkodGhpcywgYCR7U3RhY2sub2YodGhpcykuc3RhY2tOYW1lfS1zdGFjLWFwaWAsIHtcbiAgICAgIGRlZmF1bHREb21haW5NYXBwaW5nOiBwcm9wcy5zdGFjQXBpRG9tYWluTmFtZSA/IHsgXG4gICAgICAgIGRvbWFpbk5hbWU6IHByb3BzLnN0YWNBcGlEb21haW5OYW1lXG4gICAgICB9IDogdW5kZWZpbmVkLFxuICAgICAgZGVmYXVsdEludGVncmF0aW9uOiBuZXcgSHR0cExhbWJkYUludGVncmF0aW9uKFwiaW50ZWdyYXRpb25cIiwgdGhpcy5zdGFjQXBpTGFtYmRhRnVuY3Rpb24pLFxuICAgIH0pO1xuXG4gICAgdGhpcy51cmwgPSBzdGFjQXBpLnVybCE7XG5cbiAgICBuZXcgQ2ZuT3V0cHV0KHRoaXMsIFwic3RhYy1hcGktb3V0cHV0XCIsIHtcbiAgICAgIGV4cG9ydE5hbWU6IGAke1N0YWNrLm9mKHRoaXMpLnN0YWNrTmFtZX0tdXJsYCxcbiAgICAgIHZhbHVlOiB0aGlzLnVybCxcbiAgICB9KTtcbiAgfVxufVxuXG5leHBvcnQgaW50ZXJmYWNlIFBnU3RhY0FwaUxhbWJkYVByb3BzIHtcbiAgLyoqXG4gICAqIFZQQyBpbnRvIHdoaWNoIHRoZSBsYW1iZGEgc2hvdWxkIGJlIGRlcGxveWVkLlxuICAgKi9cbiAgcmVhZG9ubHkgdnBjOiBlYzIuSVZwYztcblxuICAvKipcbiAgICogUkRTIEluc3RhbmNlIHdpdGggaW5zdGFsbGVkIHBnU1RBQy5cbiAgICovXG4gIHJlYWRvbmx5IGRiOiByZHMuSURhdGFiYXNlSW5zdGFuY2U7XG5cbiAgLyoqXG4gICAqIFN1Ym5ldCBpbnRvIHdoaWNoIHRoZSBsYW1iZGEgc2hvdWxkIGJlIGRlcGxveWVkLlxuICAgKi9cbiAgcmVhZG9ubHkgc3VibmV0U2VsZWN0aW9uOiBlYzIuU3VibmV0U2VsZWN0aW9uO1xuXG4gIC8qKlxuICAgKiBTZWNyZXQgY29udGFpbmluZyBjb25uZWN0aW9uIGluZm9ybWF0aW9uIGZvciBwZ1NUQUMgZGF0YWJhc2UuXG4gICAqL1xuICByZWFkb25seSBkYlNlY3JldDogc2VjcmV0c21hbmFnZXIuSVNlY3JldDtcblxuICAvKipcbiAgICogQ3VzdG9tIGNvZGUgdG8gcnVuIGZvciBmYXN0YXBpLXBnc3RhYy5cbiAgICpcbiAgICogQGRlZmF1bHQgLSBzaW1wbGlmaWVkIHZlcnNpb24gb2YgZmFzdGFwaS1wZ3N0YWNcbiAgICovXG4gIHJlYWRvbmx5IGFwaUNvZGU/OiBBcGlFbnRyeXBvaW50O1xuXG4gIC8qKlxuICAgKiBDdXN0b21pemVkIGVudmlyb25tZW50IHZhcmlhYmxlcyB0byBzZW5kIHRvIGZhc3RhcGktcGdzdGFjIHJ1bnRpbWUuXG4gICAqL1xuICByZWFkb25seSBhcGlFbnY/OiBSZWNvcmQ8c3RyaW5nLCBzdHJpbmc+O1xuXG4gIC8qKlxuICAgKiBDdXN0b20gRG9tYWluIE5hbWUgT3B0aW9ucyBmb3IgU1RBQyBBUEksXG4gICAqL1xuICAgcmVhZG9ubHkgc3RhY0FwaURvbWFpbk5hbWU/OiBJRG9tYWluTmFtZTtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBBcGlFbnRyeXBvaW50IHtcbiAgLyoqXG4gICAqIFBhdGggdG8gdGhlIHNvdXJjZSBvZiB0aGUgZnVuY3Rpb24gb3IgdGhlIGxvY2F0aW9uIGZvciBkZXBlbmRlbmNpZXMuXG4gICAqL1xuICByZWFkb25seSBlbnRyeTogUHl0aG9uRnVuY3Rpb25Qcm9wc1tcImVudHJ5XCJdO1xuICAvKipcbiAgICogVGhlIHBhdGggKHJlbGF0aXZlIHRvIGVudHJ5KSB0byB0aGUgaW5kZXggZmlsZSBjb250YWluaW5nIHRoZSBleHBvcnRlZCBoYW5kbGVyLlxuICAgKi9cbiAgcmVhZG9ubHkgaW5kZXg6IFB5dGhvbkZ1bmN0aW9uUHJvcHNbXCJpbmRleFwiXTtcbiAgLyoqXG4gICAqIFRoZSBuYW1lIG9mIHRoZSBleHBvcnRlZCBoYW5kbGVyIGluIHRoZSBpbmRleCBmaWxlLlxuICAgKi9cbiAgcmVhZG9ubHkgaGFuZGxlcjogUHl0aG9uRnVuY3Rpb25Qcm9wc1tcImhhbmRsZXJcIl07XG59XG4iXX0=
@@ -82,5 +82,5 @@ class StacBrowser extends constructs_1.Construct {
82
82
  }
83
83
  exports.StacBrowser = StacBrowser;
84
84
  _a = JSII_RTTI_SYMBOL_1;
85
- StacBrowser[_a] = { fqn: "eoapi-cdk.StacBrowser", version: "5.3.0" };
85
+ StacBrowser[_a] = { fqn: "eoapi-cdk.StacBrowser", version: "5.4.0" };
86
86
  //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["index.ts"],"names":[],"mappings":"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","sourcesContent":["import { Stack, aws_s3 as s3, aws_s3_deployment as s3_deployment} from \"aws-cdk-lib\";\nimport { RemovalPolicy, CfnOutput } from \"aws-cdk-lib\";\nimport { PolicyStatement, ServicePrincipal, Effect } from \"aws-cdk-lib/aws-iam\";\n\nimport { Construct } from \"constructs\";\nimport { execSync } from \"child_process\";\nimport * as fs from 'fs';\n\nconst DEFAULT_CLONE_DIRECTORY = './stac-browser';\n\nexport class StacBrowser extends Construct {\n\n    public bucket: s3.IBucket;\n    public bucketDeployment: s3_deployment.BucketDeployment;\n\n    constructor(scope: Construct, id: string, props: StacBrowserProps) {\n        super(scope, id);\n\n        const buildPath = this.buildApp(props.stacCatalogUrl, props.githubRepoTag, props.cloneDirectory || DEFAULT_CLONE_DIRECTORY);\n\n        // import a bucket from props.bucketArn if defined, otherwise create a new bucket\n        if (props.bucketArn) {\n            this.bucket = s3.Bucket.fromBucketArn(this, 'Bucket', props.bucketArn);\n        } else {\n            this.bucket = new s3.Bucket(this, 'Bucket', {\n                accessControl: s3.BucketAccessControl.PRIVATE,\n                removalPolicy: RemovalPolicy.DESTROY,\n                websiteIndexDocument: props.websiteIndexDocument\n            })\n        }\n        \n        // if props.cloudFrontDistributionArn is defined and props.bucketArn is not defined, add a bucket policy to allow read access from the cloudfront distribution\n        if (props.cloudFrontDistributionArn && !props.bucketArn) {\n            this.bucket.addToResourcePolicy(new PolicyStatement({\n                        sid: 'AllowCloudFrontServicePrincipal',\n                        effect: Effect.ALLOW, \n                        actions: ['s3:GetObject'],\n                        principals: [new ServicePrincipal('cloudfront.amazonaws.com')],\n                        resources: [this.bucket.arnForObjects('*')],\n                        conditions: {\n                            'StringEquals': {\n                                'aws:SourceArn': props.cloudFrontDistributionArn\n                            }\n                        }\n                    }));\n        }\n        \n        // add the compiled code to the bucket as a bucket deployment\n        this.bucketDeployment = new s3_deployment.BucketDeployment(this, 'BucketDeployment', {\n            destinationBucket: this.bucket,\n            sources: [s3_deployment.Source.asset(buildPath)]\n          });\n\n        new CfnOutput(this, \"bucket-name\", {\n        exportName: `${Stack.of(this).stackName}-bucket-name`,\n        value: this.bucket.bucketName,\n        });\n\n    }\n\n    private buildApp(stacCatalogUrl: string, githubRepoTag: string, cloneDirectory: string): string {\n            \n        // Define where to clone and build\n        const githubRepoUrl = 'https://github.com/radiantearth/stac-browser.git';\n\n\n        // Maybe the repo already exists in cloneDirectory. Try checking out the desired version and if it fails, delete and reclone. \n        try {\n            console.log(`Checking if a valid cloned repo exists with version ${githubRepoTag}...`)\n            execSync(`git checkout tags/${githubRepoTag}`, { cwd: cloneDirectory });\n        }\n        catch (error) {\n\n            // if directory exists, raise an error\n            if (fs.existsSync(cloneDirectory)) {\n                throw new Error(`Directory ${cloneDirectory} already exists and is not a valid clone of ${githubRepoUrl}. Please delete this directory or specify a different cloneDirectory.`);\n            }\n\n            // else, we clone and check out the version.\n\n            // Clone the repo\n            console.log(`Cloning ${githubRepoUrl} into ${cloneDirectory}...`)\n            execSync(`git clone ${githubRepoUrl} ${cloneDirectory}`);\n\n            // Check out the desired version\n            console.log(`Checking out version ${githubRepoTag}...`)\n            execSync(`git checkout tags/${githubRepoTag}`, { cwd: cloneDirectory });\n\n        }\n\n        // Install the dependencies and build the application\n        console.log(`Installing dependencies`)\n        execSync('npm install', { cwd: cloneDirectory });\n\n        // Build the app with catalogUrl\n        console.log(`Building app with catalogUrl=${stacCatalogUrl} into ${cloneDirectory}`)\n        execSync(`npm run build -- --catalogUrl=${stacCatalogUrl}`, { cwd: cloneDirectory });\n\n        return './stac-browser/dist'\n\n    }\n\n\n}\n\nexport interface StacBrowserProps {\n\n    /**\n     * Bucket ARN. If specified, the identity used to deploy the stack must have the appropriate permissions to create a deployment for this bucket. \n     * In addition, if specified, `cloudFrontDistributionArn` is ignored since the policy of an imported resource can't be modified.\n     * \n     * @default - No bucket ARN. A new bucket will be created.\n     */\n\n    readonly bucketArn?: string;\n\n    /**\n     * STAC catalog URL\n     */    \n    readonly stacCatalogUrl: string;\n\n    /**\n     * Tag of the radiant earth stac-browser repo to use to build the app.\n     */\n    readonly githubRepoTag: string;\n\n\n    /**\n     * The ARN of the cloudfront distribution that will be added to the bucket policy with read access.\n     * If `bucketArn` is specified, this parameter is ignored since the policy of an imported bucket can't be modified.\n     * \n     * @default - No cloudfront distribution ARN. The bucket policy will not be modified.\n     */    \n    readonly cloudFrontDistributionArn?: string;\n    \n    /**\n     * The name of the index document (e.g. \"index.html\") for the website. Enables static website\n     * hosting for this bucket.\n     *\n     * @default - No index document.\n     */\n    readonly websiteIndexDocument?: string;\n\n    /**\n     * Location in the filesystem where to compile the browser code. \n     * \n     * @default - DEFAULT_CLONE_DIRECTORY\n     */\n    readonly cloneDirectory?: string;\n\n}\n"]}
@@ -25,9 +25,9 @@ export interface TiPgApiLambdaProps {
25
25
  */
26
26
  readonly dbSecret: secretsmanager.ISecret;
27
27
  /**
28
- * Custom code to run for fastapi-pgstac.
28
+ * Custom code to run for the application.
29
29
  *
30
- * @default - simplified version of fastapi-pgstac
30
+ * @default - simplified version of tipg.
31
31
  */
32
32
  readonly apiCode?: TiPgApiEntrypoint;
33
33
  /**
@@ -49,5 +49,5 @@ class TiPgApiLambda extends constructs_1.Construct {
49
49
  }
50
50
  exports.TiPgApiLambda = TiPgApiLambda;
51
51
  _a = JSII_RTTI_SYMBOL_1;
52
- TiPgApiLambda[_a] = { fqn: "eoapi-cdk.TiPgApiLambda", version: "5.3.0" };
53
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJpbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7OztBQUFBLDZDQVF1QjtBQUNyQiw4RUFHMEM7QUFDMUMsNEVBQXVFO0FBQ3ZFLHNHQUFxRjtBQUNyRiwyQ0FBdUM7QUFFdkMsTUFBYSxhQUFjLFNBQVEsc0JBQVM7SUFJMUMsWUFBWSxLQUFnQixFQUFFLEVBQVUsRUFBRSxLQUF5QjtRQUNqRSxLQUFLLENBQUMsS0FBSyxFQUFFLEVBQUUsQ0FBQyxDQUFDO1FBRWpCLE1BQU0sT0FBTyxHQUFHLEtBQUssQ0FBQyxPQUFPLElBQUk7WUFDL0IsS0FBSyxFQUFFLEdBQUcsU0FBUyxVQUFVO1lBQzdCLEtBQUssRUFBRSxnQkFBZ0I7WUFDdkIsT0FBTyxFQUFFLFNBQVM7U0FDbkIsQ0FBQztRQUVGLElBQUksQ0FBQyxrQkFBa0IsR0FBRyxJQUFJLHdDQUFjLENBQUMsSUFBSSxFQUFFLFVBQVUsRUFBRTtZQUM3RCxHQUFHLE9BQU87WUFDVixPQUFPLEVBQUUsd0JBQU0sQ0FBQyxPQUFPLENBQUMsV0FBVztZQUNuQyxZQUFZLEVBQUUsd0JBQU0sQ0FBQyxZQUFZLENBQUMsTUFBTTtZQUN4QyxXQUFXLEVBQUU7Z0JBQ1gsaUJBQWlCLEVBQUUsS0FBSyxDQUFDLFFBQVEsQ0FBQyxTQUFTO2dCQUMzQyxnQkFBZ0IsRUFBRSxHQUFHO2dCQUNyQixnQkFBZ0IsRUFBRSxHQUFHO2dCQUNyQixHQUFHLEtBQUssQ0FBQyxNQUFNO2FBQ2hCO1lBQ0QsR0FBRyxFQUFFLEtBQUssQ0FBQyxHQUFHO1lBQ2QsVUFBVSxFQUFFLEtBQUssQ0FBQyxlQUFlO1lBQ2pDLGlCQUFpQixFQUFFLElBQUk7WUFDdkIsVUFBVSxFQUFFLElBQUk7WUFDaEIsT0FBTyxFQUFFLHNCQUFRLENBQUMsT0FBTyxDQUFDLEVBQUUsQ0FBQztTQUM5QixDQUFDLENBQUM7UUFFSCxLQUFLLENBQUMsUUFBUSxDQUFDLFNBQVMsQ0FBQyxJQUFJLENBQUMsa0JBQWtCLENBQUMsQ0FBQztRQUNsRCxJQUFJLENBQUMsa0JBQWtCLENBQUMsV0FBVyxDQUFDLE9BQU8sQ0FBQyxLQUFLLENBQUMsRUFBRSxFQUFFLHFCQUFHLENBQUMsSUFBSSxDQUFDLEdBQUcsQ0FBQyxJQUFJLENBQUMsRUFBRSw2QkFBNkIsQ0FBQyxDQUFDO1FBRXpHLE1BQU0sT0FBTyxHQUFHLElBQUksZ0NBQU8sQ0FBQyxJQUFJLEVBQUUsR0FBRyxtQkFBSyxDQUFDLEVBQUUsQ0FBQyxJQUFJLENBQUMsQ0FBQyxTQUFTLFdBQVcsRUFBRTtZQUN4RSxvQkFBb0IsRUFBRSxLQUFLLENBQUMsaUJBQWlCLENBQUMsQ0FBQyxDQUFDO2dCQUM5QyxVQUFVLEVBQUUsS0FBSyxDQUFDLGlCQUFpQjthQUNwQyxDQUFDLENBQUMsQ0FBQyxTQUFTO1lBQ2Isa0JBQWtCLEVBQUUsSUFBSSwyREFBcUIsQ0FBQyxhQUFhLEVBQUUsSUFBSSxDQUFDLGtCQUFrQixDQUFDO1NBQ3RGLENBQUMsQ0FBQztRQUVILElBQUksQ0FBQyxHQUFHLEdBQUcsT0FBTyxDQUFDLEdBQUksQ0FBQztRQUV4QixJQUFJLHVCQUFTLENBQUMsSUFBSSxFQUFFLGlCQUFpQixFQUFFO1lBQ3JDLFVBQVUsRUFBRSxHQUFHLG1CQUFLLENBQUMsRUFBRSxDQUFDLElBQUksQ0FBQyxDQUFDLFNBQVMsVUFBVTtZQUNqRCxLQUFLLEVBQUUsSUFBSSxDQUFDLEdBQUc7U0FDaEIsQ0FBQyxDQUFDO0lBQ0wsQ0FBQzs7QUE5Q0gsc0NBK0NDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtcbiAgICBTdGFjayxcbiAgICBhd3NfZWMyIGFzIGVjMixcbiAgICBhd3NfcmRzIGFzIHJkcyxcbiAgICBhd3NfbGFtYmRhIGFzIGxhbWJkYSxcbiAgICBhd3Nfc2VjcmV0c21hbmFnZXIgYXMgc2VjcmV0c21hbmFnZXIsXG4gICAgQ2ZuT3V0cHV0LFxuICAgIER1cmF0aW9uLFxuICB9IGZyb20gXCJhd3MtY2RrLWxpYlwiO1xuICBpbXBvcnQge1xuICAgIFB5dGhvbkZ1bmN0aW9uLFxuICAgIFB5dGhvbkZ1bmN0aW9uUHJvcHMsXG4gIH0gZnJvbSBcIkBhd3MtY2RrL2F3cy1sYW1iZGEtcHl0aG9uLWFscGhhXCI7XG4gIGltcG9ydCB7IElEb21haW5OYW1lLCBIdHRwQXBpIH0gZnJvbSBcIkBhd3MtY2RrL2F3cy1hcGlnYXRld2F5djItYWxwaGFcIjtcbiAgaW1wb3J0IHsgSHR0cExhbWJkYUludGVncmF0aW9uIH0gZnJvbSBcIkBhd3MtY2RrL2F3cy1hcGlnYXRld2F5djItaW50ZWdyYXRpb25zLWFscGhhXCI7XG4gIGltcG9ydCB7IENvbnN0cnVjdCB9IGZyb20gXCJjb25zdHJ1Y3RzXCI7XG5cbiAgZXhwb3J0IGNsYXNzIFRpUGdBcGlMYW1iZGEgZXh0ZW5kcyBDb25zdHJ1Y3Qge1xuICAgIHJlYWRvbmx5IHVybDogc3RyaW5nO1xuICAgIHB1YmxpYyB0aVBnTGFtYmRhRnVuY3Rpb246IFB5dGhvbkZ1bmN0aW9uO1xuXG4gICAgY29uc3RydWN0b3Ioc2NvcGU6IENvbnN0cnVjdCwgaWQ6IHN0cmluZywgcHJvcHM6IFRpUGdBcGlMYW1iZGFQcm9wcykge1xuICAgICAgc3VwZXIoc2NvcGUsIGlkKTtcblxuICAgICAgY29uc3QgYXBpQ29kZSA9IHByb3BzLmFwaUNvZGUgfHwge1xuICAgICAgICBlbnRyeTogYCR7X19kaXJuYW1lfS9ydW50aW1lYCxcbiAgICAgICAgaW5kZXg6IFwic3JjL2hhbmRsZXIucHlcIixcbiAgICAgICAgaGFuZGxlcjogXCJoYW5kbGVyXCIsXG4gICAgICB9O1xuXG4gICAgICB0aGlzLnRpUGdMYW1iZGFGdW5jdGlvbiA9IG5ldyBQeXRob25GdW5jdGlvbih0aGlzLCBcInRpcGctYXBpXCIsIHtcbiAgICAgICAgLi4uYXBpQ29kZSxcbiAgICAgICAgcnVudGltZTogbGFtYmRhLlJ1bnRpbWUuUFlUSE9OXzNfMTAsXG4gICAgICAgIGFyY2hpdGVjdHVyZTogbGFtYmRhLkFyY2hpdGVjdHVyZS5YODZfNjQsXG4gICAgICAgIGVudmlyb25tZW50OiB7XG4gICAgICAgICAgUEdTVEFDX1NFQ1JFVF9BUk46IHByb3BzLmRiU2VjcmV0LnNlY3JldEFybixcbiAgICAgICAgICBEQl9NSU5fQ09OTl9TSVpFOiBcIjFcIixcbiAgICAgICAgICBEQl9NQVhfQ09OTl9TSVpFOiBcIjFcIixcbiAgICAgICAgICAuLi5wcm9wcy5hcGlFbnYsXG4gICAgICAgIH0sXG4gICAgICAgIHZwYzogcHJvcHMudnBjLFxuICAgICAgICB2cGNTdWJuZXRzOiBwcm9wcy5zdWJuZXRTZWxlY3Rpb24sXG4gICAgICAgIGFsbG93UHVibGljU3VibmV0OiB0cnVlLFxuICAgICAgICBtZW1vcnlTaXplOiAxMDI0LFxuICAgICAgICB0aW1lb3V0OiBEdXJhdGlvbi5zZWNvbmRzKDMwKSxcbiAgICAgIH0pO1xuXG4gICAgICBwcm9wcy5kYlNlY3JldC5ncmFudFJlYWQodGhpcy50aVBnTGFtYmRhRnVuY3Rpb24pO1xuICAgICAgdGhpcy50aVBnTGFtYmRhRnVuY3Rpb24uY29ubmVjdGlvbnMuYWxsb3dUbyhwcm9wcy5kYiwgZWMyLlBvcnQudGNwKDU0MzIpLCBcImFsbG93IGNvbm5lY3Rpb25zIGZyb20gdGlwZ1wiKTtcblxuICAgICAgY29uc3QgdGlwZ0FwaSA9IG5ldyBIdHRwQXBpKHRoaXMsIGAke1N0YWNrLm9mKHRoaXMpLnN0YWNrTmFtZX0tdGlwZy1hcGlgLCB7XG4gICAgICAgIGRlZmF1bHREb21haW5NYXBwaW5nOiBwcm9wcy50aXBnQXBpRG9tYWluTmFtZSA/IHsgXG4gICAgICAgICAgZG9tYWluTmFtZTogcHJvcHMudGlwZ0FwaURvbWFpbk5hbWVcbiAgICAgICAgfSA6IHVuZGVmaW5lZCxcbiAgICAgICAgZGVmYXVsdEludGVncmF0aW9uOiBuZXcgSHR0cExhbWJkYUludGVncmF0aW9uKFwiaW50ZWdyYXRpb25cIiwgdGhpcy50aVBnTGFtYmRhRnVuY3Rpb24pLFxuICAgICAgfSk7XG5cbiAgICAgIHRoaXMudXJsID0gdGlwZ0FwaS51cmwhO1xuXG4gICAgICBuZXcgQ2ZuT3V0cHV0KHRoaXMsIFwidGlwZy1hcGktb3V0cHV0XCIsIHtcbiAgICAgICAgZXhwb3J0TmFtZTogYCR7U3RhY2sub2YodGhpcykuc3RhY2tOYW1lfS10aXAtdXJsYCxcbiAgICAgICAgdmFsdWU6IHRoaXMudXJsLFxuICAgICAgfSk7XG4gICAgfVxuICB9XG5cbiAgZXhwb3J0IGludGVyZmFjZSBUaVBnQXBpTGFtYmRhUHJvcHMge1xuXG4gICAgLyoqXG4gICAgICogVlBDIGludG8gd2hpY2ggdGhlIGxhbWJkYSBzaG91bGQgYmUgZGVwbG95ZWQuXG4gICAgICovXG4gICAgcmVhZG9ubHkgdnBjOiBlYzIuSVZwYztcblxuICAgIC8qKlxuICAgICAqIFJEUyBJbnN0YW5jZSB3aXRoIGluc3RhbGxlZCBwZ1NUQUMuXG4gICAgICovXG4gICAgcmVhZG9ubHkgZGI6IHJkcy5JRGF0YWJhc2VJbnN0YW5jZTtcblxuICAgIC8qKlxuICAgICAqIFN1Ym5ldCBpbnRvIHdoaWNoIHRoZSBsYW1iZGEgc2hvdWxkIGJlIGRlcGxveWVkLlxuICAgICAqL1xuICAgIHJlYWRvbmx5IHN1Ym5ldFNlbGVjdGlvbjogZWMyLlN1Ym5ldFNlbGVjdGlvbjtcblxuICAgIC8qKlxuICAgICAqIFNlY3JldCBjb250YWluaW5nIGNvbm5lY3Rpb24gaW5mb3JtYXRpb24gZm9yIHBnU1RBQyBkYXRhYmFzZS5cbiAgICAgKi9cbiAgICByZWFkb25seSBkYlNlY3JldDogc2VjcmV0c21hbmFnZXIuSVNlY3JldDtcblxuICAgIC8qKlxuICAgICAqIEN1c3RvbSBjb2RlIHRvIHJ1biBmb3IgZmFzdGFwaS1wZ3N0YWMuXG4gICAgICpcbiAgICAgKiBAZGVmYXVsdCAtIHNpbXBsaWZpZWQgdmVyc2lvbiBvZiBmYXN0YXBpLXBnc3RhY1xuICAgICAqL1xuICAgIHJlYWRvbmx5IGFwaUNvZGU/OiBUaVBnQXBpRW50cnlwb2ludDtcblxuICAgIC8qKlxuICAgICAqIEN1c3RvbWl6ZWQgZW52aXJvbm1lbnQgdmFyaWFibGVzIHRvIHNlbmQgdG8gdGl0aWxlci1wZ3N0YWMgcnVudGltZS5cbiAgICAgKi9cbiAgICByZWFkb25seSBhcGlFbnY/OiBSZWNvcmQ8c3RyaW5nLCBzdHJpbmc+O1xuICBcbiAgICAvKipcbiAgICAgKiBDdXN0b20gRG9tYWluIE5hbWUgZm9yIHRpcGcgQVBJLiBJZiBkZWZpbmVkLCB3aWxsIGNyZWF0ZSB0aGUgXG4gICAgICogZG9tYWluIG5hbWUgYW5kIGludGVncmF0ZSBpdCB3aXRoIHRoZSB0aXBnIEFQSS4gXG4gICAgICogXG4gICAgICogQGRlZmF1bHQgLSB1bmRlZmluZWRcbiAgICAgKi9cbiAgICByZWFkb25seSB0aXBnQXBpRG9tYWluTmFtZT86IElEb21haW5OYW1lO1xuICB9XG5cbiAgZXhwb3J0IGludGVyZmFjZSBUaVBnQXBpRW50cnlwb2ludCB7XG4gICAgLyoqXG4gICAgICogUGF0aCB0byB0aGUgc291cmNlIG9mIHRoZSBmdW5jdGlvbiBvciB0aGUgbG9jYXRpb24gZm9yIGRlcGVuZGVuY2llcy5cbiAgICAgKi9cbiAgICByZWFkb25seSBlbnRyeTogUHl0aG9uRnVuY3Rpb25Qcm9wc1tcImVudHJ5XCJdO1xuICAgIC8qKlxuICAgICAqIFRoZSBwYXRoIChyZWxhdGl2ZSB0byBlbnRyeSkgdG8gdGhlIGluZGV4IGZpbGUgY29udGFpbmluZyB0aGUgZXhwb3J0ZWQgaGFuZGxlci5cbiAgICAgKi9cbiAgICByZWFkb25seSBpbmRleDogUHl0aG9uRnVuY3Rpb25Qcm9wc1tcImluZGV4XCJdO1xuICAgIC8qKlxuICAgICAqIFRoZSBuYW1lIG9mIHRoZSBleHBvcnRlZCBoYW5kbGVyIGluIHRoZSBpbmRleCBmaWxlLlxuICAgICAqL1xuICAgIHJlYWRvbmx5IGhhbmRsZXI6IFB5dGhvbkZ1bmN0aW9uUHJvcHNbXCJoYW5kbGVyXCJdO1xuICB9XG4iXX0=
52
+ TiPgApiLambda[_a] = { fqn: "eoapi-cdk.TiPgApiLambda", version: "5.4.0" };
53
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJpbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7OztBQUFBLDZDQVF1QjtBQUNyQiw4RUFHMEM7QUFDMUMsNEVBQXVFO0FBQ3ZFLHNHQUFxRjtBQUNyRiwyQ0FBdUM7QUFFdkMsTUFBYSxhQUFjLFNBQVEsc0JBQVM7SUFJMUMsWUFBWSxLQUFnQixFQUFFLEVBQVUsRUFBRSxLQUF5QjtRQUNqRSxLQUFLLENBQUMsS0FBSyxFQUFFLEVBQUUsQ0FBQyxDQUFDO1FBRWpCLE1BQU0sT0FBTyxHQUFHLEtBQUssQ0FBQyxPQUFPLElBQUk7WUFDL0IsS0FBSyxFQUFFLEdBQUcsU0FBUyxVQUFVO1lBQzdCLEtBQUssRUFBRSxnQkFBZ0I7WUFDdkIsT0FBTyxFQUFFLFNBQVM7U0FDbkIsQ0FBQztRQUVGLElBQUksQ0FBQyxrQkFBa0IsR0FBRyxJQUFJLHdDQUFjLENBQUMsSUFBSSxFQUFFLFVBQVUsRUFBRTtZQUM3RCxHQUFHLE9BQU87WUFDVixPQUFPLEVBQUUsd0JBQU0sQ0FBQyxPQUFPLENBQUMsV0FBVztZQUNuQyxZQUFZLEVBQUUsd0JBQU0sQ0FBQyxZQUFZLENBQUMsTUFBTTtZQUN4QyxXQUFXLEVBQUU7Z0JBQ1gsaUJBQWlCLEVBQUUsS0FBSyxDQUFDLFFBQVEsQ0FBQyxTQUFTO2dCQUMzQyxnQkFBZ0IsRUFBRSxHQUFHO2dCQUNyQixnQkFBZ0IsRUFBRSxHQUFHO2dCQUNyQixHQUFHLEtBQUssQ0FBQyxNQUFNO2FBQ2hCO1lBQ0QsR0FBRyxFQUFFLEtBQUssQ0FBQyxHQUFHO1lBQ2QsVUFBVSxFQUFFLEtBQUssQ0FBQyxlQUFlO1lBQ2pDLGlCQUFpQixFQUFFLElBQUk7WUFDdkIsVUFBVSxFQUFFLElBQUk7WUFDaEIsT0FBTyxFQUFFLHNCQUFRLENBQUMsT0FBTyxDQUFDLEVBQUUsQ0FBQztTQUM5QixDQUFDLENBQUM7UUFFSCxLQUFLLENBQUMsUUFBUSxDQUFDLFNBQVMsQ0FBQyxJQUFJLENBQUMsa0JBQWtCLENBQUMsQ0FBQztRQUNsRCxJQUFJLENBQUMsa0JBQWtCLENBQUMsV0FBVyxDQUFDLE9BQU8sQ0FBQyxLQUFLLENBQUMsRUFBRSxFQUFFLHFCQUFHLENBQUMsSUFBSSxDQUFDLEdBQUcsQ0FBQyxJQUFJLENBQUMsRUFBRSw2QkFBNkIsQ0FBQyxDQUFDO1FBRXpHLE1BQU0sT0FBTyxHQUFHLElBQUksZ0NBQU8sQ0FBQyxJQUFJLEVBQUUsR0FBRyxtQkFBSyxDQUFDLEVBQUUsQ0FBQyxJQUFJLENBQUMsQ0FBQyxTQUFTLFdBQVcsRUFBRTtZQUN4RSxvQkFBb0IsRUFBRSxLQUFLLENBQUMsaUJBQWlCLENBQUMsQ0FBQyxDQUFDO2dCQUM5QyxVQUFVLEVBQUUsS0FBSyxDQUFDLGlCQUFpQjthQUNwQyxDQUFDLENBQUMsQ0FBQyxTQUFTO1lBQ2Isa0JBQWtCLEVBQUUsSUFBSSwyREFBcUIsQ0FBQyxhQUFhLEVBQUUsSUFBSSxDQUFDLGtCQUFrQixDQUFDO1NBQ3RGLENBQUMsQ0FBQztRQUVILElBQUksQ0FBQyxHQUFHLEdBQUcsT0FBTyxDQUFDLEdBQUksQ0FBQztRQUV4QixJQUFJLHVCQUFTLENBQUMsSUFBSSxFQUFFLGlCQUFpQixFQUFFO1lBQ3JDLFVBQVUsRUFBRSxHQUFHLG1CQUFLLENBQUMsRUFBRSxDQUFDLElBQUksQ0FBQyxDQUFDLFNBQVMsVUFBVTtZQUNqRCxLQUFLLEVBQUUsSUFBSSxDQUFDLEdBQUc7U0FDaEIsQ0FBQyxDQUFDO0lBQ0wsQ0FBQzs7QUE5Q0gsc0NBK0NDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtcbiAgICBTdGFjayxcbiAgICBhd3NfZWMyIGFzIGVjMixcbiAgICBhd3NfcmRzIGFzIHJkcyxcbiAgICBhd3NfbGFtYmRhIGFzIGxhbWJkYSxcbiAgICBhd3Nfc2VjcmV0c21hbmFnZXIgYXMgc2VjcmV0c21hbmFnZXIsXG4gICAgQ2ZuT3V0cHV0LFxuICAgIER1cmF0aW9uLFxuICB9IGZyb20gXCJhd3MtY2RrLWxpYlwiO1xuICBpbXBvcnQge1xuICAgIFB5dGhvbkZ1bmN0aW9uLFxuICAgIFB5dGhvbkZ1bmN0aW9uUHJvcHMsXG4gIH0gZnJvbSBcIkBhd3MtY2RrL2F3cy1sYW1iZGEtcHl0aG9uLWFscGhhXCI7XG4gIGltcG9ydCB7IElEb21haW5OYW1lLCBIdHRwQXBpIH0gZnJvbSBcIkBhd3MtY2RrL2F3cy1hcGlnYXRld2F5djItYWxwaGFcIjtcbiAgaW1wb3J0IHsgSHR0cExhbWJkYUludGVncmF0aW9uIH0gZnJvbSBcIkBhd3MtY2RrL2F3cy1hcGlnYXRld2F5djItaW50ZWdyYXRpb25zLWFscGhhXCI7XG4gIGltcG9ydCB7IENvbnN0cnVjdCB9IGZyb20gXCJjb25zdHJ1Y3RzXCI7XG5cbiAgZXhwb3J0IGNsYXNzIFRpUGdBcGlMYW1iZGEgZXh0ZW5kcyBDb25zdHJ1Y3Qge1xuICAgIHJlYWRvbmx5IHVybDogc3RyaW5nO1xuICAgIHB1YmxpYyB0aVBnTGFtYmRhRnVuY3Rpb246IFB5dGhvbkZ1bmN0aW9uO1xuXG4gICAgY29uc3RydWN0b3Ioc2NvcGU6IENvbnN0cnVjdCwgaWQ6IHN0cmluZywgcHJvcHM6IFRpUGdBcGlMYW1iZGFQcm9wcykge1xuICAgICAgc3VwZXIoc2NvcGUsIGlkKTtcblxuICAgICAgY29uc3QgYXBpQ29kZSA9IHByb3BzLmFwaUNvZGUgfHwge1xuICAgICAgICBlbnRyeTogYCR7X19kaXJuYW1lfS9ydW50aW1lYCxcbiAgICAgICAgaW5kZXg6IFwic3JjL2hhbmRsZXIucHlcIixcbiAgICAgICAgaGFuZGxlcjogXCJoYW5kbGVyXCIsXG4gICAgICB9O1xuXG4gICAgICB0aGlzLnRpUGdMYW1iZGFGdW5jdGlvbiA9IG5ldyBQeXRob25GdW5jdGlvbih0aGlzLCBcInRpcGctYXBpXCIsIHtcbiAgICAgICAgLi4uYXBpQ29kZSxcbiAgICAgICAgcnVudGltZTogbGFtYmRhLlJ1bnRpbWUuUFlUSE9OXzNfMTAsXG4gICAgICAgIGFyY2hpdGVjdHVyZTogbGFtYmRhLkFyY2hpdGVjdHVyZS5YODZfNjQsXG4gICAgICAgIGVudmlyb25tZW50OiB7XG4gICAgICAgICAgUEdTVEFDX1NFQ1JFVF9BUk46IHByb3BzLmRiU2VjcmV0LnNlY3JldEFybixcbiAgICAgICAgICBEQl9NSU5fQ09OTl9TSVpFOiBcIjFcIixcbiAgICAgICAgICBEQl9NQVhfQ09OTl9TSVpFOiBcIjFcIixcbiAgICAgICAgICAuLi5wcm9wcy5hcGlFbnYsXG4gICAgICAgIH0sXG4gICAgICAgIHZwYzogcHJvcHMudnBjLFxuICAgICAgICB2cGNTdWJuZXRzOiBwcm9wcy5zdWJuZXRTZWxlY3Rpb24sXG4gICAgICAgIGFsbG93UHVibGljU3VibmV0OiB0cnVlLFxuICAgICAgICBtZW1vcnlTaXplOiAxMDI0LFxuICAgICAgICB0aW1lb3V0OiBEdXJhdGlvbi5zZWNvbmRzKDMwKSxcbiAgICAgIH0pO1xuXG4gICAgICBwcm9wcy5kYlNlY3JldC5ncmFudFJlYWQodGhpcy50aVBnTGFtYmRhRnVuY3Rpb24pO1xuICAgICAgdGhpcy50aVBnTGFtYmRhRnVuY3Rpb24uY29ubmVjdGlvbnMuYWxsb3dUbyhwcm9wcy5kYiwgZWMyLlBvcnQudGNwKDU0MzIpLCBcImFsbG93IGNvbm5lY3Rpb25zIGZyb20gdGlwZ1wiKTtcblxuICAgICAgY29uc3QgdGlwZ0FwaSA9IG5ldyBIdHRwQXBpKHRoaXMsIGAke1N0YWNrLm9mKHRoaXMpLnN0YWNrTmFtZX0tdGlwZy1hcGlgLCB7XG4gICAgICAgIGRlZmF1bHREb21haW5NYXBwaW5nOiBwcm9wcy50aXBnQXBpRG9tYWluTmFtZSA/IHsgXG4gICAgICAgICAgZG9tYWluTmFtZTogcHJvcHMudGlwZ0FwaURvbWFpbk5hbWVcbiAgICAgICAgfSA6IHVuZGVmaW5lZCxcbiAgICAgICAgZGVmYXVsdEludGVncmF0aW9uOiBuZXcgSHR0cExhbWJkYUludGVncmF0aW9uKFwiaW50ZWdyYXRpb25cIiwgdGhpcy50aVBnTGFtYmRhRnVuY3Rpb24pLFxuICAgICAgfSk7XG5cbiAgICAgIHRoaXMudXJsID0gdGlwZ0FwaS51cmwhO1xuXG4gICAgICBuZXcgQ2ZuT3V0cHV0KHRoaXMsIFwidGlwZy1hcGktb3V0cHV0XCIsIHtcbiAgICAgICAgZXhwb3J0TmFtZTogYCR7U3RhY2sub2YodGhpcykuc3RhY2tOYW1lfS10aXAtdXJsYCxcbiAgICAgICAgdmFsdWU6IHRoaXMudXJsLFxuICAgICAgfSk7XG4gICAgfVxuICB9XG5cbiAgZXhwb3J0IGludGVyZmFjZSBUaVBnQXBpTGFtYmRhUHJvcHMge1xuXG4gICAgLyoqXG4gICAgICogVlBDIGludG8gd2hpY2ggdGhlIGxhbWJkYSBzaG91bGQgYmUgZGVwbG95ZWQuXG4gICAgICovXG4gICAgcmVhZG9ubHkgdnBjOiBlYzIuSVZwYztcblxuICAgIC8qKlxuICAgICAqIFJEUyBJbnN0YW5jZSB3aXRoIGluc3RhbGxlZCBwZ1NUQUMuXG4gICAgICovXG4gICAgcmVhZG9ubHkgZGI6IHJkcy5JRGF0YWJhc2VJbnN0YW5jZTtcblxuICAgIC8qKlxuICAgICAqIFN1Ym5ldCBpbnRvIHdoaWNoIHRoZSBsYW1iZGEgc2hvdWxkIGJlIGRlcGxveWVkLlxuICAgICAqL1xuICAgIHJlYWRvbmx5IHN1Ym5ldFNlbGVjdGlvbjogZWMyLlN1Ym5ldFNlbGVjdGlvbjtcblxuICAgIC8qKlxuICAgICAqIFNlY3JldCBjb250YWluaW5nIGNvbm5lY3Rpb24gaW5mb3JtYXRpb24gZm9yIHBnU1RBQyBkYXRhYmFzZS5cbiAgICAgKi9cbiAgICByZWFkb25seSBkYlNlY3JldDogc2VjcmV0c21hbmFnZXIuSVNlY3JldDtcblxuICAgIC8qKlxuICAgICAqIEN1c3RvbSBjb2RlIHRvIHJ1biBmb3IgdGhlIGFwcGxpY2F0aW9uLlxuICAgICAqXG4gICAgICogQGRlZmF1bHQgLSBzaW1wbGlmaWVkIHZlcnNpb24gb2YgdGlwZy5cbiAgICAgKi9cbiAgICByZWFkb25seSBhcGlDb2RlPzogVGlQZ0FwaUVudHJ5cG9pbnQ7XG5cbiAgICAvKipcbiAgICAgKiBDdXN0b21pemVkIGVudmlyb25tZW50IHZhcmlhYmxlcyB0byBzZW5kIHRvIHRpdGlsZXItcGdzdGFjIHJ1bnRpbWUuXG4gICAgICovXG4gICAgcmVhZG9ubHkgYXBpRW52PzogUmVjb3JkPHN0cmluZywgc3RyaW5nPjtcbiAgXG4gICAgLyoqXG4gICAgICogQ3VzdG9tIERvbWFpbiBOYW1lIGZvciB0aXBnIEFQSS4gSWYgZGVmaW5lZCwgd2lsbCBjcmVhdGUgdGhlIFxuICAgICAqIGRvbWFpbiBuYW1lIGFuZCBpbnRlZ3JhdGUgaXQgd2l0aCB0aGUgdGlwZyBBUEkuIFxuICAgICAqIFxuICAgICAqIEBkZWZhdWx0IC0gdW5kZWZpbmVkXG4gICAgICovXG4gICAgcmVhZG9ubHkgdGlwZ0FwaURvbWFpbk5hbWU/OiBJRG9tYWluTmFtZTtcbiAgfVxuXG4gIGV4cG9ydCBpbnRlcmZhY2UgVGlQZ0FwaUVudHJ5cG9pbnQge1xuICAgIC8qKlxuICAgICAqIFBhdGggdG8gdGhlIHNvdXJjZSBvZiB0aGUgZnVuY3Rpb24gb3IgdGhlIGxvY2F0aW9uIGZvciBkZXBlbmRlbmNpZXMuXG4gICAgICovXG4gICAgcmVhZG9ubHkgZW50cnk6IFB5dGhvbkZ1bmN0aW9uUHJvcHNbXCJlbnRyeVwiXTtcbiAgICAvKipcbiAgICAgKiBUaGUgcGF0aCAocmVsYXRpdmUgdG8gZW50cnkpIHRvIHRoZSBpbmRleCBmaWxlIGNvbnRhaW5pbmcgdGhlIGV4cG9ydGVkIGhhbmRsZXIuXG4gICAgICovXG4gICAgcmVhZG9ubHkgaW5kZXg6IFB5dGhvbkZ1bmN0aW9uUHJvcHNbXCJpbmRleFwiXTtcbiAgICAvKipcbiAgICAgKiBUaGUgbmFtZSBvZiB0aGUgZXhwb3J0ZWQgaGFuZGxlciBpbiB0aGUgaW5kZXggZmlsZS5cbiAgICAgKi9cbiAgICByZWFkb25seSBoYW5kbGVyOiBQeXRob25GdW5jdGlvblByb3BzW1wiaGFuZGxlclwiXTtcbiAgfVxuIl19
@@ -1,4 +1,5 @@
1
- import { aws_ec2 as ec2, aws_rds as rds, aws_lambda as lambda, aws_secretsmanager as secretsmanager } from "aws-cdk-lib";
1
+ import { aws_ec2 as ec2, aws_rds as rds, aws_lambda as lambda, aws_secretsmanager as secretsmanager, BundlingOptions } from "aws-cdk-lib";
2
+ import { Runtime } from 'aws-cdk-lib/aws-lambda';
2
3
  import { IDomainName } from "@aws-cdk/aws-apigatewayv2-alpha";
3
4
  import { Construct } from "constructs";
4
5
  export declare class TitilerPgstacApiLambda extends Construct {
@@ -24,8 +25,9 @@ export interface TitilerPgStacApiLambdaProps {
24
25
  */
25
26
  readonly dbSecret: secretsmanager.ISecret;
26
27
  /**
27
- * Customized environment variables to send to titiler-pgstac runtime.
28
- */
28
+ * Customized environment variables to send to titiler-pgstac runtime. These will be merged with `defaultTitilerPgstacEnv`.
29
+ * The database secret arn is automatically added to the environment variables at deployment.
30
+ /*/
29
31
  readonly apiEnv?: Record<string, string>;
30
32
  /**
31
33
  * list of buckets the lambda will be granted access to.
@@ -33,6 +35,47 @@ export interface TitilerPgStacApiLambdaProps {
33
35
  readonly buckets?: string[];
34
36
  /**
35
37
  * Custom Domain Name Options for Titiler Pgstac API,
38
+ *
39
+ * @default - undefined.
36
40
  */
37
41
  readonly titilerPgstacApiDomainName?: IDomainName;
42
+ /**
43
+ * Optional settings for the titiler-pgstac python lambda function.
44
+ *
45
+ * @default - defined in the construct.
46
+ */
47
+ readonly pythonLambdaOptions?: TitilerPgstacPythonLambdaOptions;
48
+ }
49
+ export interface TitilerPgstacPythonLambdaOptions {
50
+ /**
51
+ * Path to the source of the function or the location for dependencies.
52
+ */
53
+ readonly entry: string;
54
+ /**
55
+ * The runtime environment. Only runtimes of the Python family are
56
+ * supported.
57
+ */
58
+ readonly runtime: Runtime;
59
+ /**
60
+ * The path (relative to entry) to the index file containing the exported handler.
61
+ *
62
+ */
63
+ readonly index: string;
64
+ /**
65
+ * The name of the exported handler in the index file.
66
+ */
67
+ readonly handler: string;
68
+ /**
69
+ * Bundling options to use for this function. Use this to specify custom bundling options like
70
+ * the bundling Docker image, asset hash type, custom hash, architecture, etc.
71
+ */
72
+ readonly bundling?: BundlingOptions;
73
+ /**
74
+ * The amount of memory, in MB, that is allocated to your Lambda function.
75
+ */
76
+ readonly memorySize: number;
77
+ /**
78
+ * The system architectures compatible with this lambda function.
79
+ */
80
+ readonly architecture: lambda.Architecture;
38
81
  }
@@ -4,41 +4,46 @@ Object.defineProperty(exports, "__esModule", { value: true });
4
4
  exports.TitilerPgstacApiLambda = void 0;
5
5
  const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
6
6
  const aws_cdk_lib_1 = require("aws-cdk-lib");
7
+ const aws_lambda_python_alpha_1 = require("@aws-cdk/aws-lambda-python-alpha");
7
8
  const aws_apigatewayv2_alpha_1 = require("@aws-cdk/aws-apigatewayv2-alpha");
8
9
  const aws_apigatewayv2_integrations_alpha_1 = require("@aws-cdk/aws-apigatewayv2-integrations-alpha");
9
10
  const constructs_1 = require("constructs");
11
+ // default settings that can be overridden by the user-provided environment.
12
+ let defaultTitilerPgstacEnv = {
13
+ "CPL_VSIL_CURL_ALLOWED_EXTENSIONS": ".tif,.TIF,.tiff",
14
+ "GDAL_CACHEMAX": "200",
15
+ "GDAL_DISABLE_READDIR_ON_OPEN": "EMPTY_DIR",
16
+ "GDAL_INGESTED_BYTES_AT_OPEN": "32768",
17
+ "GDAL_HTTP_MERGE_CONSECUTIVE_RANGES": "YES",
18
+ "GDAL_HTTP_MULTIPLEX": "YES",
19
+ "GDAL_HTTP_VERSION": "2",
20
+ "PYTHONWARNINGS": "ignore",
21
+ "VSI_CACHE": "TRUE",
22
+ "VSI_CACHE_SIZE": "5000000",
23
+ "DB_MIN_CONN_SIZE": "1",
24
+ "DB_MAX_CONN_SIZE": "1"
25
+ };
10
26
  class TitilerPgstacApiLambda extends constructs_1.Construct {
11
27
  constructor(scope, id, props) {
12
28
  super(scope, id);
13
- const titilerPgstacEnv = {
14
- "CPL_VSIL_CURL_ALLOWED_EXTENSIONS": ".tif,.TIF,.tiff",
15
- "GDAL_CACHEMAX": "200",
16
- "GDAL_DISABLE_READDIR_ON_OPEN": "EMPTY_DIR",
17
- "GDAL_INGESTED_BYTES_AT_OPEN": "32768",
18
- "GDAL_HTTP_MERGE_CONSECUTIVE_RANGES": "YES",
19
- "GDAL_HTTP_MULTIPLEX": "YES",
20
- "GDAL_HTTP_VERSION": "2",
21
- "PYTHONWARNINGS": "ignore",
22
- "VSI_CACHE": "TRUE",
23
- "VSI_CACHE_SIZE": "5000000",
24
- "DB_MIN_CONN_SIZE": "1",
25
- "DB_MAX_CONN_SIZE": "1",
26
- "PGSTAC_SECRET_ARN": props.dbSecret.secretArn,
27
- };
28
- this.titilerPgstacLambdaFunction = new aws_cdk_lib_1.aws_lambda.Function(this, "lambda", {
29
- handler: "handler.handler",
29
+ // if user provided environment variables, merge them with the defaults.
30
+ const apiEnv = props.apiEnv ? { ...defaultTitilerPgstacEnv, ...props.apiEnv, "PGSTAC_SECRET_ARN": props.dbSecret.secretArn } : defaultTitilerPgstacEnv;
31
+ const pythonLambdaOptions = props.pythonLambdaOptions ?? {
30
32
  runtime: aws_cdk_lib_1.aws_lambda.Runtime.PYTHON_3_10,
31
- code: aws_cdk_lib_1.aws_lambda.Code.fromDockerBuild(__dirname, {
32
- file: "runtime/Dockerfile",
33
- buildArgs: { PYTHON_VERSION: '3.10' },
34
- }),
35
- timeout: aws_cdk_lib_1.Duration.seconds(30),
33
+ entry: `${__dirname}/runtime`,
34
+ index: "src/handler.py",
35
+ handler: "handler",
36
+ memorySize: 3008,
37
+ architecture: aws_cdk_lib_1.aws_lambda.Architecture.X86_64
38
+ };
39
+ this.titilerPgstacLambdaFunction = new aws_lambda_python_alpha_1.PythonFunction(this, "titiler-pgstac-api", {
40
+ ...pythonLambdaOptions,
41
+ environment: apiEnv,
36
42
  vpc: props.vpc,
37
43
  vpcSubnets: props.subnetSelection,
38
44
  allowPublicSubnet: true,
39
- memorySize: 3008,
40
45
  logRetention: aws_cdk_lib_1.aws_logs.RetentionDays.ONE_WEEK,
41
- environment: titilerPgstacEnv,
46
+ timeout: aws_cdk_lib_1.Duration.seconds(30)
42
47
  });
43
48
  // grant access to buckets using addToRolePolicy
44
49
  if (props.buckets) {
@@ -66,5 +71,5 @@ class TitilerPgstacApiLambda extends constructs_1.Construct {
66
71
  }
67
72
  exports.TitilerPgstacApiLambda = TitilerPgstacApiLambda;
68
73
  _a = JSII_RTTI_SYMBOL_1;
69
- TitilerPgstacApiLambda[_a] = { fqn: "eoapi-cdk.TitilerPgstacApiLambda", version: "5.3.0" };
70
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["index.ts"],"names":[],"mappings":";;;;;AAAA,6CAUuB;AACrB,4EAAuE;AACvE,sGAAqF;AACrF,2CAAuC;AAEvC,MAAa,sBAAuB,SAAQ,sBAAS;IAInD,YAAY,KAAgB,EAAE,EAAU,EAAE,KAAkC;QAC1E,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAEjB,MAAM,gBAAgB,GAAG;YACrB,kCAAkC,EAAE,iBAAiB;YACrD,eAAe,EAAE,KAAK;YACtB,8BAA8B,EAAE,WAAW;YAC3C,6BAA6B,EAAE,OAAO;YACtC,oCAAoC,EAAE,KAAK;YAC3C,qBAAqB,EAAE,KAAK;YAC5B,mBAAmB,EAAE,GAAG;YACxB,gBAAgB,EAAE,QAAQ;YAC1B,WAAW,EAAE,MAAM;YACnB,gBAAgB,EAAE,SAAS;YAC3B,kBAAkB,EAAE,GAAG;YACvB,kBAAkB,EAAE,GAAG;YACvB,mBAAmB,EAAE,KAAK,CAAC,QAAQ,CAAC,SAAS;SAChD,CAAA;QAGD,IAAI,CAAC,2BAA2B,GAAG,IAAI,wBAAM,CAAC,QAAQ,CAAC,IAAI,EAAE,QAAQ,EAAE;YACrE,OAAO,EAAE,iBAAiB;YAC1B,OAAO,EAAE,wBAAM,CAAC,OAAO,CAAC,WAAW;YACnC,IAAI,EAAE,wBAAM,CAAC,IAAI,CAAC,eAAe,CAAC,SAAS,EAAE;gBAC3C,IAAI,EAAE,oBAAoB;gBAC1B,SAAS,EAAE,EAAE,cAAc,EAAE,MAAM,EAAE;aACtC,CAAC;YACF,OAAO,EAAE,sBAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;YAC7B,GAAG,EAAE,KAAK,CAAC,GAAG;YACd,UAAU,EAAE,KAAK,CAAC,eAAe;YACjC,iBAAiB,EAAE,IAAI;YACvB,UAAU,EAAE,IAAI;YAChB,YAAY,EAAE,sBAAQ,CAAC,aAAa,CAAC,QAAQ;YAC7C,WAAW,EAAE,gBAAgB;SAC9B,CAAC,CAAC;QAEH,gDAAgD;QAChD,IAAI,KAAK,CAAC,OAAO,EAAE;YACjB,KAAK,CAAC,OAAO,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE;gBAC7B,IAAI,CAAC,2BAA2B,CAAC,eAAe,CAAC,IAAI,qBAAG,CAAC,eAAe,CAAC;oBACvE,OAAO,EAAE,CAAC,cAAc,CAAC;oBACzB,SAAS,EAAE,CAAC,gBAAgB,MAAM,IAAI,CAAC;iBACxC,CAAC,CAAC,CAAC;YACN,CAAC,CAAC,CAAC;SACJ;QAED,KAAK,CAAC,QAAQ,CAAC,SAAS,CAAC,IAAI,CAAC,2BAA2B,CAAC,CAAC;QAC3D,IAAI,CAAC,2BAA2B,CAAC,WAAW,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE,EAAE,qBAAG,CAAC,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,gCAAgC,CAAC,CAAC;QAErH,MAAM,OAAO,GAAG,IAAI,gCAAO,CAAC,IAAI,EAAE,GAAG,mBAAK,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,SAAS,qBAAqB,EAAE;YAClF,oBAAoB,EAAE,KAAK,CAAC,0BAA0B,CAAC,CAAC,CAAC;gBACvD,UAAU,EAAE,KAAK,CAAC,0BAA0B;aAC7C,CAAC,CAAC,CAAC,SAAS;YACb,kBAAkB,EAAE,IAAI,2DAAqB,CAAC,aAAa,EAAE,IAAI,CAAC,2BAA2B,CAAC;SAC/F,CAAC,CAAC;QAEH,IAAI,CAAC,GAAG,GAAG,OAAO,CAAC,GAAI,CAAC;QAExB,IAAI,uBAAS,CAAC,IAAI,EAAE,2BAA2B,EAAE;YAC/C,UAAU,EAAE,GAAG,mBAAK,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,SAAS,qBAAqB;YAC5D,KAAK,EAAE,IAAI,CAAC,GAAG;SAChB,CAAC,CAAC;IACL,CAAC;;AAlEH,wDAmEC","sourcesContent":["import {\n    Stack,\n    aws_iam as iam,\n    aws_ec2 as ec2,\n    aws_rds as rds,\n    aws_lambda as lambda,\n    aws_secretsmanager as secretsmanager,\n    CfnOutput,\n    Duration,\n    aws_logs,\n  } from \"aws-cdk-lib\";\n  import { IDomainName, HttpApi } from \"@aws-cdk/aws-apigatewayv2-alpha\";\n  import { HttpLambdaIntegration } from \"@aws-cdk/aws-apigatewayv2-integrations-alpha\";\n  import { Construct } from \"constructs\";\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      const titilerPgstacEnv = {\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          \"PGSTAC_SECRET_ARN\": props.dbSecret.secretArn,\n      }\n    \n      \n      this.titilerPgstacLambdaFunction = new lambda.Function(this, \"lambda\", {\n        handler: \"handler.handler\",\n        runtime: lambda.Runtime.PYTHON_3_10,\n        code: lambda.Code.fromDockerBuild(__dirname, {\n          file: \"runtime/Dockerfile\",\n          buildArgs: { PYTHON_VERSION: '3.10' },\n        }),\n        timeout: Duration.seconds(30),\n        vpc: props.vpc,\n        vpcSubnets: props.subnetSelection,\n        allowPublicSubnet: true,\n        memorySize: 3008,\n        logRetention: aws_logs.RetentionDays.ONE_WEEK,\n        environment: titilerPgstacEnv,\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      this.titilerPgstacLambdaFunction.connections.allowTo(props.db, ec2.Port.tcp(5432), \"allow connections from titiler\");\n  \n      const stacApi = new HttpApi(this, `${Stack.of(this).stackName}-titiler-pgstac-api`, {\n        defaultDomainMapping: props.titilerPgstacApiDomainName ? { \n          domainName: props.titilerPgstacApiDomainName \n        } : undefined,\n        defaultIntegration: new HttpLambdaIntegration(\"integration\", this.titilerPgstacLambdaFunction),\n      });\n  \n      this.url = stacApi.url!;\n  \n      new CfnOutput(this, \"titiler-pgstac-api-output\", {\n        exportName: `${Stack.of(this).stackName}-titiler-pgstac-url`,\n        value: this.url,\n      });\n    }\n  }\n  \n  export interface TitilerPgStacApiLambdaProps {\n\n    /**\n     * VPC into which the lambda should be deployed.\n     */\n    readonly vpc: ec2.IVpc;\n  \n    /**\n     * RDS Instance with installed pgSTAC.\n     */\n    readonly db: rds.IDatabaseInstance;\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     * 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    readonly titilerPgstacApiDomainName?: IDomainName;\n  }\n"]}
74
+ TitilerPgstacApiLambda[_a] = { fqn: "eoapi-cdk.TitilerPgstacApiLambda", version: "5.4.0" };
75
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["index.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n    Stack,\n    aws_iam as iam,\n    aws_ec2 as ec2,\n    aws_rds as rds,\n    aws_lambda as lambda,\n    aws_secretsmanager as secretsmanager,\n    CfnOutput,\n    Duration,\n    aws_logs,\n    BundlingOptions\n  } from \"aws-cdk-lib\";\n  import { Runtime } from 'aws-cdk-lib/aws-lambda';\n  import {PythonFunction} from \"@aws-cdk/aws-lambda-python-alpha\";\n  import { IDomainName, HttpApi } from \"@aws-cdk/aws-apigatewayv2-alpha\";\n  import { HttpLambdaIntegration } from \"@aws-cdk/aws-apigatewayv2-integrations-alpha\";\n  import { Construct } from \"constructs\";\n  \n\n  // default settings that can be overridden by the user-provided environment. \n  let defaultTitilerPgstacEnv :{ [key: string]: any } = {\n    \"CPL_VSIL_CURL_ALLOWED_EXTENSIONS\": \".tif,.TIF,.tiff\",\n    \"GDAL_CACHEMAX\": \"200\", \n    \"GDAL_DISABLE_READDIR_ON_OPEN\": \"EMPTY_DIR\",\n    \"GDAL_INGESTED_BYTES_AT_OPEN\": \"32768\",\n    \"GDAL_HTTP_MERGE_CONSECUTIVE_RANGES\": \"YES\",\n    \"GDAL_HTTP_MULTIPLEX\": \"YES\",\n    \"GDAL_HTTP_VERSION\": \"2\",\n    \"PYTHONWARNINGS\": \"ignore\",\n    \"VSI_CACHE\": \"TRUE\",\n    \"VSI_CACHE_SIZE\": \"5000000\", \n    \"DB_MIN_CONN_SIZE\": \"1\",\n    \"DB_MAX_CONN_SIZE\": \"1\"\n  }\n\n  export class TitilerPgstacApiLambda extends Construct {\n    readonly url: string;\n    public titilerPgstacLambdaFunction: lambda.Function;\n  \n    constructor(scope: Construct, id: string, props: TitilerPgStacApiLambdaProps) {\n      super(scope, id);\n\n\n      // if user provided environment variables, merge them with the defaults.\n      const apiEnv = props.apiEnv ? { ...defaultTitilerPgstacEnv, ...props.apiEnv, \"PGSTAC_SECRET_ARN\": props.dbSecret.secretArn } : defaultTitilerPgstacEnv;\n\n      const pythonLambdaOptions: TitilerPgstacPythonLambdaOptions = props.pythonLambdaOptions ?? {\n        runtime: lambda.Runtime.PYTHON_3_10,\n        entry: `${__dirname}/runtime`,\n        index: \"src/handler.py\",\n        handler: \"handler\",\n        memorySize: 3008,\n        architecture: lambda.Architecture.X86_64\n      }\n\n      this.titilerPgstacLambdaFunction = new PythonFunction(this, \"titiler-pgstac-api\", {\n        ...pythonLambdaOptions,\n        environment: apiEnv,\n        vpc: props.vpc,\n        vpcSubnets: props.subnetSelection,\n        allowPublicSubnet: true,\n        logRetention: aws_logs.RetentionDays.ONE_WEEK,\n        timeout: Duration.seconds(30)\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      this.titilerPgstacLambdaFunction.connections.allowTo(props.db, ec2.Port.tcp(5432), \"allow connections from titiler\");\n  \n      const stacApi = new HttpApi(this, `${Stack.of(this).stackName}-titiler-pgstac-api`, {\n        defaultDomainMapping: props.titilerPgstacApiDomainName ? { \n          domainName: props.titilerPgstacApiDomainName \n        } : undefined,\n        defaultIntegration: new HttpLambdaIntegration(\"integration\", this.titilerPgstacLambdaFunction),\n      });\n  \n      this.url = stacApi.url!;\n  \n      new CfnOutput(this, \"titiler-pgstac-api-output\", {\n        exportName: `${Stack.of(this).stackName}-titiler-pgstac-url`,\n        value: this.url,\n      });\n    }\n  }\n  \n  export interface TitilerPgStacApiLambdaProps {\n\n    /**\n     * VPC into which the lambda should be deployed.\n     */\n    readonly vpc: ec2.IVpc;\n  \n    /**\n     * RDS Instance with installed pgSTAC.\n     */\n    readonly db: rds.IDatabaseInstance;\n  \n    /**\n     * Subnet into which the lambda should be deployed.\n     */\n    readonly subnetSelection: ec2.SubnetSelection;\n  \n    /**\n     * Secret containing connection information for pgSTAC database.\n     */\n    readonly dbSecret: secretsmanager.ISecret;\n\n    /**\n     * Customized environment variables to send to titiler-pgstac runtime. These will be merged with `defaultTitilerPgstacEnv`.\n     * The database secret arn is automatically added to the environment variables at deployment. \n    /*/\n    readonly apiEnv?: Record<string, string>;\n\n    /**\n     * list of buckets the lambda will be granted access to. \n     */\n    readonly buckets?: string[];\n\n    /**\n     * Custom Domain Name Options for Titiler Pgstac API,\n     * \n     * @default - undefined. \n     */\n    readonly titilerPgstacApiDomainName?: IDomainName;\n\n    /**\n     * Optional settings for the titiler-pgstac python lambda function.\n     *\n     * @default - defined in the construct.\n     */\n    readonly pythonLambdaOptions?: TitilerPgstacPythonLambdaOptions;\n\n  }\n\n\n  export interface TitilerPgstacPythonLambdaOptions {\n\n    /**\n     * Path to the source of the function or the location for dependencies.\n     */\n    readonly entry: string;\n    /**\n     * The runtime environment. Only runtimes of the Python family are\n     * supported.\n     */\n    readonly runtime: Runtime;\n\n    /**\n     * The path (relative to entry) to the index file containing the exported handler.\n     *\n     */\n    readonly index: string;\n    /**\n     * The name of the exported handler in the index file.\n     */\n    readonly handler: string;\n\n    /**\n     * Bundling options to use for this function. Use this to specify custom bundling options like\n     * the bundling Docker image, asset hash type, custom hash, architecture, etc.\n     */\n    readonly bundling?: BundlingOptions;\n\n    /**\n     * The amount of memory, in MB, that is allocated to your Lambda function.\n     */\n    readonly memorySize: number;\n\n    /**\n     * The system architectures compatible with this lambda function.\n     */\n    readonly architecture: lambda.Architecture;\n\n  }\n"]}
@@ -1 +1,2 @@
1
- uvicorn
1
+ uvicorn
2
+ boto3>=1.26.139
@@ -1,3 +1,3 @@
1
1
  titiler.pgstac==0.5.1
2
- boto3>=1.26.139
3
2
  psycopg[binary, pool]
3
+ mangum>=0.14,<0.15
@@ -5,7 +5,7 @@ Handler for AWS Lambda.
5
5
  import asyncio
6
6
  import os
7
7
  from mangum import Mangum
8
- from utils import get_secret_dict
8
+ from src.utils import get_secret_dict
9
9
 
10
10
  pgstac_secret_arn = os.environ["PGSTAC_SECRET_ARN"]
11
11
 
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "eoapi-cdk",
3
- "version": "5.3.0",
3
+ "version": "5.4.0",
4
4
  "description": "A set of constructs deploying pgSTAC with CDK",
5
5
  "main": "lib/index.js",
6
6
  "types": "lib/index.d.ts",
@@ -6574,7 +6574,7 @@
6574
6574
  "affectsGlobalScope": false
6575
6575
  },
6576
6576
  "./lib/ingestor-api/index.ts": {
6577
- "version": "2fe2e8014a09f3380c17267d665dfac70070d7aca4c108f8a4a64f010a2451b3",
6577
+ "version": "b5a0d936e7135736494119f1d79c5c940edd1ff455c170c17ebc08edeb08dfda",
6578
6578
  "affectsGlobalScope": false
6579
6579
  },
6580
6580
  "./node_modules/@aws-cdk/aws-apigatewayv2-alpha/lib/common/api.d.ts": {
@@ -6726,7 +6726,7 @@
6726
6726
  "affectsGlobalScope": false
6727
6727
  },
6728
6728
  "./lib/titiler-pgstac-api/index.ts": {
6729
- "version": "dd16f73144a3c7dd2fd759d876d5a6940ba8250dca9f930a324d5637c07a25da",
6729
+ "version": "6cb642e3767927a0095d8866e4a87da7ffb9aa070339ef11779442c694fc28eb",
6730
6730
  "affectsGlobalScope": false
6731
6731
  },
6732
6732
  "./lib/stac-browser/index.ts": {
@@ -6734,7 +6734,7 @@
6734
6734
  "affectsGlobalScope": false
6735
6735
  },
6736
6736
  "./lib/tipg-api/index.ts": {
6737
- "version": "6d028789fd0d34f24dd7d44bd1c5e4c224476c75ca8d6874a4ae6a9f6c37e9f6",
6737
+ "version": "63467193cd44122f3a12d2485cf391eccba9eeaf2daa2cbe9734605a07785a51",
6738
6738
  "affectsGlobalScope": false
6739
6739
  },
6740
6740
  "./lib/index.ts": {
@@ -7161,7 +7161,9 @@
7161
7161
  "./lib/titiler-pgstac-api/index.ts": [
7162
7162
  "./node_modules/@aws-cdk/aws-apigatewayv2-alpha/lib/index.d.ts",
7163
7163
  "./node_modules/@aws-cdk/aws-apigatewayv2-integrations-alpha/lib/index.d.ts",
7164
+ "./node_modules/@aws-cdk/aws-lambda-python-alpha/lib/index.d.ts",
7164
7165
  "./node_modules/@types/node/ts4.8/util.d.ts",
7166
+ "./node_modules/aws-cdk-lib/aws-lambda/index.d.ts",
7165
7167
  "./node_modules/aws-cdk-lib/index.d.ts",
7166
7168
  "./node_modules/constructs/lib/index.d.ts"
7167
7169
  ],
@@ -1,20 +0,0 @@
1
- ARG PYTHON_VERSION
2
-
3
- FROM --platform=linux/amd64 public.ecr.aws/lambda/python:${PYTHON_VERSION}
4
-
5
- WORKDIR /tmp
6
- RUN python -m pip install pip -U
7
-
8
- COPY runtime/requirements.txt requirements.txt
9
- RUN python -m pip install -r requirements.txt "mangum>=0.14,<0.15" -t /asset
10
-
11
- # Reduce package size and remove useless files
12
- RUN find /asset -type f -name '*.pyc' | while read f; do n=$(echo $f | sed 's/__pycache__\///' | sed 's/.cpython-[0-9]*//'); cp $f $n; done;
13
- RUN find /asset -type d -name '__pycache__' -print0 | xargs -0 rm -rf
14
- RUN find /asset -type f -name '*.py' -print0 | xargs -0 rm -f
15
- RUN find /asset -type d -name 'tests' -print0 | xargs -0 rm -rf
16
- RUN rm -rdf /asset/numpy/doc/ /asset/boto3* /asset/botocore* /asset/bin /asset/geos_license /asset/Misc
17
-
18
- COPY runtime/src/*.py /asset/
19
-
20
- CMD ["echo", "hello world"]