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.
@@ -0,0 +1,247 @@
1
+ import { Model } from '../model';
2
+ import {
3
+ addQueryOn,
4
+ BelongsToNestedInsert,
5
+ BelongsToNestedUpdate,
6
+ BelongsToRelation,
7
+ isQueryReturnsAll,
8
+ Query,
9
+ QueryBase,
10
+ WhereArg,
11
+ WhereResult,
12
+ } from 'pqb';
13
+ import { RelationData, RelationThunkBase } from './relations';
14
+
15
+ export interface BelongsTo extends RelationThunkBase {
16
+ type: 'belongsTo';
17
+ returns: 'one';
18
+ options: BelongsToRelation['options'];
19
+ }
20
+
21
+ export type BelongsToInfo<
22
+ T extends Model,
23
+ Relation extends BelongsTo,
24
+ FK extends string = Relation['options']['foreignKey'],
25
+ > = {
26
+ params: Record<FK, T['columns']['shape'][FK]['type']>;
27
+ populate: never;
28
+ chainedCreate: false;
29
+ chainedDelete: false;
30
+ };
31
+
32
+ export const makeBelongsToMethod = (
33
+ relation: BelongsTo,
34
+ query: Query,
35
+ ): RelationData => {
36
+ const { primaryKey, foreignKey } = relation.options;
37
+
38
+ return {
39
+ returns: 'one',
40
+ method: (params: Record<string, unknown>) => {
41
+ return query.findBy({ [primaryKey]: params[foreignKey] });
42
+ },
43
+ nestedInsert: (async (q, data) => {
44
+ const connectOrCreate = data.filter(
45
+ (
46
+ item,
47
+ ): item is {
48
+ connectOrCreate: {
49
+ where: WhereArg<QueryBase>;
50
+ create: Record<string, unknown>;
51
+ };
52
+ } => Boolean(item.connectOrCreate),
53
+ );
54
+
55
+ const t = query.transacting(q);
56
+
57
+ let connectOrCreated: unknown[];
58
+ if (connectOrCreate.length) {
59
+ connectOrCreated = await Promise.all(
60
+ connectOrCreate.map((item) =>
61
+ t.findBy(item.connectOrCreate.where)._takeOptional(),
62
+ ),
63
+ );
64
+ } else {
65
+ connectOrCreated = [];
66
+ }
67
+
68
+ let connectOrCreatedI = 0;
69
+ const create = data.filter(
70
+ (
71
+ item,
72
+ ): item is
73
+ | {
74
+ create: Record<string, unknown>;
75
+ }
76
+ | {
77
+ connectOrCreate: {
78
+ where: WhereArg<QueryBase>;
79
+ create: Record<string, unknown>;
80
+ };
81
+ } => {
82
+ if (item.connectOrCreate) {
83
+ return !connectOrCreated[connectOrCreatedI++];
84
+ } else {
85
+ return Boolean(item.create);
86
+ }
87
+ },
88
+ );
89
+
90
+ let created: unknown[];
91
+ if (create.length) {
92
+ created = (await t
93
+ .select(primaryKey)
94
+ ._createMany(
95
+ create.map((item) =>
96
+ 'create' in item ? item.create : item.connectOrCreate.create,
97
+ ),
98
+ )) as unknown[];
99
+ } else {
100
+ created = [];
101
+ }
102
+
103
+ const connect = data.filter(
104
+ (
105
+ item,
106
+ ): item is {
107
+ connect: WhereArg<QueryBase>;
108
+ } => Boolean(item.connect),
109
+ );
110
+
111
+ let connected: unknown[];
112
+ if (connect.length) {
113
+ connected = await Promise.all(
114
+ connect.map((item) => t.findBy(item.connect)._take()),
115
+ );
116
+ } else {
117
+ connected = [];
118
+ }
119
+
120
+ let createdI = 0;
121
+ let connectedI = 0;
122
+ connectOrCreatedI = 0;
123
+ return data.map((item) =>
124
+ item.connectOrCreate
125
+ ? connectOrCreated[connectOrCreatedI++] || created[createdI++]
126
+ : item.connect
127
+ ? connected[connectedI++]
128
+ : created[createdI++],
129
+ );
130
+ }) as BelongsToNestedInsert,
131
+ nestedUpdate: ((q, update, params, state) => {
132
+ if (params.upsert && isQueryReturnsAll(q)) {
133
+ throw new Error('`upsert` option is not allowed in a batch update');
134
+ }
135
+
136
+ let idForDelete: unknown;
137
+
138
+ q._beforeUpdate(async (q) => {
139
+ if (params.disconnect) {
140
+ update[foreignKey] = null;
141
+ } else if (params.set) {
142
+ if (primaryKey in params.set) {
143
+ update[foreignKey] =
144
+ params.set[primaryKey as keyof typeof params.set];
145
+ } else {
146
+ update[foreignKey] = await query
147
+ .transacting(q)
148
+ ._findBy(params.set)
149
+ ._get(primaryKey);
150
+ }
151
+ } else if (params.create) {
152
+ update[foreignKey] = await query
153
+ .transacting(q)
154
+ ._get(primaryKey)
155
+ ._create(params.create);
156
+ } else if (params.delete) {
157
+ const selectQuery = q.transacting(q);
158
+ selectQuery.query.type = undefined;
159
+ idForDelete = await selectQuery._getOptional(foreignKey);
160
+ update[foreignKey] = null;
161
+ }
162
+ });
163
+
164
+ const { upsert } = params;
165
+ if (upsert || params.update || params.delete) {
166
+ if (
167
+ !q.query.select?.includes('*') &&
168
+ !q.query.select?.includes(foreignKey)
169
+ ) {
170
+ q._select(foreignKey);
171
+ }
172
+ }
173
+
174
+ if (upsert) {
175
+ if (!state.updateLater) {
176
+ state.updateLater = {};
177
+ state.updateLaterPromises = [];
178
+ }
179
+
180
+ const { handleResult } = q.query;
181
+ q.query.handleResult = async (q, queryResult) => {
182
+ const data = (await handleResult(q, queryResult)) as Record<
183
+ string,
184
+ unknown
185
+ >[];
186
+
187
+ const id = data[0][foreignKey];
188
+ if (id !== null) {
189
+ await query
190
+ .transacting(q)
191
+ ._findBy({ [primaryKey]: id })
192
+ ._update<WhereResult<Query>>(upsert.update);
193
+ } else {
194
+ (state.updateLaterPromises as Promise<void>[]).push(
195
+ query
196
+ .transacting(q)
197
+ ._select(primaryKey)
198
+ ._create(upsert.create)
199
+ .then((result) => {
200
+ (state.updateLater as Record<string, unknown>)[foreignKey] = (
201
+ result as Record<string, unknown>
202
+ )[primaryKey];
203
+ }) as unknown as Promise<void>,
204
+ );
205
+ }
206
+
207
+ return data;
208
+ };
209
+ } else if (params.delete || params.update) {
210
+ q._afterQuery(async (q, data) => {
211
+ const id = params.delete
212
+ ? idForDelete
213
+ : Array.isArray(data)
214
+ ? data.length === 0
215
+ ? null
216
+ : {
217
+ in: data
218
+ .map((item) => item[foreignKey])
219
+ .filter((id) => id !== null),
220
+ }
221
+ : (data as Record<string, unknown>)[foreignKey];
222
+
223
+ if (id !== undefined && id !== null) {
224
+ const t = query.transacting(q)._findBy({
225
+ [primaryKey]: id,
226
+ });
227
+
228
+ if (params.delete) {
229
+ await t._delete();
230
+ } else if (params.update) {
231
+ await t._update<WhereResult<Query>>(params.update);
232
+ }
233
+ }
234
+ });
235
+ }
236
+
237
+ return !params.update && !params.upsert;
238
+ }) as BelongsToNestedUpdate,
239
+ joinQuery(fromQuery, toQuery) {
240
+ return addQueryOn(toQuery, fromQuery, toQuery, primaryKey, foreignKey);
241
+ },
242
+ reverseJoin(fromQuery, toQuery) {
243
+ return addQueryOn(fromQuery, toQuery, fromQuery, foreignKey, primaryKey);
244
+ },
245
+ primaryKey,
246
+ };
247
+ };