/* eslint-disable @typescript-eslint/no-non-null-assertion */ import type { Column, InferSelectModel } from "drizzle-orm"; import type { ForeignKey as MysqlForeignKey, MySqlTableWithColumns } from "drizzle-orm/mysql-core"; import type { PgTableWithColumns, ForeignKey as PostgresqlForeignKey } from "drizzle-orm/pg-core"; import type { ForeignKey as SqliteForeignKey, SQLiteTableWithColumns } from "drizzle-orm/sqlite-core"; import { expect, expectTypeOf, test } from "vitest"; import { objectEntries } from "@homarr/common"; import * as mysqlSchema from "../schema/mysql"; import * as postgresqlSchema from "../schema/postgresql"; import * as sqliteSchema from "../schema/sqlite"; // We need the following three types as there is currently no support for Buffer in mysql & pg and // so we use a custom type which results in the config beeing different type FixedMysqlConfig = { [key in keyof MysqlConfig]: { [column in keyof MysqlConfig[key]]: { [property in Exclude]: MysqlConfig[key][column][property]; } & { dataType: MysqlConfig[key][column]["data"] extends Buffer ? "buffer" : MysqlConfig[key][column]["dataType"]; data: MysqlConfig[key][column]["data"] extends Buffer ? Buffer : MysqlConfig[key][column]["data"]; }; }; }; type FixedPostgresqlConfig = { [key in keyof PostgreisqlConfig]: { [column in keyof PostgreisqlConfig[key]]: { [property in Exclude< keyof PostgreisqlConfig[key][column], "dataType" | "data" >]: PostgreisqlConfig[key][column][property]; } & { dataType: PostgreisqlConfig[key][column]["data"] extends Buffer ? "buffer" : PostgreisqlConfig[key][column]["dataType"]; data: PostgreisqlConfig[key][column]["data"] extends Buffer ? Buffer : PostgreisqlConfig[key][column]["data"]; }; }; }; type FixedSqliteConfig = { [key in keyof SqliteConfig]: { [column in keyof SqliteConfig[key]]: { [property in Exclude]: SqliteConfig[key][column][property]; } & { dataType: SqliteConfig[key][column]["dataType"] extends Buffer ? "buffer" : SqliteConfig[key][column]["dataType"]; data: SqliteConfig[key][column]["data"] extends Buffer ? Buffer : SqliteConfig[key][column]["data"]; }; }; }; test("schemas should match", () => { expectTypeOf().toEqualTypeOf(); expectTypeOf().toEqualTypeOf(); expectTypeOf().toEqualTypeOf(); expectTypeOf().toEqualTypeOf(); objectEntries(sqliteSchema).forEach(([tableName, sqliteTable]) => { Object.entries(sqliteTable).forEach(([columnName, sqliteColumn]: [string, object]) => { if (!("isUnique" in sqliteColumn)) return; if (!("uniqueName" in sqliteColumn)) return; if (!("primary" in sqliteColumn)) return; const mysqlTable = mysqlSchema[tableName]; const mysqlColumn = mysqlTable[columnName as keyof typeof mysqlTable] as object; if (!("isUnique" in mysqlColumn)) return; if (!("uniqueName" in mysqlColumn)) return; if (!("primary" in mysqlColumn)) return; expect( sqliteColumn.isUnique, `expect unique of column ${columnName} in table ${tableName} to be the same for both schemas`, ).toEqual(mysqlColumn.isUnique); expect( sqliteColumn.uniqueName, `expect uniqueName of column ${columnName} in table ${tableName} to be the same for both schemas`, ).toEqual(mysqlColumn.uniqueName); expect( sqliteColumn.primary, `expect primary of column ${columnName} in table ${tableName} to be the same for both schemas`, ).toEqual(mysqlColumn.primary); }); const mysqlTable = mysqlSchema[tableName]; const sqliteForeignKeys = sqliteTable[Symbol.for("drizzle:SQLiteInlineForeignKeys") as keyof typeof sqliteTable] as | SqliteForeignKey[] | undefined; const mysqlForeignKeys = mysqlTable[Symbol.for("drizzle:MySqlInlineForeignKeys") as keyof typeof mysqlTable] as | MysqlForeignKey[] | undefined; if (!sqliteForeignKeys && !mysqlForeignKeys) return; expect(mysqlForeignKeys, `mysql foreign key for ${tableName} to be defined`).toBeDefined(); expect(sqliteForeignKeys, `sqlite foreign key for ${tableName} to be defined`).toBeDefined(); expect( sqliteForeignKeys!.length, `expect number of foreign keys in table ${tableName} to be the same for both schemas`, ).toEqual(mysqlForeignKeys!.length); sqliteForeignKeys?.forEach((sqliteForeignKey) => { sqliteForeignKey.getName(); const mysqlForeignKey = mysqlForeignKeys!.find((key) => key.getName() === sqliteForeignKey.getName()); expect( mysqlForeignKey, `expect foreign key ${sqliteForeignKey.getName()} to be defined in mysql schema`, ).toBeDefined(); expect( sqliteForeignKey.onDelete, `expect foreign key (${sqliteForeignKey.getName()}) onDelete to be the same for both schemas`, ).toEqual(mysqlForeignKey!.onDelete); expect( sqliteForeignKey.onUpdate, `expect foreign key (${sqliteForeignKey.getName()}) onUpdate to be the same for both schemas`, ).toEqual(mysqlForeignKey!.onUpdate); sqliteForeignKey.reference().foreignColumns.forEach((column) => { expect( mysqlForeignKey!.reference().foreignColumns.map((column) => column.name), `expect foreign key (${sqliteForeignKey.getName()}) columns to be the same for both schemas`, ).toContainEqual(column.name); }); expect( Object.keys(sqliteForeignKey.reference().foreignTable), `expect foreign key (${sqliteForeignKey.getName()}) table to be the same for both schemas`, ).toEqual(Object.keys(mysqlForeignKey!.reference().foreignTable)); }); }); }); test("schemas should match for postgresql", () => { expectTypeOf().toEqualTypeOf(); expectTypeOf().toEqualTypeOf(); expectTypeOf().toEqualTypeOf(); expectTypeOf().toEqualTypeOf(); objectEntries(sqliteSchema).forEach(([tableName, sqliteTable]) => { // keys of sqliteSchema and postgresqlSchema are the same, so we can safely use tableName as key // skipcq: JS-E1007 const postgresqlTable = postgresqlSchema[tableName]; Object.entries(sqliteTable).forEach(([columnName, sqliteColumn]: [string, object]) => { if (!("isUnique" in sqliteColumn)) return; if (!("uniqueName" in sqliteColumn)) return; if (!("primary" in sqliteColumn)) return; const postgresqlColumn = postgresqlTable[columnName as keyof typeof postgresqlTable] as object; if (!("isUnique" in postgresqlColumn)) return; if (!("uniqueName" in postgresqlColumn)) return; if (!("primary" in postgresqlColumn)) return; expect( sqliteColumn.isUnique, `expect unique of column ${columnName} in table ${tableName} to be the same for both schemas`, ).toEqual(postgresqlColumn.isUnique); expect( sqliteColumn.uniqueName, `expect uniqueName of column ${columnName} in table ${tableName} to be the same for both schemas`, ).toEqual(postgresqlColumn.uniqueName); expect( sqliteColumn.primary, `expect primary of column ${columnName} in table ${tableName} to be the same for both schemas`, ).toEqual(postgresqlColumn.primary); }); const sqliteForeignKeys = sqliteTable[Symbol.for("drizzle:SQLiteInlineForeignKeys") as keyof typeof sqliteTable] as | SqliteForeignKey[] | undefined; const postgresqlForeignKeys = postgresqlTable[ Symbol.for("drizzle:PgInlineForeignKeys") as keyof typeof postgresqlTable ] as PostgresqlForeignKey[] | undefined; if (!sqliteForeignKeys && !postgresqlForeignKeys) return; expect(postgresqlForeignKeys, `postgresql foreign key for ${tableName} to be defined`).toBeDefined(); expect(sqliteForeignKeys, `sqlite foreign key for ${tableName} to be defined`).toBeDefined(); expect( sqliteForeignKeys!.length, `expect number of foreign keys in table ${tableName} to be the same for both schemas`, ).toEqual(postgresqlForeignKeys?.length); sqliteForeignKeys?.forEach((sqliteForeignKey) => { sqliteForeignKey.getName(); const postgresqlForeignKey = postgresqlForeignKeys!.find((key) => key.getName() === sqliteForeignKey.getName()); expect( postgresqlForeignKey, `expect foreign key ${sqliteForeignKey.getName()} to be defined in postgresql schema`, ).toBeDefined(); // In PostgreSql, onDelete is "no action" by default, so it is treated as undefined to match Sqlite. expect( sqliteForeignKey.onDelete, `expect foreign key (${sqliteForeignKey.getName()}) onDelete to be the same for both schemas`, ).toEqual(postgresqlForeignKey!.onDelete === "no action" ? undefined : postgresqlForeignKey!.onDelete); // In PostgreSql, onUpdate is "no action" by default, so it is treated as undefined to match Sqlite. expect( sqliteForeignKey.onUpdate, `expect foreign key (${sqliteForeignKey.getName()}) onUpdate to be the same for both schemas`, ).toEqual(postgresqlForeignKey!.onUpdate === "no action" ? undefined : postgresqlForeignKey!.onUpdate); sqliteForeignKey.reference().foreignColumns.forEach((column) => { expect( postgresqlForeignKey!.reference().foreignColumns.map((column) => column.name), `expect foreign key (${sqliteForeignKey.getName()}) columns to be the same for both schemas`, ).toContainEqual(column.name); }); expect( Object.keys(sqliteForeignKey.reference().foreignTable), `expect foreign key (${sqliteForeignKey.getName()}) table to be the same for both schemas`, ).toEqual(Object.keys(postgresqlForeignKey!.reference().foreignTable).filter((key) => key !== "enableRLS")); }); }); }); type SqliteTables = { // eslint-disable-next-line @typescript-eslint/no-explicit-any [K in keyof typeof sqliteSchema]: (typeof sqliteSchema)[K] extends SQLiteTableWithColumns ? InferSelectModel<(typeof sqliteSchema)[K]> : never; }; type MysqlTables = { // eslint-disable-next-line @typescript-eslint/no-explicit-any [K in keyof typeof mysqlSchema]: (typeof mysqlSchema)[K] extends MySqlTableWithColumns ? InferSelectModel<(typeof mysqlSchema)[K]> : never; }; type PostgresqlTables = { // eslint-disable-next-line @typescript-eslint/no-explicit-any [K in keyof typeof postgresqlSchema]: (typeof postgresqlSchema)[K] extends PgTableWithColumns ? InferSelectModel<(typeof postgresqlSchema)[K]> : never; }; // eslint-disable-next-line @typescript-eslint/no-explicit-any type InferColumnConfig> = T extends Column ? Omit : never; type SqliteConfig = { // eslint-disable-next-line @typescript-eslint/no-explicit-any [K in keyof typeof sqliteSchema]: (typeof sqliteSchema)[K] extends SQLiteTableWithColumns ? { [C in keyof (typeof sqliteSchema)[K]["_"]["config"]["columns"]]: InferColumnConfig< (typeof sqliteSchema)[K]["_"]["config"]["columns"][C] >; } : never; }; type MysqlConfig = { // eslint-disable-next-line @typescript-eslint/no-explicit-any [K in keyof typeof mysqlSchema]: (typeof mysqlSchema)[K] extends MySqlTableWithColumns ? { [C in keyof (typeof mysqlSchema)[K]["_"]["config"]["columns"]]: InferColumnConfig< (typeof mysqlSchema)[K]["_"]["config"]["columns"][C] >; } : never; }; type PostgreisqlConfig = { // eslint-disable-next-line @typescript-eslint/no-explicit-any [K in keyof typeof postgresqlSchema]: (typeof postgresqlSchema)[K] extends PgTableWithColumns ? { [C in keyof (typeof postgresqlSchema)[K]["_"]["config"]["columns"]]: InferColumnConfig< (typeof postgresqlSchema)[K]["_"]["config"]["columns"][C] >; } : never; };