@casekit/orm 0.0.1-alpha.6 → 0.0.1-alpha.8

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 (151) hide show
  1. package/lib/migrate/commands/implode.js +1 -1
  2. package/lib/migrate/commands/implode.js.map +1 -1
  3. package/lib/migrate/sql/createTableSql.test.js +1 -0
  4. package/lib/migrate/sql/createTableSql.test.js.map +1 -1
  5. package/lib/orm.d.ts.map +1 -1
  6. package/lib/orm.js +4 -2
  7. package/lib/orm.js.map +1 -1
  8. package/lib/queries/clauses/where/buildWhereClauses.d.ts +1 -1
  9. package/lib/queries/clauses/where/buildWhereClauses.d.ts.map +1 -1
  10. package/lib/queries/clauses/where/buildWhereClauses.js.map +1 -1
  11. package/lib/queries/clauses/where/buildWhereClauses.test.js +1 -1
  12. package/lib/queries/clauses/where/buildWhereClauses.test.js.map +1 -1
  13. package/lib/queries/count/buildCount.d.ts.map +1 -1
  14. package/lib/queries/count/buildCount.js +33 -29
  15. package/lib/queries/count/buildCount.js.map +1 -1
  16. package/lib/queries/count/countToSql.d.ts.map +1 -1
  17. package/lib/queries/count/countToSql.js +12 -17
  18. package/lib/queries/count/countToSql.js.map +1 -1
  19. package/lib/queries/count/types/CountBuilder.d.ts +12 -4
  20. package/lib/queries/count/types/CountBuilder.d.ts.map +1 -1
  21. package/lib/queries/count.d.ts.map +1 -1
  22. package/lib/queries/count.js +3 -1
  23. package/lib/queries/count.js.map +1 -1
  24. package/lib/queries/create/buildCreate.d.ts +1 -1
  25. package/lib/queries/create/buildCreate.d.ts.map +1 -1
  26. package/lib/queries/create/buildCreate.js +1 -1
  27. package/lib/queries/create/buildCreate.js.map +1 -1
  28. package/lib/queries/create/createResultSchema.d.ts.map +1 -1
  29. package/lib/queries/create/createResultSchema.js +3 -1
  30. package/lib/queries/create/createResultSchema.js.map +1 -1
  31. package/lib/queries/create/createToSql.d.ts.map +1 -1
  32. package/lib/queries/create/createToSql.js +2 -2
  33. package/lib/queries/create/createToSql.js.map +1 -1
  34. package/lib/queries/createMany.d.ts.map +1 -1
  35. package/lib/queries/createMany.js +3 -1
  36. package/lib/queries/createMany.js.map +1 -1
  37. package/lib/queries/createOne.d.ts.map +1 -1
  38. package/lib/queries/createOne.js +6 -0
  39. package/lib/queries/createOne.js.map +1 -1
  40. package/lib/queries/delete/buildDelete.d.ts +1 -1
  41. package/lib/queries/delete/buildDelete.d.ts.map +1 -1
  42. package/lib/queries/delete/buildDelete.js +1 -1
  43. package/lib/queries/delete/buildDelete.js.map +1 -1
  44. package/lib/queries/delete/deleteResultSchema.d.ts.map +1 -1
  45. package/lib/queries/delete/deleteResultSchema.js +3 -1
  46. package/lib/queries/delete/deleteResultSchema.js.map +1 -1
  47. package/lib/queries/delete/deleteToSql.js +1 -1
  48. package/lib/queries/delete/deleteToSql.js.map +1 -1
  49. package/lib/queries/deleteMany.d.ts.map +1 -1
  50. package/lib/queries/deleteMany.js +3 -1
  51. package/lib/queries/deleteMany.js.map +1 -1
  52. package/lib/queries/find/buildFind.d.ts.map +1 -1
  53. package/lib/queries/find/buildFind.js +34 -30
  54. package/lib/queries/find/buildFind.js.map +1 -1
  55. package/lib/queries/find/findResultSchema.d.ts.map +1 -1
  56. package/lib/queries/find/findResultSchema.js +9 -2
  57. package/lib/queries/find/findResultSchema.js.map +1 -1
  58. package/lib/queries/find/findToSql.d.ts.map +1 -1
  59. package/lib/queries/find/findToSql.js +12 -17
  60. package/lib/queries/find/findToSql.js.map +1 -1
  61. package/lib/queries/find/tests/findMany.nullable-relations.test.d.ts +2 -0
  62. package/lib/queries/find/tests/findMany.nullable-relations.test.d.ts.map +1 -0
  63. package/lib/queries/find/tests/findMany.nullable-relations.test.js +118 -0
  64. package/lib/queries/find/tests/findMany.nullable-relations.test.js.map +1 -0
  65. package/lib/queries/find/types/FindBuilder.d.ts +12 -4
  66. package/lib/queries/find/types/FindBuilder.d.ts.map +1 -1
  67. package/lib/queries/findMany.d.ts.map +1 -1
  68. package/lib/queries/findMany.js +17 -9
  69. package/lib/queries/findMany.js.map +1 -1
  70. package/lib/queries/update/buildUpdate.d.ts +1 -1
  71. package/lib/queries/update/buildUpdate.d.ts.map +1 -1
  72. package/lib/queries/update/buildUpdate.js +1 -1
  73. package/lib/queries/update/buildUpdate.js.map +1 -1
  74. package/lib/queries/update/updateResultSchema.d.ts.map +1 -1
  75. package/lib/queries/update/updateResultSchema.js +3 -1
  76. package/lib/queries/update/updateResultSchema.js.map +1 -1
  77. package/lib/queries/update/updateToSql.js +1 -1
  78. package/lib/queries/update/updateToSql.js.map +1 -1
  79. package/lib/queries/updateMany.d.ts.map +1 -1
  80. package/lib/queries/updateMany.js +3 -1
  81. package/lib/queries/updateMany.js.map +1 -1
  82. package/lib/queries/util/rowToObject.d.ts.map +1 -1
  83. package/lib/queries/util/rowToObject.js +1 -1
  84. package/lib/queries/util/rowToObject.js.map +1 -1
  85. package/lib/schema/types/base/BaseRelation.d.ts +1 -0
  86. package/lib/schema/types/base/BaseRelation.d.ts.map +1 -1
  87. package/lib/schema/types/loose/LooseRelationDefinition.d.ts +1 -0
  88. package/lib/schema/types/loose/LooseRelationDefinition.d.ts.map +1 -1
  89. package/lib/schema/types/relations/OneToManyRelation.d.ts +1 -0
  90. package/lib/schema/types/relations/OneToManyRelation.d.ts.map +1 -1
  91. package/lib/test/db/index.d.ts +40 -0
  92. package/lib/test/db/index.d.ts.map +1 -1
  93. package/lib/test/db/models/post.model.d.ts +9 -0
  94. package/lib/test/db/models/post.model.d.ts.map +1 -1
  95. package/lib/test/db/models/post.model.js +6 -0
  96. package/lib/test/db/models/post.model.js.map +1 -1
  97. package/lib/test/db/models/post.relations.d.ts +6 -0
  98. package/lib/test/db/models/post.relations.d.ts.map +1 -1
  99. package/lib/test/db/models/post.relations.js +6 -0
  100. package/lib/test/db/models/post.relations.js.map +1 -1
  101. package/lib/test/db/models/user.model.d.ts +9 -0
  102. package/lib/test/db/models/user.model.d.ts.map +1 -1
  103. package/lib/test/db/models/user.model.js +6 -0
  104. package/lib/test/db/models/user.model.js.map +1 -1
  105. package/lib/test/db/models/user.relations.d.ts +16 -0
  106. package/lib/test/db/models/user.relations.d.ts.map +1 -1
  107. package/lib/test/db/models/user.relations.js +16 -0
  108. package/lib/test/db/models/user.relations.js.map +1 -1
  109. package/lib/test/db/models.d.ts +18 -0
  110. package/lib/test/db/models.d.ts.map +1 -1
  111. package/lib/test/db/relations.d.ts +22 -0
  112. package/lib/test/db/relations.d.ts.map +1 -1
  113. package/lib/test/seed/index.d.ts +40 -0
  114. package/lib/test/seed/index.d.ts.map +1 -1
  115. package/package.json +1 -1
  116. package/src/migrate/commands/implode.ts +1 -1
  117. package/src/migrate/sql/createTableSql.test.ts +1 -0
  118. package/src/orm.ts +5 -2
  119. package/src/queries/clauses/where/buildWhereClauses.test.ts +1 -1
  120. package/src/queries/clauses/where/buildWhereClauses.ts +1 -1
  121. package/src/queries/count/buildCount.ts +39 -33
  122. package/src/queries/count/countToSql.ts +27 -32
  123. package/src/queries/count/types/CountBuilder.ts +23 -6
  124. package/src/queries/count.ts +4 -1
  125. package/src/queries/create/buildCreate.ts +2 -2
  126. package/src/queries/create/createResultSchema.ts +3 -1
  127. package/src/queries/create/createToSql.ts +6 -2
  128. package/src/queries/createMany.ts +4 -1
  129. package/src/queries/createOne.ts +9 -0
  130. package/src/queries/delete/buildDelete.ts +2 -2
  131. package/src/queries/delete/deleteResultSchema.ts +3 -1
  132. package/src/queries/delete/deleteToSql.ts +1 -1
  133. package/src/queries/deleteMany.ts +4 -1
  134. package/src/queries/find/buildFind.ts +40 -34
  135. package/src/queries/find/findResultSchema.ts +9 -2
  136. package/src/queries/find/findToSql.ts +27 -32
  137. package/src/queries/find/tests/findMany.nullable-relations.test.ts +127 -0
  138. package/src/queries/find/types/FindBuilder.ts +23 -6
  139. package/src/queries/findMany.ts +25 -17
  140. package/src/queries/update/buildUpdate.ts +2 -2
  141. package/src/queries/update/updateResultSchema.ts +3 -1
  142. package/src/queries/update/updateToSql.ts +1 -1
  143. package/src/queries/updateMany.ts +4 -1
  144. package/src/queries/util/rowToObject.ts +5 -1
  145. package/src/schema/types/base/BaseRelation.ts +1 -0
  146. package/src/schema/types/loose/LooseRelationDefinition.ts +1 -1
  147. package/src/schema/types/relations/OneToManyRelation.ts +1 -0
  148. package/src/test/db/models/post.model.ts +6 -0
  149. package/src/test/db/models/post.relations.ts +6 -0
  150. package/src/test/db/models/user.model.ts +6 -0
  151. package/src/test/db/models/user.relations.ts +16 -0
