@prisma-next/mongo-contract 0.0.1 → 0.3.0-dev.147

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.
@@ -0,0 +1,158 @@
1
+ import { Contract, ContractField, ContractModel, ContractValueObject, StorageBase } from "@prisma-next/contract/types";
2
+
3
+ //#region src/contract-types.d.ts
4
+ type MongoIndexFieldValue = 1 | -1 | 'text' | '2dsphere' | '2d' | 'hashed';
5
+ type MongoIndexFields = Record<string, MongoIndexFieldValue>;
6
+ type MongoJsonPrimitive = string | number | boolean | null;
7
+ type MongoJsonValue = MongoJsonPrimitive | readonly MongoJsonValue[] | MongoJsonObject;
8
+ type MongoJsonObject = {
9
+ readonly [key: string]: MongoJsonValue;
10
+ };
11
+ type MongoCollationCaseFirst = 'off' | 'upper' | 'lower';
12
+ type MongoCollationStrength = 1 | 2 | 3 | 4 | 5;
13
+ type MongoCollationAlternate = 'non-ignorable' | 'shifted';
14
+ type MongoCollationMaxVariable = 'punct' | 'space';
15
+ type MongoCollationOptions = {
16
+ readonly locale: string;
17
+ readonly caseLevel?: boolean;
18
+ readonly caseFirst?: MongoCollationCaseFirst;
19
+ readonly strength?: MongoCollationStrength;
20
+ readonly numericOrdering?: boolean;
21
+ readonly alternate?: MongoCollationAlternate;
22
+ readonly maxVariable?: MongoCollationMaxVariable;
23
+ readonly backwards?: boolean;
24
+ readonly normalization?: boolean;
25
+ };
26
+ type MongoWildcardProjection = Readonly<Record<string, 0 | 1>>;
27
+ type MongoIndexOptions = {
28
+ readonly unique?: boolean;
29
+ readonly name?: string;
30
+ readonly partialFilterExpression?: MongoJsonObject;
31
+ readonly sparse?: boolean;
32
+ readonly expireAfterSeconds?: number;
33
+ readonly weights?: Readonly<Record<string, number>>;
34
+ readonly default_language?: string;
35
+ readonly language_override?: string;
36
+ readonly textIndexVersion?: number;
37
+ readonly '2dsphereIndexVersion'?: number;
38
+ readonly bits?: number;
39
+ readonly min?: number;
40
+ readonly max?: number;
41
+ readonly bucketSize?: number;
42
+ readonly hidden?: boolean;
43
+ readonly collation?: MongoCollationOptions;
44
+ readonly wildcardProjection?: MongoWildcardProjection;
45
+ };
46
+ type MongoIndex = {
47
+ readonly fields: MongoIndexFields;
48
+ readonly options?: MongoIndexOptions;
49
+ };
50
+ type MongoIndexOptionDefaults = {
51
+ readonly storageEngine?: MongoJsonObject;
52
+ };
53
+ type MongoTimeSeriesGranularity = 'seconds' | 'minutes' | 'hours';
54
+ type MongoTimeSeriesCollectionOptions = {
55
+ readonly timeField: string;
56
+ readonly metaField?: string;
57
+ readonly granularity?: MongoTimeSeriesGranularity;
58
+ readonly bucketMaxSpanSeconds?: number;
59
+ readonly bucketRoundingSeconds?: number;
60
+ };
61
+ type MongoClusteredCollectionKey = Readonly<Record<string, 1>>;
62
+ type MongoClusteredCollectionOptions = {
63
+ readonly name?: string;
64
+ readonly key: MongoClusteredCollectionKey;
65
+ readonly unique: boolean;
66
+ };
67
+ type MongoChangeStreamPreAndPostImagesOptions = {
68
+ readonly enabled: boolean;
69
+ };
70
+ type MongoCollectionOptions = {
71
+ readonly capped?: boolean;
72
+ readonly size?: number;
73
+ readonly max?: number;
74
+ readonly storageEngine?: MongoJsonObject;
75
+ readonly indexOptionDefaults?: MongoIndexOptionDefaults;
76
+ readonly collation?: MongoCollationOptions;
77
+ readonly timeseries?: MongoTimeSeriesCollectionOptions;
78
+ readonly clusteredIndex?: MongoClusteredCollectionOptions;
79
+ readonly expireAfterSeconds?: number;
80
+ readonly changeStreamPreAndPostImages?: MongoChangeStreamPreAndPostImagesOptions;
81
+ };
82
+ type MongoStorageCollection = {
83
+ readonly indexes?: readonly MongoIndex[];
84
+ readonly options?: MongoCollectionOptions;
85
+ };
86
+ type MongoStorage<THash extends string = string> = StorageBase<THash> & {
87
+ readonly collections: Record<string, MongoStorageCollection>;
88
+ };
89
+ type MongoModelStorage = {
90
+ readonly collection?: string;
91
+ readonly relations?: Record<string, {
92
+ readonly field: string;
93
+ }>;
94
+ };
95
+ type MongoModelDefinition = ContractModel<MongoModelStorage>;
96
+ type MongoContract<S extends MongoStorage = MongoStorage, M extends Record<string, MongoModelDefinition> = Record<string, MongoModelDefinition>> = Contract<S, M>;
97
+ type MongoTypeMaps<TCodecTypes extends Record<string, {
98
+ output: unknown;
99
+ }> = Record<string, {
100
+ output: unknown;
101
+ }>, TOperationTypes extends Record<string, unknown> = Record<string, never>, TFieldOutputTypes extends Record<string, Record<string, unknown>> = Record<string, Record<string, unknown>>> = {
102
+ readonly codecTypes: TCodecTypes;
103
+ readonly operationTypes: TOperationTypes;
104
+ readonly fieldOutputTypes: TFieldOutputTypes;
105
+ };
106
+ type MongoTypeMapsPhantomKey = '__@prisma-next/mongo-core/typeMaps@__';
107
+ type MongoContractWithTypeMaps<TContract, TTypeMaps> = TContract & { readonly [K in MongoTypeMapsPhantomKey]?: TTypeMaps };
108
+ type ExtractMongoTypeMaps<T> = MongoTypeMapsPhantomKey extends keyof T ? NonNullable<T[MongoTypeMapsPhantomKey & keyof T]> : never;
109
+ type ExtractMongoCodecTypes<T> = ExtractMongoTypeMaps<T> extends {
110
+ codecTypes: infer C;
111
+ } ? C extends Record<string, {
112
+ output: unknown;
113
+ }> ? C : Record<string, never> : Record<string, never>;
114
+ type ExtractMongoFieldOutputTypes<T> = ExtractMongoTypeMaps<T> extends {
115
+ fieldOutputTypes: infer F;
116
+ } ? F extends Record<string, Record<string, unknown>> ? F : Record<string, never> : Record<string, never>;
117
+ type ExtractValueObjects<TContract> = TContract extends {
118
+ valueObjects: infer VO extends Record<string, ContractValueObject>;
119
+ } ? VO : Record<never, never>;
120
+ type NormalizeContractFields<TFields> = { [K in keyof TFields]: TFields[K] extends ContractField ? TFields[K] : never };
121
+ type ExtractValueObjectFields<TValueObjects extends Record<string, ContractValueObject>, VOName$1 extends keyof TValueObjects> = NormalizeContractFields<TValueObjects[VOName$1]['fields']>;
122
+ type InferFieldBaseType<TFieldType, TValueObjects extends Record<string, ContractValueObject>, TCodecTypes extends Record<string, {
123
+ output: unknown;
124
+ }>> = TFieldType extends {
125
+ kind: 'scalar';
126
+ codecId: infer CId extends string & keyof TCodecTypes;
127
+ } ? TCodecTypes[CId]['output'] : TFieldType extends {
128
+ kind: 'valueObject';
129
+ name: infer VOName extends string;
130
+ } ? VOName extends keyof TValueObjects ? { -readonly [K in keyof ExtractValueObjectFields<TValueObjects, VOName>]: InferFieldType<ExtractValueObjectFields<TValueObjects, VOName>[K], TValueObjects, TCodecTypes> } : unknown : TFieldType extends {
131
+ kind: 'union';
132
+ members: infer TMembers extends ReadonlyArray<unknown>;
133
+ } ? TMembers[number] extends infer TMember ? InferFieldBaseType<TMember, TValueObjects, TCodecTypes> : unknown : unknown;
134
+ type InferFieldType<TField, TValueObjects extends Record<string, ContractValueObject>, TCodecTypes extends Record<string, {
135
+ output: unknown;
136
+ }>> = TField extends ContractField ? TField extends {
137
+ many: true;
138
+ } ? TField['nullable'] extends true ? InferFieldBaseType<TField['type'], TValueObjects, TCodecTypes>[] | null : InferFieldBaseType<TField['type'], TValueObjects, TCodecTypes>[] : TField['nullable'] extends true ? InferFieldBaseType<TField['type'], TValueObjects, TCodecTypes> | null : InferFieldBaseType<TField['type'], TValueObjects, TCodecTypes> : never;
139
+ type InferModelRow<TContract extends MongoContractWithTypeMaps<MongoContract, MongoTypeMaps>, ModelName extends string & keyof TContract['models'], TFields extends Record<string, ContractField> = TContract['models'][ModelName]['fields'], TCodecTypes extends Record<string, {
140
+ output: unknown;
141
+ }> = ExtractMongoCodecTypes<TContract>, TValueObjects extends Record<string, ContractValueObject> = ExtractValueObjects<TContract>> = { -readonly [FieldName in keyof TFields]: InferFieldType<TFields[FieldName], TValueObjects, TCodecTypes> };
142
+ //#endregion
143
+ //#region src/validate-mongo-contract.d.ts
144
+ interface MongoContractIndices {
145
+ readonly variantToBase: Record<string, string>;
146
+ readonly modelToVariants: Record<string, string[]>;
147
+ }
148
+ interface ValidatedMongoContract<TContract extends MongoContract> {
149
+ readonly contract: TContract;
150
+ readonly indices: MongoContractIndices;
151
+ }
152
+ declare function validateMongoContract<TContract extends MongoContract>(value: unknown): ValidatedMongoContract<TContract>;
153
+ //#endregion
154
+ //#region src/validate-storage.d.ts
155
+ declare function validateMongoStorage(contract: MongoContract): void;
156
+ //#endregion
157
+ export { type ExtractMongoCodecTypes, type ExtractMongoFieldOutputTypes, type ExtractMongoTypeMaps, type InferModelRow, type MongoChangeStreamPreAndPostImagesOptions, type MongoClusteredCollectionKey, type MongoClusteredCollectionOptions, type MongoCollationAlternate, type MongoCollationCaseFirst, type MongoCollationMaxVariable, type MongoCollationOptions, type MongoCollationStrength, type MongoCollectionOptions, type MongoContract, type MongoContractIndices, type MongoContractWithTypeMaps, type MongoIndex, type MongoIndexFieldValue, type MongoIndexFields, type MongoIndexOptionDefaults, type MongoIndexOptions, type MongoJsonObject, type MongoJsonPrimitive, type MongoJsonValue, type MongoModelDefinition, type MongoModelStorage, type MongoStorage, type MongoStorageCollection, type MongoTimeSeriesCollectionOptions, type MongoTimeSeriesGranularity, type MongoTypeMaps, type MongoTypeMapsPhantomKey, type MongoWildcardProjection, type ValidatedMongoContract, validateMongoContract, validateMongoStorage };
158
+ //# sourceMappingURL=index.d.mts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.mts","names":[],"sources":["../src/contract-types.ts","../src/validate-mongo-contract.ts","../src/validate-storage.ts"],"sourcesContent":[],"mappings":"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"}
package/dist/index.mjs ADDED
@@ -0,0 +1,266 @@
1
+ import { validateContractDomain } from "@prisma-next/contract/validate-domain";
2
+ import { type } from "arktype";
3
+
4
+ //#region src/contract-schema.ts
5
+ const ScalarFieldTypeSchema = type({
6
+ "+": "reject",
7
+ kind: "'scalar'",
8
+ codecId: "string",
9
+ "typeParams?": "Record<string, unknown>"
10
+ });
11
+ const ValueObjectFieldTypeSchema = type({
12
+ "+": "reject",
13
+ kind: "'valueObject'",
14
+ name: "string"
15
+ });
16
+ const UnionFieldTypeSchema = type({
17
+ "+": "reject",
18
+ kind: "'union'",
19
+ members: ScalarFieldTypeSchema.or(ValueObjectFieldTypeSchema).array()
20
+ });
21
+ const FieldTypeSchema = ScalarFieldTypeSchema.or(ValueObjectFieldTypeSchema).or(UnionFieldTypeSchema);
22
+ const FieldSchema = type({
23
+ "+": "reject",
24
+ type: FieldTypeSchema,
25
+ "nullable?": "boolean",
26
+ "many?": "boolean",
27
+ "dict?": "boolean"
28
+ }).pipe((field) => ({
29
+ ...field,
30
+ nullable: field.nullable ?? false
31
+ }));
32
+ const RelationOnSchema = type({
33
+ "+": "reject",
34
+ localFields: "string[]",
35
+ targetFields: "string[]"
36
+ });
37
+ const RelationSchema = type({
38
+ "+": "reject",
39
+ to: "string",
40
+ cardinality: "'1:1' | '1:N' | 'N:1'",
41
+ "on?": RelationOnSchema
42
+ });
43
+ const StorageRelationEntrySchema = type({
44
+ "+": "reject",
45
+ field: "string"
46
+ });
47
+ const MongoJsonPrimitiveSchema = type.declare().type("string | number | boolean | null");
48
+ function isMongoJsonRecord(value) {
49
+ if (typeof value !== "object" || value === null || Array.isArray(value)) return false;
50
+ const prototype = Object.getPrototypeOf(value);
51
+ return prototype === Object.prototype || prototype === null;
52
+ }
53
+ function withUnseenReference(value, seen, visit) {
54
+ if (seen.has(value)) return false;
55
+ seen.add(value);
56
+ const result = visit();
57
+ seen.delete(value);
58
+ return result;
59
+ }
60
+ function isMongoJsonObject(value, seen) {
61
+ return isMongoJsonRecord(value) && withUnseenReference(value, seen, () => Object.values(value).every((entry) => isMongoJsonValue(entry, seen)));
62
+ }
63
+ function isMongoJsonValue(value, seen = /* @__PURE__ */ new WeakSet()) {
64
+ if (MongoJsonPrimitiveSchema.allows(value)) return true;
65
+ if (Array.isArray(value)) return withUnseenReference(value, seen, () => value.every((entry) => isMongoJsonValue(entry, seen)));
66
+ return isMongoJsonObject(value, seen);
67
+ }
68
+ const MongoJsonValueSchema = type("unknown").narrow((value, ctx) => isMongoJsonValue(value) ? true : ctx.mustBe("a JSON-serializable MongoJsonValue"));
69
+ const MongoJsonObjectSchema = type({ "[string]": "unknown" }).narrow((value, ctx) => isMongoJsonRecord(value) && Object.values(value).every((entry) => MongoJsonValueSchema.allows(entry)) ? true : ctx.mustBe("a JSON object with MongoJsonValue entries"));
70
+ const NumberRecordSchema = type({ "[string]": "number" });
71
+ const IndexFieldsSchema = type({
72
+ "+": "reject",
73
+ "[string]": "1 | -1 | \"text\" | \"2dsphere\" | \"2d\" | \"hashed\""
74
+ }).narrow((fields, ctx) => Object.keys(fields).length > 0 ? true : ctx.mustBe("an index field map with at least one entry"));
75
+ const CollationSchema = type({
76
+ "+": "reject",
77
+ locale: "string",
78
+ "caseLevel?": "boolean",
79
+ "caseFirst?": "\"off\" | \"upper\" | \"lower\"",
80
+ "strength?": "1 | 2 | 3 | 4 | 5",
81
+ "numericOrdering?": "boolean",
82
+ "alternate?": "\"non-ignorable\" | \"shifted\"",
83
+ "maxVariable?": "\"punct\" | \"space\"",
84
+ "backwards?": "boolean",
85
+ "normalization?": "boolean"
86
+ });
87
+ const WildcardProjectionSchema = type({
88
+ "+": "reject",
89
+ "[string]": "0 | 1"
90
+ });
91
+ const IndexOptionsSchema = type({
92
+ "+": "reject",
93
+ "unique?": "boolean",
94
+ "name?": "string",
95
+ "partialFilterExpression?": MongoJsonObjectSchema,
96
+ "sparse?": "boolean",
97
+ "expireAfterSeconds?": "number",
98
+ "weights?": NumberRecordSchema,
99
+ "default_language?": "string",
100
+ "language_override?": "string",
101
+ "textIndexVersion?": "number",
102
+ "2dsphereIndexVersion?": "number",
103
+ "bits?": "number",
104
+ "min?": "number",
105
+ "max?": "number",
106
+ "bucketSize?": "number",
107
+ "hidden?": "boolean",
108
+ "collation?": CollationSchema,
109
+ "wildcardProjection?": WildcardProjectionSchema
110
+ });
111
+ const IndexSchema = type({
112
+ "+": "reject",
113
+ fields: IndexFieldsSchema,
114
+ "options?": IndexOptionsSchema
115
+ });
116
+ const IndexOptionDefaultsSchema = type({
117
+ "+": "reject",
118
+ "storageEngine?": MongoJsonObjectSchema
119
+ });
120
+ const TimeSeriesCollectionOptionsSchema = type({
121
+ "+": "reject",
122
+ timeField: "string",
123
+ "metaField?": "string",
124
+ "granularity?": "\"seconds\" | \"minutes\" | \"hours\"",
125
+ "bucketMaxSpanSeconds?": "number",
126
+ "bucketRoundingSeconds?": "number"
127
+ });
128
+ const ClusteredCollectionKeySchema = type({
129
+ "+": "reject",
130
+ "[string]": "1"
131
+ }).narrow((key, ctx) => Object.keys(key).length > 0 ? true : ctx.mustBe("a clustered index key map with at least one entry"));
132
+ const ClusteredCollectionOptionsSchema = type({
133
+ "+": "reject",
134
+ "name?": "string",
135
+ key: ClusteredCollectionKeySchema,
136
+ unique: "boolean"
137
+ });
138
+ const ChangeStreamPreAndPostImagesSchema = type({
139
+ "+": "reject",
140
+ enabled: "boolean"
141
+ });
142
+ const CollectionOptionsSchema = type({
143
+ "+": "reject",
144
+ "capped?": "boolean",
145
+ "size?": "number",
146
+ "max?": "number",
147
+ "storageEngine?": MongoJsonObjectSchema,
148
+ "indexOptionDefaults?": IndexOptionDefaultsSchema,
149
+ "collation?": CollationSchema,
150
+ "timeseries?": TimeSeriesCollectionOptionsSchema,
151
+ "clusteredIndex?": ClusteredCollectionOptionsSchema,
152
+ "expireAfterSeconds?": "number",
153
+ "changeStreamPreAndPostImages?": ChangeStreamPreAndPostImagesSchema
154
+ });
155
+ const ModelStorageSchema = type({
156
+ "+": "reject",
157
+ "collection?": "string",
158
+ "relations?": type({ "[string]": StorageRelationEntrySchema })
159
+ });
160
+ const DiscriminatorSchema = type({
161
+ "+": "reject",
162
+ field: "string"
163
+ });
164
+ const VariantEntrySchema = type({
165
+ "+": "reject",
166
+ value: "string"
167
+ });
168
+ const ModelDefinitionSchema = type({
169
+ "+": "reject",
170
+ fields: type({ "[string]": FieldSchema }),
171
+ storage: ModelStorageSchema,
172
+ "relations?": type({ "[string]": RelationSchema }),
173
+ "discriminator?": DiscriminatorSchema,
174
+ "variants?": type({ "[string]": VariantEntrySchema }),
175
+ "base?": "string",
176
+ "owner?": "string"
177
+ });
178
+ const StorageCollectionSchema = type({
179
+ "+": "reject",
180
+ "indexes?": IndexSchema.array(),
181
+ "options?": CollectionOptionsSchema
182
+ });
183
+ const MongoContractSchema = type({
184
+ "+": "reject",
185
+ targetFamily: "'mongo'",
186
+ "schemaVersion?": "string",
187
+ "target?": "string",
188
+ "storageHash?": "string",
189
+ "profileHash?": "string",
190
+ roots: "Record<string, string>",
191
+ "capabilities?": "Record<string, unknown>",
192
+ "extensionPacks?": "Record<string, unknown>",
193
+ "meta?": "Record<string, unknown>",
194
+ "sources?": "Record<string, unknown>",
195
+ "_generated?": "Record<string, unknown>",
196
+ storage: type({
197
+ "+": "reject",
198
+ collections: type({ "[string]": StorageCollectionSchema }),
199
+ "storageHash?": "string"
200
+ }),
201
+ models: type({ "[string]": ModelDefinitionSchema }),
202
+ "valueObjects?": type({ "[string]": type({
203
+ "+": "reject",
204
+ fields: type({ "[string]": FieldSchema })
205
+ }) })
206
+ });
207
+
208
+ //#endregion
209
+ //#region src/validate-storage.ts
210
+ function validateMongoStorage(contract) {
211
+ const errors = [];
212
+ for (const [modelName, model] of Object.entries(contract.models)) {
213
+ if (model.storage.collection && !(model.storage.collection in contract.storage.collections)) errors.push(`Model "${modelName}" references collection "${model.storage.collection}" which is not in storage.collections`);
214
+ if (model.base) {
215
+ const baseModel = contract.models[model.base];
216
+ if (baseModel) {
217
+ const variantCollection = model.storage.collection;
218
+ const baseCollection = baseModel.storage.collection;
219
+ if (variantCollection !== baseCollection) errors.push(`Mongo does not support multi-table inheritance; variant "${modelName}" must share its base's collection ("${baseCollection ?? "(none)"}"), but has "${variantCollection ?? "(none)"}"`);
220
+ }
221
+ }
222
+ for (const [relName, relation] of Object.entries(model.relations ?? {})) {
223
+ const targetModel = contract.models[relation.to];
224
+ if (targetModel?.owner) {
225
+ if (targetModel.owner !== modelName) errors.push(`Embed relation "${relName}" targets "${relation.to}" which is owned by "${targetModel.owner}", not "${modelName}"`);
226
+ if (targetModel.storage.collection) errors.push(`Embed relation "${relName}" targets "${relation.to}" which must not have a collection`);
227
+ } else if ("on" in relation && relation.on) {
228
+ for (const localField of relation.on.localFields) if (!(localField in model.fields)) errors.push(`Reference relation "${relName}": localField "${localField}" is not a field on model "${modelName}"`);
229
+ if (targetModel) {
230
+ for (const targetField of relation.on.targetFields) if (!(targetField in targetModel.fields)) errors.push(`Reference relation "${relName}": targetField "${targetField}" is not a field on model "${relation.to}"`);
231
+ }
232
+ }
233
+ }
234
+ }
235
+ if (errors.length > 0) throw new Error(`Contract storage validation failed:\n- ${errors.join("\n- ")}`);
236
+ }
237
+
238
+ //#endregion
239
+ //#region src/validate-mongo-contract.ts
240
+ function validateMongoContract(value) {
241
+ const parsed = MongoContractSchema(value);
242
+ if (parsed instanceof type.errors) throw new Error(`Contract structural validation failed: ${parsed.summary}`);
243
+ const contract = parsed;
244
+ validateContractDomain(contract);
245
+ validateMongoStorage(contract);
246
+ return {
247
+ contract,
248
+ indices: buildIndices(contract)
249
+ };
250
+ }
251
+ function buildIndices(contract) {
252
+ const variantToBase = {};
253
+ const modelToVariants = {};
254
+ for (const [modelName, model] of Object.entries(contract.models)) {
255
+ if (model.base) variantToBase[modelName] = model.base;
256
+ if (model.variants) modelToVariants[modelName] = Object.keys(model.variants);
257
+ }
258
+ return {
259
+ variantToBase,
260
+ modelToVariants
261
+ };
262
+ }
263
+
264
+ //#endregion
265
+ export { validateMongoContract, validateMongoStorage };
266
+ //# sourceMappingURL=index.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.mjs","names":["errors: string[]","arktypeType","variantToBase: Record<string, string>","modelToVariants: Record<string, string[]>"],"sources":["../src/contract-schema.ts","../src/validate-storage.ts","../src/validate-mongo-contract.ts"],"sourcesContent":["import { type } from 'arktype';\nimport type { MongoJsonObject, MongoJsonPrimitive, MongoJsonValue } from './contract-types';\n\nconst ScalarFieldTypeSchema = type({\n '+': 'reject',\n kind: \"'scalar'\",\n codecId: 'string',\n 'typeParams?': 'Record<string, unknown>',\n});\n\nconst ValueObjectFieldTypeSchema = type({\n '+': 'reject',\n kind: \"'valueObject'\",\n name: 'string',\n});\n\nconst UnionFieldTypeSchema = type({\n '+': 'reject',\n kind: \"'union'\",\n members: ScalarFieldTypeSchema.or(ValueObjectFieldTypeSchema).array(),\n});\n\nconst FieldTypeSchema = ScalarFieldTypeSchema.or(ValueObjectFieldTypeSchema).or(\n UnionFieldTypeSchema,\n);\n\nconst RawFieldSchema = type({\n '+': 'reject',\n type: FieldTypeSchema,\n 'nullable?': 'boolean',\n 'many?': 'boolean',\n 'dict?': 'boolean',\n});\n\nconst FieldSchema = RawFieldSchema.pipe((field) => ({\n ...field,\n nullable: field.nullable ?? false,\n}));\n\nconst RelationOnSchema = type({\n '+': 'reject',\n localFields: 'string[]',\n targetFields: 'string[]',\n});\n\nconst RelationSchema = type({\n '+': 'reject',\n to: 'string',\n cardinality: \"'1:1' | '1:N' | 'N:1'\",\n 'on?': RelationOnSchema,\n});\n\nconst StorageRelationEntrySchema = type({\n '+': 'reject',\n field: 'string',\n});\n\nconst MongoJsonPrimitiveSchema = type\n .declare<MongoJsonPrimitive>()\n .type('string | number | boolean | null');\n\nfunction isMongoJsonRecord(value: unknown): value is Record<string, unknown> {\n if (typeof value !== 'object' || value === null || Array.isArray(value)) {\n return false;\n }\n const prototype = Object.getPrototypeOf(value);\n return prototype === Object.prototype || prototype === null;\n}\n\nfunction withUnseenReference(value: object, seen: WeakSet<object>, visit: () => boolean): boolean {\n if (seen.has(value)) {\n return false;\n }\n\n seen.add(value);\n const result = visit();\n seen.delete(value);\n return result;\n}\n\nfunction isMongoJsonObject(value: unknown, seen: WeakSet<object>): value is MongoJsonObject {\n return (\n isMongoJsonRecord(value) &&\n withUnseenReference(value, seen, () =>\n Object.values(value).every((entry) => isMongoJsonValue(entry, seen)),\n )\n );\n}\n\nfunction isMongoJsonValue(value: unknown, seen = new WeakSet<object>()): value is MongoJsonValue {\n if (MongoJsonPrimitiveSchema.allows(value)) {\n return true;\n }\n if (Array.isArray(value)) {\n return withUnseenReference(value, seen, () =>\n value.every((entry) => isMongoJsonValue(entry, seen)),\n );\n }\n return isMongoJsonObject(value, seen);\n}\n\nconst MongoJsonValueSchema = type('unknown').narrow((value, ctx) =>\n isMongoJsonValue(value) ? true : ctx.mustBe('a JSON-serializable MongoJsonValue'),\n);\n\nconst MongoJsonObjectSchema = type({ '[string]': 'unknown' }).narrow((value, ctx) =>\n isMongoJsonRecord(value) &&\n Object.values(value).every((entry) => MongoJsonValueSchema.allows(entry))\n ? true\n : ctx.mustBe('a JSON object with MongoJsonValue entries'),\n);\n\nconst NumberRecordSchema = type({ '[string]': 'number' });\n\nconst IndexFieldsSchema = type({\n '+': 'reject',\n '[string]': '1 | -1 | \"text\" | \"2dsphere\" | \"2d\" | \"hashed\"',\n}).narrow((fields, ctx) =>\n Object.keys(fields).length > 0 ? true : ctx.mustBe('an index field map with at least one entry'),\n);\n\nconst CollationSchema = type({\n '+': 'reject',\n locale: 'string',\n 'caseLevel?': 'boolean',\n 'caseFirst?': '\"off\" | \"upper\" | \"lower\"',\n 'strength?': '1 | 2 | 3 | 4 | 5',\n 'numericOrdering?': 'boolean',\n 'alternate?': '\"non-ignorable\" | \"shifted\"',\n 'maxVariable?': '\"punct\" | \"space\"',\n 'backwards?': 'boolean',\n 'normalization?': 'boolean',\n});\n\nconst WildcardProjectionSchema = type({\n '+': 'reject',\n '[string]': '0 | 1',\n});\n\nconst IndexOptionsSchema = type({\n '+': 'reject',\n 'unique?': 'boolean',\n 'name?': 'string',\n 'partialFilterExpression?': MongoJsonObjectSchema,\n 'sparse?': 'boolean',\n 'expireAfterSeconds?': 'number',\n 'weights?': NumberRecordSchema,\n 'default_language?': 'string',\n 'language_override?': 'string',\n 'textIndexVersion?': 'number',\n '2dsphereIndexVersion?': 'number',\n 'bits?': 'number',\n 'min?': 'number',\n 'max?': 'number',\n 'bucketSize?': 'number',\n 'hidden?': 'boolean',\n 'collation?': CollationSchema,\n 'wildcardProjection?': WildcardProjectionSchema,\n});\n\nconst IndexSchema = type({\n '+': 'reject',\n fields: IndexFieldsSchema,\n 'options?': IndexOptionsSchema,\n});\n\nconst IndexOptionDefaultsSchema = type({\n '+': 'reject',\n 'storageEngine?': MongoJsonObjectSchema,\n});\n\nconst TimeSeriesCollectionOptionsSchema = type({\n '+': 'reject',\n timeField: 'string',\n 'metaField?': 'string',\n 'granularity?': '\"seconds\" | \"minutes\" | \"hours\"',\n 'bucketMaxSpanSeconds?': 'number',\n 'bucketRoundingSeconds?': 'number',\n});\n\nconst ClusteredCollectionKeySchema = type({\n '+': 'reject',\n '[string]': '1',\n}).narrow((key, ctx) =>\n Object.keys(key).length > 0\n ? true\n : ctx.mustBe('a clustered index key map with at least one entry'),\n);\n\nconst ClusteredCollectionOptionsSchema = type({\n '+': 'reject',\n 'name?': 'string',\n key: ClusteredCollectionKeySchema,\n unique: 'boolean',\n});\n\nconst ChangeStreamPreAndPostImagesSchema = type({\n '+': 'reject',\n enabled: 'boolean',\n});\n\nconst CollectionOptionsSchema = type({\n '+': 'reject',\n 'capped?': 'boolean',\n 'size?': 'number',\n 'max?': 'number',\n 'storageEngine?': MongoJsonObjectSchema,\n 'indexOptionDefaults?': IndexOptionDefaultsSchema,\n 'collation?': CollationSchema,\n 'timeseries?': TimeSeriesCollectionOptionsSchema,\n 'clusteredIndex?': ClusteredCollectionOptionsSchema,\n 'expireAfterSeconds?': 'number',\n 'changeStreamPreAndPostImages?': ChangeStreamPreAndPostImagesSchema,\n});\n\nconst ModelStorageSchema = type({\n '+': 'reject',\n 'collection?': 'string',\n 'relations?': type({ '[string]': StorageRelationEntrySchema }),\n});\n\nconst DiscriminatorSchema = type({\n '+': 'reject',\n field: 'string',\n});\n\nconst VariantEntrySchema = type({\n '+': 'reject',\n value: 'string',\n});\n\nconst ModelDefinitionSchema = type({\n '+': 'reject',\n fields: type({ '[string]': FieldSchema }),\n storage: ModelStorageSchema,\n 'relations?': type({ '[string]': RelationSchema }),\n 'discriminator?': DiscriminatorSchema,\n 'variants?': type({ '[string]': VariantEntrySchema }),\n 'base?': 'string',\n 'owner?': 'string',\n});\n\nconst StorageCollectionSchema = type({\n '+': 'reject',\n 'indexes?': IndexSchema.array(),\n 'options?': CollectionOptionsSchema,\n});\n\nexport const MongoContractSchema = type({\n '+': 'reject',\n targetFamily: \"'mongo'\",\n 'schemaVersion?': 'string',\n 'target?': 'string',\n 'storageHash?': 'string',\n 'profileHash?': 'string',\n roots: 'Record<string, string>',\n 'capabilities?': 'Record<string, unknown>',\n 'extensionPacks?': 'Record<string, unknown>',\n 'meta?': 'Record<string, unknown>',\n 'sources?': 'Record<string, unknown>',\n '_generated?': 'Record<string, unknown>',\n storage: type({\n '+': 'reject',\n collections: type({ '[string]': StorageCollectionSchema }),\n 'storageHash?': 'string',\n }),\n models: type({ '[string]': ModelDefinitionSchema }),\n 'valueObjects?': type({\n '[string]': type({ '+': 'reject', fields: type({ '[string]': FieldSchema }) }),\n }),\n});\n","import type { MongoContract } from './contract-types';\n\nexport function validateMongoStorage(contract: MongoContract): void {\n const errors: string[] = [];\n\n for (const [modelName, model] of Object.entries(contract.models)) {\n if (model.storage.collection && !(model.storage.collection in contract.storage.collections)) {\n errors.push(\n `Model \"${modelName}\" references collection \"${model.storage.collection}\" which is not in storage.collections`,\n );\n }\n\n // Mongo does not support multi-table inheritance (ADR 2): all variants of a base\n // must share the same collection (single-table inheritance only).\n if (model.base) {\n const baseModel = contract.models[model.base];\n if (baseModel) {\n const variantCollection = model.storage.collection;\n const baseCollection = baseModel.storage.collection;\n if (variantCollection !== baseCollection) {\n errors.push(\n `Mongo does not support multi-table inheritance; variant \"${modelName}\" must share its base's collection (\"${baseCollection ?? '(none)'}\"), but has \"${variantCollection ?? '(none)'}\"`,\n );\n }\n }\n }\n\n for (const [relName, relation] of Object.entries(model.relations ?? {})) {\n const targetModel = contract.models[relation.to];\n\n if (targetModel?.owner) {\n if (targetModel.owner !== modelName) {\n errors.push(\n `Embed relation \"${relName}\" targets \"${relation.to}\" which is owned by \"${targetModel.owner}\", not \"${modelName}\"`,\n );\n }\n if (targetModel.storage.collection) {\n errors.push(\n `Embed relation \"${relName}\" targets \"${relation.to}\" which must not have a collection`,\n );\n }\n } else if ('on' in relation && relation.on) {\n for (const localField of relation.on.localFields) {\n if (!(localField in model.fields)) {\n errors.push(\n `Reference relation \"${relName}\": localField \"${localField}\" is not a field on model \"${modelName}\"`,\n );\n }\n }\n\n if (targetModel) {\n for (const targetField of relation.on.targetFields) {\n if (!(targetField in targetModel.fields)) {\n errors.push(\n `Reference relation \"${relName}\": targetField \"${targetField}\" is not a field on model \"${relation.to}\"`,\n );\n }\n }\n }\n }\n }\n }\n\n if (errors.length > 0) {\n throw new Error(`Contract storage validation failed:\\n- ${errors.join('\\n- ')}`);\n }\n}\n","import { validateContractDomain } from '@prisma-next/contract/validate-domain';\nimport { type as arktypeType } from 'arktype';\nimport { MongoContractSchema } from './contract-schema';\nimport type { MongoContract } from './contract-types';\nimport { validateMongoStorage } from './validate-storage';\n\nexport interface MongoContractIndices {\n readonly variantToBase: Record<string, string>;\n readonly modelToVariants: Record<string, string[]>;\n}\n\nexport interface ValidatedMongoContract<TContract extends MongoContract> {\n readonly contract: TContract;\n readonly indices: MongoContractIndices;\n}\n\nexport function validateMongoContract<TContract extends MongoContract>(\n value: unknown,\n): ValidatedMongoContract<TContract> {\n const parsed = MongoContractSchema(value);\n if (parsed instanceof arktypeType.errors) {\n throw new Error(`Contract structural validation failed: ${parsed.summary}`);\n }\n\n const contract = parsed as unknown as TContract;\n\n validateContractDomain(contract);\n validateMongoStorage(contract);\n\n const indices = buildIndices(contract);\n\n return { contract, indices };\n}\n\nfunction buildIndices(contract: MongoContract): MongoContractIndices {\n const variantToBase: Record<string, string> = {};\n const modelToVariants: Record<string, string[]> = {};\n\n for (const [modelName, model] of Object.entries(contract.models)) {\n if (model.base) {\n variantToBase[modelName] = model.base;\n }\n if (model.variants) {\n modelToVariants[modelName] = Object.keys(model.variants);\n }\n }\n\n return { variantToBase, modelToVariants };\n}\n"],"mappings":"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"}
package/package.json CHANGED
@@ -1,20 +1,20 @@
1
1
  {
2
2
  "name": "@prisma-next/mongo-contract",
3
- "version": "0.0.1",
3
+ "version": "0.3.0-dev.147",
4
4
  "type": "module",
5
5
  "sideEffects": false,
6
6
  "description": "Contract types and validation for Prisma Next MongoDB support",
7
7
  "dependencies": {
8
8
  "arktype": "^2.1.29",
9
- "@prisma-next/contract": "0.0.1"
9
+ "@prisma-next/contract": "0.3.0-dev.147"
10
10
  },
11
11
  "devDependencies": {
12
12
  "tsdown": "0.18.4",
13
13
  "typescript": "5.9.3",
14
14
  "vitest": "4.0.17",
15
- "@prisma-next/test-utils": "0.0.1",
16
15
  "@prisma-next/tsconfig": "0.0.0",
17
- "@prisma-next/adapter-mongo": "0.0.1",
16
+ "@prisma-next/test-utils": "0.0.1",
17
+ "@prisma-next/adapter-mongo": "0.3.0-dev.147",
18
18
  "@prisma-next/tsdown": "0.0.0"
19
19
  },
20
20
  "files": [
@@ -1,4 +1,5 @@
1
1
  import { type } from 'arktype';
2
+ import type { MongoJsonObject, MongoJsonPrimitive, MongoJsonValue } from './contract-types';
2
3
 
3
4
  const ScalarFieldTypeSchema = type({
4
5
  '+': 'reject',
@@ -54,6 +55,164 @@ const StorageRelationEntrySchema = type({
54
55
  field: 'string',
55
56
  });
56
57
 
58
+ const MongoJsonPrimitiveSchema = type
59
+ .declare<MongoJsonPrimitive>()
60
+ .type('string | number | boolean | null');
61
+
62
+ function isMongoJsonRecord(value: unknown): value is Record<string, unknown> {
63
+ if (typeof value !== 'object' || value === null || Array.isArray(value)) {
64
+ return false;
65
+ }
66
+ const prototype = Object.getPrototypeOf(value);
67
+ return prototype === Object.prototype || prototype === null;
68
+ }
69
+
70
+ function withUnseenReference(value: object, seen: WeakSet<object>, visit: () => boolean): boolean {
71
+ if (seen.has(value)) {
72
+ return false;
73
+ }
74
+
75
+ seen.add(value);
76
+ const result = visit();
77
+ seen.delete(value);
78
+ return result;
79
+ }
80
+
81
+ function isMongoJsonObject(value: unknown, seen: WeakSet<object>): value is MongoJsonObject {
82
+ return (
83
+ isMongoJsonRecord(value) &&
84
+ withUnseenReference(value, seen, () =>
85
+ Object.values(value).every((entry) => isMongoJsonValue(entry, seen)),
86
+ )
87
+ );
88
+ }
89
+
90
+ function isMongoJsonValue(value: unknown, seen = new WeakSet<object>()): value is MongoJsonValue {
91
+ if (MongoJsonPrimitiveSchema.allows(value)) {
92
+ return true;
93
+ }
94
+ if (Array.isArray(value)) {
95
+ return withUnseenReference(value, seen, () =>
96
+ value.every((entry) => isMongoJsonValue(entry, seen)),
97
+ );
98
+ }
99
+ return isMongoJsonObject(value, seen);
100
+ }
101
+
102
+ const MongoJsonValueSchema = type('unknown').narrow((value, ctx) =>
103
+ isMongoJsonValue(value) ? true : ctx.mustBe('a JSON-serializable MongoJsonValue'),
104
+ );
105
+
106
+ const MongoJsonObjectSchema = type({ '[string]': 'unknown' }).narrow((value, ctx) =>
107
+ isMongoJsonRecord(value) &&
108
+ Object.values(value).every((entry) => MongoJsonValueSchema.allows(entry))
109
+ ? true
110
+ : ctx.mustBe('a JSON object with MongoJsonValue entries'),
111
+ );
112
+
113
+ const NumberRecordSchema = type({ '[string]': 'number' });
114
+
115
+ const IndexFieldsSchema = type({
116
+ '+': 'reject',
117
+ '[string]': '1 | -1 | "text" | "2dsphere" | "2d" | "hashed"',
118
+ }).narrow((fields, ctx) =>
119
+ Object.keys(fields).length > 0 ? true : ctx.mustBe('an index field map with at least one entry'),
120
+ );
121
+
122
+ const CollationSchema = type({
123
+ '+': 'reject',
124
+ locale: 'string',
125
+ 'caseLevel?': 'boolean',
126
+ 'caseFirst?': '"off" | "upper" | "lower"',
127
+ 'strength?': '1 | 2 | 3 | 4 | 5',
128
+ 'numericOrdering?': 'boolean',
129
+ 'alternate?': '"non-ignorable" | "shifted"',
130
+ 'maxVariable?': '"punct" | "space"',
131
+ 'backwards?': 'boolean',
132
+ 'normalization?': 'boolean',
133
+ });
134
+
135
+ const WildcardProjectionSchema = type({
136
+ '+': 'reject',
137
+ '[string]': '0 | 1',
138
+ });
139
+
140
+ const IndexOptionsSchema = type({
141
+ '+': 'reject',
142
+ 'unique?': 'boolean',
143
+ 'name?': 'string',
144
+ 'partialFilterExpression?': MongoJsonObjectSchema,
145
+ 'sparse?': 'boolean',
146
+ 'expireAfterSeconds?': 'number',
147
+ 'weights?': NumberRecordSchema,
148
+ 'default_language?': 'string',
149
+ 'language_override?': 'string',
150
+ 'textIndexVersion?': 'number',
151
+ '2dsphereIndexVersion?': 'number',
152
+ 'bits?': 'number',
153
+ 'min?': 'number',
154
+ 'max?': 'number',
155
+ 'bucketSize?': 'number',
156
+ 'hidden?': 'boolean',
157
+ 'collation?': CollationSchema,
158
+ 'wildcardProjection?': WildcardProjectionSchema,
159
+ });
160
+
161
+ const IndexSchema = type({
162
+ '+': 'reject',
163
+ fields: IndexFieldsSchema,
164
+ 'options?': IndexOptionsSchema,
165
+ });
166
+
167
+ const IndexOptionDefaultsSchema = type({
168
+ '+': 'reject',
169
+ 'storageEngine?': MongoJsonObjectSchema,
170
+ });
171
+
172
+ const TimeSeriesCollectionOptionsSchema = type({
173
+ '+': 'reject',
174
+ timeField: 'string',
175
+ 'metaField?': 'string',
176
+ 'granularity?': '"seconds" | "minutes" | "hours"',
177
+ 'bucketMaxSpanSeconds?': 'number',
178
+ 'bucketRoundingSeconds?': 'number',
179
+ });
180
+
181
+ const ClusteredCollectionKeySchema = type({
182
+ '+': 'reject',
183
+ '[string]': '1',
184
+ }).narrow((key, ctx) =>
185
+ Object.keys(key).length > 0
186
+ ? true
187
+ : ctx.mustBe('a clustered index key map with at least one entry'),
188
+ );
189
+
190
+ const ClusteredCollectionOptionsSchema = type({
191
+ '+': 'reject',
192
+ 'name?': 'string',
193
+ key: ClusteredCollectionKeySchema,
194
+ unique: 'boolean',
195
+ });
196
+
197
+ const ChangeStreamPreAndPostImagesSchema = type({
198
+ '+': 'reject',
199
+ enabled: 'boolean',
200
+ });
201
+
202
+ const CollectionOptionsSchema = type({
203
+ '+': 'reject',
204
+ 'capped?': 'boolean',
205
+ 'size?': 'number',
206
+ 'max?': 'number',
207
+ 'storageEngine?': MongoJsonObjectSchema,
208
+ 'indexOptionDefaults?': IndexOptionDefaultsSchema,
209
+ 'collation?': CollationSchema,
210
+ 'timeseries?': TimeSeriesCollectionOptionsSchema,
211
+ 'clusteredIndex?': ClusteredCollectionOptionsSchema,
212
+ 'expireAfterSeconds?': 'number',
213
+ 'changeStreamPreAndPostImages?': ChangeStreamPreAndPostImagesSchema,
214
+ });
215
+
57
216
  const ModelStorageSchema = type({
58
217
  '+': 'reject',
59
218
  'collection?': 'string',
@@ -81,7 +240,11 @@ const ModelDefinitionSchema = type({
81
240
  'owner?': 'string',
82
241
  });
83
242
 
84
- const StorageCollectionSchema = type({ '+': 'reject' });
243
+ const StorageCollectionSchema = type({
244
+ '+': 'reject',
245
+ 'indexes?': IndexSchema.array(),
246
+ 'options?': CollectionOptionsSchema,
247
+ });
85
248
 
86
249
  export const MongoContractSchema = type({
87
250
  '+': 'reject',
@@ -6,7 +6,108 @@ import type {
6
6
  StorageBase,
7
7
  } from '@prisma-next/contract/types';
8
8
 
9
- export type MongoStorageCollection = Record<string, never>;
9
+ export type MongoIndexFieldValue = 1 | -1 | 'text' | '2dsphere' | '2d' | 'hashed';
10
+
11
+ export type MongoIndexFields = Record<string, MongoIndexFieldValue>;
12
+
13
+ export type MongoJsonPrimitive = string | number | boolean | null;
14
+
15
+ export type MongoJsonValue = MongoJsonPrimitive | readonly MongoJsonValue[] | MongoJsonObject;
16
+
17
+ export type MongoJsonObject = {
18
+ readonly [key: string]: MongoJsonValue;
19
+ };
20
+
21
+ export type MongoCollationCaseFirst = 'off' | 'upper' | 'lower';
22
+
23
+ export type MongoCollationStrength = 1 | 2 | 3 | 4 | 5;
24
+
25
+ export type MongoCollationAlternate = 'non-ignorable' | 'shifted';
26
+
27
+ export type MongoCollationMaxVariable = 'punct' | 'space';
28
+
29
+ export type MongoCollationOptions = {
30
+ readonly locale: string;
31
+ readonly caseLevel?: boolean;
32
+ readonly caseFirst?: MongoCollationCaseFirst;
33
+ readonly strength?: MongoCollationStrength;
34
+ readonly numericOrdering?: boolean;
35
+ readonly alternate?: MongoCollationAlternate;
36
+ readonly maxVariable?: MongoCollationMaxVariable;
37
+ readonly backwards?: boolean;
38
+ readonly normalization?: boolean;
39
+ };
40
+
41
+ export type MongoWildcardProjection = Readonly<Record<string, 0 | 1>>;
42
+
43
+ export type MongoIndexOptions = {
44
+ readonly unique?: boolean;
45
+ readonly name?: string;
46
+ readonly partialFilterExpression?: MongoJsonObject;
47
+ readonly sparse?: boolean;
48
+ readonly expireAfterSeconds?: number;
49
+ readonly weights?: Readonly<Record<string, number>>;
50
+ readonly default_language?: string;
51
+ readonly language_override?: string;
52
+ readonly textIndexVersion?: number;
53
+ readonly '2dsphereIndexVersion'?: number;
54
+ readonly bits?: number;
55
+ readonly min?: number;
56
+ readonly max?: number;
57
+ readonly bucketSize?: number;
58
+ readonly hidden?: boolean;
59
+ readonly collation?: MongoCollationOptions;
60
+ readonly wildcardProjection?: MongoWildcardProjection;
61
+ };
62
+
63
+ export type MongoIndex = {
64
+ readonly fields: MongoIndexFields;
65
+ readonly options?: MongoIndexOptions;
66
+ };
67
+
68
+ export type MongoIndexOptionDefaults = {
69
+ readonly storageEngine?: MongoJsonObject;
70
+ };
71
+
72
+ export type MongoTimeSeriesGranularity = 'seconds' | 'minutes' | 'hours';
73
+
74
+ export type MongoTimeSeriesCollectionOptions = {
75
+ readonly timeField: string;
76
+ readonly metaField?: string;
77
+ readonly granularity?: MongoTimeSeriesGranularity;
78
+ readonly bucketMaxSpanSeconds?: number;
79
+ readonly bucketRoundingSeconds?: number;
80
+ };
81
+
82
+ export type MongoClusteredCollectionKey = Readonly<Record<string, 1>>;
83
+
84
+ export type MongoClusteredCollectionOptions = {
85
+ readonly name?: string;
86
+ readonly key: MongoClusteredCollectionKey;
87
+ readonly unique: boolean;
88
+ };
89
+
90
+ export type MongoChangeStreamPreAndPostImagesOptions = {
91
+ readonly enabled: boolean;
92
+ };
93
+
94
+ export type MongoCollectionOptions = {
95
+ readonly capped?: boolean;
96
+ readonly size?: number;
97
+ readonly max?: number;
98
+ readonly storageEngine?: MongoJsonObject;
99
+ readonly indexOptionDefaults?: MongoIndexOptionDefaults;
100
+ readonly collation?: MongoCollationOptions;
101
+ readonly timeseries?: MongoTimeSeriesCollectionOptions;
102
+ readonly clusteredIndex?: MongoClusteredCollectionOptions;
103
+ readonly expireAfterSeconds?: number;
104
+ readonly changeStreamPreAndPostImages?: MongoChangeStreamPreAndPostImagesOptions;
105
+ };
106
+
107
+ export type MongoStorageCollection = {
108
+ readonly indexes?: readonly MongoIndex[];
109
+ readonly options?: MongoCollectionOptions;
110
+ };
10
111
 
11
112
  export type MongoStorage<THash extends string = string> = StorageBase<THash> & {
12
113
  readonly collections: Record<string, MongoStorageCollection>;
@@ -67,6 +168,15 @@ type ExtractValueObjects<TContract> = TContract extends {
67
168
  ? VO
68
169
  : Record<never, never>;
69
170
 
171
+ type NormalizeContractFields<TFields> = {
172
+ [K in keyof TFields]: TFields[K] extends ContractField ? TFields[K] : never;
173
+ };
174
+
175
+ type ExtractValueObjectFields<
176
+ TValueObjects extends Record<string, ContractValueObject>,
177
+ VOName extends keyof TValueObjects,
178
+ > = NormalizeContractFields<TValueObjects[VOName]['fields']>;
179
+
70
180
  type InferFieldBaseType<
71
181
  TFieldType,
72
182
  TValueObjects extends Record<string, ContractValueObject>,
@@ -76,8 +186,8 @@ type InferFieldBaseType<
76
186
  : TFieldType extends { kind: 'valueObject'; name: infer VOName extends string }
77
187
  ? VOName extends keyof TValueObjects
78
188
  ? {
79
- -readonly [K in keyof TValueObjects[VOName]['fields']]: InferFieldType<
80
- TValueObjects[VOName]['fields'][K],
189
+ -readonly [K in keyof ExtractValueObjectFields<TValueObjects, VOName>]: InferFieldType<
190
+ ExtractValueObjectFields<TValueObjects, VOName>[K],
81
191
  TValueObjects,
82
192
  TCodecTypes
83
193
  >;
@@ -93,16 +203,18 @@ type InferFieldBaseType<
93
203
  : unknown;
94
204
 
95
205
  type InferFieldType<
96
- TField extends ContractField,
206
+ TField,
97
207
  TValueObjects extends Record<string, ContractValueObject>,
98
208
  TCodecTypes extends Record<string, { output: unknown }>,
99
- > = TField extends { many: true }
100
- ? TField['nullable'] extends true
101
- ? InferFieldBaseType<TField['type'], TValueObjects, TCodecTypes>[] | null
102
- : InferFieldBaseType<TField['type'], TValueObjects, TCodecTypes>[]
103
- : TField['nullable'] extends true
104
- ? InferFieldBaseType<TField['type'], TValueObjects, TCodecTypes> | null
105
- : InferFieldBaseType<TField['type'], TValueObjects, TCodecTypes>;
209
+ > = TField extends ContractField
210
+ ? TField extends { many: true }
211
+ ? TField['nullable'] extends true
212
+ ? InferFieldBaseType<TField['type'], TValueObjects, TCodecTypes>[] | null
213
+ : InferFieldBaseType<TField['type'], TValueObjects, TCodecTypes>[]
214
+ : TField['nullable'] extends true
215
+ ? InferFieldBaseType<TField['type'], TValueObjects, TCodecTypes> | null
216
+ : InferFieldBaseType<TField['type'], TValueObjects, TCodecTypes>
217
+ : never;
106
218
 
107
219
  export type InferModelRow<
108
220
  TContract extends MongoContractWithTypeMaps<MongoContract, MongoTypeMaps>,
@@ -3,14 +3,34 @@ export type {
3
3
  ExtractMongoFieldOutputTypes,
4
4
  ExtractMongoTypeMaps,
5
5
  InferModelRow,
6
+ MongoChangeStreamPreAndPostImagesOptions,
7
+ MongoClusteredCollectionKey,
8
+ MongoClusteredCollectionOptions,
9
+ MongoCollationAlternate,
10
+ MongoCollationCaseFirst,
11
+ MongoCollationMaxVariable,
12
+ MongoCollationOptions,
13
+ MongoCollationStrength,
14
+ MongoCollectionOptions,
6
15
  MongoContract,
7
16
  MongoContractWithTypeMaps,
17
+ MongoIndex,
18
+ MongoIndexFields,
19
+ MongoIndexFieldValue,
20
+ MongoIndexOptionDefaults,
21
+ MongoIndexOptions,
22
+ MongoJsonObject,
23
+ MongoJsonPrimitive,
24
+ MongoJsonValue,
8
25
  MongoModelDefinition,
9
26
  MongoModelStorage,
10
27
  MongoStorage,
11
28
  MongoStorageCollection,
29
+ MongoTimeSeriesCollectionOptions,
30
+ MongoTimeSeriesGranularity,
12
31
  MongoTypeMaps,
13
32
  MongoTypeMapsPhantomKey,
33
+ MongoWildcardProjection,
14
34
  } from '../contract-types';
15
35
  export type { MongoContractIndices, ValidatedMongoContract } from '../validate-mongo-contract';
16
36
  export { validateMongoContract } from '../validate-mongo-contract';