eoapi-cdk 8.2.3 → 8.3.1

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.
@@ -179,5 +179,5 @@ class StacIngestor extends constructs_1.Construct {
179
179
  }
180
180
  exports.StacIngestor = StacIngestor;
181
181
  _a = JSII_RTTI_SYMBOL_1;
182
- StacIngestor[_a] = { fqn: "eoapi-cdk.StacIngestor", version: "8.2.3" };
182
+ StacIngestor[_a] = { fqn: "eoapi-cdk.StacIngestor", version: "8.3.1" };
183
183
  //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["index.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  aws_apigateway as apigateway,\n  aws_dynamodb as dynamodb,\n  aws_ec2 as ec2,\n  aws_iam as iam,\n  aws_lambda as lambda,\n  aws_logs,\n  aws_lambda_event_sources as events,\n  aws_secretsmanager as secretsmanager,\n  aws_ssm as ssm,\n  Duration,\n  RemovalPolicy,\n  Stack,\n} from \"aws-cdk-lib\";\nimport { Construct } from \"constructs\";\nimport { CustomLambdaFunctionProps, DEFAULT_PGSTAC_VERSION } from \"../utils\";\n\nexport class StacIngestor extends Construct {\n  table: dynamodb.Table;\n  public handlerRole: iam.Role;\n\n  constructor(scope: Construct, id: string, props: StacIngestorProps) {\n    super(scope, id);\n\n    this.table = this.buildTable();\n\n    const env: Record<string, string> = {\n      DYNAMODB_TABLE: this.table.tableName,\n      ROOT_PATH: `/${props.stage}`,\n      NO_PYDANTIC_SSM_SETTINGS: \"1\",\n      STAC_URL: props.stacUrl,\n      DATA_ACCESS_ROLE: props.dataAccessRole.roleArn,\n      ...props.apiEnv,\n    };\n\n    this.handlerRole = new iam.Role(this, \"execution-role\", {\n      description:\n        \"Role used by STAC Ingestor. Manually defined so that we can choose a name that is supported by the data access roles trust policy\",\n      assumedBy: new iam.ServicePrincipal(\"lambda.amazonaws.com\"),\n      managedPolicies: [\n        iam.ManagedPolicy.fromAwsManagedPolicyName(\n          \"service-role/AWSLambdaBasicExecutionRole\"\n        ),\n        iam.ManagedPolicy.fromAwsManagedPolicyName(\n          \"service-role/AWSLambdaVPCAccessExecutionRole\"\n        ),\n      ],\n    });\n\n    const handler = this.buildApiLambda({\n      table: this.table,\n      env,\n      dataAccessRole: props.dataAccessRole,\n      stage: props.stage,\n      dbSecret: props.stacDbSecret,\n      dbVpc: props.vpc,\n      dbSecurityGroup: props.stacDbSecurityGroup,\n      subnetSelection: props.subnetSelection,\n      lambdaFunctionOptions: props.apiLambdaFunctionOptions,\n      pgstacVersion: props.pgstacVersion,\n    });\n\n    this.buildApiEndpoint({\n      handler,\n      stage: props.stage,\n      endpointConfiguration: props.apiEndpointConfiguration,\n      policy: props.apiPolicy,\n      ingestorDomainNameOptions: props.ingestorDomainNameOptions,\n    });\n\n    this.buildIngestor({\n      table: this.table,\n      env: env,\n      dbSecret: props.stacDbSecret,\n      dbVpc: props.vpc,\n      dbSecurityGroup: props.stacDbSecurityGroup,\n      subnetSelection: props.subnetSelection,\n      lambdaFunctionOptions: props.ingestorLambdaFunctionOptions,\n      pgstacVersion: props.pgstacVersion,\n    });\n\n    this.registerSsmParameter({\n      name: \"dynamodb_table\",\n      value: this.table.tableName,\n      description: \"Name of table used to store ingestions\",\n    });\n  }\n\n  private buildTable(): dynamodb.Table {\n    const table = new dynamodb.Table(this, \"ingestions-table\", {\n      partitionKey: { name: \"created_by\", type: dynamodb.AttributeType.STRING },\n      sortKey: { name: \"id\", type: dynamodb.AttributeType.STRING },\n      billingMode: dynamodb.BillingMode.PAY_PER_REQUEST,\n      removalPolicy: RemovalPolicy.DESTROY,\n      stream: dynamodb.StreamViewType.NEW_IMAGE,\n    });\n\n    table.addGlobalSecondaryIndex({\n      indexName: \"status\",\n      partitionKey: { name: \"status\", type: dynamodb.AttributeType.STRING },\n      sortKey: { name: \"created_at\", type: dynamodb.AttributeType.STRING },\n    });\n\n    return table;\n  }\n\n  private buildApiLambda(props: {\n    table: dynamodb.ITable;\n    env: Record<string, string>;\n    dataAccessRole: iam.IRole;\n    stage: string;\n    dbSecret: secretsmanager.ISecret;\n    dbVpc: undefined | ec2.IVpc;\n    dbSecurityGroup: ec2.ISecurityGroup;\n    subnetSelection: undefined | ec2.SubnetSelection;\n    lambdaFunctionOptions?: CustomLambdaFunctionProps;\n    pgstacVersion?: string;\n  }): lambda.Function {\n    const handler = new lambda.Function(this, \"api-handler\", {\n      // defaults\n      runtime: lambda.Runtime.PYTHON_3_11,\n      handler: \"src.handler.handler\",\n      memorySize: 2048,\n      logRetention: aws_logs.RetentionDays.ONE_WEEK,\n      timeout: Duration.seconds(30),\n      code: lambda.Code.fromDockerBuild(__dirname, {\n        file: \"runtime/Dockerfile\",\n        buildArgs: {\n          PYTHON_VERSION: \"3.11\",\n          PGSTAC_VERSION: props.pgstacVersion || DEFAULT_PGSTAC_VERSION,\n        },\n      }),\n      allowPublicSubnet: true,\n      vpc: props.dbVpc,\n      vpcSubnets: props.subnetSelection,\n      environment: { DB_SECRET_ARN: props.dbSecret.secretArn, ...props.env },\n      role: this.handlerRole,\n      // overwrites defaults with user-provided configurable properties\n      ...props.lambdaFunctionOptions,\n    });\n\n    // Allow handler to read DB secret\n    props.dbSecret.grantRead(handler);\n\n    // Allow handler to connect to DB\n\n    if (props.dbVpc) {\n      props.dbSecurityGroup.addIngressRule(\n        handler.connections.securityGroups[0],\n        ec2.Port.tcp(5432),\n        \"Allow connections from STAC Ingestor\"\n      );\n    }\n\n    props.table.grantReadWriteData(handler);\n\n    return handler;\n  }\n\n  private buildIngestor(props: {\n    table: dynamodb.ITable;\n    env: Record<string, string>;\n    dbSecret: secretsmanager.ISecret;\n    dbVpc: undefined | ec2.IVpc;\n    dbSecurityGroup: ec2.ISecurityGroup;\n    subnetSelection: undefined | ec2.SubnetSelection;\n    lambdaFunctionOptions?: CustomLambdaFunctionProps;\n    pgstacVersion?: string;\n  }): lambda.Function {\n    const handler = new lambda.Function(this, \"stac-ingestor\", {\n      // defaults\n      runtime: lambda.Runtime.PYTHON_3_11,\n      handler: \"src.ingestor.handler\",\n      memorySize: 2048,\n      logRetention: aws_logs.RetentionDays.ONE_WEEK,\n      timeout: Duration.seconds(180),\n      code: lambda.Code.fromDockerBuild(__dirname, {\n        file: \"runtime/Dockerfile\",\n        buildArgs: {\n          PYTHON_VERSION: \"3.11\",\n          PGSTAC_VERSION: props.pgstacVersion || DEFAULT_PGSTAC_VERSION,\n        },\n      }),\n      vpc: props.dbVpc,\n      vpcSubnets: props.subnetSelection,\n      allowPublicSubnet: true,\n      environment: { DB_SECRET_ARN: props.dbSecret.secretArn, ...props.env },\n      role: this.handlerRole,\n      // overwrites defaults with user-provided configurable properties\n      ...props.lambdaFunctionOptions,\n    });\n\n    // Allow handler to read DB secret\n    props.dbSecret.grantRead(handler);\n\n    // Allow handler to connect to DB\n    if (props.dbVpc) {\n      props.dbSecurityGroup.addIngressRule(\n        handler.connections.securityGroups[0],\n        ec2.Port.tcp(5432),\n        \"Allow connections from STAC Ingestor\"\n      );\n    }\n\n    // Allow handler to write results back to DB\n    props.table.grantWriteData(handler);\n\n    // Trigger handler from writes to DynamoDB table\n    handler.addEventSource(\n      new events.DynamoEventSource(props.table, {\n        // Read when batches reach size...\n        batchSize: 1000,\n        // ... or when window is reached.\n        maxBatchingWindow: Duration.seconds(10),\n        // Read oldest data first.\n        startingPosition: lambda.StartingPosition.TRIM_HORIZON,\n        retryAttempts: 1,\n      })\n    );\n\n    return handler;\n  }\n\n  private buildApiEndpoint(props: {\n    handler: lambda.IFunction;\n    stage: string;\n    policy?: iam.PolicyDocument;\n    endpointConfiguration?: apigateway.EndpointConfiguration;\n    ingestorDomainNameOptions?: apigateway.DomainNameOptions;\n  }): apigateway.LambdaRestApi {\n    return new apigateway.LambdaRestApi(\n      this,\n      `${Stack.of(this).stackName}-ingestor-api`,\n      {\n        handler: props.handler,\n        proxy: true,\n\n        cloudWatchRole: true,\n        deployOptions: { stageName: props.stage },\n        endpointExportName: `${Stack.of(this)}-ingestor-api`,\n\n        endpointConfiguration: props.endpointConfiguration,\n        policy: props.policy,\n\n        domainName: props.ingestorDomainNameOptions\n          ? {\n              domainName: props.ingestorDomainNameOptions.domainName,\n              certificate: props.ingestorDomainNameOptions.certificate,\n            }\n          : undefined,\n      }\n    );\n  }\n\n  private registerSsmParameter(props: {\n    name: string;\n    value: string;\n    description: string;\n  }): ssm.IStringParameter {\n    const parameterNamespace = Stack.of(this).stackName;\n    return new ssm.StringParameter(\n      this,\n      `${props.name.replace(\"_\", \"-\")}-parameter`,\n      {\n        description: props.description,\n        parameterName: `/${parameterNamespace}/${props.name}`,\n        stringValue: props.value,\n      }\n    );\n  }\n}\n\nexport interface StacIngestorProps {\n  /**\n   * ARN of AWS Role used to validate access to S3 data\n   */\n  readonly dataAccessRole: iam.IRole;\n\n  /**\n   * URL of STAC API\n   */\n  readonly stacUrl: string;\n\n  /**\n   * Stage of deployment (e.g. `dev`, `prod`)\n   */\n  readonly stage: string;\n\n  /**\n   * Secret containing pgSTAC DB connection information\n   */\n  readonly stacDbSecret: secretsmanager.ISecret;\n\n  /**\n   * VPC running pgSTAC DB\n   */\n  readonly vpc?: ec2.IVpc;\n\n  /**\n   * Security Group used by pgSTAC DB\n   */\n  readonly stacDbSecurityGroup: ec2.ISecurityGroup;\n\n  /**\n   * Subnet into which the lambda should be deployed if using a VPC\n   */\n  readonly subnetSelection?: ec2.SubnetSelection;\n\n  /**\n   * Environment variables to be sent to Lambda.\n   */\n  readonly apiEnv?: Record<string, string>;\n\n  /**\n   * API Endpoint Configuration, useful for creating private APIs.\n   */\n  readonly apiEndpointConfiguration?: apigateway.EndpointConfiguration;\n\n  /**\n   * API Policy Document, useful for creating private APIs.\n   */\n  readonly apiPolicy?: iam.PolicyDocument;\n\n  /**\n   * Custom Domain Name Options for Ingestor API\n   */\n  readonly ingestorDomainNameOptions?: apigateway.DomainNameOptions;\n\n  /**\n   * Can be used to override the default lambda function properties.\n   *\n   * @default - default settings are defined in the construct.\n   */\n  readonly apiLambdaFunctionOptions?: CustomLambdaFunctionProps;\n\n  /**\n   * Can be used to override the default lambda function properties.\n   *\n   * @default - default settings are defined in the construct.\n   */\n  readonly ingestorLambdaFunctionOptions?: CustomLambdaFunctionProps;\n\n  /**\n   * pgstac version - must match the version installed on the pgstac database\n   *\n   * @default - default settings are defined in the construct\n   */\n  readonly pgstacVersion?: string;\n}\n"]}