@@ -0,0 +1,127 @@
1
+ import { describe, expect, test } from "vitest";
2
+
3
+ import { db } from "../../../test/db";
4
+ import { seed } from "../../../test/seed";
5
+
6
+ describe("findMany", () => {
7
+ test("optional N:1 relations don't prevent records being returned", async () => {
8
+ await db.transact(
9
+ async (db) => {
10
+ await seed(db, {
11
+ users: [
12
+ {
13
+ username: "Russell",
14
+ tenants: [
15
+ { name: "WFMA", posts: 1 },
16
+ { name: "Popova Park", posts: 0 },
17
+ ],
18
+ },
19
+ {
20
+ username: "Dan",
21
+ tenants: [{ name: "WFMA", posts: 2 }],
22
+ },
23
+ {
24
+ username: "Fairooz",
25
+ tenants: [{ name: "WFMA", posts: 0 }],
26
+ },
27
+ ],
28
+ });
29
+ const results = await db.findMany("post", {
30
+ select: ["id", "title"],
31
+ include: {
32
+ reviewedBy: {
33
+ select: ["id", "username"],
34
+ include: {
35
+ posts: { select: ["id", "title"] },
36
+ invitedBy: { select: ["id", "username"] },
37
+ },
38
+ },
39
+ },
40
+ orderBy: ["title"],
41
+ limit: 2,
42
+ });
43
+
44
+ expect(results.map((p) => [p.title, p.reviewedBy])).toEqual([
45
+ ["Post a", null],
46
+ ["Post b", null],
47
+ ]);
48
+ },
49
+ { rollback: true },
50
+ );
51
+ });
52
+
53
+ test("optional N:1 relations are returned if they exist", async () => {
54
+ await db.transact(
55
+ async (db) => {
56
+ const { users } = await seed(db, {
57
+ users: [
58
+ {
59
+ username: "Russell",
60
+ tenants: [
61
+ { name: "WFMA", posts: 1 },
62
+ { name: "Popova Park", posts: 0 },
63
+ ],
64
+ },
65
+ {
66
+ username: "Dan",
67
+ tenants: [{ name: "WFMA", posts: 2 }],
68
+ },
69
+ {
70
+ username: "Fairooz",
71
+ tenants: [{ name: "WFMA", posts: 0 }],
72
+ },
73
+ ],
74
+ });
75
+ await db.updateOne("user", {
76
+ where: { id: users["Dan"].id },
77
+ values: {
78
+ invitedById: users["Russell"].id,
79
+ },
80
+ });
81
+ const results = await db.findMany("user", {
82
+ select: ["username"],
83
+ include: {
84
+ invitedBy: {
85
+ select: ["username"],
86
+ },
87
+ tenants: {
88
+ select: ["name"],
89
+ orderBy: ["name"],
90
+ },
91
+ reviewedPosts: {
92
+ select: ["title"],
93
+ },
94
+ invitedUsers: {
95
+ select: ["username"],
96
+ },
97
+ },
98
+ orderBy: ["username"],
99
+ });
100
+ expect(results).toEqual([
101
+ {
102
+ username: "Dan",
103
+ invitedBy: { username: "Russell" },
104
+ tenants: [{ name: "WFMA" }],
105
+ reviewedPosts: [],
106
+ invitedUsers: [],
107
+ },
108
+ {
109
+ username: "Fairooz",
110
+ invitedBy: null,
111
+ tenants: [{ name: "WFMA" }],
112
+ reviewedPosts: [],
113
+ invitedUsers: [],
114
+ },
115
+ {
116
+ username: "Russell",
117
+ invitedBy: null,
118
+ tenants: [{ name: "Popova Park" }, { name: "WFMA" }],
119
+ reviewedPosts: [],
120
+ invitedUsers: [{ username: "Dan" }],
121
+ },
122
+ ]);
123
+ },
124
+ { rollback: true },
125
+ );
126
+ });
127
+ });
@@ -12,12 +12,12 @@ export type FindBuilder = {
12
12
  path: string[];
13
13
  }[];
