UNPKG

@aws-cdk/aws-s3tables-alpha

Version:

CDK Constructs for S3 Tables

261 lines 37.2 kB
"use strict"; 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"]}