@chaim-tools/cdk-lib 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (39) hide show
  1. package/README.md +238 -0
  2. package/lib/binders/base-chaim-binder.d.ts +144 -0
  3. package/lib/binders/base-chaim-binder.js +532 -0
  4. package/lib/binders/chaim-dynamodb-binder.d.ts +95 -0
  5. package/lib/binders/chaim-dynamodb-binder.js +292 -0
  6. package/lib/config/chaim-endpoints.d.ts +47 -0
  7. package/lib/config/chaim-endpoints.js +51 -0
  8. package/lib/index.d.ts +15 -0
  9. package/lib/index.js +43 -0
  10. package/lib/lambda-handler/.test-temp/snapshot.json +1 -0
  11. package/lib/lambda-handler/handler.js +513 -0
  12. package/lib/lambda-handler/handler.test.ts +365 -0
  13. package/lib/lambda-handler/package-lock.json +1223 -0
  14. package/lib/lambda-handler/package.json +14 -0
  15. package/lib/services/ingestion-service.d.ts +50 -0
  16. package/lib/services/ingestion-service.js +81 -0
  17. package/lib/services/os-cache-paths.d.ts +52 -0
  18. package/lib/services/os-cache-paths.js +123 -0
  19. package/lib/services/schema-service.d.ts +11 -0
  20. package/lib/services/schema-service.js +67 -0
  21. package/lib/services/snapshot-cleanup.d.ts +78 -0
  22. package/lib/services/snapshot-cleanup.js +220 -0
  23. package/lib/types/base-binder-props.d.ts +32 -0
  24. package/lib/types/base-binder-props.js +17 -0
  25. package/lib/types/credentials.d.ts +57 -0
  26. package/lib/types/credentials.js +83 -0
  27. package/lib/types/data-store-metadata.d.ts +67 -0
  28. package/lib/types/data-store-metadata.js +4 -0
  29. package/lib/types/failure-mode.d.ts +16 -0
  30. package/lib/types/failure-mode.js +21 -0
  31. package/lib/types/ingest-contract.d.ts +110 -0
  32. package/lib/types/ingest-contract.js +12 -0
  33. package/lib/types/snapshot-cache-policy.d.ts +52 -0
  34. package/lib/types/snapshot-cache-policy.js +57 -0
  35. package/lib/types/snapshot-payload.d.ts +245 -0
  36. package/lib/types/snapshot-payload.js +3 -0
  37. package/lib/types/table-binding-config.d.ts +43 -0
  38. package/lib/types/table-binding-config.js +57 -0
  39. package/package.json +67 -0