14
14
 
15
- tables: {
16
- name: string;
15
+ table: {
16
+ table: string;
17
17
  model: string;
18
18
  schema: string;
19
19
  alias: string;
20
- joins?: Join[];
20
+ joins: Join[];
21
21
  conditions?: WhereClause<
22
22
  LooseModelDefinitions,
23
23
  ModelName<LooseModelDefinitions>
@@ -26,7 +26,7 @@ export type FindBuilder = {
26
26
  LooseModelDefinitions,
27
27
  ModelName<LooseModelDefinitions>
28
28
  >;
29
- }[];
29
+ };
30
30
 
31
31
  lateralBy?: {
32
32
  groupTable: string;
@@ -49,6 +49,23 @@ export type FindBuilder = {
49
49
  offset?: number;
50
50
  };
51
51
  export type Join = {
52
- from: { table: string; columns: string[] };
53
- to: { table: string; columns: string[] };
52
+ from: {
53
+ schema: string;
54
+ table: string;
55
+ alias: string;
56
+ model: string;
57
+ columns: string[];
58
+ };
59
+ to: {
60
+ schema: string;
61
+ table: string;
62
+ alias: string;
63
+ model: string;
64
+ columns: string[];
65
+ };
66
+ where?: WhereClause<
67
+ LooseModelDefinitions,
68
+ ModelName<LooseModelDefinitions>
69
+ >;
70
+ type?: "inner" | "left";
54
71
  };
@@ -27,7 +27,10 @@ export const findMany = async (
27
27
  values: statement.values,
28
28
  });
29
29
 
30
- if (!process.env.CI) console.log(statement.text);
30
+ if (process.env.ORM_VERBOSE_LOGGING) {
31
+ console.log(statement.text);
32
+ console.log(statement.values);
33
+ }
31
34
 
32
35
  const results = await conn
33
36
  .query(statement)
@@ -38,7 +41,6 @@ export const findMany = async (
38
41
  m,
39
42
  query,
40
43
  ).map(({ model, relation, query, path }) => {
41
- console.log("Fetching 1:N relation");
42
44
  const pk = config.models[model].primaryKey;
43
45
  const fk = ensureArray(relation.foreignKey);
44
46
  const lateralBy = fk.map((c, index) => ({
@@ -55,11 +57,14 @@ export const findMany = async (
55
57
  return hash(fk.map((c) => result[c]));
56
58
  });
57
59
  for (const result of results) {
58
- const key = hash(
59
- pk.map((c) => get(result, [...dropRight(path, 1), c])),
60
- );
61
- console.log([...path]);
62
- set(result, [...path], lookup[key] ?? []);
60
+ const parent =
61
+ path.length === 1
62
+ ? result
63
+ : get(result, dropRight(path, 1));
64
+ if (parent !== undefined) {
65
+ const key = hash(pk.map((c) => get(parent, c)));
66
+ set(result, path, lookup[key] ?? []);
67
+ }
63
68
  }
64
69
  });
65
70
  });
@@ -69,7 +74,6 @@ export const findMany = async (
69
74
  m,
70
75
  query,
71
76
  ).map(({ model, relation, query, path }) => {
72
- console.log("Fetching N:N relation");
73
77
  const joinFrom = Object.entries(config.relations[model]).find(
74
78
  ([, rel]) => rel.type === "1:N" && rel.model === relation.through,
75
79
  )?.[0];
@@ -102,15 +106,19 @@ export const findMany = async (
102
106
  return hash(fk.map((c) => result[c]));
103
107
  });
104
108
  for (const result of results) {
105
- const key = hash(
106
- pk.map((c) => get(result, [...dropRight(path, 1), c])),
107
- );
108
- console.log([...path]);
109
- set(
110
- result,
111
- [...path],
112
- (lookup[key] ?? []).map((r) => r[joinTo] ?? []),
113
- );
109
+ const parent =
110
+ path.length === 1
111
+ ? result
112
+ : get(result, dropRight(path, 1));
113
+
114
+ if (parent !== undefined) {
115
+ const key = hash(pk.map((c) => get(parent, c)));
116
+ set(
117
+ result,
118
+ path,
119
+ (lookup[key] ?? []).map((r) => r[joinTo] ?? []),
120
+ );
121
+ }
114
122
  }
115
123
  });