@@ -92,5 +92,5 @@ class PgStacApiLambda extends constructs_1.Construct {
92
92
  }
93
93
  exports.PgStacApiLambda = PgStacApiLambda;
94
94
  _a = JSII_RTTI_SYMBOL_1;
95
- PgStacApiLambda[_a] = { fqn: "eoapi-cdk.PgStacApiLambda", version: "8.2.3" };
95
+ PgStacApiLambda[_a] = { fqn: "eoapi-cdk.PgStacApiLambda", version: "8.3.1" };
96
96
  //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["index.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Stack,\n  aws_apigatewayv2 as apigatewayv2,\n  aws_apigatewayv2_integrations as apigatewayv2_integrations,\n  aws_ec2 as ec2,\n  aws_rds as rds,\n  aws_lambda as lambda,\n  aws_secretsmanager as secretsmanager,\n  CfnOutput,\n  Duration,\n  aws_logs,\n} from \"aws-cdk-lib\";\nimport { Construct } from \"constructs\";\nimport { CustomLambdaFunctionProps } from \"../utils\";\nimport * as path from \"path\";\n\nexport const EXTENSIONS = {\n  QUERY: \"query\",\n  SORT: \"sort\",\n  FIELDS: \"fields\",\n  FILTER: \"filter\",\n  FREE_TEXT: \"free_text\",\n  PAGINATION: \"pagination\",\n  COLLECTION_SEARCH: \"collection_search\",\n  TRANSACTION: \"transaction\",\n  BULK_TRANSACTION: \"bulk_transactions\",\n} as const;\n\ntype ExtensionType = (typeof EXTENSIONS)[keyof typeof EXTENSIONS];\n\n/**\n * Validates if a given string is a valid STAC extension\n */\nfunction isValidExtension(value: string): value is ExtensionType {\n  return Object.values(EXTENSIONS).includes(value as any);\n}\n\nexport class PgStacApiLambda extends Construct {\n  readonly url: string;\n  public stacApiLambdaFunction: lambda.Function;\n\n  constructor(scope: Construct, id: string, props: PgStacApiLambdaProps) {\n    super(scope, id);\n\n    const defaultExtensions: ExtensionType[] = [\n      EXTENSIONS.QUERY,\n      EXTENSIONS.SORT,\n      EXTENSIONS.FIELDS,\n      EXTENSIONS.FILTER,\n      EXTENSIONS.FREE_TEXT,\n      EXTENSIONS.PAGINATION,\n      EXTENSIONS.COLLECTION_SEARCH,\n    ];\n\n    if (props.enabledExtensions) {\n      for (const ext of props.enabledExtensions) {\n        if (!isValidExtension(ext)) {\n          throw new Error(\n            `Invalid extension: \"${ext}\". Must be one of: ${Object.values(\n              EXTENSIONS\n            ).join(\", \")}`\n          );\n        }\n      }\n    }\n\n    const enabledExtensions = props.enabledExtensions || defaultExtensions;\n\n    this.stacApiLambdaFunction = new lambda.Function(this, \"lambda\", {\n      // defaults\n      runtime: lambda.Runtime.PYTHON_3_11,\n      handler: \"handler.handler\",\n      memorySize: 8192,\n      logRetention: aws_logs.RetentionDays.ONE_WEEK,\n      timeout: Duration.seconds(30),\n      code: lambda.Code.fromDockerBuild(path.join(__dirname, \"..\"), {\n        file: \"stac-api/runtime/Dockerfile\",\n        buildArgs: { PYTHON_VERSION: \"3.11\" },\n      }),\n      vpc: props.vpc,\n      vpcSubnets: props.subnetSelection,\n      allowPublicSubnet: true,\n      environment: {\n        PGSTAC_SECRET_ARN: props.dbSecret.secretArn,\n        DB_MIN_CONN_SIZE: \"0\",\n        DB_MAX_CONN_SIZE: \"1\",\n        ENABLED_EXTENSIONS: enabledExtensions.join(\",\"),\n        ...props.apiEnv,\n      },\n      // overwrites defaults with user-provided configurable properties\n      ...props.lambdaFunctionOptions,\n    });\n\n    props.dbSecret.grantRead(this.stacApiLambdaFunction);\n\n    if (props.vpc) {\n      this.stacApiLambdaFunction.connections.allowTo(\n        props.db,\n        ec2.Port.tcp(5432),\n        \"allow connections from stac-fastapi-pgstac\"\n      );\n    }\n\n    const stacApi = new apigatewayv2.HttpApi(\n      this,\n      `${Stack.of(this).stackName}-stac-api`,\n      {\n        defaultDomainMapping: props.stacApiDomainName\n          ? {\n              domainName: props.stacApiDomainName,\n            }\n          : undefined,\n        defaultIntegration: new apigatewayv2_integrations.HttpLambdaIntegration(\n          \"integration\",\n          this.stacApiLambdaFunction,\n          props.stacApiDomainName\n            ? {\n                parameterMapping:\n                  new apigatewayv2.ParameterMapping().overwriteHeader(\n                    \"host\",\n                    apigatewayv2.MappingValue.custom(\n                      props.stacApiDomainName.name\n                    )\n                  ),\n              }\n            : undefined\n        ),\n      }\n    );\n\n    this.url = stacApi.url!;\n\n    new CfnOutput(this, \"stac-api-output\", {\n      exportName: `${Stack.of(this).stackName}-url`,\n      value: this.url,\n    });\n  }\n}\n\nexport interface PgStacApiLambdaProps {\n  /**\n   * VPC into which the lambda should be deployed.\n   */\n  readonly vpc?: ec2.IVpc;\n\n  /**\n   * RDS Instance with installed pgSTAC or pgbouncer server.\n   */\n  readonly db: rds.IDatabaseInstance | ec2.IInstance;\n\n  /**\n   * Subnet into which the lambda should be deployed.\n   */\n  readonly subnetSelection?: ec2.SubnetSelection;\n\n  /**\n   * Secret containing connection information for pgSTAC database.\n   */\n  readonly dbSecret: secretsmanager.ISecret;\n\n  /**\n   * Customized environment variables to send to fastapi-pgstac runtime.\n   */\n  readonly apiEnv?: Record<string, string>;\n\n  /**\n   * Custom Domain Name Options for STAC API,\n   */\n  readonly stacApiDomainName?: apigatewayv2.IDomainName;\n\n  /**\n   * List of STAC API extensions to enable.\n   *\n   * @default - query, sort, fields, filter, free_text, pagniation, collection_search\n   */\n  readonly enabledExtensions?: ExtensionType[];\n\n  /**\n   * Can be used to override the default lambda function properties.\n   *\n   * @default - defined in the construct.\n   */\n  readonly lambdaFunctionOptions?: CustomLambdaFunctionProps;\n}\n"]}
