@aws-cdk/aws-s3tables-alpha
Version:
CDK Constructs for S3 Tables
261 lines • 37.2 kB
JavaScript
;
var __esDecorate = (this && this.__esDecorate) || function (ctor, descriptorIn, decorators, contextIn, initializers, extraInitializers) {
function accept(f) { if (f !== void 0 && typeof f !== "function") throw new TypeError("Function expected"); return f; }
var kind = contextIn.kind, key = kind === "getter" ? "get" : kind === "setter" ? "set" : "value";
var target = !descriptorIn && ctor ? contextIn["static"] ? ctor : ctor.prototype : null;
var descriptor = descriptorIn || (target ? Object.getOwnPropertyDescriptor(target, contextIn.name) : {});
var _, done = false;
for (var i = decorators.length - 1; i >= 0; i--) {
var context = {};
for (var p in contextIn) context[p] = p === "access" ? {} : contextIn[p];
for (var p in contextIn.access) context.access[p] = contextIn.access[p];
context.addInitializer = function (f) { if (done) throw new TypeError("Cannot add initializers after decoration has completed"); extraInitializers.push(accept(f || null)); };
var result = (0, decorators[i])(kind === "accessor" ? { get: descriptor.get, set: descriptor.set } : descriptor[key], context);
if (kind === "accessor") {
if (result === void 0) continue;
if (result === null || typeof result !== "object") throw new TypeError("Object expected");
if (_ = accept(result.get)) descriptor.get = _;
if (_ = accept(result.set)) descriptor.set = _;
if (_ = accept(result.init)) initializers.unshift(_);
}
else if (_ = accept(result)) {
if (kind === "field") initializers.unshift(_);
else descriptor[key] = _;
}
}
if (target) Object.defineProperty(target, contextIn.name, descriptor);
done = true;
};
var __runInitializers = (this && this.__runInitializers) || function (thisArg, initializers, value) {
var useValue = arguments.length > 2;
for (var i = 0; i < initializers.length; i++) {
value = useValue ? initializers[i].call(thisArg, value) : initializers[i].call(thisArg);
}
return useValue ? value : void 0;
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.Table = exports.Status = exports.OpenTableFormat = void 0;
const jsiiDeprecationWarnings = require("../.warnings.jsii.js");
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
const core_1 = require("aws-cdk-lib/core");
const aws_s3tables_1 = require("aws-cdk-lib/aws-s3tables");
const metadata_resource_1 = require("aws-cdk-lib/core/lib/metadata-resource");
const prop_injectable_1 = require("aws-cdk-lib/core/lib/prop-injectable");
const iam = require("aws-cdk-lib/aws-iam");
const perms = require("./permissions");
const os_1 = require("os");
/**
* Base class for Table implementations.
*/
class TableBase extends core_1.Resource {
addToResourcePolicy(statement) {
if (!this.tablePolicy && this.autoCreatePolicy) {
this.tablePolicy = new aws_s3tables_1.CfnTablePolicy(this, 'DefaultPolicy', {
tableArn: this.tableArn,
resourcePolicy: new iam.PolicyDocument({}),
});
}
if (this.tablePolicy) {
this.tablePolicy.resourcePolicy.addStatements(statement);
return { statementAdded: true, policyDependable: this.tablePolicy };
}
return { statementAdded: false };
}
grantRead(identity) {
return this.grant(identity, perms.TABLE_READ_ACCESS, this.tableArn);
}
grantWrite(identity) {
return this.grant(identity, perms.TABLE_WRITE_ACCESS, this.tableArn);
}
grantReadWrite(identity) {
return this.grant(identity, perms.TABLE_READ_WRITE_ACCESS, this.tableArn);
}
/**
* Grants the given s3tables permissions to the provided principal
* @returns Grant object
*/
grant(grantee, tableActions, resourceArn, ...otherResourceArns) {
const resources = [resourceArn, ...otherResourceArns].filter(arn => arn != undefined);
const grant = iam.Grant.addToPrincipalOrResource({
grantee,
actions: tableActions,
resourceArns: resources,
resource: this,
});
return grant;
}
}
/**
* Supported open table formats.
*/
var OpenTableFormat;
(function (OpenTableFormat) {
/**
* Apache Iceberg table format.
*/
OpenTableFormat["ICEBERG"] = "ICEBERG";
})(OpenTableFormat || (exports.OpenTableFormat = OpenTableFormat = {}));
/**
* Status values for maintenance actions.
*/
var Status;
(function (Status) {
/**
* Enable the maintenance action.
*/
Status["ENABLED"] = "enabled";
/**
* Disable the maintenance action.
*/
Status["DISABLED"] = "disabled";
})(Status || (exports.Status = Status = {}));
/**
* An S3 Table with helpers.
*/
let Table = (() => {
let _classDecorators = [prop_injectable_1.propertyInjectable];
let _classDescriptor;
let _classExtraInitializers = [];
let _classThis;
let _classSuper = TableBase;
var Table = class extends _classSuper {
static { _classThis = this; }
static {
const _metadata = typeof Symbol === "function" && Symbol.metadata ? Object.create(_classSuper[Symbol.metadata] ?? null) : void 0;
__esDecorate(null, _classDescriptor = { value: _classThis }, _classDecorators, { kind: "class", name: _classThis.name, metadata: _metadata }, null, _classExtraInitializers);
Table = _classThis = _classDescriptor.value;
if (_metadata) Object.defineProperty(_classThis, Symbol.metadata, { enumerable: true, configurable: true, writable: true, value: _metadata });
}
static [JSII_RTTI_SYMBOL_1] = { fqn: "@aws-cdk/aws-s3tables-alpha.Table", version: "2.223.0-alpha.0" };
/** Uniquely identifies this class. */
static PROPERTY_INJECTION_ID = '@aws-cdk.aws-s3tables-alpha.Table';
/**
* Defines a Table construct that represents an external table.
*
* @param scope The parent creating construct (usually `this`).
* @param id The construct's name.
* @param attrs A `TableAttributes` object containing the table name and ARN.
*/
static fromTableAttributes(scope, id, attrs) {
try {
jsiiDeprecationWarnings._aws_cdk_aws_s3tables_alpha_TableAttributes(attrs);
}
catch (error) {
if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
Error.captureStackTrace(error, this.fromTableAttributes);
}
throw error;
}
const tableArn = attrs.tableArn;
Table.validateTableName(attrs.tableName);
class Import extends TableBase {
tableName = attrs.tableName;
tableArn = tableArn;
tablePolicy;
autoCreatePolicy = false;
/**
* Exports this from the stack.
*/
export() {
return attrs;
}
}
return new Import(scope, id, {
environmentFromArn: tableArn,
physicalName: tableArn,
});
}
/**
* See https://docs.aws.amazon.com/AmazonS3/latest/userguide/s3-tables-buckets-naming.html
* @param tableName Name of the table
* @throws UnscopedValidationError if any naming errors are detected
*/
static validateTableName(tableName) {
if (tableName == undefined || core_1.Token.isUnresolved(tableName)) {
// the name is a late-bound value, not a defined string, so skip validation
return;
}
const errors = [];
// Length validation
if (tableName.length < 1 || tableName.length > 255) {
errors.push('Table name must be at least 1 and no more than 255 characters');
}
// Character set validation
const illegalCharsetRegEx = /[^a-z0-9_]/;
const allowedEdgeCharsetRegEx = /[a-z0-9]/;
const illegalCharMatch = tableName.match(illegalCharsetRegEx);
if (illegalCharMatch) {
errors.push('Table name must only contain lowercase characters, numbers, and underscores (_)' +
` (offset: ${illegalCharMatch.index})`);
}
// Edge character validation
if (!allowedEdgeCharsetRegEx.test(tableName.charAt(0))) {
errors.push('Table name must start with a lowercase letter or number (offset: 0)');
}
if (!allowedEdgeCharsetRegEx.test(tableName.charAt(tableName.length - 1))) {
errors.push(`Table name must end with a lowercase letter or number (offset: ${tableName.length - 1})`);
}
if (errors.length > 0) {
throw new core_1.UnscopedValidationError(`Invalid S3 table name (value: ${tableName})${os_1.EOL}${errors.join(os_1.EOL)}`);
}
}
/**
* The unique Amazon Resource Name (arn) of this table
*/
tableArn;
/**
* The underlying CfnTable L1 resource
* @internal
*/
_resource;
/**
* The name of this table
*/
tableName;
/**
* The namespace containing this table
*/
namespace;
/**
* The resource policy for this table.
*/
tablePolicy;
autoCreatePolicy = true;
constructor(scope, id, props) {
super(scope, id, {});
try {
jsiiDeprecationWarnings._aws_cdk_aws_s3tables_alpha_TableProps(props);
}
catch (error) {
if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
Error.captureStackTrace(error, Table);
}
throw error;
}
// Enhanced CDK Analytics Telemetry
(0, metadata_resource_1.addConstructMetadata)(this, props);
Table.validateTableName(props.tableName);
this._resource = new aws_s3tables_1.CfnTable(this, id, {
tableName: props.tableName,
openTableFormat: props.openTableFormat,
tableBucketArn: props.namespace.tableBucket.tableBucketArn,
namespace: props.namespace.namespaceName,
compaction: props.compaction,
icebergMetadata: props.icebergMetadata,
snapshotManagement: props.snapshotManagement,
withoutMetadata: props.withoutMetadata ? 'Yes' : undefined,
});
this.namespace = props.namespace;
this.tableName = props.tableName;
this.tableArn = this._resource.attrTableArn;
this._resource.applyRemovalPolicy(props.removalPolicy);
this.node.addDependency(this.namespace);
}
static {
__runInitializers(_classThis, _classExtraInitializers);
}
};
return Table = _classThis;
})();
exports.Table = Table;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"table.js","sourceRoot":"","sources":["table.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Resource,\n  IResource,\n  RemovalPolicy,\n  UnscopedValidationError,\n  Token,\n} from 'aws-cdk-lib/core';\nimport { INamespace } from './namespace';\nimport { CfnTable, CfnTablePolicy } from 'aws-cdk-lib/aws-s3tables';\nimport { addConstructMetadata } from 'aws-cdk-lib/core/lib/metadata-resource';\nimport { propertyInjectable } from 'aws-cdk-lib/core/lib/prop-injectable';\nimport { Construct } from 'constructs';\nimport * as iam from 'aws-cdk-lib/aws-iam';\nimport * as perms from './permissions';\nimport { EOL } from 'os';\n\n/**\n * Represents an S3 Table.\n */\nexport interface ITable extends IResource {\n  /**\n   * The ARN of this table.\n   * @attribute\n   */\n  readonly tableArn: string;\n\n  /**\n   * The name of this table.\n   * @attribute\n   */\n  readonly tableName: string;\n\n  /**\n   * The accountId containing this table.\n   * @attribute\n   */\n  readonly account?: string;\n\n  /**\n   * The region containing this table.\n   * @attribute\n   */\n  readonly region?: string;\n\n  /**\n   * Adds a statement to the resource policy for a principal (i.e.\n   * account/role/service) to perform actions on this table.\n   *\n   * Note that the policy statement may or may not be added to the policy.\n   * For example, when an `ITable` is created from an existing table,\n   * it's not possible to tell whether the table already has a policy\n   * attached, let alone to re-use that policy to add more statements to it.\n   * So it's safest to do nothing in these cases.\n   *\n   * @param statement the policy statement to be added to the table's\n   * policy.\n   * @returns metadata about the execution of this method. If the policy\n   * was not added, the value of `statementAdded` will be `false`. You\n   * should always check this value to make sure that the operation was\n   * actually carried out. Otherwise, synthesis and deploy will terminate\n   * silently, which may be confusing.\n   */\n  addToResourcePolicy(statement: iam.PolicyStatement): iam.AddToResourcePolicyResult;\n\n  /**\n   * Grant read permissions for this table to an IAM principal (Role/Group/User).\n   *\n   * If the parent TableBucket of this table has encryption,\n   * you should grant kms:Decrypt permission to use this key to the same principal.\n   *\n   * @param identity The principal to allow read permissions to\n   */\n  grantRead(identity: iam.IGrantable): iam.Grant;\n\n  /**\n   * Grant write permissions for this table to an IAM principal (Role/Group/User).\n   *\n   * If the parent TableBucket of this table has encryption,\n   * you should grant kms:GenerateDataKey and kms:Decrypt permission\n   * to use this key to the same principal.\n   *\n   * @param identity The principal to allow write permissions to\n   */\n  grantWrite(identity: iam.IGrantable): iam.Grant;\n\n  /**\n   * Grant read and write permissions for this table to an IAM principal (Role/Group/User).\n   *\n   * If the parent TableBucket of this table has encryption,\n   * you should grant kms:GenerateDataKey and kms:Decrypt permission\n   * to use this key to the same principal.\n   *\n   * @param identity The principal to allow read and write permissions to\n   */\n  grantReadWrite(identity: iam.IGrantable): iam.Grant;\n}\n\n/**\n * Base class for Table implementations.\n */\nabstract class TableBase extends Resource implements ITable {\n  public abstract readonly tableName: string;\n  public abstract readonly tableArn: string;\n\n  /**\n   * The resource policy associated with this table.\n   *\n   * If `autoCreatePolicy` is true, a `TablePolicy` will be created upon the\n   * first call to addToResourcePolicy(s).\n   */\n  public abstract tablePolicy?: CfnTablePolicy;\n\n  /**\n   * Indicates if a table resource policy should automatically created upon\n   * the first call to `addToResourcePolicy`.\n   */\n  protected abstract autoCreatePolicy: boolean;\n\n  public addToResourcePolicy(\n    statement: iam.PolicyStatement,\n  ): iam.AddToResourcePolicyResult {\n    if (!this.tablePolicy && this.autoCreatePolicy) {\n      this.tablePolicy = new CfnTablePolicy(this, 'DefaultPolicy', {\n        tableArn: this.tableArn,\n        resourcePolicy: new iam.PolicyDocument({}),\n      });\n    }\n\n    if (this.tablePolicy) {\n      this.tablePolicy.resourcePolicy.addStatements(statement);\n      return { statementAdded: true, policyDependable: this.tablePolicy };\n    }\n\n    return { statementAdded: false };\n  }\n\n  public grantRead(identity: iam.IGrantable) {\n    return this.grant(\n      identity,\n      perms.TABLE_READ_ACCESS,\n      this.tableArn,\n    );\n  }\n\n  public grantWrite(identity: iam.IGrantable) {\n    return this.grant(\n      identity,\n      perms.TABLE_WRITE_ACCESS,\n      this.tableArn,\n    );\n  }\n\n  public grantReadWrite(identity: iam.IGrantable) {\n    return this.grant(\n      identity,\n      perms.TABLE_READ_WRITE_ACCESS,\n      this.tableArn,\n    );\n  }\n\n  /**\n   * Grants the given s3tables permissions to the provided principal\n   * @returns Grant object\n   */\n  private grant(\n    grantee: iam.IGrantable,\n    tableActions: string[],\n    resourceArn: string,\n    ...otherResourceArns: (string | undefined)[]) {\n    const resources = [resourceArn, ...otherResourceArns].filter(arn => arn != undefined);\n\n    const grant = iam.Grant.addToPrincipalOrResource({\n      grantee,\n      actions: tableActions,\n      resourceArns: resources,\n      resource: this,\n    });\n\n    return grant;\n  }\n}\n\n/**\n * Properties for creating a new S3 Table.\n */\nexport interface TableProps {\n  /**\n   * Name of this table, unique within the namespace\n   */\n  readonly tableName: string;\n  /**\n   * The namespace under which this table is created\n   */\n  readonly namespace: INamespace;\n  /**\n   * Format of this table. Currently, the only supported value is OpenTableFormat.ICEBERG.\n   */\n  readonly openTableFormat: OpenTableFormat;\n  /**\n   * Settings governing the Compaction maintenance action.\n   * @default Amazon S3 selects the best compaction strategy based on your table sort order.\n   * @see https://docs.aws.amazon.com/AmazonS3/latest/userguide/s3-tables-maintenance.html\n   */\n  readonly compaction?: CompactionProperty;\n  /**\n   * Contains details about the metadata for an Iceberg table.\n   * @default table is created without any metadata\n   */\n  readonly icebergMetadata?: IcebergMetadataProperty;\n  /**\n   * Contains details about the snapshot management settings for an Iceberg table.\n   * @default enabled: MinimumSnapshots is 1 by default and MaximumSnapshotAge is 120 hours by default.\n   */\n  readonly snapshotManagement?: SnapshotManagementProperty;\n  /**\n   * Controls what happens to this table it it stoped being managed by cloudformation.\n   *\n   * @default RETAIN\n   */\n  readonly removalPolicy?: RemovalPolicy;\n  /**\n   * If true, indicates that you don't want to specify a schema for the table.\n   *\n   * This property is mutually exclusive to 'IcebergMetadata'.\n   *\n   * @default false\n   */\n  readonly withoutMetadata?: boolean;\n}\n\n/**\n * Supported open table formats.\n */\nexport enum OpenTableFormat {\n  /**\n   * Apache Iceberg table format.\n   */\n  ICEBERG = 'ICEBERG',\n}\n\n/**\n * Settings governing the Compaction maintenance action.\n *\n * @default - No compaction settings\n */\nexport interface CompactionProperty {\n  /**\n   * Status of the compaction maintenance action.\n   */\n  readonly status: Status;\n  /**\n   * Target file size in megabytes for compaction.\n   */\n  readonly targetFileSizeMb: number;\n}\n\n/**\n * Status values for maintenance actions.\n */\nexport enum Status {\n  /**\n   * Enable the maintenance action.\n   */\n  ENABLED = 'enabled',\n  /**\n   * Disable the maintenance action.\n   */\n  DISABLED = 'disabled',\n}\n\n/**\n * Contains details about the metadata for an Iceberg table.\n * @see http://docs.aws.amazon.com/AWSCloudFormation/latest/UserGuide/aws-properties-s3tables-table-icebergmetadata.html\n */\nexport interface IcebergMetadataProperty {\n  /**\n   * Contains details about the schema for an Iceberg table.\n   *\n   * @see http://docs.aws.amazon.com/AWSCloudFormation/latest/UserGuide/aws-properties-s3tables-table-icebergmetadata.html#cfn-s3tables-table-icebergmetadata-icebergschema\n   */\n  readonly icebergSchema: IcebergSchemaProperty;\n}\n\n/**\n * Contains details about the schema for an Iceberg table.\n */\nexport interface IcebergSchemaProperty {\n  /**\n   * Contains details about the schema for an Iceberg table.\n   */\n  readonly schemaFieldList: SchemaFieldProperty[];\n}\n\n/**\n * Contains details about a schema field.\n */\nexport interface SchemaFieldProperty {\n  /**\n   * The name of the field.\n   */\n  readonly name: string;\n\n  /**\n   * A Boolean value that specifies whether values are required for each row in this field.\n   *\n   * By default, this is `false` and null values are allowed in the field. If this is `true`, the field does not allow null values.\n   *\n   * @default false\n   */\n  readonly required?: boolean;\n\n  /**\n   * The field type.\n   *\n   * S3 Tables supports all Apache Iceberg primitive types. For more information, see the [Apache Iceberg documentation](https://docs.aws.amazon.com/https://iceberg.apache.org/spec/#primitive-types).\n   */\n  readonly type: string;\n}\n\n/**\n * Contains details about the snapshot management settings for an Iceberg table.\n *\n * A snapshot is expired when it exceeds MinSnapshotsToKeep and MaxSnapshotAgeHours.\n *\n * @default - No snapshot management settings\n */\nexport interface SnapshotManagementProperty {\n  /**\n   * The maximum age of a snapshot before it can be expired.\n   *\n   * @default - No maximum age\n   */\n  readonly maxSnapshotAgeHours?: number;\n\n  /**\n   * The minimum number of snapshots to keep.\n   *\n   * @default - No minimum number\n   */\n  readonly minSnapshotsToKeep?: number;\n\n  /**\n   * Indicates whether the SnapshotManagement maintenance action is enabled.\n   *\n   * @default - Not specified\n   */\n  readonly status?: Status;\n}\n\n/**\n * A reference to a table outside this stack\n *\n * The tableName, region, and account can be provided explicitly\n * or will be inferred from the tableArn\n */\nexport interface TableAttributes {\n  /**\n   * Name of this table\n   */\n  readonly tableName: string;\n\n  /**\n   * The table's ARN.\n   */\n  readonly tableArn: string;\n}\n\n/**\n * An S3 Table with helpers.\n */\n@propertyInjectable\nexport class Table extends TableBase {\n  /** Uniquely identifies this class. */\n  public static readonly PROPERTY_INJECTION_ID: string = '@aws-cdk.aws-s3tables-alpha.Table';\n\n  /**\n   * Defines a Table construct that represents an external table.\n   *\n   * @param scope The parent creating construct (usually `this`).\n   * @param id The construct's name.\n   * @param attrs A `TableAttributes` object containing the table name and ARN.\n   */\n  public static fromTableAttributes(\n    scope: Construct,\n    id: string,\n    attrs: TableAttributes,\n  ): ITable {\n    const tableArn = attrs.tableArn;\n    Table.validateTableName(attrs.tableName);\n\n    class Import extends TableBase {\n      public readonly tableName = attrs.tableName;\n      public readonly tableArn = tableArn;\n      public readonly tablePolicy?: CfnTablePolicy;\n      protected autoCreatePolicy: boolean = false;\n\n      /**\n       * Exports this  from the stack.\n       */\n      public export() {\n        return attrs;\n      }\n    }\n\n    return new Import(scope, id, {\n      environmentFromArn: tableArn,\n      physicalName: tableArn,\n    });\n  }\n\n  /**\n   * See https://docs.aws.amazon.com/AmazonS3/latest/userguide/s3-tables-buckets-naming.html\n   * @param tableName Name of the table\n   * @throws UnscopedValidationError if any naming errors are detected\n   */\n  public static validateTableName(tableName: string) {\n    if (tableName == undefined || Token.isUnresolved(tableName)) {\n      // the name is a late-bound value, not a defined string, so skip validation\n      return;\n    }\n\n    const errors: string[] = [];\n\n    // Length validation\n    if (tableName.length < 1 || tableName.length > 255) {\n      errors.push(\n        'Table name must be at least 1 and no more than 255 characters',\n      );\n    }\n\n    // Character set validation\n    const illegalCharsetRegEx = /[^a-z0-9_]/;\n    const allowedEdgeCharsetRegEx = /[a-z0-9]/;\n\n    const illegalCharMatch = tableName.match(illegalCharsetRegEx);\n    if (illegalCharMatch) {\n      errors.push(\n        'Table name must only contain lowercase characters, numbers, and underscores (_)' +\n          ` (offset: ${illegalCharMatch.index})`,\n      );\n    }\n\n    // Edge character validation\n    if (!allowedEdgeCharsetRegEx.test(tableName.charAt(0))) {\n      errors.push(\n        'Table name must start with a lowercase letter or number (offset: 0)',\n      );\n    }\n    if (!allowedEdgeCharsetRegEx.test(tableName.charAt(tableName.length - 1))) {\n      errors.push(\n        `Table name must end with a lowercase letter or number (offset: ${\n          tableName.length - 1\n        })`,\n      );\n    }\n\n    if (errors.length > 0) {\n      throw new UnscopedValidationError(\n        `Invalid S3 table name (value: ${tableName})${EOL}${errors.join(EOL)}`,\n      );\n    }\n  }\n\n  /**\n   * The unique Amazon Resource Name (arn) of this table\n   */\n  public readonly tableArn: string;\n\n  /**\n   * The underlying CfnTable L1 resource\n   * @internal\n   */\n  private readonly _resource: CfnTable;\n\n  /**\n   * The name of this table\n   */\n  public readonly tableName: string;\n\n  /**\n   * The namespace containing this table\n   */\n  public readonly namespace: INamespace;\n\n  /**\n   * The resource policy for this table.\n   */\n  public readonly tablePolicy?: CfnTablePolicy;\n\n  protected autoCreatePolicy: boolean = true;\n\n  constructor(scope: Construct, id: string, props: TableProps) {\n    super(scope, id, {});\n\n    // Enhanced CDK Analytics Telemetry\n    addConstructMetadata(this, props);\n\n    Table.validateTableName(props.tableName);\n\n    this._resource = new CfnTable(this, id, {\n      tableName: props.tableName,\n      openTableFormat: props.openTableFormat,\n      tableBucketArn: props.namespace.tableBucket.tableBucketArn,\n      namespace: props.namespace.namespaceName,\n      compaction: props.compaction,\n      icebergMetadata: props.icebergMetadata,\n      snapshotManagement: props.snapshotManagement,\n      withoutMetadata: props.withoutMetadata ? 'Yes' : undefined,\n    });\n\n    this.namespace = props.namespace;\n    this.tableName = props.tableName;\n    this.tableArn = this._resource.attrTableArn;\n    this._resource.applyRemovalPolicy(props.removalPolicy);\n    this.node.addDependency(this.namespace);\n  }\n}\n"]}