Skip to content

Commit f39cf23

Browse files
Merge pull request #31 from developmentseed/stac-browser
Add CloudFront distribution for STAC Browser
2 parents 14c1853 + 0d86184 commit f39cf23

File tree

4 files changed

+132
-16
lines changed

4 files changed

+132
-16
lines changed

.gitignore

+1
Original file line numberDiff line numberDiff line change
@@ -169,3 +169,4 @@ node_modules/
169169

170170
.ruff_cache/
171171
.env-cdk
172+
config.yaml

browser_config.js

+34
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,34 @@
1+
module.exports = {
2+
catalogUrl: "https://stac.eoapi.dev",
3+
catalogTitle: "eoAPI STAC Browser",
4+
allowExternalAccess: true, // Must be true if catalogUrl is not given
5+
allowedDomains: [],
6+
detectLocaleFromBrowser: true,
7+
storeLocale: true,
8+
locale: "en",
9+
fallbackLocale: "en",
10+
supportedLocales: ["de", "es", "en", "fr", "it", "ro"],
11+
apiCatalogPriority: null,
12+
useTileLayerAsFallback: true,
13+
displayGeoTiffByDefault: false,
14+
buildTileUrlTemplate: ({ href, asset }) =>
15+
"https://raster.eoapi.dev/external/tiles/WebMercatorQuad/{z}/{x}/{y}@2x?url=" +
16+
encodeURIComponent(asset.href.startsWith("/vsi") ? asset.href : href),
17+
stacProxyUrl: null,
18+
pathPrefix: "/",
19+
historyMode: "history",
20+
cardViewMode: "cards",
21+
cardViewSort: "asc",
22+
showThumbnailsAsAssets: false,
23+
stacLint: true,
24+
geoTiffResolution: 128,
25+
redirectLegacyUrls: false,
26+
itemsPerPage: 12,
27+
defaultThumbnailSize: null,
28+
maxPreviewsOnMap: 50,
29+
crossOriginMedia: null,
30+
requestHeaders: {},
31+
requestQueryParameters: {},
32+
preprocessSTAC: null,
33+
authConfig: null,
34+
};

infrastructure/app.py

+73-8
Original file line numberDiff line numberDiff line change
@@ -7,10 +7,14 @@
77
RemovalPolicy,
88
Stack,
99
aws_certificatemanager,
10+
aws_cloudfront,
11+
aws_cloudfront_origins,
1012
aws_ec2,
1113
aws_iam,
1214
aws_lambda,
1315
aws_rds,
16+
aws_route53,
17+
aws_route53_targets,
1418
aws_s3,
1519
)
1620
from aws_cdk.aws_apigateway import DomainNameOptions
@@ -352,29 +356,90 @@ def __init__(
352356
)
353357