@@ -91,5 +91,5 @@ class StacBrowser extends constructs_1.Construct {
91
91
  }
92
92
  exports.StacBrowser = StacBrowser;
93
93
  _a = JSII_RTTI_SYMBOL_1;
94
- StacBrowser[_a] = { fqn: "eoapi-cdk.StacBrowser", version: "8.2.3" };
94
+ StacBrowser[_a] = { fqn: "eoapi-cdk.StacBrowser", version: "8.3.1" };
95
95
  //# 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, 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(props: StacBrowserProps, 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 ${props.githubRepoTag}...`)\n            execSync(`git checkout tags/${props.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 ${props.githubRepoTag}...`)\n            execSync(`git checkout tags/${props.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        // If a config file is provided, copy it to the stac-browser directory at \"config.js\", replaces the default config.js.\n        if (props.configFilePath) {\n            // check that the file exists at this location. if not, raise an error and print current working directory.\n            if (!fs.existsSync(props.configFilePath)) {\n                throw new Error(`Config file ${props.configFilePath} does not exist. Current working directory is ${process.cwd()}`);\n            }\n            console.log(`Copying config file ${props.configFilePath} to ${cloneDirectory}/config.js`)\n            fs.copyFileSync(props.configFilePath, `${cloneDirectory}/config.js`);\n        }\n\n        // Build the app with catalogUrl\n        console.log(`Building app with catalogUrl=${props.stacCatalogUrl} into ${cloneDirectory}`)\n        execSync(`npm run build -- --catalogUrl=${props.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. Overrides the catalog URL in the stac-browser configuration.\n     */\n    readonly stacCatalogUrl: string;\n\n    /**\n     * Path to config file for the STAC browser. If not provided, default configuration in the STAC browser\n     * repository is used.\n     */\n    readonly configFilePath?: 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"]}
