orchid-orm 0.0.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.env.example +1 -0
- package/.rush/temp/shrinkwrap-deps.json +327 -0
- package/README.md +5 -0
- package/dist/index.d.ts +228 -0
- package/dist/index.esm.js +1116 -0
- package/dist/index.esm.js.map +1 -0
- package/dist/index.js +1122 -0
- package/dist/index.js.map +1 -0
- package/jest-setup.ts +7 -0
- package/package.json +58 -0
- package/rollup.config.js +3 -0
- package/src/index.ts +3 -0
- package/src/model.test.ts +92 -0
- package/src/model.ts +169 -0
- package/src/orm.test.ts +73 -0
- package/src/orm.ts +74 -0
- package/src/relations/belongsTo.test.ts +890 -0
- package/src/relations/belongsTo.ts +247 -0
- package/src/relations/hasAndBelongsToMany.test.ts +1122 -0
- package/src/relations/hasAndBelongsToMany.ts +398 -0
- package/src/relations/hasMany.test.ts +2003 -0
- package/src/relations/hasMany.ts +332 -0
- package/src/relations/hasOne.test.ts +1219 -0
- package/src/relations/hasOne.ts +278 -0
- package/src/relations/relations.test.ts +37 -0
- package/src/relations/relations.ts +316 -0
- package/src/relations/utils.ts +12 -0
- package/src/repo.test.ts +200 -0
- package/src/repo.ts +119 -0
- package/src/test-utils/test-db.ts +32 -0
- package/src/test-utils/test-models.ts +194 -0
- package/src/test-utils/test-utils.ts +69 -0
- package/src/transaction.test.ts +45 -0
- package/src/transaction.ts +27 -0
- package/tsconfig.json +13 -0
|
@@ -0,0 +1,332 @@
|
|
|
1
|
+
import {
|
|
2
|
+
RelationData,
|
|
3
|
+
RelationInfo,
|
|
4
|
+
RelationThunkBase,
|
|
5
|
+
RelationThunks,
|
|
6
|
+
} from './relations';
|
|
7
|
+
import { Model } from '../model';
|
|
8
|
+
import {
|
|
9
|
+
addQueryOn,
|
|
10
|
+
getQueryAs,
|
|
11
|
+
HasManyNestedInsert,
|
|
12
|
+
HasManyNestedUpdate,
|
|
13
|
+
HasManyRelation,
|
|
14
|
+
CreateData,
|
|
15
|
+
JoinCallback,
|
|
16
|
+
MaybeArray,
|
|
17
|
+
Query,
|
|
18
|
+
QueryBase,
|
|
19
|
+
toSqlCacheKey,
|
|
20
|
+
WhereArg,
|
|
21
|
+
WhereResult,
|
|
22
|
+
InsertQueryData,
|
|
23
|
+
isQueryReturnsAll,
|
|
24
|
+
} from 'pqb';
|
|
25
|
+
import { getSourceRelation, getThroughRelation } from './utils';
|
|
26
|
+
|
|
27
|
+
export interface HasMany extends RelationThunkBase {
|
|
28
|
+
type: 'hasMany';
|
|
29
|
+
returns: 'many';
|
|
30
|
+
options: HasManyRelation['options'];
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
export type HasManyInfo<
|
|
34
|
+
T extends Model,
|
|
35
|
+
Relations extends RelationThunks,
|
|
36
|
+
Relation extends HasMany,
|
|
37
|
+
> = {
|
|
38
|
+
params: Relation['options'] extends { primaryKey: string }
|
|
39
|
+
? Record<
|
|
40
|
+
Relation['options']['primaryKey'],
|
|
41
|
+
T['columns']['shape'][Relation['options']['primaryKey']]['type']
|
|
42
|
+
>
|
|
43
|
+
: Relation['options'] extends { through: string }
|
|
44
|
+
? RelationInfo<
|
|
45
|
+
T,
|
|
46
|
+
Relations,
|
|
47
|
+
Relations[Relation['options']['through']]
|
|
48
|
+
>['params']
|
|
49
|
+
: never;
|
|
50
|
+
populate: Relation['options'] extends { foreignKey: string }
|
|
51
|
+
? Relation['options']['foreignKey']
|
|
52
|
+
: never;
|
|
53
|
+
chainedCreate: Relation['options'] extends { primaryKey: string }
|
|
54
|
+
? true
|
|
55
|
+
: false;
|
|
56
|
+
chainedDelete: true;
|
|
57
|
+
};
|
|
58
|
+
|
|
59
|
+
export const makeHasManyMethod = (
|
|
60
|
+
model: Query,
|
|
61
|
+
relation: HasMany,
|
|
62
|
+
relationName: string,
|
|
63
|
+
query: Query,
|
|
64
|
+
): RelationData => {
|
|
65
|
+
if ('through' in relation.options) {
|
|
66
|
+
const { through, source } = relation.options;
|
|
67
|
+
|
|
68
|
+
type ModelWithQueryMethod = Record<
|
|
69
|
+
string,
|
|
70
|
+
(params: Record<string, unknown>) => Query
|
|
71
|
+
>;
|
|
72
|
+
|
|
73
|
+
const throughRelation = getThroughRelation(model, through);
|
|
74
|
+
const sourceRelation = getSourceRelation(throughRelation, source);
|
|
75
|
+
const sourceRelationQuery = sourceRelation.query.as(relationName);
|
|
76
|
+
const sourceQuery = sourceRelation.joinQuery(
|
|
77
|
+
throughRelation.query,
|
|
78
|
+
sourceRelationQuery,
|
|
79
|
+
);
|
|
80
|
+
|
|
81
|
+
const whereExistsCallback = () => sourceQuery;
|
|
82
|
+
|
|
83
|
+
return {
|
|
84
|
+
returns: 'many',
|
|
85
|
+
method: (params: Record<string, unknown>) => {
|
|
86
|
+
const throughQuery = (model as unknown as ModelWithQueryMethod)[
|
|
87
|
+
through
|
|
88
|
+
](params);
|
|
89
|
+
|
|
90
|
+
return query.whereExists<Query, Query>(
|
|
91
|
+
throughQuery,
|
|
92
|
+
whereExistsCallback as unknown as JoinCallback<Query, Query>,
|
|
93
|
+
);
|
|
94
|
+
},
|
|
95
|
+
nestedInsert: undefined,
|
|
96
|
+
nestedUpdate: undefined,
|
|
97
|
+
joinQuery(fromQuery, toQuery) {
|
|
98
|
+
return toQuery.whereExists<Query, Query>(
|
|
99
|
+
throughRelation.joinQuery(fromQuery, throughRelation.query),
|
|
100
|
+
(() => {
|
|
101
|
+
const as = getQueryAs(toQuery);
|
|
102
|
+
return sourceRelation.joinQuery(
|
|
103
|
+
throughRelation.query,
|
|
104
|
+
sourceRelation.query.as(as),
|
|
105
|
+
);
|
|
106
|
+
}) as unknown as JoinCallback<Query, Query>,
|
|
107
|
+
);
|
|
108
|
+
},
|
|
109
|
+
reverseJoin(fromQuery, toQuery) {
|
|
110
|
+
return fromQuery.whereExists<Query, Query>(
|
|
111
|
+
throughRelation.joinQuery(fromQuery, throughRelation.query),
|
|
112
|
+
(() => {
|
|
113
|
+
const as = getQueryAs(toQuery);
|
|
114
|
+
return sourceRelation.joinQuery(
|
|
115
|
+
throughRelation.query,
|
|
116
|
+
sourceRelation.query.as(as),
|
|
117
|
+
);
|
|
118
|
+
}) as unknown as JoinCallback<Query, Query>,
|
|
119
|
+
);
|
|
120
|
+
},
|
|
121
|
+
primaryKey: sourceRelation.primaryKey,
|
|
122
|
+
};
|
|
123
|
+
}
|
|
124
|
+
|
|
125
|
+
const { primaryKey, foreignKey } = relation.options;
|
|
126
|
+
|
|
127
|
+
const fromQuerySelect = [{ selectAs: { [foreignKey]: primaryKey } }];
|
|
128
|
+
|
|
129
|
+
return {
|
|
130
|
+
returns: 'many',
|
|
131
|
+
method: (params: Record<string, unknown>) => {
|
|
132
|
+
const values = { [foreignKey]: params[primaryKey] };
|
|
133
|
+
return query.where(values)._defaults(values);
|
|
134
|
+
},
|
|
135
|
+
nestedInsert: (async (q, data) => {
|
|
136
|
+
const connect = data.filter(
|
|
137
|
+
(
|
|
138
|
+
item,
|
|
139
|
+
): item is [
|
|
140
|
+
Record<string, unknown>,
|
|
141
|
+
{ connect: WhereArg<QueryBase>[] },
|
|
142
|
+
] => Boolean(item[1].connect),
|
|
143
|
+
);
|
|
144
|
+
|
|
145
|
+
const t = query.transacting(q);
|
|
146
|
+
|
|
147
|
+
if (connect.length) {
|
|
148
|
+
await Promise.all(
|
|
149
|
+
connect.flatMap(([selfData, { connect }]) =>
|
|
150
|
+
connect.map((item) =>
|
|
151
|
+
t
|
|
152
|
+
.where<Query>(item)
|
|
153
|
+
._updateOrThrow({ [foreignKey]: selfData[primaryKey] }),
|
|
154
|
+
),
|
|
155
|
+
),
|
|
156
|
+
);
|
|
157
|
+
}
|
|
158
|
+
|
|
159
|
+
const connectOrCreate = data.filter(
|
|
160
|
+
(
|
|
161
|
+
item,
|
|
162
|
+
): item is [
|
|
163
|
+
Record<string, unknown>,
|
|
164
|
+
{
|
|
165
|
+
connectOrCreate: {
|
|
166
|
+
where: WhereArg<QueryBase>;
|
|
167
|
+
create: Record<string, unknown>;
|
|
168
|
+
}[];
|
|
169
|
+
},
|
|
170
|
+
] => Boolean(item[1].connectOrCreate),
|
|
171
|
+
);
|
|
172
|
+
|
|
173
|
+
let connected: number[];
|
|
174
|
+
if (connectOrCreate.length) {
|
|
175
|
+
connected = await Promise.all(
|
|
176
|
+
connectOrCreate.flatMap(([selfData, { connectOrCreate }]) =>
|
|
177
|
+
connectOrCreate.map((item) =>
|
|
178
|
+
(
|
|
179
|
+
t.where(item.where) as WhereResult<Query & { hasSelect: false }>
|
|
180
|
+
)._update({
|
|
181
|
+
[foreignKey]: selfData[primaryKey],
|
|
182
|
+
}),
|
|
183
|
+
),
|
|
184
|
+
),
|
|
185
|
+
);
|
|
186
|
+
} else {
|
|
187
|
+
connected = [];
|
|
188
|
+
}
|
|
189
|
+
|
|
190
|
+
let connectedI = 0;
|
|
191
|
+
const create = data.filter(
|
|
192
|
+
(
|
|
193
|
+
item,
|
|
194
|
+
): item is [
|
|
195
|
+
Record<string, unknown>,
|
|
196
|
+
{
|
|
197
|
+
create?: Record<string, unknown>[];
|
|
198
|
+
connectOrCreate?: {
|
|
199
|
+
where: WhereArg<QueryBase>;
|
|
200
|
+
create: Record<string, unknown>;
|
|
201
|
+
}[];
|
|
202
|
+
},
|
|
203
|
+
] => {
|
|
204
|
+
if (item[1].connectOrCreate) {
|
|
205
|
+
const length = item[1].connectOrCreate.length;
|
|
206
|
+
connectedI += length;
|
|
207
|
+
for (let i = length; i > 0; i--) {
|
|
208
|
+
if (connected[connectedI - i] === 0) return true;
|
|
209
|
+
}
|
|
210
|
+
}
|
|
211
|
+
return Boolean(item[1].create);
|
|
212
|
+
},
|
|
213
|
+
);
|
|
214
|
+
|
|
215
|
+
connectedI = 0;
|
|
216
|
+
if (create.length) {
|
|
217
|
+
await t._createMany(
|
|
218
|
+
create.flatMap(
|
|
219
|
+
([selfData, { create = [], connectOrCreate = [] }]) => {
|
|
220
|
+
return [
|
|
221
|
+
...create.map((item) => ({
|
|
222
|
+
[foreignKey]: selfData[primaryKey],
|
|
223
|
+
...item,
|
|
224
|
+
})),
|
|
225
|
+
...connectOrCreate
|
|
226
|
+
.filter(() => connected[connectedI++] === 0)
|
|
227
|
+
.map((item) => ({
|
|
228
|
+
[foreignKey]: selfData[primaryKey],
|
|
229
|
+
...item.create,
|
|
230
|
+
})),
|
|
231
|
+
];
|
|
232
|
+
},
|
|
233
|
+
) as CreateData<Query>[],
|
|
234
|
+
);
|
|
235
|
+
}
|
|
236
|
+
}) as HasManyNestedInsert,
|
|
237
|
+
nestedUpdate: (async (q, data, params) => {
|
|
238
|
+
if ((params.set || params.create) && isQueryReturnsAll(q)) {
|
|
239
|
+
const key = params.set ? 'set' : 'create';
|
|
240
|
+
throw new Error(`\`${key}\` option is not allowed in a batch update`);
|
|
241
|
+
}
|
|
242
|
+
|
|
243
|
+
const t = query.transacting(q);
|
|
244
|
+
if (params.create) {
|
|
245
|
+
await t._count()._createMany(
|
|
246
|
+
params.create.map((create) => ({
|
|
247
|
+
...create,
|
|
248
|
+
[foreignKey]: data[0][primaryKey],
|
|
249
|
+
})),
|
|
250
|
+
);
|
|
251
|
+
delete t.query[toSqlCacheKey];
|
|
252
|
+
}
|
|
253
|
+
|
|
254
|
+
if (params.disconnect || params.set) {
|
|
255
|
+
await t
|
|
256
|
+
.where<Query>(
|
|
257
|
+
getWhereForNestedUpdate(
|
|
258
|
+
data,
|
|
259
|
+
params.disconnect,
|
|
260
|
+
primaryKey,
|
|
261
|
+
foreignKey,
|
|
262
|
+
),
|
|
263
|
+
)
|
|
264
|
+
._update({ [foreignKey]: null });
|
|
265
|
+
|
|
266
|
+
if (params.set) {
|
|
267
|
+
delete t.query[toSqlCacheKey];
|
|
268
|
+
await t
|
|
269
|
+
.where<Query>(
|
|
270
|
+
Array.isArray(params.set)
|
|
271
|
+
? {
|
|
272
|
+
OR: params.set,
|
|
273
|
+
}
|
|
274
|
+
: params.set,
|
|
275
|
+
)
|
|
276
|
+
._update({ [foreignKey]: data[0][primaryKey] });
|
|
277
|
+
}
|
|
278
|
+
}
|
|
279
|
+
|
|
280
|
+
if (params.delete || params.update) {
|
|
281
|
+
delete t.query[toSqlCacheKey];
|
|
282
|
+
const q = t._where(
|
|
283
|
+
getWhereForNestedUpdate(
|
|
284
|
+
data,
|
|
285
|
+
params.delete || params.update?.where,
|
|
286
|
+
primaryKey,
|
|
287
|
+
foreignKey,
|
|
288
|
+
),
|
|
289
|
+
);
|
|
290
|
+
|
|
291
|
+
if (params.delete) {
|
|
292
|
+
await q._delete();
|
|
293
|
+
} else if (params.update) {
|
|
294
|
+
await q._update<WhereResult<Query>>(params.update.data);
|
|
295
|
+
}
|
|
296
|
+
}
|
|
297
|
+
}) as HasManyNestedUpdate,
|
|
298
|
+
joinQuery(fromQuery, toQuery) {
|
|
299
|
+
return addQueryOn(toQuery, fromQuery, toQuery, foreignKey, primaryKey);
|
|
300
|
+
},
|
|
301
|
+
reverseJoin(fromQuery, toQuery) {
|
|
302
|
+
return addQueryOn(fromQuery, toQuery, fromQuery, primaryKey, foreignKey);
|
|
303
|
+
},
|
|
304
|
+
primaryKey,
|
|
305
|
+
modifyRelatedQuery(relationQuery) {
|
|
306
|
+
return (query) => {
|
|
307
|
+
const fromQuery = query.clone();
|
|
308
|
+
fromQuery.query.select = fromQuerySelect;
|
|
309
|
+
(relationQuery.query as InsertQueryData).fromQuery = fromQuery;
|
|
310
|
+
};
|
|
311
|
+
},
|
|
312
|
+
};
|
|
313
|
+
};
|
|
314
|
+
|
|
315
|
+
const getWhereForNestedUpdate = (
|
|
316
|
+
data: Record<string, unknown>[],
|
|
317
|
+
params: MaybeArray<WhereArg<QueryBase>> | undefined,
|
|
318
|
+
primaryKey: string,
|
|
319
|
+
foreignKey: string,
|
|
320
|
+
) => {
|
|
321
|
+
const where: WhereArg<Query> = {
|
|
322
|
+
[foreignKey]: { in: data.map((item) => item[primaryKey]) },
|
|
323
|
+
};
|
|
324
|
+
if (params) {
|
|
325
|
+
if (Array.isArray(params)) {
|
|
326
|
+
where.OR = params;
|
|
327
|
+
} else {
|
|
328
|
+
Object.assign(where, params);
|
|
329
|
+
}
|
|
330
|
+
}
|
|
331
|
+
return where;
|
|
332
|
+
};
|