354358
if app_config.stac_browser_version:
359+
if not (
360+
app_config.hosted_zone_id
361+
and app_config.hosted_zone_name
362+
and app_config.stac_browser_custom_domain
363+
and app_config.stac_browser_certificate_arn
364+
):
365+
raise ValueError(
366+
"to deploy STAC browser you must provide config parameters for hosted_zone_id and stac_browser_custom_domain and stac_browser_certificate_arn"
367+
)
368+
355369
stac_browser_bucket = aws_s3.Bucket(
356370
self,
357371
"stac-browser-bucket",
358372
bucket_name=app_config.build_service_name("stac-browser"),
359373
removal_policy=RemovalPolicy.DESTROY,
360374
auto_delete_objects=True,
361-
website_index_document="index.html",
362-
public_read_access=True,
363-
block_public_access=aws_s3.BlockPublicAccess(
364-
block_public_acls=False,
365-
block_public_policy=False,
366-
ignore_public_acls=False,
367-
restrict_public_buckets=False,
375+
block_public_access=aws_s3.BlockPublicAccess.BLOCK_ALL,
376+
enforce_ssl=True,
377+
)
378+
379+
distribution = aws_cloudfront.Distribution(
380+
self,
381+
"stac-browser-distribution",
382+
default_behavior=aws_cloudfront.BehaviorOptions(
383+
origin=aws_cloudfront_origins.S3Origin(stac_browser_bucket),
384+
viewer_protocol_policy=aws_cloudfront.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,
385+
allowed_methods=aws_cloudfront.AllowedMethods.ALLOW_GET_HEAD,
386+
cached_methods=aws_cloudfront.CachedMethods.CACHE_GET_HEAD,
368387
),
369-
object_ownership=aws_s3.ObjectOwnership.OBJECT_WRITER,
388+
default_root_object="index.html",
389+
error_responses=[
390+
aws_cloudfront.ErrorResponse(
391+
http_status=404,
392+
response_http_status=200,
393+
response_page_path="/index.html",
394+
)
395+
],
396+
certificate=aws_certificatemanager.Certificate.from_certificate_arn(
397+
self,
398+
"stac-browser-certificate",
399+
app_config.stac_browser_certificate_arn,
400+
),
401+
domain_names=[app_config.stac_browser_custom_domain],
370402
)
403+
404+
account_id = Stack.of(self).account
405+
distribution_arn = f"arn:aws:cloudfront::${account_id}:distribution/${distribution.distribution_id}"
406+
407+
stac_browser_bucket.add_to_resource_policy(
408+
aws_iam.PolicyStatement(
409+
actions=["s3:GetObject"],
410+
resources=[stac_browser_bucket.arn_for_objects("*")],
411+
principals=[aws_iam.ServicePrincipal("cloudfront.amazonaws.com")],
412+
conditions={"StringEquals": {"AWS:SourceArn": distribution_arn}},
413+
)
414+
)
415+
416+
hosted_zone = aws_route53.HostedZone.from_hosted_zone_attributes(
417+
self,
418+
"stac-browser-hosted-zone",
419+
hosted_zone_id=app_config.hosted_zone_id,
420+
zone_name=app_config.hosted_zone_name,
421+
)
422+
423+
aws_route53.ARecord(
424+
self,
425+
"stac-browser-alias",
426+
zone=hosted_zone,
427+
target=aws_route53.RecordTarget.from_alias(
428+
aws_route53_targets.CloudFrontTarget(distribution)
429+
),
430+
record_name=app_config.stac_browser_custom_domain,
431+
)
432+
371433
StacBrowser(
372434
self,
373435
"stac-browser",
374436
github_repo_tag=app_config.stac_browser_version,
375437
stac_catalog_url=f"https://{app_config.stac_api_custom_domain}",
376438
website_index_document="index.html",
377439
bucket_arn=stac_browser_bucket.bucket_arn,
440+
config_file_path=os.path.join(
441+
os.path.abspath(context_dir), "browser_config.js"
442+
),
378443
)
379444

380445
def _create_data_access_role(self) -> aws_iam.Role:

infrastructure/config.py

+24-8
Original file line numberDiff line numberDiff line change
@@ -116,6 +116,22 @@ class AppConfig(BaseSettings):
116116
as it will be used as a backend.""",
117117
default=None,
118118
)
119+
stac_browser_custom_domain: Optional[str] = Field(
120+
description="Custom domain name for the STAC Browser site",
121+
default=None,
122+
)
123+
stac_browser_certificate_arn: Optional[str] = Field(
124+
description="Arn for the STAC Browser custom domain name (must be in us-east-1)",
125+
default=None,
126+
)
127+
hosted_zone_id: Optional[str] = Field(
128+
description="Hosted Zone ID for custom domains",
129+
default=None,
130+
)
131+
hosted_zone_name: Optional[str] = Field(
132+
description="Hosted Zone Name for custom domains",
133+
default=None,
134+
)
119135

120136
model_config = SettingsConfigDict(
121137
env_file=".env-cdk", yaml_file="config.yaml", extra="allow"
@@ -137,14 +153,14 @@ def validate_model(self) -> Self:
137153
and therefore `nat_gateway_count` has to be > 0."""
138154
)
139155

140-
if (
141-
self.stac_browser_version is not None
142-
and self.stac_api_custom_domain is None
143-
):
144-
raise ValueError(
145-
"""If a STAC browser version is provided,
146-
a custom domain must be provided for the STAC API"""
147-
)
156+
# if (
157+
# self.stac_browser_version is not None
158+
# and self.stac_api_custom_domain is None
159+
# ):
160+
# raise ValueError(
161+
# """If a STAC browser version is provided,
162+
# a custom domain must be provided for the STAC API"""
163+
# )
148164

149165
if self.acm_certificate_arn is None and any(
150166
[

0 commit comments

Comments
 (0)