@@ -3,30 +3,30 @@ import { Construct } from "constructs";
3
3
  import { PgStacDatabase } from "../database";
4
4
  import { CustomLambdaFunctionProps } from "../utils";
5
5
  /**
6
- * Configuration properties for the StacItemLoader construct.
6
+ * Configuration properties for the StacLoader construct.
7
7
  *
8
- * The StacItemLoader is part of a two-phase serverless STAC ingestion pipeline
9
- * that loads STAC items into a pgstac database. This construct creates
10
- * the infrastructure for receiving STAC items from multiple sources:
8
+ * The StacLoader is part of a two-phase serverless STAC ingestion pipeline
9
+ * that loads STAC collections and items into a pgstac database. This construct creates
10
+ * the infrastructure for receiving STAC objects from multiple sources:
11
11
  * 1. SNS messages containing STAC metadata (direct ingestion)
12
- * 2. S3 event notifications for STAC items uploaded to S3 buckets
12
+ * 2. S3 event notifications for STAC objects uploaded to S3 buckets
13
13
  *
14
- * Items from both sources are batched and inserted into PostgreSQL with the pgstac extension.
14
+ * Objects from both sources are batched and inserted into PostgreSQL with the pgstac extension.
15
15
  *
16
16
  * @example
17
- * const loader = new StacItemLoader(this, 'ItemLoader', {
17
+ * const loader = new StacLoader(this, 'StacLoader', {
18
18
  * pgstacDb: database,
19
19
  * batchSize: 1000,
20
20
  * maxBatchingWindowMinutes: 1,
21
21
  * lambdaTimeoutSeconds: 300
22
22
  * });
23
23
  */