@@ -0,0 +1,95 @@
1
+ import * as dynamodb from 'aws-cdk-lib/aws-dynamodb';
2
+ import { Construct } from 'constructs';
3
+ import { BaseChaimBinder } from './base-chaim-binder';
4
+ import { BaseBinderProps } from '../types/base-binder-props';
5
+ import { DynamoDBMetadata } from '../types/data-store-metadata';
6
+ /**
7
+ * Properties for ChaimDynamoDBBinder construct.
8
+ */
9
+ export interface ChaimDynamoDBBinderProps extends BaseBinderProps {
10
+ /** DynamoDB table to bind with the schema */
11
+ table: dynamodb.ITable;
12
+ }
13
+ /**
14
+ * CDK construct for binding a .bprint schema to a DynamoDB table.
15
+ *
16
+ * Publishes schema and table metadata to Chaim SaaS platform via
17
+ * S3 presigned upload and snapshot-ref commit.
18
+ *
19
+ * @example
20
+ * ```typescript
21
+ * import { ChaimDynamoDBBinder, ChaimCredentials, FailureMode } from '@chaim-tools/cdk-lib';
22
+ *
23
+ * const table = new dynamodb.Table(this, 'UsersTable', {
24
+ * partitionKey: { name: 'userId', type: dynamodb.AttributeType.STRING },
25
+ * });
26
+ *
27
+ * // failureMode defaults to BEST_EFFORT
28
+ * new ChaimDynamoDBBinder(this, 'UserSchema', {
29
+ * schemaPath: './schemas/user.bprint',
30
+ * table,
31
+ * appId: 'my-app',
32
+ * credentials: ChaimCredentials.fromSecretsManager('chaim/api-credentials'),
33
+ * failureMode: FailureMode.STRICT, // Optional - rolls back on failure
34
+ * });
35
+ * ```
36
+ */
37
+ export declare class ChaimDynamoDBBinder extends BaseChaimBinder {
38
+ /** The DynamoDB table being bound */
39
+ readonly table: dynamodb.ITable;
40
+ /** DynamoDB-specific metadata */
41
+ readonly dynamoDBMetadata: DynamoDBMetadata;
42
+ private readonly dynamoDBProps;
43
+ constructor(scope: Construct, id: string, props: ChaimDynamoDBBinderProps);
44
+ /**
45
+ * Override to resolve the actual table name when possible.
46
+ *
47
+ * Uses stack.resolve() to convert CDK tokens to actual values for explicit table names
48
+ * (e.g., 'acme-product-catalog') and dynamic names (e.g., `${stack.stackName}-orders`).
49
+ * Falls back to construct node ID for auto-generated names or cross-stack references.
50
+ *
51
+ * Note: This is called from BaseChaimBinder constructor before this.table is set,
52
+ * so we access the table from baseProps instead.
53
+ */
54
+ protected getResourceName(): string;
55
+ /**
56
+ * Extract DynamoDB table metadata.
57
+ */
58
+ protected extractMetadata(): DynamoDBMetadata;
59
+ /**
60
+ * Validate that the table is a concrete DynamoDB Table construct.
61
+ *
62
+ * Note: We use duck typing for instanceof check to handle cross-package
63
+ * type compatibility issues in monorepo setups (e.g., pnpm with isolated node_modules).
64
+ */
65
+ private validateTable;
66
+ /**
67
+ * Get the underlying CloudFormation table resource.
68
+ * Uses duck typing for cross-package compatibility.
69
+ */
70
+ private getCfnTable;
71
+ /**
72
+ * Extract partition key and sort key from key schema.
73
+ */
74
+ private extractKeySchema;
75
+ /**
76
+ * Extract Global Secondary Index metadata.
77
+ */
78
+ private extractGSIs;
79
+ /**
80
+ * Extract Local Secondary Index metadata.
81
+ */
82
+ private extractLSIs;
83
+ /**
84
+ * Extract TTL attribute name.
85
+ */
86
+ private extractTTL;
87
+ /**
88
+ * Extract stream configuration.
89
+ */
90
+ private extractStreamInfo;
91
+ /**
92
+ * Extract billing mode.
93
+ */
94
+ private extractBillingMode;
95
+ }
@@ -0,0 +1,292 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || function (mod) {
19
+ if (mod && mod.__esModule) return mod;
20
+ var result = {};
21
+ if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
22
+ __setModuleDefault(result, mod);
23
+ return result;
24
+ };
25
+ Object.defineProperty(exports, "__esModule", { value: true });
26
+ exports.ChaimDynamoDBBinder = void 0;
27
+ const cdk = __importStar(require("aws-cdk-lib"));
28
+ const dynamodb = __importStar(require("aws-cdk-lib/aws-dynamodb"));
29
+ const base_chaim_binder_1 = require("./base-chaim-binder");
30
+ /**
31
+ * CDK construct for binding a .bprint schema to a DynamoDB table.
32
+ *
33
+ * Publishes schema and table metadata to Chaim SaaS platform via
34
+ * S3 presigned upload and snapshot-ref commit.
35
+ *
36
+ * @example
37
+ * ```typescript
38
+ * import { ChaimDynamoDBBinder, ChaimCredentials, FailureMode } from '@chaim-tools/cdk-lib';
39
+ *
40
+ * const table = new dynamodb.Table(this, 'UsersTable', {
41
+ * partitionKey: { name: 'userId', type: dynamodb.AttributeType.STRING },
42
+ * });
43
+ *
44
+ * // failureMode defaults to BEST_EFFORT
45
+ * new ChaimDynamoDBBinder(this, 'UserSchema', {
46
+ * schemaPath: './schemas/user.bprint',
47
+ * table,
48
+ * appId: 'my-app',
49
+ * credentials: ChaimCredentials.fromSecretsManager('chaim/api-credentials'),
50
+ * failureMode: FailureMode.STRICT, // Optional - rolls back on failure
51
+ * });
52
+ * ```
53
+ */
54
+ class ChaimDynamoDBBinder extends base_chaim_binder_1.BaseChaimBinder {
55
+ /** The DynamoDB table being bound */
56
+ table;
57
+ /** DynamoDB-specific metadata */
58
+ dynamoDBMetadata;
59
+ dynamoDBProps;
60
+ constructor(scope, id, props) {
61
+ // Store props before calling super (which calls extractMetadata)
62
+ super(scope, id, props);
63
+ this.dynamoDBProps = props;
64
+ this.table = props.table;
65
+ this.dynamoDBMetadata = this.dataStoreMetadata;
66
+ }
67
+ /**
68
+ * Override to resolve the actual table name when possible.
69
+ *
70
+ * Uses stack.resolve() to convert CDK tokens to actual values for explicit table names
71
+ * (e.g., 'acme-product-catalog') and dynamic names (e.g., `${stack.stackName}-orders`).
72
+ * Falls back to construct node ID for auto-generated names or cross-stack references.
73
+ *
74
+ * Note: This is called from BaseChaimBinder constructor before this.table is set,
75
+ * so we access the table from baseProps instead.
76
+ */
77
+ getResourceName() {
78
+ const props = this.baseProps;
79
+ const cfnTable = props.table.node.defaultChild;
80
+ const stack = cdk.Stack.of(this);
81
+ // Try to resolve the table name token
82
+ // This works for explicit names like 'acme-product-catalog'
83
+ // and dynamic names like `${stack.stackName}-orders`
84
+ const resolvedName = stack.resolve(cfnTable.tableName);
85
+ // Check if it's still an unresolved token
86
+ if (!resolvedName || cdk.Token.isUnresolved(resolvedName)) {
87
+ // Fallback to construct ID for auto-generated names
88
+ return props.table.node.id;
89
+ }
90
+ // Return the actual resolved table name
91
+ return resolvedName;
92
+ }
93
+ /**
94
+ * Extract DynamoDB table metadata.
95
+ */
96
+ extractMetadata() {
97
+ const props = this.baseProps;
98
+ const table = props.table;
99
+ const stack = cdk.Stack.of(this);
100
+ // Validate table
101
+ this.validateTable(table);
102
+ // Get CloudFormation resource for detailed metadata
103
+ const cfnTable = this.getCfnTable(table);
104
+ // Extract key schema
105
+ const { partitionKey, sortKey } = this.extractKeySchema(cfnTable);
106
+ // Extract indexes
107
+ const globalSecondaryIndexes = this.extractGSIs(cfnTable);
108
+ const localSecondaryIndexes = this.extractLSIs(cfnTable);
109
+ // Extract TTL
110
+ const ttlAttribute = this.extractTTL(cfnTable);
111
+ // Extract stream info
112
+ const { streamEnabled, streamViewType } = this.extractStreamInfo(cfnTable);
113
+ // Extract billing mode
114
+ const billingMode = this.extractBillingMode(cfnTable);
115
+ // Resolve table name from token (same logic as getResourceName)
116
+ const resolvedTableName = stack.resolve(cfnTable.tableName);
117
+ const tableName = (!resolvedTableName || cdk.Token.isUnresolved(resolvedTableName))
118
+ ? table.tableName // Keep token if can't resolve
119
+ : resolvedTableName;
120
+ return {
121
+ type: 'dynamodb',
122
+ // Removed duplicate fields in v1.1:
123
+ // - arn (use tableArn instead)
124
+ // - name (use tableName instead)
125
+ // - account (use top-level accountId instead)
126
+ tableName,
127
+ tableArn: table.tableArn,
128
+ region: stack.region,
129
+ partitionKey,
130
+ sortKey,
131
+ globalSecondaryIndexes,
132
+ localSecondaryIndexes,
133
+ ttlAttribute,
134
+ streamEnabled,
135
+ streamViewType,
136
+ billingMode,
137
+ encryptionKeyArn: table.encryptionKey?.keyArn,
138
+ };
139
+ }
140
+ /**
141
+ * Validate that the table is a concrete DynamoDB Table construct.
142
+ *
143
+ * Note: We use duck typing for instanceof check to handle cross-package
144
+ * type compatibility issues in monorepo setups (e.g., pnpm with isolated node_modules).
145
+ */
146
+ validateTable(table) {
147
+ if (!table) {
148
+ throw new Error('DynamoDB table is required');
149
+ }
150
+ // Duck typing check: a concrete Table has node.defaultChild (CfnTable)
151
+ // This handles cross-package type compatibility in monorepos
152
+ const isConcreteTable = table instanceof dynamodb.Table ||
153
+ (table.node && table.node.defaultChild &&
154
+ table.node.defaultChild.cfnResourceType === 'AWS::DynamoDB::Table');
155
+ if (!isConcreteTable) {
156
+ throw new Error('Table must be a concrete DynamoDB Table construct. Imported tables are not supported.');
157
+ }
158
+ if (!table.tableName) {
159
+ throw new Error('Table must have a valid table name');
160
+ }
161
+ if (!table.tableArn) {
162
+ throw new Error('Table must have a valid table ARN');
163
+ }
164
+ }
165
+ /**
166
+ * Get the underlying CloudFormation table resource.
167
+ * Uses duck typing for cross-package compatibility.
168
+ */
169
+ getCfnTable(table) {
170
+ // Use duck typing check for cross-package compatibility
171
+ const cfnTable = table.node?.defaultChild;
172
+ if (!cfnTable || cfnTable.cfnResourceType !== 'AWS::DynamoDB::Table') {
173
+ throw new Error('Cannot access CloudFormation resource for imported table');
174
+ }
175
+ return cfnTable;
176
+ }
177
+ /**
178
+ * Extract partition key and sort key from key schema.
179
+ */
180
+ extractKeySchema(cfnTable) {
181
+ const keySchema = cfnTable.keySchema;
182
+ if (!keySchema || !Array.isArray(keySchema) || keySchema.length === 0) {
183
+ throw new Error('Table must have a key schema');
184
+ }
185
+ let partitionKey;
186
+ let sortKey;
187
+ for (const key of keySchema) {
188
+ if (typeof key === 'object' && 'attributeName' in key && 'keyType' in key) {
189
+ if (key.keyType === 'HASH') {
190
+ partitionKey = key.attributeName;
191
+ }
192
+ else if (key.keyType === 'RANGE') {
193
+ sortKey = key.attributeName;
194
+ }
195
+ }
196
+ }
197
+ if (!partitionKey) {
198
+ throw new Error('Cannot extract partition key from table key schema');
199
+ }
200
+ return { partitionKey, sortKey };
201
+ }
202
+ /**
203
+ * Extract Global Secondary Index metadata.
204
+ */
205
+ extractGSIs(cfnTable) {
206
+ const gsis = cfnTable.globalSecondaryIndexes;
207
+ if (!gsis || !Array.isArray(gsis) || gsis.length === 0) {
208
+ return undefined;
209
+ }
210
+ return gsis.map((gsi) => {
211
+ const keySchema = gsi.keySchema || [];
212
+ let partitionKey = '';
213
+ let sortKey;
214
+ for (const key of keySchema) {
215
+ if (key.keyType === 'HASH') {
216
+ partitionKey = key.attributeName;
217
+ }
218
+ else if (key.keyType === 'RANGE') {
219
+ sortKey = key.attributeName;
220
+ }
221
+ }
222
+ return {
223
+ indexName: gsi.indexName,
224
+ partitionKey,
225
+ sortKey,
226
+ projectionType: gsi.projection?.projectionType || 'ALL',
227
+ nonKeyAttributes: gsi.projection?.nonKeyAttributes,
228
+ };
229
+ });
230
+ }
231
+ /**
232
+ * Extract Local Secondary Index metadata.
233
+ */
234
+ extractLSIs(cfnTable) {
235
+ const lsis = cfnTable.localSecondaryIndexes;
236
+ if (!lsis || !Array.isArray(lsis) || lsis.length === 0) {
237
+ return undefined;
238
+ }
239
+ return lsis.map((lsi) => {
240
+ const keySchema = lsi.keySchema || [];
241
+ let sortKey = '';
242
+ for (const key of keySchema) {
243
+ if (key.keyType === 'RANGE') {
244
+ sortKey = key.attributeName;
245
+ }
246
+ }
247
+ return {
248
+ indexName: lsi.indexName,
249
+ sortKey,
250
+ projectionType: lsi.projection?.projectionType || 'ALL',
251
+ nonKeyAttributes: lsi.projection?.nonKeyAttributes,
252
+ };
253
+ });
254
+ }
255
+ /**
256
+ * Extract TTL attribute name.
257
+ */
258
+ extractTTL(cfnTable) {
259
+ const ttlSpec = cfnTable.timeToLiveSpecification;
260
+ if (ttlSpec && typeof ttlSpec === 'object' && 'enabled' in ttlSpec) {
261
+ if (ttlSpec.enabled && 'attributeName' in ttlSpec) {
262
+ return ttlSpec.attributeName;
263
+ }
264
+ }
265
+ return undefined;
266
+ }
267
+ /**
268
+ * Extract stream configuration.
269
+ */
270
+ extractStreamInfo(cfnTable) {
271
+ const streamSpec = cfnTable.streamSpecification;
272
+ if (streamSpec && typeof streamSpec === 'object' && 'streamViewType' in streamSpec) {
273
+ return {
274
+ streamEnabled: true,
275
+ streamViewType: streamSpec.streamViewType,
276
+ };
277
+ }
278
+ return { streamEnabled: false };
279
+ }
280
+ /**
281
+ * Extract billing mode.
282
+ */
283
+ extractBillingMode(cfnTable) {
284
+ const billingMode = cfnTable.billingMode;
285
+ if (billingMode === 'PAY_PER_REQUEST' || billingMode === 'PROVISIONED') {
286
+ return billingMode;
287
+ }
288
+ return undefined;
289
+ }
290
+ }
291
+ exports.ChaimDynamoDBBinder = ChaimDynamoDBBinder;
292
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"chaim-dynamodb-binder.js","sourceRoot":"","sources":["../../src/binders/chaim-dynamodb-binder.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cdk from 'aws-cdk-lib';\nimport * as dynamodb from 'aws-cdk-lib/aws-dynamodb';\nimport { Construct } from 'constructs';\n\nimport { BaseChaimBinder } from './base-chaim-binder';\nimport { BaseBinderProps } from '../types/base-binder-props';\nimport {\n  DynamoDBMetadata,\n  GSIMetadata,\n  LSIMetadata,\n} from '../types/data-store-metadata';\n\n/**\n * Properties for ChaimDynamoDBBinder construct.\n */\nexport interface ChaimDynamoDBBinderProps extends BaseBinderProps {\n  /** DynamoDB table to bind with the schema */\n  table: dynamodb.ITable;\n}\n\n/**\n * CDK construct for binding a .bprint schema to a DynamoDB table.\n *\n * Publishes schema and table metadata to Chaim SaaS platform via\n * S3 presigned upload and snapshot-ref commit.\n *\n * @example\n * ```typescript\n * import { ChaimDynamoDBBinder, ChaimCredentials, FailureMode } from '@chaim-tools/cdk-lib';\n *\n * const table = new dynamodb.Table(this, 'UsersTable', {\n *   partitionKey: { name: 'userId', type: dynamodb.AttributeType.STRING },\n * });\n *\n * // failureMode defaults to BEST_EFFORT\n * new ChaimDynamoDBBinder(this, 'UserSchema', {\n *   schemaPath: './schemas/user.bprint',\n *   table,\n *   appId: 'my-app',\n *   credentials: ChaimCredentials.fromSecretsManager('chaim/api-credentials'),\n *   failureMode: FailureMode.STRICT,  // Optional - rolls back on failure\n * });\n * ```\n */\nexport class ChaimDynamoDBBinder extends BaseChaimBinder {\n  /** The DynamoDB table being bound */\n  public readonly table: dynamodb.ITable;\n\n  /** DynamoDB-specific metadata */\n  public readonly dynamoDBMetadata: DynamoDBMetadata;\n\n  private readonly dynamoDBProps: ChaimDynamoDBBinderProps;\n\n  constructor(scope: Construct, id: string, props: ChaimDynamoDBBinderProps) {\n    // Store props before calling super (which calls extractMetadata)\n    super(scope, id, props);\n\n    this.dynamoDBProps = props;\n    this.table = props.table;\n    this.dynamoDBMetadata = this.dataStoreMetadata as DynamoDBMetadata;\n  }\n\n  /**\n   * Override to resolve the actual table name when possible.\n   * \n   * Uses stack.resolve() to convert CDK tokens to actual values for explicit table names\n   * (e.g., 'acme-product-catalog') and dynamic names (e.g., `${stack.stackName}-orders`).\n   * Falls back to construct node ID for auto-generated names or cross-stack references.\n   * \n   * Note: This is called from BaseChaimBinder constructor before this.table is set,\n   * so we access the table from baseProps instead.\n   */\n  protected getResourceName(): string {\n    const props = this.baseProps as ChaimDynamoDBBinderProps;\n    const cfnTable = props.table.node.defaultChild as dynamodb.CfnTable;\n    const stack = cdk.Stack.of(this);\n    \n    // Try to resolve the table name token\n    // This works for explicit names like 'acme-product-catalog'\n    // and dynamic names like `${stack.stackName}-orders`\n    const resolvedName = stack.resolve(cfnTable.tableName);\n    \n    // Check if it's still an unresolved token\n    if (!resolvedName || cdk.Token.isUnresolved(resolvedName)) {\n      // Fallback to construct ID for auto-generated names\n      return props.table.node.id;\n    }\n    \n    // Return the actual resolved table name\n    return resolvedName;\n  }\n\n  /**\n   * Extract DynamoDB table metadata.\n   */\n  protected extractMetadata(): DynamoDBMetadata {\n    const props = this.baseProps as ChaimDynamoDBBinderProps;\n    const table = props.table;\n    const stack = cdk.Stack.of(this);\n\n    // Validate table\n    this.validateTable(table);\n\n    // Get CloudFormation resource for detailed metadata\n    const cfnTable = this.getCfnTable(table);\n\n    // Extract key schema\n    const { partitionKey, sortKey } = this.extractKeySchema(cfnTable);\n\n    // Extract indexes\n    const globalSecondaryIndexes = this.extractGSIs(cfnTable);\n    const localSecondaryIndexes = this.extractLSIs(cfnTable);\n\n    // Extract TTL\n    const ttlAttribute = this.extractTTL(cfnTable);\n\n    // Extract stream info\n    const { streamEnabled, streamViewType } = this.extractStreamInfo(cfnTable);\n\n    // Extract billing mode\n    const billingMode = this.extractBillingMode(cfnTable);\n\n    // Resolve table name from token (same logic as getResourceName)\n    const resolvedTableName = stack.resolve(cfnTable.tableName);\n    const tableName = (!resolvedTableName || cdk.Token.isUnresolved(resolvedTableName)) \n      ? table.tableName  // Keep token if can't resolve\n      : resolvedTableName;\n\n    return {\n      type: 'dynamodb',\n      // Removed duplicate fields in v1.1:\n      // - arn (use tableArn instead)\n      // - name (use tableName instead)\n      // - account (use top-level accountId instead)\n      tableName,\n      tableArn: table.tableArn,\n      region: stack.region,\n      partitionKey,\n      sortKey,\n      globalSecondaryIndexes,\n      localSecondaryIndexes,\n      ttlAttribute,\n      streamEnabled,\n      streamViewType,\n      billingMode,\n      encryptionKeyArn: table.encryptionKey?.keyArn,\n    };\n  }\n\n  /**\n   * Validate that the table is a concrete DynamoDB Table construct.\n   * \n   * Note: We use duck typing for instanceof check to handle cross-package\n   * type compatibility issues in monorepo setups (e.g., pnpm with isolated node_modules).\n   */\n  private validateTable(table: dynamodb.ITable): void {\n    if (!table) {\n      throw new Error('DynamoDB table is required');\n    }\n\n    // Duck typing check: a concrete Table has node.defaultChild (CfnTable)\n    // This handles cross-package type compatibility in monorepos\n    const isConcreteTable = table instanceof dynamodb.Table || \n      (table.node && table.node.defaultChild && \n       (table.node.defaultChild as any).cfnResourceType === 'AWS::DynamoDB::Table');\n\n    if (!isConcreteTable) {\n      throw new Error(\n        'Table must be a concrete DynamoDB Table construct. Imported tables are not supported.'\n      );\n    }\n\n    if (!table.tableName) {\n      throw new Error('Table must have a valid table name');\n    }\n\n    if (!table.tableArn) {\n      throw new Error('Table must have a valid table ARN');\n    }\n  }\n\n  /**\n   * Get the underlying CloudFormation table resource.\n   * Uses duck typing for cross-package compatibility.\n   */\n  private getCfnTable(table: dynamodb.ITable): dynamodb.CfnTable {\n    // Use duck typing check for cross-package compatibility\n    const cfnTable = table.node?.defaultChild as dynamodb.CfnTable | undefined;\n    \n    if (!cfnTable || (cfnTable as any).cfnResourceType !== 'AWS::DynamoDB::Table') {\n      throw new Error('Cannot access CloudFormation resource for imported table');\n    }\n\n    return cfnTable;\n  }\n\n  /**\n   * Extract partition key and sort key from key schema.\n   */\n  private extractKeySchema(cfnTable: dynamodb.CfnTable): {\n    partitionKey: string;\n    sortKey?: string;\n  } {\n    const keySchema = cfnTable.keySchema;\n\n    if (!keySchema || !Array.isArray(keySchema) || keySchema.length === 0) {\n      throw new Error('Table must have a key schema');\n    }\n\n    let partitionKey: string | undefined;\n    let sortKey: string | undefined;\n\n    for (const key of keySchema) {\n      if (typeof key === 'object' && 'attributeName' in key && 'keyType' in key) {\n        if (key.keyType === 'HASH') {\n          partitionKey = key.attributeName;\n        } else if (key.keyType === 'RANGE') {\n          sortKey = key.attributeName;\n        }\n      }\n    }\n\n    if (!partitionKey) {\n      throw new Error('Cannot extract partition key from table key schema');\n    }\n\n    return { partitionKey, sortKey };\n  }\n\n  /**\n   * Extract Global Secondary Index metadata.\n   */\n  private extractGSIs(cfnTable: dynamodb.CfnTable): GSIMetadata[] | undefined {\n    const gsis = cfnTable.globalSecondaryIndexes;\n    if (!gsis || !Array.isArray(gsis) || gsis.length === 0) {\n      return undefined;\n    }\n\n    return gsis.map((gsi: any) => {\n      const keySchema = gsi.keySchema || [];\n      let partitionKey = '';\n      let sortKey: string | undefined;\n\n      for (const key of keySchema) {\n        if (key.keyType === 'HASH') {\n          partitionKey = key.attributeName;\n        } else if (key.keyType === 'RANGE') {\n          sortKey = key.attributeName;\n        }\n      }\n\n      return {\n        indexName: gsi.indexName,\n        partitionKey,\n        sortKey,\n        projectionType: gsi.projection?.projectionType || 'ALL',\n        nonKeyAttributes: gsi.projection?.nonKeyAttributes,\n      };\n    });\n  }\n\n  /**\n   * Extract Local Secondary Index metadata.\n   */\n  private extractLSIs(cfnTable: dynamodb.CfnTable): LSIMetadata[] | undefined {\n    const lsis = cfnTable.localSecondaryIndexes;\n    if (!lsis || !Array.isArray(lsis) || lsis.length === 0) {\n      return undefined;\n    }\n\n    return lsis.map((lsi: any) => {\n      const keySchema = lsi.keySchema || [];\n      let sortKey = '';\n\n      for (const key of keySchema) {\n        if (key.keyType === 'RANGE') {\n          sortKey = key.attributeName;\n        }\n      }\n\n      return {\n        indexName: lsi.indexName,\n        sortKey,\n        projectionType: lsi.projection?.projectionType || 'ALL',\n        nonKeyAttributes: lsi.projection?.nonKeyAttributes,\n      };\n    });\n  }\n\n  /**\n   * Extract TTL attribute name.\n   */\n  private extractTTL(cfnTable: dynamodb.CfnTable): string | undefined {\n    const ttlSpec = cfnTable.timeToLiveSpecification;\n    if (ttlSpec && typeof ttlSpec === 'object' && 'enabled' in ttlSpec) {\n      if (ttlSpec.enabled && 'attributeName' in ttlSpec) {\n        return ttlSpec.attributeName as string;\n      }\n    }\n    return undefined;\n  }\n\n  /**\n   * Extract stream configuration.\n   */\n  private extractStreamInfo(cfnTable: dynamodb.CfnTable): {\n    streamEnabled?: boolean;\n    streamViewType?: string;\n  } {\n    const streamSpec = cfnTable.streamSpecification;\n    if (streamSpec && typeof streamSpec === 'object' && 'streamViewType' in streamSpec) {\n      return {\n        streamEnabled: true,\n        streamViewType: streamSpec.streamViewType as string,\n      };\n    }\n    return { streamEnabled: false };\n  }\n\n  /**\n   * Extract billing mode.\n   */\n  private extractBillingMode(cfnTable: dynamodb.CfnTable): 'PAY_PER_REQUEST' | 'PROVISIONED' | undefined {\n    const billingMode = cfnTable.billingMode;\n    if (billingMode === 'PAY_PER_REQUEST' || billingMode === 'PROVISIONED') {\n      return billingMode;\n    }\n    return undefined;\n  }\n}\n"]}
@@ -0,0 +1,47 @@
1
+ /**
2
+ * Chaim API endpoint configuration.
3
+ *
4
+ * This is the single source of truth for all Chaim API URLs.
5
+ */
6
+ /**
7
+ * Default Chaim API base URL.
8
+ * Can be overridden via:
9
+ * - CDK context: `chaimApiBaseUrl`
10
+ * - Environment variable: `CHAIM_API_BASE_URL`
11
+ */
12
+ export declare const DEFAULT_CHAIM_API_BASE_URL = "https://api.chaim.co";
13
+ /**
14
+ * Chaim ingestion API endpoints (relative paths).
15
+ */
16
+ export declare const CHAIM_ENDPOINTS: {
17
+ /** Request presigned S3 upload URL with HMAC authentication */
18
+ readonly PRESIGN: "/ingest/presign";
19
+ };
20
+ /**
21
+ * Default request timeout in milliseconds.
22
+ */
23
+ export declare const DEFAULT_REQUEST_TIMEOUT_MS = 30000;
24
+ /**
25
+ * Default maximum snapshot size in bytes (10MB).
26
+ */
27
+ export declare const DEFAULT_MAX_SNAPSHOT_BYTES: number;
28
+ /**
29
+ * Current schema version for snapshot payloads.
30
+ * Increment when making changes to LocalSnapshotPayload.
31
+ *
32
+ * Version history:
33
+ * - 1.0: Initial schema
34
+ * - 1.1: Added operation identity, normalized identity fields (bindingId, tableId, entityId),
35
+ * enhanced DELETE metadata (reason, scope), content hashing (contentHash, schemaHash),
36
+ * producer metadata (component, version, runtime, mode), and removed duplicate
37
+ * dataStore fields (arn, name, account)
38
+ * - 2.0: BREAKING - Removed redundant top-level fields (appId, stackName, datastoreType, tableId, entityId),
39
+ * made identity object required, all binding metadata now in identity object only
40
+ * - 3.0: BREAKING - Complete restructuring into logical sections: providerIdentity (cloud/infra metadata),
41
+ * identity (binding metadata with moved stableResourceKey/resourceId/entityId), operation (event tracking),
42
+ * resolution (token status), resource (simplified, provider-focused, renamed from dataStore),
43
+ * producer (explicit component tracking). Renamed schemaVersion to snapshotVersion.
44
+ *
45
+ * @see LocalSnapshotPayload.snapshotVersion
46
+ */
47
+ export declare const SNAPSHOT_SCHEMA_VERSION: "3.0";
@@ -0,0 +1,51 @@
1
+ "use strict";
2
+ /**
3
+ * Chaim API endpoint configuration.
4
+ *
5
+ * This is the single source of truth for all Chaim API URLs.
6
+ */
7
+ Object.defineProperty(exports, "__esModule", { value: true });
8
+ exports.SNAPSHOT_SCHEMA_VERSION = exports.DEFAULT_MAX_SNAPSHOT_BYTES = exports.DEFAULT_REQUEST_TIMEOUT_MS = exports.CHAIM_ENDPOINTS = exports.DEFAULT_CHAIM_API_BASE_URL = void 0;
9
+ /**
10
+ * Default Chaim API base URL.
11
+ * Can be overridden via:
12
+ * - CDK context: `chaimApiBaseUrl`
13
+ * - Environment variable: `CHAIM_API_BASE_URL`
14
+ */
15
+ exports.DEFAULT_CHAIM_API_BASE_URL = 'https://api.chaim.co';
16
+ /**
17
+ * Chaim ingestion API endpoints (relative paths).
18
+ */
19
+ exports.CHAIM_ENDPOINTS = {
20
+ /** Request presigned S3 upload URL with HMAC authentication */
21
+ PRESIGN: '/ingest/presign',
22
+ };
23
+ /**
24
+ * Default request timeout in milliseconds.
25
+ */
26
+ exports.DEFAULT_REQUEST_TIMEOUT_MS = 30000;
27
+ /**
28
+ * Default maximum snapshot size in bytes (10MB).
29
+ */
30
+ exports.DEFAULT_MAX_SNAPSHOT_BYTES = 10 * 1024 * 1024;
31
+ /**
32
+ * Current schema version for snapshot payloads.
33
+ * Increment when making changes to LocalSnapshotPayload.
34
+ *
35
+ * Version history:
36
+ * - 1.0: Initial schema
37
+ * - 1.1: Added operation identity, normalized identity fields (bindingId, tableId, entityId),
38
+ * enhanced DELETE metadata (reason, scope), content hashing (contentHash, schemaHash),
39
+ * producer metadata (component, version, runtime, mode), and removed duplicate
40
+ * dataStore fields (arn, name, account)
41
+ * - 2.0: BREAKING - Removed redundant top-level fields (appId, stackName, datastoreType, tableId, entityId),
42
+ * made identity object required, all binding metadata now in identity object only
43
+ * - 3.0: BREAKING - Complete restructuring into logical sections: providerIdentity (cloud/infra metadata),
44
+ * identity (binding metadata with moved stableResourceKey/resourceId/entityId), operation (event tracking),
45
+ * resolution (token status), resource (simplified, provider-focused, renamed from dataStore),
46
+ * producer (explicit component tracking). Renamed schemaVersion to snapshotVersion.
47
+ *
48
+ * @see LocalSnapshotPayload.snapshotVersion
49
+ */
50
+ exports.SNAPSHOT_SCHEMA_VERSION = '3.0';
51
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2hhaW0tZW5kcG9pbnRzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2NvbmZpZy9jaGFpbS1lbmRwb2ludHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBOzs7O0dBSUc7OztBQUVIOzs7OztHQUtHO0FBQ1UsUUFBQSwwQkFBMEIsR0FBRyxzQkFBc0IsQ0FBQztBQUVqRTs7R0FFRztBQUNVLFFBQUEsZUFBZSxHQUFHO0lBQzdCLCtEQUErRDtJQUMvRCxPQUFPLEVBQUUsaUJBQWlCO0NBQ2xCLENBQUM7QUFFWDs7R0FFRztBQUNVLFFBQUEsMEJBQTBCLEdBQUcsS0FBSyxDQUFDO0FBRWhEOztHQUVHO0FBQ1UsUUFBQSwwQkFBMEIsR0FBRyxFQUFFLEdBQUcsSUFBSSxHQUFHLElBQUksQ0FBQztBQUUzRDs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBa0JHO0FBQ1UsUUFBQSx1QkFBdUIsR0FBRyxLQUFjLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIENoYWltIEFQSSBlbmRwb2ludCBjb25maWd1cmF0aW9uLlxuICogXG4gKiBUaGlzIGlzIHRoZSBzaW5nbGUgc291cmNlIG9mIHRydXRoIGZvciBhbGwgQ2hhaW0gQVBJIFVSTHMuXG4gKi9cblxuLyoqXG4gKiBEZWZhdWx0IENoYWltIEFQSSBiYXNlIFVSTC5cbiAqIENhbiBiZSBvdmVycmlkZGVuIHZpYTpcbiAqIC0gQ0RLIGNvbnRleHQ6IGBjaGFpbUFwaUJhc2VVcmxgXG4gKiAtIEVudmlyb25tZW50IHZhcmlhYmxlOiBgQ0hBSU1fQVBJX0JBU0VfVVJMYFxuICovXG5leHBvcnQgY29uc3QgREVGQVVMVF9DSEFJTV9BUElfQkFTRV9VUkwgPSAnaHR0cHM6Ly9hcGkuY2hhaW0uY28nO1xuXG4vKipcbiAqIENoYWltIGluZ2VzdGlvbiBBUEkgZW5kcG9pbnRzIChyZWxhdGl2ZSBwYXRocykuXG4gKi9cbmV4cG9ydCBjb25zdCBDSEFJTV9FTkRQT0lOVFMgPSB7XG4gIC8qKiBSZXF1ZXN0IHByZXNpZ25lZCBTMyB1cGxvYWQgVVJMIHdpdGggSE1BQyBhdXRoZW50aWNhdGlvbiAqL1xuICBQUkVTSUdOOiAnL2luZ2VzdC9wcmVzaWduJyxcbn0gYXMgY29uc3Q7XG5cbi8qKlxuICogRGVmYXVsdCByZXF1ZXN0IHRpbWVvdXQgaW4gbWlsbGlzZWNvbmRzLlxuICovXG5leHBvcnQgY29uc3QgREVGQVVMVF9SRVFVRVNUX1RJTUVPVVRfTVMgPSAzMDAwMDtcblxuLyoqXG4gKiBEZWZhdWx0IG1heGltdW0gc25hcHNob3Qgc2l6ZSBpbiBieXRlcyAoMTBNQikuXG4gKi9cbmV4cG9ydCBjb25zdCBERUZBVUxUX01BWF9TTkFQU0hPVF9CWVRFUyA9IDEwICogMTAyNCAqIDEwMjQ7XG5cbi8qKlxuICogQ3VycmVudCBzY2hlbWEgdmVyc2lvbiBmb3Igc25hcHNob3QgcGF5bG9hZHMuXG4gKiBJbmNyZW1lbnQgd2hlbiBtYWtpbmcgY2hhbmdlcyB0byBMb2NhbFNuYXBzaG90UGF5bG9hZC5cbiAqIFxuICogVmVyc2lvbiBoaXN0b3J5OlxuICogLSAxLjA6IEluaXRpYWwgc2NoZW1hXG4gKiAtIDEuMTogQWRkZWQgb3BlcmF0aW9uIGlkZW50aXR5LCBub3JtYWxpemVkIGlkZW50aXR5IGZpZWxkcyAoYmluZGluZ0lkLCB0YWJsZUlkLCBlbnRpdHlJZCksXG4gKiAgICAgICAgZW5oYW5jZWQgREVMRVRFIG1ldGFkYXRhIChyZWFzb24sIHNjb3BlKSwgY29udGVudCBoYXNoaW5nIChjb250ZW50SGFzaCwgc2NoZW1hSGFzaCksXG4gKiAgICAgICAgcHJvZHVjZXIgbWV0YWRhdGEgKGNvbXBvbmVudCwgdmVyc2lvbiwgcnVudGltZSwgbW9kZSksIGFuZCByZW1vdmVkIGR1cGxpY2F0ZVxuICogICAgICAgIGRhdGFTdG9yZSBmaWVsZHMgKGFybiwgbmFtZSwgYWNjb3VudClcbiAqIC0gMi4wOiBCUkVBS0lORyAtIFJlbW92ZWQgcmVkdW5kYW50IHRvcC1sZXZlbCBmaWVsZHMgKGFwcElkLCBzdGFja05hbWUsIGRhdGFzdG9yZVR5cGUsIHRhYmxlSWQsIGVudGl0eUlkKSxcbiAqICAgICAgICBtYWRlIGlkZW50aXR5IG9iamVjdCByZXF1aXJlZCwgYWxsIGJpbmRpbmcgbWV0YWRhdGEgbm93IGluIGlkZW50aXR5IG9iamVjdCBvbmx5XG4gKiAtIDMuMDogQlJFQUtJTkcgLSBDb21wbGV0ZSByZXN0cnVjdHVyaW5nIGludG8gbG9naWNhbCBzZWN0aW9uczogcHJvdmlkZXJJZGVudGl0eSAoY2xvdWQvaW5mcmEgbWV0YWRhdGEpLFxuICogICAgICAgIGlkZW50aXR5IChiaW5kaW5nIG1ldGFkYXRhIHdpdGggbW92ZWQgc3RhYmxlUmVzb3VyY2VLZXkvcmVzb3VyY2VJZC9lbnRpdHlJZCksIG9wZXJhdGlvbiAoZXZlbnQgdHJhY2tpbmcpLFxuICogICAgICAgIHJlc29sdXRpb24gKHRva2VuIHN0YXR1cyksIHJlc291cmNlIChzaW1wbGlmaWVkLCBwcm92aWRlci1mb2N1c2VkLCByZW5hbWVkIGZyb20gZGF0YVN0b3JlKSxcbiAqICAgICAgICBwcm9kdWNlciAoZXhwbGljaXQgY29tcG9uZW50IHRyYWNraW5nKS4gUmVuYW1lZCBzY2hlbWFWZXJzaW9uIHRvIHNuYXBzaG90VmVyc2lvbi5cbiAqIFxuICogQHNlZSBMb2NhbFNuYXBzaG90UGF5bG9hZC5zbmFwc2hvdFZlcnNpb25cbiAqL1xuZXhwb3J0IGNvbnN0IFNOQVBTSE9UX1NDSEVNQV9WRVJTSU9OID0gJzMuMCcgYXMgY29uc3Q7Il19
package/lib/index.d.ts ADDED
@@ -0,0 +1,15 @@
1
+ export { ChaimDynamoDBBinder, ChaimDynamoDBBinderProps } from './binders/chaim-dynamodb-binder';
2
+ export { BaseChaimBinder } from './binders/base-chaim-binder';
3
+ export { ChaimCredentials, IChaimCredentials, } from './types/credentials';
4
+ export { FailureMode } from './types/failure-mode';
5
+ export { BaseBinderProps, validateBinderProps, } from './types/base-binder-props';
6
+ export { TableBindingConfig } from './types/table-binding-config';
7
+ export { BaseDataStoreMetadata, DynamoDBMetadata, GSIMetadata, LSIMetadata, DataStoreMetadata, } from './types/data-store-metadata';
8
+ export { LocalSnapshotPayload, PublishedSnapshotPayload, ProviderIdentity, BindingIdentity, OperationMetadata, ResolutionMetadata, DeleteMetadata, HashMetadata, ResourceMetadata, ProducerMetadata, IngestResponse, CustomResourceResponseData, } from './types/snapshot-payload';
9
+ export { SnapshotAction, UploadUrlRequest, UploadUrlResponse, SnapshotRefUpsertRequest, SnapshotRefDeleteRequest, SnapshotRefRequest, SnapshotRefResponse, CloudFormationRequestType, } from './types/ingest-contract';
10
+ export { DEFAULT_CHAIM_API_BASE_URL, CHAIM_ENDPOINTS, DEFAULT_REQUEST_TIMEOUT_MS, DEFAULT_MAX_SNAPSHOT_BYTES, SNAPSHOT_SCHEMA_VERSION, } from './config/chaim-endpoints';
11
+ export { SchemaService } from './services/schema-service';
12
+ export { IngestionService, IngestionConfig, INGESTION_ENDPOINTS, DEFAULT_INGESTION_CONFIG, } from './services/ingestion-service';
13
+ export { pruneStackSnapshots, pruneOldSnapshots, CleanupOptions, TTLCleanupOptions, CleanupResult, } from './services/snapshot-cleanup';
14
+ export { SnapshotCachePolicy, DEFAULT_SNAPSHOT_CACHE_POLICY, SNAPSHOT_CACHE_POLICY_CONTEXT_KEY, } from './types/snapshot-cache-policy';
15
+ export { SchemaData, Field, PrimaryKey } from '@chaim-tools/chaim-bprint-spec';
package/lib/index.js ADDED
@@ -0,0 +1,43 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.SNAPSHOT_CACHE_POLICY_CONTEXT_KEY = exports.DEFAULT_SNAPSHOT_CACHE_POLICY = exports.SnapshotCachePolicy = exports.pruneOldSnapshots = exports.pruneStackSnapshots = exports.DEFAULT_INGESTION_CONFIG = exports.INGESTION_ENDPOINTS = exports.IngestionService = exports.SchemaService = exports.SNAPSHOT_SCHEMA_VERSION = exports.DEFAULT_MAX_SNAPSHOT_BYTES = exports.DEFAULT_REQUEST_TIMEOUT_MS = exports.CHAIM_ENDPOINTS = exports.DEFAULT_CHAIM_API_BASE_URL = exports.TableBindingConfig = exports.validateBinderProps = exports.FailureMode = exports.ChaimCredentials = exports.BaseChaimBinder = exports.ChaimDynamoDBBinder = void 0;
4
+ // Main construct exports
5
+ var chaim_dynamodb_binder_1 = require("./binders/chaim-dynamodb-binder");
6
+ Object.defineProperty(exports, "ChaimDynamoDBBinder", { enumerable: true, get: function () { return chaim_dynamodb_binder_1.ChaimDynamoDBBinder; } });
7
+ // Base class export (for extension by future data store binders)
8
+ var base_chaim_binder_1 = require("./binders/base-chaim-binder");
9
+ Object.defineProperty(exports, "BaseChaimBinder", { enumerable: true, get: function () { return base_chaim_binder_1.BaseChaimBinder; } });
10
+ // Credentials factory
11
+ var credentials_1 = require("./types/credentials");
12
+ Object.defineProperty(exports, "ChaimCredentials", { enumerable: true, get: function () { return credentials_1.ChaimCredentials; } });
13
+ // Failure mode enum
14
+ var failure_mode_1 = require("./types/failure-mode");
15
+ Object.defineProperty(exports, "FailureMode", { enumerable: true, get: function () { return failure_mode_1.FailureMode; } });
16
+ // Type exports
17
+ var base_binder_props_1 = require("./types/base-binder-props");
18
+ Object.defineProperty(exports, "validateBinderProps", { enumerable: true, get: function () { return base_binder_props_1.validateBinderProps; } });
19
+ // Binding configuration
20
+ var table_binding_config_1 = require("./types/table-binding-config");
21
+ Object.defineProperty(exports, "TableBindingConfig", { enumerable: true, get: function () { return table_binding_config_1.TableBindingConfig; } });
22
+ // Config exports
23
+ var chaim_endpoints_1 = require("./config/chaim-endpoints");
24
+ Object.defineProperty(exports, "DEFAULT_CHAIM_API_BASE_URL", { enumerable: true, get: function () { return chaim_endpoints_1.DEFAULT_CHAIM_API_BASE_URL; } });
25
+ Object.defineProperty(exports, "CHAIM_ENDPOINTS", { enumerable: true, get: function () { return chaim_endpoints_1.CHAIM_ENDPOINTS; } });
26
+ Object.defineProperty(exports, "DEFAULT_REQUEST_TIMEOUT_MS", { enumerable: true, get: function () { return chaim_endpoints_1.DEFAULT_REQUEST_TIMEOUT_MS; } });
27
+ Object.defineProperty(exports, "DEFAULT_MAX_SNAPSHOT_BYTES", { enumerable: true, get: function () { return chaim_endpoints_1.DEFAULT_MAX_SNAPSHOT_BYTES; } });
28
+ Object.defineProperty(exports, "SNAPSHOT_SCHEMA_VERSION", { enumerable: true, get: function () { return chaim_endpoints_1.SNAPSHOT_SCHEMA_VERSION; } });
29
+ // Service exports
30
+ var schema_service_1 = require("./services/schema-service");
31
+ Object.defineProperty(exports, "SchemaService", { enumerable: true, get: function () { return schema_service_1.SchemaService; } });
32
+ var ingestion_service_1 = require("./services/ingestion-service");
33
+ Object.defineProperty(exports, "IngestionService", { enumerable: true, get: function () { return ingestion_service_1.IngestionService; } });
34
+ Object.defineProperty(exports, "INGESTION_ENDPOINTS", { enumerable: true, get: function () { return ingestion_service_1.INGESTION_ENDPOINTS; } });
35
+ Object.defineProperty(exports, "DEFAULT_INGESTION_CONFIG", { enumerable: true, get: function () { return ingestion_service_1.DEFAULT_INGESTION_CONFIG; } });
36
+ var snapshot_cleanup_1 = require("./services/snapshot-cleanup");
37
+ Object.defineProperty(exports, "pruneStackSnapshots", { enumerable: true, get: function () { return snapshot_cleanup_1.pruneStackSnapshots; } });
38
+ Object.defineProperty(exports, "pruneOldSnapshots", { enumerable: true, get: function () { return snapshot_cleanup_1.pruneOldSnapshots; } });
39
+ var snapshot_cache_policy_1 = require("./types/snapshot-cache-policy");
40
+ Object.defineProperty(exports, "SnapshotCachePolicy", { enumerable: true, get: function () { return snapshot_cache_policy_1.SnapshotCachePolicy; } });
41
+ Object.defineProperty(exports, "DEFAULT_SNAPSHOT_CACHE_POLICY", { enumerable: true, get: function () { return snapshot_cache_policy_1.DEFAULT_SNAPSHOT_CACHE_POLICY; } });
42
+ Object.defineProperty(exports, "SNAPSHOT_CACHE_POLICY_CONTEXT_KEY", { enumerable: true, get: function () { return snapshot_cache_policy_1.SNAPSHOT_CACHE_POLICY_CONTEXT_KEY; } });
43
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEseUJBQXlCO0FBQ3pCLHlFQUFnRztBQUF2Riw0SEFBQSxtQkFBbUIsT0FBQTtBQUU1QixpRUFBaUU7QUFDakUsaUVBQThEO0FBQXJELG9IQUFBLGVBQWUsT0FBQTtBQUV4QixzQkFBc0I7QUFDdEIsbURBRzZCO0FBRjNCLCtHQUFBLGdCQUFnQixPQUFBO0FBSWxCLG9CQUFvQjtBQUNwQixxREFBbUQ7QUFBMUMsMkdBQUEsV0FBVyxPQUFBO0FBRXBCLGVBQWU7QUFDZiwrREFHbUM7QUFEakMsd0hBQUEsbUJBQW1CLE9BQUE7QUFHckIsd0JBQXdCO0FBQ3hCLHFFQUFrRTtBQUF6RCwwSEFBQSxrQkFBa0IsT0FBQTtBQXNDM0IsaUJBQWlCO0FBQ2pCLDREQU1rQztBQUxoQyw2SEFBQSwwQkFBMEIsT0FBQTtBQUMxQixrSEFBQSxlQUFlLE9BQUE7QUFDZiw2SEFBQSwwQkFBMEIsT0FBQTtBQUMxQiw2SEFBQSwwQkFBMEIsT0FBQTtBQUMxQiwwSEFBQSx1QkFBdUIsT0FBQTtBQUd6QixrQkFBa0I7QUFDbEIsNERBQTBEO0FBQWpELCtHQUFBLGFBQWEsT0FBQTtBQUN0QixrRUFLc0M7QUFKcEMscUhBQUEsZ0JBQWdCLE9BQUE7QUFFaEIsd0hBQUEsbUJBQW1CLE9BQUE7QUFDbkIsNkhBQUEsd0JBQXdCLE9BQUE7QUFFMUIsZ0VBTXFDO0FBTG5DLHVIQUFBLG1CQUFtQixPQUFBO0FBQ25CLHFIQUFBLGlCQUFpQixPQUFBO0FBS25CLHVFQUl1QztBQUhyQyw0SEFBQSxtQkFBbUIsT0FBQTtBQUNuQixzSUFBQSw2QkFBNkIsT0FBQTtBQUM3QiwwSUFBQSxpQ0FBaUMsT0FBQSIsInNvdXJjZXNDb250ZW50IjpbIi8vIE1haW4gY29uc3RydWN0IGV4cG9ydHNcbmV4cG9ydCB7IENoYWltRHluYW1vREJCaW5kZXIsIENoYWltRHluYW1vREJCaW5kZXJQcm9wcyB9IGZyb20gJy4vYmluZGVycy9jaGFpbS1keW5hbW9kYi1iaW5kZXInO1xuXG4vLyBCYXNlIGNsYXNzIGV4cG9ydCAoZm9yIGV4dGVuc2lvbiBieSBmdXR1cmUgZGF0YSBzdG9yZSBiaW5kZXJzKVxuZXhwb3J0IHsgQmFzZUNoYWltQmluZGVyIH0gZnJvbSAnLi9iaW5kZXJzL2Jhc2UtY2hhaW0tYmluZGVyJztcblxuLy8gQ3JlZGVudGlhbHMgZmFjdG9yeVxuZXhwb3J0IHtcbiAgQ2hhaW1DcmVkZW50aWFscyxcbiAgSUNoYWltQ3JlZGVudGlhbHMsXG59IGZyb20gJy4vdHlwZXMvY3JlZGVudGlhbHMnO1xuXG4vLyBGYWlsdXJlIG1vZGUgZW51bVxuZXhwb3J0IHsgRmFpbHVyZU1vZGUgfSBmcm9tICcuL3R5cGVzL2ZhaWx1cmUtbW9kZSc7XG5cbi8vIFR5cGUgZXhwb3J0c1xuZXhwb3J0IHtcbiAgQmFzZUJpbmRlclByb3BzLFxuICB2YWxpZGF0ZUJpbmRlclByb3BzLFxufSBmcm9tICcuL3R5cGVzL2Jhc2UtYmluZGVyLXByb3BzJztcblxuLy8gQmluZGluZyBjb25maWd1cmF0aW9uXG5leHBvcnQgeyBUYWJsZUJpbmRpbmdDb25maWcgfSBmcm9tICcuL3R5cGVzL3RhYmxlLWJpbmRpbmctY29uZmlnJztcblxuZXhwb3J0IHtcbiAgQmFzZURhdGFTdG9yZU1ldGFkYXRhLFxuICBEeW5hbW9EQk1ldGFkYXRhLFxuICBHU0lNZXRhZGF0YSxcbiAgTFNJTWV0YWRhdGEsXG4gIERhdGFTdG9yZU1ldGFkYXRhLFxufSBmcm9tICcuL3R5cGVzL2RhdGEtc3RvcmUtbWV0YWRhdGEnO1xuXG4vLyBTbmFwc2hvdCBwYXlsb2FkIHR5cGVzXG5leHBvcnQge1xuICBMb2NhbFNuYXBzaG90UGF5bG9hZCxcbiAgUHVibGlzaGVkU25hcHNob3RQYXlsb2FkLFxuICBQcm92aWRlcklkZW50aXR5LFxuICBCaW5kaW5nSWRlbnRpdHksXG4gIE9wZXJhdGlvbk1ldGFkYXRhLFxuICBSZXNvbHV0aW9uTWV0YWRhdGEsXG4gIERlbGV0ZU1ldGFkYXRhLFxuICBIYXNoTWV0YWRhdGEsXG4gIFJlc291cmNlTWV0YWRhdGEsXG4gIFByb2R1Y2VyTWV0YWRhdGEsXG4gIEluZ2VzdFJlc3BvbnNlLFxuICBDdXN0b21SZXNvdXJjZVJlc3BvbnNlRGF0YSxcbn0gZnJvbSAnLi90eXBlcy9zbmFwc2hvdC1wYXlsb2FkJztcblxuLy8gSW5nZXN0IGNvbnRyYWN0IHR5cGVzXG5leHBvcnQge1xuICBTbmFwc2hvdEFjdGlvbixcbiAgVXBsb2FkVXJsUmVxdWVzdCxcbiAgVXBsb2FkVXJsUmVzcG9uc2UsXG4gIFNuYXBzaG90UmVmVXBzZXJ0UmVxdWVzdCxcbiAgU25hcHNob3RSZWZEZWxldGVSZXF1ZXN0LFxuICBTbmFwc2hvdFJlZlJlcXVlc3QsXG4gIFNuYXBzaG90UmVmUmVzcG9uc2UsXG4gIENsb3VkRm9ybWF0aW9uUmVxdWVzdFR5cGUsXG59IGZyb20gJy4vdHlwZXMvaW5nZXN0LWNvbnRyYWN0JztcblxuLy8gQ29uZmlnIGV4cG9ydHNcbmV4cG9ydCB7XG4gIERFRkFVTFRfQ0hBSU1fQVBJX0JBU0VfVVJMLFxuICBDSEFJTV9FTkRQT0lOVFMsXG4gIERFRkFVTFRfUkVRVUVTVF9USU1FT1VUX01TLFxuICBERUZBVUxUX01BWF9TTkFQU0hPVF9CWVRFUyxcbiAgU05BUFNIT1RfU0NIRU1BX1ZFUlNJT04sXG59IGZyb20gJy4vY29uZmlnL2NoYWltLWVuZHBvaW50cyc7XG5cbi8vIFNlcnZpY2UgZXhwb3J0c1xuZXhwb3J0IHsgU2NoZW1hU2VydmljZSB9IGZyb20gJy4vc2VydmljZXMvc2NoZW1hLXNlcnZpY2UnO1xuZXhwb3J0IHtcbiAgSW5nZXN0aW9uU2VydmljZSxcbiAgSW5nZXN0aW9uQ29uZmlnLFxuICBJTkdFU1RJT05fRU5EUE9JTlRTLFxuICBERUZBVUxUX0lOR0VTVElPTl9DT05GSUcsXG59IGZyb20gJy4vc2VydmljZXMvaW5nZXN0aW9uLXNlcnZpY2UnO1xuZXhwb3J0IHtcbiAgcHJ1bmVTdGFja1NuYXBzaG90cyxcbiAgcHJ1bmVPbGRTbmFwc2hvdHMsXG4gIENsZWFudXBPcHRpb25zLFxuICBUVExDbGVhbnVwT3B0aW9ucyxcbiAgQ2xlYW51cFJlc3VsdCxcbn0gZnJvbSAnLi9zZXJ2aWNlcy9zbmFwc2hvdC1jbGVhbnVwJztcbmV4cG9ydCB7XG4gIFNuYXBzaG90Q2FjaGVQb2xpY3ksXG4gIERFRkFVTFRfU05BUFNIT1RfQ0FDSEVfUE9MSUNZLFxuICBTTkFQU0hPVF9DQUNIRV9QT0xJQ1lfQ09OVEVYVF9LRVksXG59IGZyb20gJy4vdHlwZXMvc25hcHNob3QtY2FjaGUtcG9saWN5JztcblxuLy8gUmUtZXhwb3J0IHNjaGVtYSB0eXBlcyBmcm9tIGJwcmludC1zcGVjIGZvciBjb252ZW5pZW5jZVxuZXhwb3J0IHsgU2NoZW1hRGF0YSwgRmllbGQsIFByaW1hcnlLZXkgfSBmcm9tICdAY2hhaW0tdG9vbHMvY2hhaW0tYnByaW50LXNwZWMnO1xuIl19
@@ -0,0 +1 @@
1
+ {"provider":"aws","accountId":"123456789012","region":"us-east-1","stackName":"TestStack","datastoreType":"dynamodb","resourceName":"TestTable","resourceId":"TestTable__User","appId":"test-app","schema":{"schemaVersion":"1.0","namespace":"test"},"dataStore":{"type":"dynamodb","arn":"arn:aws:dynamodb:us-east-1:123456789012:table/TestTable","name":"TestTable"},"context":{"account":"123456789012","region":"us-east-1"},"capturedAt":"2024-01-01T00:00:00.000Z"}