116
124
  });
@@ -9,7 +9,7 @@ import { BaseUpdateParams } from "./types/BaseUpdateParams";
9
9
 
10
10
  export type UpdateBuilder = {
11
11
  tableIndex: number;
12
- table: { name: string; model: string; alias: string; schema: string };
12
+ table: { table: string; model: string; alias: string; schema: string };
13
13
  where: WhereClause<LooseModelDefinitions, ModelName<LooseModelDefinitions>>;
14
14
  values: { name: string; value: unknown }[];
15
15
  returning: { name: string; path: string; alias: string }[];
@@ -24,7 +24,7 @@ export const buildUpdate = (
24
24
  const builder: UpdateBuilder = {
25
25
  tableIndex: _tableIndex,
26
26
  table: {
27
- name: config.models[m].table,
27
+ table: config.models[m].table,
28
28
  schema: config.models[m].schema,
29
29
  model: m,
30
30
  alias: tableAlias(_tableIndex++),
@@ -14,7 +14,9 @@ export const updateResultSchema = (
14
14
 
15
15
  params.returning?.forEach((s) => {
16
16
  const col = config.models[m].columns[s];
17
- obj[s] = col.nullable ? col.zodSchema.nullable() : col.zodSchema;
17
+ obj[s] = col.nullable
18
+ ? col.zodSchema.nullish().transform((x) => x ?? null)
19
+ : col.zodSchema;
18
20
  });
19
21
 
20
22
  return z.object(obj);
@@ -14,7 +14,7 @@ export const updateToSql = (
14
14
 
15
15
  const frag = sql`UPDATE %I.%I AS %I\nSET`.withIdentifiers(
16
16
  table.schema,
17
- table.name,
17
+ table.table,
18
18
  table.alias,
19
19
  );
20
20
 
@@ -21,7 +21,10 @@ export const updateMany = async (
21
21
  values: statement.values,
22
22
  });
23
23
 
24
- if (!process.env.CI) console.log(statement.text);
24
+ if (process.env.ORM_VERBOSE_LOGGING) {
25
+ console.log(statement.text);
26
+ console.log(statement.values);
27
+ }
25
28
 
26
29
  const result = await conn.query(statement);
27
30
  return params.returning
@@ -5,5 +5,9 @@ export type ColumnMapping = { path: string | string[]; alias: string };
5
5
  export const rowToObject =
6
6
  (columns: ColumnMapping[]) =>
7
7
  (row: Record<string, unknown>): Record<string, unknown> => {
8
- return columns.reduce((acc, c) => set(acc, c.path, row[c.alias]), {});
8
+ return columns.reduce(
9
+ (acc, c) =>
10
+ row[c.alias] !== null ? set(acc, c.path, row[c.alias]) : acc,
11
+ {},
12
+ );
9
13
  };
@@ -10,6 +10,7 @@ export type BaseRelation =
10
10
  model: string;
11
11
  type: "N:1";
12
12
  foreignKey: string | string[];
13
+ optional?: boolean;
13
14
  }
14
15
  | {
15
16
  model: string;
@@ -1,6 +1,6 @@
1
1
  export type LooseRelationDefinition =
2
2
  | { model: string; type: "1:N"; foreignKey: string }
3
- | { model: string; type: "N:1"; foreignKey: string }
3
+ | { model: string; type: "N:1"; foreignKey: string; optional?: true }
4
4
  | {
5
5
  model: string;
6
6
  type: "N:N";
@@ -7,5 +7,6 @@ export type OneToManyRelation<Models extends LooseModelDefinitions> = {
7
7
  model: M2;
8
8
  type: "1:N";
9
9
  foreignKey: ColumnName<Models, M2> | ColumnName<Models, M2>[];
10
+ optional?: boolean;
10
11
  };
11
12
  }[ModelName<Models>];
@@ -15,6 +15,12 @@ export const post = {
15
15
  type: "uuid",
16
16
  references: { table: "user", column: "id" },
17
17
  },
18
+ reviewedById: {
19
+ name: "reviewed_by_id",
20
+ type: "uuid",
21
+ nullable: true,
22
+ references: { table: "user", column: "id" },
23
+ },
18
24
  publishedAt: { type: "timestamp", nullable: true },
19
25
  tags: { type: "text[][][][]", nullable: true },
20
26
  tenantId: {
@@ -7,6 +7,12 @@ export const post = {
7
7
  type: "N:1",
8
8
  foreignKey: "authorId",
9
9
  },
10
+ reviewedBy: {
11
+ model: "user",
12
+ type: "N:1",
13
+ foreignKey: "reviewedById",
14
+ optional: true,
15
+ },
10
16
  tenant: {
11
17
  model: "tenant",
12
18
  type: "N:1",
@@ -7,6 +7,12 @@ export const user = {
7
7
  columns: {
8
8
  id: { type: "uuid", default: sql`uuid_generate_v4()` },
9
9
  username: { zodSchema: z.string(), type: "text" },
10
+ invitedById: {
11
+ name: "invited_by_id",
12
+ type: "uuid",
13
+ nullable: true,
14
+ references: { table: "user", column: "id" },
15
+ },
10
16
  joinedAt: {
11
17
  name: "created_at",
12
18
  zodSchema: z.date(),
@@ -7,6 +7,22 @@ export const user = {
7
7
  model: "post",
8
8
  foreignKey: "authorId",
9
9
  },
10
+ reviewedPosts: {
11
+ type: "1:N",
12
+ model: "post",
13
+ foreignKey: "reviewedById",
14
+ },
15
+ invitedBy: {
16
+ type: "N:1",
17
+ model: "user",
18
+ foreignKey: "invitedById",
19
+ optional: true,
20
+ },
21
+ invitedUsers: {
22
+ type: "1:N",
23
+ model: "user",
24
+ foreignKey: "invitedById",
25
+ },
10
26
  tenancies: {
11
27
  type: "1:N",
12
28
  model: "tenantUser",