24
- export interface StacItemLoaderProps {
24
+ export interface StacLoaderProps {
25
25
  /**
26
- * The PgSTAC database instance to load items into.
26
+ * The PgSTAC database instance to load data into.
27
27
  *
28
28
  * This database must have the pgstac extension installed and be properly
29
- * configured with collections before items can be loaded. The loader will
29
+ * configured with collections before objects can be loaded. The loader will
30
30
  * use AWS Secrets Manager to securely access database credentials.
31
31
  */
32
32
  readonly pgstacDb: PgStacDatabase;
@@ -52,7 +52,7 @@ export interface StacItemLoaderProps {
52
52
  * The timeout for the item load lambda in seconds.
53
53
  *
54
54
  * This should accommodate the time needed to process up to `batchSize`
55
- * items and perform database insertions. The SQS visibility timeout
55
+ * objects and perform database insertions. The SQS visibility timeout
56
56
  * will be set to this value plus 10 seconds.
57
57
  *
58
58
  * @default 300
@@ -62,7 +62,7 @@ export interface StacItemLoaderProps {
62
62
  * Memory size for the lambda function in MB.
63
63
  *
64
64
  * Higher memory allocation may improve performance when processing
65
- * large batches of STAC items, especially for memory-intensive
65
+ * large batches of STAC objects, especially for memory-intensive
66
66
  * database operations.
67
67
  *
68
68
  * @default 1024
@@ -71,7 +71,7 @@ export interface StacItemLoaderProps {
71
71
  /**
72
72
  * SQS batch size for lambda event source.
73
73
  *
74
- * This determines the maximum number of STAC items that will be
74
+ * This determines the maximum number of STAC objects that will be
75
75
  * processed together in a single lambda invocation. Larger batch
76
76
  * sizes improve database insertion efficiency but require more
77
77
  * memory and longer processing time.
@@ -88,19 +88,19 @@ export interface StacItemLoaderProps {
88
88
  * Maximum batching window in minutes.
89
89
  *
90
90
  * Even if the batch size isn't reached, the lambda will be triggered
91
- * after this time period to ensure timely processing of items.
92
- * This prevents items from waiting indefinitely in low-volume scenarios.
91
+ * after this time period to ensure timely processing of objects.
92
+ * This prevents objects from waiting indefinitely in low-volume scenarios.
93
93
  *
94
94
  * **Important**: This timeout works in conjunction with batchSize - SQS
95
95
  * will trigger the Lambda when EITHER the batch size is reached OR this
96
- * time window expires, ensuring items are processed in a timely manner
96
+ * time window expires, ensuring objects are processed in a timely manner
97
97
  * regardless of volume.
98
98
  *
99
99
  * @default 1
100
100
  */
101
101
  readonly maxBatchingWindowMinutes?: number;
102
102
  /**
103
- * Maximum concurrent executions for the StacItemLoader Lambda function
103
+ * Maximum concurrent executions for the StacLoader Lambda function
104
104
  *
105
105
  * This limit will be applied to the Lambda function and will control how
106
106
  * many concurrent batches will be released from the SQS queue.
@@ -129,35 +129,35 @@ export interface StacItemLoaderProps {
129
129
  readonly lambdaFunctionOptions?: CustomLambdaFunctionProps;
130
130
  }
131
131
  /**
132
- * AWS CDK Construct for STAC Item Loading Infrastructure
132
+ * AWS CDK Construct for STAC Object Loading Infrastructure
133
133
  *
134
- * The StacItemLoader creates a serverless, event-driven system for loading
135
- * STAC (SpatioTemporal Asset Catalog) items into a PostgreSQL database with
134
+ * The StacLoader creates a serverless, event-driven system for loading
135
+ * STAC (SpatioTemporal Asset Catalog) objects into a PostgreSQL database with
136
136
  * the pgstac extension. This construct supports multiple ingestion pathways
137
- * for flexible STAC item loading.
137
+ * for flexible STAC object loading.
138
138
  *
139
139
  * ## Architecture Overview
140
140
  *
141
141
  * This construct creates the following AWS resources:
142
- * - **SNS Topic**: Entry point for STAC items and S3 event notifications
142
+ * - **SNS Topic**: Entry point for STAC objects and S3 event notifications
143
143
  * - **SQS Queue**: Buffers and batches messages before processing (60-second visibility timeout)
144
144
  * - **Dead Letter Queue**: Captures failed loading attempts after 5 retries
145
- * - **Lambda Function**: Python function that processes batches and inserts items into pgstac
145
+ * - **Lambda Function**: Python function that processes batches and inserts objects into pgstac
146
146
  *
147
147
  * ## Data Flow
148
148
  *
149
149
  * The loader supports two primary data ingestion patterns:
150
150
  *
151
- * ### Direct STAC Item Publishing
152
- * 1. STAC items (JSON) are published directly to the SNS topic in message bodies
153
- * 2. The SQS queue collects messages and batches them (up to {batchSize} items or 1 minute window)
154
- * 3. The Lambda function receives batches, validates items, and inserts into pgstac
151
+ * ### Direct STAC Object Publishing
152
+ * 1. STAC objects (JSON) are published directly to the SNS topic in message bodies
153
+ * 2. The SQS queue collects messages and batches them (up to {batchSize} objects or 1 minute window)
154
+ * 3. The Lambda function receives batches, validates objects, and inserts into pgstac
155
155
  *
156
156
  * ### S3 Event-Driven Loading
157
157
  * 1. An S3 bucket is configured to send notifications to the SNS topic when json files are created
158
- * 2. STAC items are uploaded to S3 buckets as JSON/GeoJSON files
159
- * 3. S3 event notifications are sent to the SNS topic when items are uploaded
160
- * 4. The Lambda function receives S3 events in the SQS message batch, fetches items from S3, and loads into pgstac
158
+ * 2. STAC objects are uploaded to S3 buckets as JSON/GeoJSON files
159
+ * 3. S3 event notifications are sent to the SNS topic when objects are uploaded
160
+ * 4. The Lambda function receives S3 events in the SQS message batch, fetches objects from S3, and loads into pgstac
161
161
  *
162
162
  * ## Batching Behavior
163
163
  *
@@ -168,7 +168,7 @@ export interface StacItemLoaderProps {
168
168
  * triggers after `maxBatchingWindow` minutes (default: 1 minute)
169
169
  * - **Trigger Condition**: Lambda executes when EITHER condition is met first
170
170
  * - **Concurrency**: Limited to `maxConcurrency` concurrent executions to prevent database overload
171
- * - **Partial Failures**: Uses `reportBatchItemFailures` to retry only failed items
171
+ * - **Partial Failures**: Uses `reportBatchItemFailures` to retry only failed objects
172
172
  *
173
173
  * This approach balances throughput (larger batches = fewer database connections)
174
174
  * with latency (time-based triggers prevent indefinite waiting).
@@ -177,13 +177,13 @@ export interface StacItemLoaderProps {
177
177
  *
178
178
  * Failed messages are sent to the dead letter queue after 5 processing attempts.
179
179
  * **Important**: This construct provides NO automated handling of dead letter queue
180
- * messages - monitoring, inspection, and reprocessing of failed items is the
180
+ * messages - monitoring, inspection, and reprocessing of failed objects is the
181
181
  * responsibility of the implementing application.
182
182
  *
183
183
  * Consider implementing:
184
184
  * - CloudWatch alarms on dead letter queue depth
185
185
  * - Manual or automated reprocessing workflows
186
- * - Logging and alerting for failed items
186
+ * - Logging and alerting for failed objects
187
187
  * - Regular cleanup of old dead letter messages (14-day retention)
188
188
  *
189
189
  * ## Operational Characteristics
@@ -197,7 +197,7 @@ export interface StacItemLoaderProps {
197
197
  * ## Prerequisites
198
198
  *
199
199
  * Before using this construct, ensure:
200
- * - The pgstac database has collections loaded (items require existing collection IDs)
200
+ * - The pgstac database has collections loaded (objects require existing collection IDs)
201
201
  * - Database credentials are stored in AWS Secrets Manager
202
202
  * - The pgstac extension is properly installed and configured
203
203
  *
@@ -209,26 +209,39 @@ export interface StacItemLoaderProps {
209
209
  * pgstacVersion: '0.9.5'
210
210
  * });
211
211
  *
212
- * // Create item loader
213
- * const loader = new StacItemLoader(this, 'ItemLoader', {
212
+ * // Create Object loader
213
+ * const loader = new StacLoader(this, 'StacLoader', {
214
214
  * pgstacDb: database,
215
- * batchSize: 1000, // Process up to 1000 items per batch
215
+ * batchSize: 1000, // Process up to 1000 objects per batch
216
216
  * maxBatchingWindowMinutes: 1, // Wait max 1 minute to fill batch
217
217
  * lambdaTimeoutSeconds: 300 // Allow up to 300 seconds for database operations
218
218
  * });
219
219
  *
220
- * // The topic ARN can be used by other services to publish items
220
+ * // The topic ARN can be used by other services to publish objects
221
221
  * new CfnOutput(this, 'LoaderTopicArn', {
222
222
  * value: loader.topic.topicArn
223
223
  * });
224
224
  * ```
225
225
  *
226
- * ## Direct Item Publishing
226
+ * ## Direct Object Publishing
227
227
  *
228
- * External services can publish STAC items directly to the topic:
228
+ * External services can publish STAC objects directly to the topic:
229
229
  *
230
230
  * ```bash
231
- * aws sns publish --topic-arn $ITEM_LOAD_TOPIC --message '{
231
+ * aws sns publish --topic-arn $STAC_LOAD_TOPIC --message '{
232
+ * "id": "example-collection",
233
+ * "type": "Collection",
234
+ * "title": "Example Collection",
235
+ * "description": "An example collection",
236
+ * "license": "proprietary",
237
+ * "extent": {
238
+ * "spatial": {"bbox": [[-180, -90, 180, 90]]},
239
+ * "temporal": {"interval": [[null, null]]},
240
+ * },
241
+ * "stac_version": "1.1.0",
242
+ * }'
243
+ *
244
+ * aws sns publish --topic-arn $STAC_LOAD_TOPIC --message '{
232
245
  * "type": "Feature",
233
246
  * "stac_version": "1.0.0",
234
247
  * "id": "example-item",
@@ -236,12 +249,14 @@ export interface StacItemLoaderProps {
236
249
  * "geometry": {"type": "Polygon", "coordinates": [...]},
237
250
  * "collection": "example-collection"
238
251
  * }'
252
+ *
253
+ *
239
254
  * ```
240
255
  *
241
256
  * ## S3 Event Configuration
242
257
  *
243
258
  * To enable S3 event-driven loading, configure S3 bucket notifications to send
244
- * events to the SNS topic when STAC items (.json or .geojson files) are uploaded:
259
+ * events to the SNS topic when STAC objects (.json or .geojson files) are uploaded:
245
260
  *
246
261
  * ```typescript
247
262
  * // Configure S3 bucket to send notifications to the loader topic
@@ -258,16 +273,16 @@ export interface StacItemLoaderProps {
258
273
  * );
259
274
  * ```
260
275
  *
261
- * When STAC items are uploaded to the configured S3 bucket, the loader will:
276
+ * When STAC objects are uploaded to the configured S3 bucket, the loader will:
262
277
  * 1. Receive S3 event notifications via SNS
263
- * 2. Fetch the STAC item JSON from S3
264
- * 3. Validate and load the item into the pgstac database
278
+ * 2. Fetch the STAC JSON from S3
279
+ * 3. Validate and load the objects into the pgstac database
265
280
  *
266
281
  * ## Monitoring and Troubleshooting
267
282
  *
268
283
  * - Monitor Lambda logs: `/aws/lambda/{FunctionName}`
269
- * - **Dead Letter Queue**: Check for failed items - **no automated handling provided**
270
- * - Use batch item failure reporting for partial batch processing
284
+ * - **Dead Letter Queue**: Check for failed objects - **no automated handling provided**
285
+ * - Use batch objects failure reporting for partial batch processing
271
286
  * - CloudWatch metrics available for queue depth and Lambda performance
272
287
  *
273
288
  * ### Dead Letter Queue Management
@@ -289,13 +304,13 @@ export interface StacItemLoaderProps {
289
304
  * ```
290
305
  *
291
306
  */
292
- export declare class StacItemLoader extends Construct {
307
+ export declare class StacLoader extends Construct {
293
308
  /**
294
- * The SNS topic that receives STAC items and S3 event notifications for loading.
309
+ * The SNS topic that receives STAC objects and S3 event notifications for loading.
295
310
  *
296
311
  * This topic serves as the entry point for two types of events:
297
- * 1. Direct STAC item JSON documents published by external services
298
- * 2. S3 event notifications when STAC items are uploaded to configured buckets
312
+ * 1. Direct STAC JSON documents published by external services
313
+ * 2. S3 event notifications when STAC objects are uploaded to configured buckets
299
314
  *
300
315
  * The topic fans out to the SQS queue for batched processing.
301
316
  */
@@ -303,14 +318,14 @@ export declare class StacItemLoader extends Construct {
303
318
  /**
304
319
  * The SQS queue that buffers messages before processing.
305
320
  *
306
- * This queue collects both direct STAC items from SNS and S3 event
321
+ * This queue collects both direct STAC objects from SNS and S3 event
307
322
  * notifications, batching them for efficient database operations.
308
323
  * Configured with a visibility timeout that accommodates Lambda
309
324
  * processing time plus buffer.
310
325
  */
311
326
  readonly queue: sqs.Queue;
312
327
  /**
313
- * Dead letter queue for failed item loading attempts.
328
+ * Dead letter queue for failed objects loading attempts.
314
329
  *
315
330
  * Messages that fail processing after 5 attempts are sent here
316
331
  * for inspection and potential replay. Retains messages for 14 days
@@ -326,16 +341,27 @@ export declare class StacItemLoader extends Construct {
326
341
  */
327
342
  readonly deadLetterQueue: sqs.Queue;
328
343
  /**
329
- * The Lambda function that loads STAC items into the pgstac database.
344
+ * The Lambda function that loads STAC objects into the pgstac database.
330
345
  *
331
346
  * This Python function receives batches of messages from SQS and processes
332
347
  * them based on their type:
333
- * - Direct STAC items: Validates and loads directly into pgstac
334
- * - S3 events: Fetches STAC items from S3, validates, and loads into pgstac
348
+ * - Direct STAC objects: Validates and loads directly into pgstac
349
+ * - S3 events: Fetches STAC JSON from S3, validates, and loads into pgstac
335
350
  *
336
351
  * The function connects to PostgreSQL using credentials from Secrets Manager
337
352
  * and uses pypgstac for efficient database operations.
338
353
  */
339
354
  readonly lambdaFunction: lambda.Function;
340
- constructor(scope: Construct, id: string, props: StacItemLoaderProps);
355
+ constructor(scope: Construct, id: string, props: StacLoaderProps);
356
+ }
357
+ /**
358
+ * @deprecated Use StacLoader instead. StacItemLoader will be removed in a future version.
359
+ */
360
+ export declare class StacItemLoader extends StacLoader {
361
+ constructor(scope: Construct, id: string, props: StacLoaderProps);
362
+ }
363
+ /**
364
+ * @deprecated Use StacLoaderProps instead. StacItemLoaderProps will be removed in a future version.
365
+ */
366
+ export interface StacItemLoaderProps extends StacLoaderProps {
341
367
  }