From a9bb8ab82b8bea46debabc27017eb8fc309b8f29 Mon Sep 17 00:00:00 2001 From: Mario564 Date: Tue, 10 Dec 2024 09:42:24 -0800 Subject: [PATCH 01/32] Remove unnecessary @internal declarations --- drizzle-typebox/src/column.ts | 1 - drizzle-valibot/src/column.ts | 1 - drizzle-zod/src/column.ts | 1 - 3 files changed, 3 deletions(-) diff --git a/drizzle-typebox/src/column.ts b/drizzle-typebox/src/column.ts index 80e6ff39d..510ba7bbe 100644 --- a/drizzle-typebox/src/column.ts +++ b/drizzle-typebox/src/column.ts @@ -55,7 +55,6 @@ export function mapEnumValues(values: string[]) { return Object.fromEntries(values.map((value) => [value, value])); } -/** @internal */ export function columnToSchema(column: Column, t: typeof typebox): TSchema { let schema!: TSchema; diff --git a/drizzle-valibot/src/column.ts b/drizzle-valibot/src/column.ts index e5716fe1e..5d7a1784e 100644 --- a/drizzle-valibot/src/column.ts +++ b/drizzle-valibot/src/column.ts @@ -55,7 +55,6 @@ export function mapEnumValues(values: string[]) { return Object.fromEntries(values.map((value) => [value, value])); } -/** @internal */ export function columnToSchema(column: Column): v.GenericSchema { let schema!: v.GenericSchema; diff --git a/drizzle-zod/src/column.ts b/drizzle-zod/src/column.ts index 4aae40e7e..d6fff3445 100644 --- a/drizzle-zod/src/column.ts +++ b/drizzle-zod/src/column.ts @@ -50,7 +50,6 @@ export const jsonSchema: z.ZodType = z.lazy(() => ); export const bufferSchema: z.ZodType = z.custom((v) => v instanceof Buffer); // eslint-disable-line no-instanceof/no-instanceof -/** @internal */ export function columnToSchema(column: Column, z: typeof zod): z.ZodTypeAny { let schema!: z.ZodTypeAny; From 1686686d5111648a301680bae7f26d605aabf3de Mon Sep 17 00:00:00 2001 From: Mario564 Date: Tue, 10 Dec 2024 09:43:17 -0800 Subject: [PATCH 02/32] Fix type refinement type mapping in drizzle-zod --- drizzle-zod/src/schema.types.internal.ts | 18 ++++++++++++------ 1 file changed, 12 insertions(+), 6 deletions(-) diff --git a/drizzle-zod/src/schema.types.internal.ts b/drizzle-zod/src/schema.types.internal.ts index 5732e2e0f..8b89187f2 100644 --- a/drizzle-zod/src/schema.types.internal.ts +++ b/drizzle-zod/src/schema.types.internal.ts @@ -39,13 +39,20 @@ export type BuildRefine< : never; type HandleRefinement< + TType extends 'select' | 'insert' | 'update', TRefinement extends z.ZodTypeAny | ((schema: z.ZodTypeAny) => z.ZodTypeAny), TColumn extends Column, -> = TRefinement extends (schema: z.ZodTypeAny) => z.ZodTypeAny - ? TColumn['_']['notNull'] extends true ? ReturnType - : z.ZodNullable> +> = TRefinement extends (schema: any) => z.ZodTypeAny ? (TColumn['_']['notNull'] extends true ? ReturnType + : z.ZodNullable>) extends infer TSchema + ? TType extends 'update' ? z.ZodOptional> : TSchema + : z.ZodTypeAny : TRefinement; +type IsRefinementDefined = TKey extends keyof TRefinements + ? TRefinements[TKey] extends z.ZodTypeAny | ((schema: any) => any) ? true + : false + : false; + export type BuildSchema< TType extends 'select' | 'insert' | 'update', TColumns extends Record, @@ -56,9 +63,8 @@ export type BuildSchema< { [K in keyof TColumns]: TColumns[K] extends infer TColumn extends Column ? TRefinements extends object - ? TRefinements[Assume] extends - infer TRefinement extends z.ZodTypeAny | ((schema: z.ZodTypeAny) => z.ZodTypeAny) - ? HandleRefinement + ? IsRefinementDefined> extends true + ? HandleRefinement], TColumn> : HandleColumn : HandleColumn : TColumns[K] extends infer TObject extends SelectedFieldsFlat | Table | View ? BuildSchema< From 2265f01dabd24649283af1548df46db3865ef382 Mon Sep 17 00:00:00 2001 From: Mario564 Date: Tue, 10 Dec 2024 10:09:47 -0800 Subject: [PATCH 03/32] Fix `drizzle-typebox` --- drizzle-typebox/src/schema.types.internal.ts | 18 ++++++++++++------ 1 file changed, 12 insertions(+), 6 deletions(-) diff --git a/drizzle-typebox/src/schema.types.internal.ts b/drizzle-typebox/src/schema.types.internal.ts index beccef94b..e48b764f4 100644 --- a/drizzle-typebox/src/schema.types.internal.ts +++ b/drizzle-typebox/src/schema.types.internal.ts @@ -40,13 +40,20 @@ export type BuildRefine< : never; type HandleRefinement< + TType extends 'select' | 'insert' | 'update', TRefinement extends t.TSchema | ((schema: t.TSchema) => t.TSchema), TColumn extends Column, -> = TRefinement extends (schema: t.TSchema) => t.TSchema - ? TColumn['_']['notNull'] extends true ? ReturnType - : t.TTuple<[ReturnType, t.TNull]> +> = TRefinement extends (schema: any) => t.TSchema ? (TColumn['_']['notNull'] extends true ? ReturnType + : t.TUnion<[ReturnType, t.TNull]>) extends infer TSchema + ? TType extends 'update' ? t.TOptional> : TSchema + : t.TSchema : TRefinement; +type IsRefinementDefined = TKey extends keyof TRefinements + ? TRefinements[TKey] extends t.TSchema | ((schema: any) => any) ? true + : false + : false; + export type BuildSchema< TType extends 'select' | 'insert' | 'update', TColumns extends Record, @@ -57,9 +64,8 @@ export type BuildSchema< { [K in keyof TColumns]: TColumns[K] extends infer TColumn extends Column ? TRefinements extends object - ? TRefinements[Assume] extends - infer TRefinement extends t.TSchema | ((schema: t.TSchema) => t.TSchema) - ? HandleRefinement + ? IsRefinementDefined> extends true + ? HandleRefinement], TColumn> : HandleColumn : HandleColumn : TColumns[K] extends infer TObject extends SelectedFieldsFlat | Table | View ? BuildSchema< From 7af7983c5567e681138d1d96a8ae364768e511c3 Mon Sep 17 00:00:00 2001 From: Mario564 Date: Wed, 11 Dec 2024 09:27:55 -0800 Subject: [PATCH 04/32] Add additional tests to validators --- drizzle-typebox/tests/mysql.test.ts | 28 +++++++++++++++++++- drizzle-typebox/tests/pg.test.ts | 38 +++++++++++++++++++++++++++- drizzle-typebox/tests/sqlite.test.ts | 28 +++++++++++++++++++- drizzle-valibot/tests/mysql.test.ts | 28 +++++++++++++++++++- drizzle-valibot/tests/pg.test.ts | 38 +++++++++++++++++++++++++++- drizzle-valibot/tests/sqlite.test.ts | 28 +++++++++++++++++++- drizzle-zod/tests/mysql.test.ts | 28 +++++++++++++++++++- drizzle-zod/tests/pg.test.ts | 38 +++++++++++++++++++++++++++- drizzle-zod/tests/sqlite.test.ts | 28 +++++++++++++++++++- 9 files changed, 273 insertions(+), 9 deletions(-) diff --git a/drizzle-typebox/tests/mysql.test.ts b/drizzle-typebox/tests/mysql.test.ts index 213240368..8863f0fb9 100644 --- a/drizzle-typebox/tests/mysql.test.ts +++ b/drizzle-typebox/tests/mysql.test.ts @@ -1,6 +1,6 @@ import { Type as t } from '@sinclair/typebox'; import { type Equal, sql } from 'drizzle-orm'; -import { int, mysqlSchema, mysqlTable, mysqlView, serial, text } from 'drizzle-orm/mysql-core'; +import { customType, int, mysqlSchema, mysqlTable, mysqlView, serial, text } from 'drizzle-orm/mysql-core'; import { test } from 'vitest'; import { jsonSchema } from '~/column.ts'; import { CONSTANTS } from '~/constants.ts'; @@ -207,6 +207,32 @@ test('refine table - select', (tc) => { Expect>(); }); +test('refine table - select with custom data type', (tc) => { + const customText = customType({ dataType: () => 'text' }); + const table = mysqlTable('test', { + c1: int(), + c2: int().notNull(), + c3: int().notNull(), + c4: customText(), + }); + + const customTextSchema = t.String({ minLength: 1, maxLength: 100 }); + const result = createSelectSchema(table, { + c2: (schema) => t.Integer({ minimum: schema.minimum, maximum: 1000 }), + c3: t.Integer({ minimum: 1, maximum: 10 }), + c4: customTextSchema, + }); + const expected = t.Object({ + c1: t.Union([intSchema, t.Null()]), + c2: t.Integer({ minimum: CONSTANTS.INT32_MIN, maximum: 1000 }), + c3: t.Integer({ minimum: 1, maximum: 10 }), + c4: customTextSchema, + }); + + expectSchemaShape(tc, expected).from(result); + Expect>(); +}); + test('refine table - insert', (tc) => { const table = mysqlTable('test', { c1: int(), diff --git a/drizzle-typebox/tests/pg.test.ts b/drizzle-typebox/tests/pg.test.ts index 3e9769aef..8fd8148d8 100644 --- a/drizzle-typebox/tests/pg.test.ts +++ b/drizzle-typebox/tests/pg.test.ts @@ -1,6 +1,16 @@ import { Type as t } from '@sinclair/typebox'; import { type Equal, sql } from 'drizzle-orm'; -import { integer, pgEnum, pgMaterializedView, pgSchema, pgTable, pgView, serial, text } from 'drizzle-orm/pg-core'; +import { + customType, + integer, + pgEnum, + pgMaterializedView, + pgSchema, + pgTable, + pgView, + serial, + text, +} from 'drizzle-orm/pg-core'; import { test } from 'vitest'; import { jsonSchema } from '~/column.ts'; import { CONSTANTS } from '~/constants.ts'; @@ -233,6 +243,32 @@ test('refine table - select', (tc) => { Expect>(); }); +test('refine table - select with custom data type', (tc) => { + const customText = customType({ dataType: () => 'text' }); + const table = pgTable('test', { + c1: integer(), + c2: integer().notNull(), + c3: integer().notNull(), + c4: customText(), + }); + + const customTextSchema = t.String({ minLength: 1, maxLength: 100 }); + const result = createSelectSchema(table, { + c2: (schema) => t.Integer({ minimum: schema.minimum, maximum: 1000 }), + c3: t.Integer({ minimum: 1, maximum: 10 }), + c4: customTextSchema, + }); + const expected = t.Object({ + c1: t.Union([integerSchema, t.Null()]), + c2: t.Integer({ minimum: CONSTANTS.INT32_MIN, maximum: 1000 }), + c3: t.Integer({ minimum: 1, maximum: 10 }), + c4: customTextSchema, + }); + + expectSchemaShape(tc, expected).from(result); + Expect>(); +}); + test('refine table - insert', (tc) => { const table = pgTable('test', { c1: integer(), diff --git a/drizzle-typebox/tests/sqlite.test.ts b/drizzle-typebox/tests/sqlite.test.ts index ba2b55002..bce2d8c63 100644 --- a/drizzle-typebox/tests/sqlite.test.ts +++ b/drizzle-typebox/tests/sqlite.test.ts @@ -1,6 +1,6 @@ import { Type as t } from '@sinclair/typebox'; import { type Equal, sql } from 'drizzle-orm'; -import { int, sqliteTable, sqliteView, text } from 'drizzle-orm/sqlite-core'; +import { customType, int, sqliteTable, sqliteView, text } from 'drizzle-orm/sqlite-core'; import { test } from 'vitest'; import { bufferSchema, jsonSchema } from '~/column.ts'; import { CONSTANTS } from '~/constants.ts'; @@ -186,6 +186,32 @@ test('refine table - select', (tc) => { Expect>(); }); +test('refine table - select with custom data type', (tc) => { + const customText = customType({ dataType: () => 'text' }); + const table = sqliteTable('test', { + c1: int(), + c2: int().notNull(), + c3: int().notNull(), + c4: customText(), + }); + + const customTextSchema = t.String({ minLength: 1, maxLength: 100 }); + const result = createSelectSchema(table, { + c2: (schema) => t.Integer({ minimum: schema.minimum, maximum: 1000 }), + c3: t.Integer({ minimum: 1, maximum: 10 }), + c4: customTextSchema, + }); + const expected = t.Object({ + c1: t.Union([intSchema, t.Null()]), + c2: t.Integer({ minimum: CONSTANTS.INT32_MIN, maximum: 1000 }), + c3: t.Integer({ minimum: 1, maximum: 10 }), + c4: customTextSchema, + }); + + expectSchemaShape(tc, expected).from(result); + Expect>(); +}); + test('refine table - insert', (tc) => { const table = sqliteTable('test', { c1: int(), diff --git a/drizzle-valibot/tests/mysql.test.ts b/drizzle-valibot/tests/mysql.test.ts index 5bf9520cb..6578729a4 100644 --- a/drizzle-valibot/tests/mysql.test.ts +++ b/drizzle-valibot/tests/mysql.test.ts @@ -1,5 +1,5 @@ import { type Equal, sql } from 'drizzle-orm'; -import { int, mysqlSchema, mysqlTable, mysqlView, serial, text } from 'drizzle-orm/mysql-core'; +import { customType, int, mysqlSchema, mysqlTable, mysqlView, serial, text } from 'drizzle-orm/mysql-core'; import * as v from 'valibot'; import { test } from 'vitest'; import { jsonSchema } from '~/column.ts'; @@ -210,6 +210,32 @@ test('refine table - select', (t) => { Expect>(); }); +test('refine table - select with custom data type', (t) => { + const customText = customType({ dataType: () => 'text' }); + const table = mysqlTable('test', { + c1: int(), + c2: int().notNull(), + c3: int().notNull(), + c4: customText(), + }); + + const customTextSchema = v.pipe(v.string(), v.minLength(1), v.maxLength(100)); + const result = createSelectSchema(table, { + c2: (schema) => v.pipe(schema, v.maxValue(1000)), + c3: v.pipe(v.string(), v.transform(Number)), + c4: customTextSchema, + }); + const expected = v.object({ + c1: v.nullable(intSchema), + c2: v.pipe(intSchema, v.maxValue(1000)), + c3: v.pipe(v.string(), v.transform(Number)), + c4: customTextSchema, + }); + + expectSchemaShape(t, expected).from(result); + Expect>(); +}); + test('refine table - insert', (t) => { const table = mysqlTable('test', { c1: int(), diff --git a/drizzle-valibot/tests/pg.test.ts b/drizzle-valibot/tests/pg.test.ts index 4d1651a7c..ea2bf2dd0 100644 --- a/drizzle-valibot/tests/pg.test.ts +++ b/drizzle-valibot/tests/pg.test.ts @@ -1,5 +1,15 @@ import { type Equal, sql } from 'drizzle-orm'; -import { integer, pgEnum, pgMaterializedView, pgSchema, pgTable, pgView, serial, text } from 'drizzle-orm/pg-core'; +import { + customType, + integer, + pgEnum, + pgMaterializedView, + pgSchema, + pgTable, + pgView, + serial, + text, +} from 'drizzle-orm/pg-core'; import * as v from 'valibot'; import { test } from 'vitest'; import { jsonSchema } from '~/column.ts'; @@ -234,6 +244,32 @@ test('refine table - select', (t) => { Expect>(); }); +test('refine table - select with custom data type', (t) => { + const customText = customType({ dataType: () => 'text' }); + const table = pgTable('test', { + c1: integer(), + c2: integer().notNull(), + c3: integer().notNull(), + c4: customText(), + }); + + const customTextSchema = v.pipe(v.string(), v.minLength(1), v.maxLength(100)); + const result = createSelectSchema(table, { + c2: (schema) => v.pipe(schema, v.maxValue(1000)), + c3: v.pipe(v.string(), v.transform(Number)), + c4: customTextSchema, + }); + const expected = v.object({ + c1: v.nullable(integerSchema), + c2: v.pipe(integerSchema, v.maxValue(1000)), + c3: v.pipe(v.string(), v.transform(Number)), + c4: customTextSchema, + }); + + expectSchemaShape(t, expected).from(result); + Expect>(); +}); + test('refine table - insert', (t) => { const table = pgTable('test', { c1: integer(), diff --git a/drizzle-valibot/tests/sqlite.test.ts b/drizzle-valibot/tests/sqlite.test.ts index 7eb5fc7bf..14e6b4bd6 100644 --- a/drizzle-valibot/tests/sqlite.test.ts +++ b/drizzle-valibot/tests/sqlite.test.ts @@ -1,5 +1,5 @@ import { type Equal, sql } from 'drizzle-orm'; -import { int, sqliteTable, sqliteView, text } from 'drizzle-orm/sqlite-core'; +import { customType, int, sqliteTable, sqliteView, text } from 'drizzle-orm/sqlite-core'; import * as v from 'valibot'; import { test } from 'vitest'; import { bufferSchema, jsonSchema } from '~/column.ts'; @@ -187,6 +187,32 @@ test('refine table - select', (t) => { Expect>(); }); +test('refine table - select with custom data type', (t) => { + const customText = customType({ dataType: () => 'text' }); + const table = sqliteTable('test', { + c1: int(), + c2: int().notNull(), + c3: int().notNull(), + c4: customText(), + }); + + const customTextSchema = v.pipe(v.string(), v.minLength(1), v.maxLength(100)); + const result = createSelectSchema(table, { + c2: (schema) => v.pipe(schema, v.maxValue(1000)), + c3: v.pipe(v.string(), v.transform(Number)), + c4: customTextSchema, + }); + const expected = v.object({ + c1: v.nullable(intSchema), + c2: v.pipe(intSchema, v.maxValue(1000)), + c3: v.pipe(v.string(), v.transform(Number)), + c4: customTextSchema, + }); + + expectSchemaShape(t, expected).from(result); + Expect>(); +}); + test('refine table - insert', (t) => { const table = sqliteTable('test', { c1: int(), diff --git a/drizzle-zod/tests/mysql.test.ts b/drizzle-zod/tests/mysql.test.ts index 37c9b7e64..73ba48dae 100644 --- a/drizzle-zod/tests/mysql.test.ts +++ b/drizzle-zod/tests/mysql.test.ts @@ -1,5 +1,5 @@ import { type Equal, sql } from 'drizzle-orm'; -import { int, mysqlSchema, mysqlTable, mysqlView, serial, text } from 'drizzle-orm/mysql-core'; +import { customType, int, mysqlSchema, mysqlTable, mysqlView, serial, text } from 'drizzle-orm/mysql-core'; import { test } from 'vitest'; import { z } from 'zod'; import { jsonSchema } from '~/column.ts'; @@ -201,6 +201,32 @@ test('refine table - select', (t) => { Expect>(); }); +test('refine table - select with custom data type', (t) => { + const customText = customType({ dataType: () => 'text' }); + const table = mysqlTable('test', { + c1: int(), + c2: int().notNull(), + c3: int().notNull(), + c4: customText(), + }); + + const customTextSchema = z.string().min(1).max(100); + const result = createSelectSchema(table, { + c2: (schema) => schema.max(1000), + c3: z.string().transform(Number), + c4: customTextSchema, + }); + const expected = z.object({ + c1: intSchema.nullable(), + c2: intSchema.max(1000), + c3: z.string().transform(Number), + c4: customTextSchema, + }); + + expectSchemaShape(t, expected).from(result); + Expect>(); +}); + test('refine table - insert', (t) => { const table = mysqlTable('test', { c1: int(), diff --git a/drizzle-zod/tests/pg.test.ts b/drizzle-zod/tests/pg.test.ts index dc703b4fc..7964f65d6 100644 --- a/drizzle-zod/tests/pg.test.ts +++ b/drizzle-zod/tests/pg.test.ts @@ -1,5 +1,15 @@ import { type Equal, sql } from 'drizzle-orm'; -import { integer, pgEnum, pgMaterializedView, pgSchema, pgTable, pgView, serial, text } from 'drizzle-orm/pg-core'; +import { + customType, + integer, + pgEnum, + pgMaterializedView, + pgSchema, + pgTable, + pgView, + serial, + text, +} from 'drizzle-orm/pg-core'; import { test } from 'vitest'; import { z } from 'zod'; import { jsonSchema } from '~/column.ts'; @@ -234,6 +244,32 @@ test('refine table - select', (t) => { Expect>(); }); +test('refine table - select with custom data type', (t) => { + const customText = customType({ dataType: () => 'text' }); + const table = pgTable('test', { + c1: integer(), + c2: integer().notNull(), + c3: integer().notNull(), + c4: customText(), + }); + + const customTextSchema = z.string().min(1).max(100); + const result = createSelectSchema(table, { + c2: (schema) => schema.max(1000), + c3: z.string().transform(Number), + c4: customTextSchema, + }); + const expected = z.object({ + c1: integerSchema.nullable(), + c2: integerSchema.max(1000), + c3: z.string().transform(Number), + c4: customTextSchema, + }); + + expectSchemaShape(t, expected).from(result); + Expect>(); +}); + test('refine table - insert', (t) => { const table = pgTable('test', { c1: integer(), diff --git a/drizzle-zod/tests/sqlite.test.ts b/drizzle-zod/tests/sqlite.test.ts index 45e64bbde..bb0f254b5 100644 --- a/drizzle-zod/tests/sqlite.test.ts +++ b/drizzle-zod/tests/sqlite.test.ts @@ -1,5 +1,5 @@ import { type Equal, sql } from 'drizzle-orm'; -import { int, sqliteTable, sqliteView, text } from 'drizzle-orm/sqlite-core'; +import { customType, int, sqliteTable, sqliteView, text } from 'drizzle-orm/sqlite-core'; import { test } from 'vitest'; import { z } from 'zod'; import { bufferSchema, jsonSchema } from '~/column.ts'; @@ -182,6 +182,32 @@ test('refine table - select', (t) => { Expect>(); }); +test('refine table - select with custom data type', (t) => { + const customText = customType({ dataType: () => 'text' }); + const table = sqliteTable('test', { + c1: int(), + c2: int().notNull(), + c3: int().notNull(), + c4: customText(), + }); + + const customTextSchema = z.string().min(1).max(100); + const result = createSelectSchema(table, { + c2: (schema) => schema.max(1000), + c3: z.string().transform(Number), + c4: customTextSchema, + }); + const expected = z.object({ + c1: intSchema.nullable(), + c2: intSchema.max(1000), + c3: z.string().transform(Number), + c4: customTextSchema, + }); + + expectSchemaShape(t, expected).from(result); + Expect>(); +}); + test('refine table - insert', (t) => { const table = sqliteTable('test', { c1: int(), From 2e49ebc0a987e4526683b508c0580af240613b9b Mon Sep 17 00:00:00 2001 From: Mario564 Date: Wed, 11 Dec 2024 09:51:11 -0800 Subject: [PATCH 05/32] Fix Typebox SQLite test --- drizzle-typebox/tests/sqlite.test.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/drizzle-typebox/tests/sqlite.test.ts b/drizzle-typebox/tests/sqlite.test.ts index bce2d8c63..2b5083b92 100644 --- a/drizzle-typebox/tests/sqlite.test.ts +++ b/drizzle-typebox/tests/sqlite.test.ts @@ -203,7 +203,7 @@ test('refine table - select with custom data type', (tc) => { }); const expected = t.Object({ c1: t.Union([intSchema, t.Null()]), - c2: t.Integer({ minimum: CONSTANTS.INT32_MIN, maximum: 1000 }), + c2: t.Integer({ minimum: Number.MIN_SAFE_INTEGER, maximum: 1000 }), c3: t.Integer({ minimum: 1, maximum: 10 }), c4: customTextSchema, }); From e13844c7f32a4fc046b3a9f4b56c2f3988a2919e Mon Sep 17 00:00:00 2001 From: OleksiiKH0240 Date: Fri, 13 Dec 2024 18:02:03 +0200 Subject: [PATCH 06/32] support varchar for postgres using sql column type --- drizzle-seed/src/datasets/adjectives.ts | 2 + drizzle-seed/src/datasets/cityNames.ts | 2 + .../src/datasets/companyNameSuffixes.ts | 2 + drizzle-seed/src/datasets/countries.ts | 2 + drizzle-seed/src/datasets/emailDomains.ts | 2 + drizzle-seed/src/datasets/firstNames.ts | 2 + drizzle-seed/src/datasets/jobsTitles.ts | 2 + drizzle-seed/src/datasets/lastNames.ts | 2 + .../src/datasets/loremIpsumSentences.ts | 2 + drizzle-seed/src/datasets/states.ts | 2 + drizzle-seed/src/datasets/streetSuffix.ts | 2 + drizzle-seed/src/index.ts | 55 ++- .../src/services/GeneratorsWrappers.ts | 110 +++-- drizzle-seed/src/services/SeedService.ts | 392 ++++++++---------- drizzle-seed/src/types/tables.ts | 6 + 15 files changed, 332 insertions(+), 253 deletions(-) diff --git a/drizzle-seed/src/datasets/adjectives.ts b/drizzle-seed/src/datasets/adjectives.ts index c2b152af0..880e52636 100644 --- a/drizzle-seed/src/datasets/adjectives.ts +++ b/drizzle-seed/src/datasets/adjectives.ts @@ -4844,3 +4844,5 @@ export default [ 'zonked', 'zoological', ]; + +export const maxStringLength = 22; diff --git a/drizzle-seed/src/datasets/cityNames.ts b/drizzle-seed/src/datasets/cityNames.ts index 780b55213..3ea80747e 100644 --- a/drizzle-seed/src/datasets/cityNames.ts +++ b/drizzle-seed/src/datasets/cityNames.ts @@ -42857,3 +42857,5 @@ export default [ 'Garches', 'Chemini', ]; + +export const maxStringLength = 49; diff --git a/drizzle-seed/src/datasets/companyNameSuffixes.ts b/drizzle-seed/src/datasets/companyNameSuffixes.ts index ae8ce6163..1ce31a9c3 100644 --- a/drizzle-seed/src/datasets/companyNameSuffixes.ts +++ b/drizzle-seed/src/datasets/companyNameSuffixes.ts @@ -25,3 +25,5 @@ export default [ 'Co.', 'SCC', ]; + +export const maxStringLength = 7; diff --git a/drizzle-seed/src/datasets/countries.ts b/drizzle-seed/src/datasets/countries.ts index 4808fc5e5..eb1c001d0 100644 --- a/drizzle-seed/src/datasets/countries.ts +++ b/drizzle-seed/src/datasets/countries.ts @@ -169,3 +169,5 @@ export default [ 'Yemen', 'Zambia', ]; + +export const maxStringLength = 30; diff --git a/drizzle-seed/src/datasets/emailDomains.ts b/drizzle-seed/src/datasets/emailDomains.ts index 9904aad3e..ea323ed41 100644 --- a/drizzle-seed/src/datasets/emailDomains.ts +++ b/drizzle-seed/src/datasets/emailDomains.ts @@ -22,3 +22,5 @@ export default [ 'ymail.com', 'libero.it', ]; + +export const maxStringLength = 14; diff --git a/drizzle-seed/src/datasets/firstNames.ts b/drizzle-seed/src/datasets/firstNames.ts index 7ca0ff928..d719aa4e2 100644 --- a/drizzle-seed/src/datasets/firstNames.ts +++ b/drizzle-seed/src/datasets/firstNames.ts @@ -30277,3 +30277,5 @@ export default [ 'Lavasia', 'Laniqua', ]; + +export const maxStringLength = 15; diff --git a/drizzle-seed/src/datasets/jobsTitles.ts b/drizzle-seed/src/datasets/jobsTitles.ts index 3a38e3244..e7993da2a 100644 --- a/drizzle-seed/src/datasets/jobsTitles.ts +++ b/drizzle-seed/src/datasets/jobsTitles.ts @@ -150,3 +150,5 @@ export default [ 'Legal secretary', 'Market analyst', ]; + +export const maxStringLength = 35; diff --git a/drizzle-seed/src/datasets/lastNames.ts b/drizzle-seed/src/datasets/lastNames.ts index 117c5fe28..9d35f7cf7 100644 --- a/drizzle-seed/src/datasets/lastNames.ts +++ b/drizzle-seed/src/datasets/lastNames.ts @@ -50001,3 +50001,5 @@ export default [ 'Thagard', 'Leavelle', ]; + +export const maxStringLength = 15; diff --git a/drizzle-seed/src/datasets/loremIpsumSentences.ts b/drizzle-seed/src/datasets/loremIpsumSentences.ts index f03277d86..64fe59f71 100644 --- a/drizzle-seed/src/datasets/loremIpsumSentences.ts +++ b/drizzle-seed/src/datasets/loremIpsumSentences.ts @@ -1637,3 +1637,5 @@ export default [ 'Sed gravida enim quis nunc interdum imperdiet.', 'Proin cursus odio ac dolor blandit, quis sollicitudin ante rutrum.', ]; + +export const maxStringLength = 190; diff --git a/drizzle-seed/src/datasets/states.ts b/drizzle-seed/src/datasets/states.ts index 1de77160d..cd66cf330 100644 --- a/drizzle-seed/src/datasets/states.ts +++ b/drizzle-seed/src/datasets/states.ts @@ -50,3 +50,5 @@ export default [ 'Wisconsin', 'Wyoming', ]; + +export const maxStringLength = 14; diff --git a/drizzle-seed/src/datasets/streetSuffix.ts b/drizzle-seed/src/datasets/streetSuffix.ts index e9b20c392..90a70c2c6 100644 --- a/drizzle-seed/src/datasets/streetSuffix.ts +++ b/drizzle-seed/src/datasets/streetSuffix.ts @@ -198,3 +198,5 @@ export default [ 'Well', 'Wells', ]; + +export const maxStringLength = 10; diff --git a/drizzle-seed/src/index.ts b/drizzle-seed/src/index.ts index 8596863fb..4f72080ab 100644 --- a/drizzle-seed/src/index.ts +++ b/drizzle-seed/src/index.ts @@ -222,6 +222,8 @@ export async function seedForDrizzleStudio( name: col.name, dataType: 'string', columnType: col.type, + // TODO: revise later + typeParams: {}, default: col.default, hasDefault: col.default === undefined ? false : true, isUnique: col.isUnique === undefined ? false : col.isUnique, @@ -322,7 +324,7 @@ export async function seedForDrizzleStudio( export function seed< DB extends | PgDatabase - | MySqlDatabase + | MySqlDatabase | BaseSQLiteDatabase, SCHEMA extends { [key: string]: @@ -415,7 +417,7 @@ const seedFunc = async ( export async function reset< DB extends | PgDatabase - | MySqlDatabase + | MySqlDatabase | BaseSQLiteDatabase, SCHEMA extends { [key: string]: @@ -604,7 +606,8 @@ const getPostgresInfo = (schema: { [key: string]: PgTable }) => { ): Column['baseColumn'] => { const baseColumnResult: Column['baseColumn'] = { name: baseColumn.name, - columnType: baseColumn.columnType.replace('Pg', '').toLowerCase(), + columnType: baseColumn.getSQLType(), + typeParams: getTypeParams(baseColumn.getSQLType()), dataType: baseColumn.dataType, size: (baseColumn as PgArray).size, hasDefault: baseColumn.hasDefault, @@ -619,12 +622,54 @@ const getPostgresInfo = (schema: { [key: string]: PgTable }) => { return baseColumnResult; }; + const getTypeParams = (sqlType: string) => { + // get type params and set only type + const typeParams: Column['typeParams'] = {}; + + // handle dimensions + if (sqlType.includes('[')) { + const match = sqlType.match(/\[\w*]/g); + if (match) { + typeParams['dimensions'] = match.length; + } + } + + if ( + sqlType.startsWith('numeric') + || sqlType.startsWith('decimal') + || sqlType.startsWith('double precision') + || sqlType.startsWith('real') + ) { + const match = sqlType.match(/\((\d+),(\d+)\)/); + if (match) { + typeParams['precision'] = Number(match[1]); + typeParams['scale'] = Number(match[2]); + } + } else if ( + sqlType.startsWith('varchar') + || sqlType.startsWith('bpchar') + || sqlType.startsWith('char') + || sqlType.startsWith('bit') + || sqlType.startsWith('time') + || sqlType.startsWith('timestamp') + || sqlType.startsWith('interval') + ) { + const match = sqlType.match(/\((\d+)\)/); + if (match) { + typeParams['length'] = Number(match[1]); + } + } + + return typeParams; + }; + // console.log(tableConfig.columns); tables.push({ name: dbToTsTableNamesMap[tableConfig.name] as string, columns: tableConfig.columns.map((column) => ({ name: dbToTsColumnNamesMap[column.name] as string, - columnType: column.columnType.replace('Pg', '').toLowerCase(), + columnType: column.getSQLType(), + typeParams: getTypeParams(column.getSQLType()), dataType: column.dataType, size: (column as PgArray).size, hasDefault: column.hasDefault, @@ -852,6 +897,7 @@ const getMySqlInfo = (schema: { [key: string]: MySqlTable }) => { columns: tableConfig.columns.map((column) => ({ name: dbToTsColumnNamesMap[column.name] as string, columnType: column.columnType.replace('MySql', '').toLowerCase(), + typeParams: {}, dataType: column.dataType, hasDefault: column.hasDefault, default: column.default, @@ -1039,6 +1085,7 @@ const getSqliteInfo = (schema: { [key: string]: SQLiteTable }) => { columns: tableConfig.columns.map((column) => ({ name: dbToTsColumnNamesMap[column.name] as string, columnType: column.columnType.replace('SQLite', '').toLowerCase(), + typeParams: {}, dataType: column.dataType, hasDefault: column.hasDefault, default: column.default, diff --git a/drizzle-seed/src/services/GeneratorsWrappers.ts b/drizzle-seed/src/services/GeneratorsWrappers.ts index 06d6adeb5..a1a751f9a 100644 --- a/drizzle-seed/src/services/GeneratorsWrappers.ts +++ b/drizzle-seed/src/services/GeneratorsWrappers.ts @@ -24,6 +24,7 @@ export abstract class AbstractGenerator { public timeSpent?: number; public arraySize?: number; public baseColumnDataType?: string; + public length?: number; constructor(public params: T) {} @@ -1337,14 +1338,26 @@ export class GenerateString extends AbstractGenerator<{ }> { static override readonly [entityKind]: string = 'GenerateString'; - private state: { rng: prand.RandomGenerator } | undefined; + private state: { + rng: prand.RandomGenerator; + minStringLength: number; + maxStringLength: number; + } | undefined; override uniqueVersionOfGen = GenerateUniqueString; override init({ count, seed }: { count: number; seed: number }) { super.init({ count, seed }); + let minStringLength = 8; + let maxStringLength = 20; + if (this.length !== undefined) { + maxStringLength = this.length; + if (maxStringLength === 1) minStringLength = maxStringLength; + if (maxStringLength < minStringLength) minStringLength = 1; + } + const rng = prand.xoroshiro128plus(seed); - this.state = { rng }; + this.state = { rng, minStringLength, maxStringLength }; } generate() { @@ -1352,8 +1365,8 @@ export class GenerateString extends AbstractGenerator<{ throw new Error('state is not defined.'); } - const minStringLength = 7; - const maxStringLength = 20; + const minStringLength = this.state.minStringLength, + maxStringLength = this.state.maxStringLength; const stringChars = '1234567890abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ'; let idx: number, strLength: number, @@ -1380,12 +1393,31 @@ export class GenerateString extends AbstractGenerator<{ export class GenerateUniqueString extends AbstractGenerator<{ isUnique?: boolean }> { static override readonly [entityKind]: string = 'GenerateUniqueString'; - private state: { rng: prand.RandomGenerator } | undefined; + private state: { + rng: prand.RandomGenerator; + minStringLength: number; + maxStringLength: number; + } | undefined; public override isUnique = true; - override init({ seed }: { seed: number }) { + override init({ seed, count }: { seed: number; count: number }) { const rng = prand.xoroshiro128plus(seed); - this.state = { rng }; + + let minStringLength = 8; + let maxStringLength = 20; + // TODO: revise later + if (this.length !== undefined) { + maxStringLength = this.length; + if (maxStringLength === 1 || maxStringLength < minStringLength) minStringLength = maxStringLength; + } + + if (maxStringLength < count.toString(16).length) { + throw new Error( + `You can't generate ${count} unique strings, with a maximum string length of ${maxStringLength}.`, + ); + } + + this.state = { rng, minStringLength, maxStringLength }; } generate({ i }: { i: number }) { @@ -1393,8 +1425,8 @@ export class GenerateUniqueString extends AbstractGenerator<{ isUnique?: boolean throw new Error('state is not defined.'); } - const minStringLength = 7; - const maxStringLength = 20; + const minStringLength = this.state.minStringLength, + maxStringLength = this.state.maxStringLength; const stringChars = '1234567890abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ'; let idx: number, strLength: number; @@ -1416,7 +1448,7 @@ export class GenerateUniqueString extends AbstractGenerator<{ isUnique?: boolean currStr += stringChars[idx]; } - return currStr.slice(0, 4) + uniqueStr + currStr.slice(4); + return uniqueStr + currStr; } } @@ -2871,7 +2903,7 @@ export const generatorsFuncs = { /** * generates same given value each time the generator is called. * @param defaultValue - value you want to generate - * @param arraySize - number of elements in each one-dimensional array. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) * * @example * ```ts @@ -2890,7 +2922,7 @@ export const generatorsFuncs = { * generates values from given array * @param values - array of values you want to generate. can be array of weighted values. * @param isUnique - property that controls if generated values gonna be unique or not. - * @param arraySize - number of elements in each one-dimensional array. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) * * @example * ```ts @@ -2952,7 +2984,7 @@ export const generatorsFuncs = { * precision equals 10 means that values will be accurate to one tenth (1.2, 34.6); * precision equals 100 means that values will be accurate to one hundredth (1.23, 34.67). * @param isUnique - property that controls if generated values gonna be unique or not. - * @param arraySize - number of elements in each one-dimensional array. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) * * @example * ```ts @@ -2974,7 +3006,7 @@ export const generatorsFuncs = { * @param minValue - lower border of range. * @param maxValue - upper border of range. * @param isUnique - property that controls if generated values gonna be unique or not. - * @param arraySize - number of elements in each one-dimensional array. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) * * @example * ```ts @@ -2993,7 +3025,7 @@ export const generatorsFuncs = { /** * generates boolean values(true or false) - * @param arraySize - number of elements in each one-dimensional array. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) * * @example * ```ts @@ -3013,7 +3045,7 @@ export const generatorsFuncs = { * generates date within given range. * @param minDate - lower border of range. * @param maxDate - upper border of range. - * @param arraySize - number of elements in each one-dimensional array. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) * * @example * ```ts @@ -3031,7 +3063,7 @@ export const generatorsFuncs = { /** * generates time in 24 hours style. - * @param arraySize - number of elements in each one-dimensional array. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) * * @example * ```ts @@ -3049,7 +3081,7 @@ export const generatorsFuncs = { /** * generates timestamps. - * @param arraySize - number of elements in each one-dimensional array. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) * * @example * ```ts @@ -3067,7 +3099,7 @@ export const generatorsFuncs = { /** * generates datetime objects. - * @param arraySize - number of elements in each one-dimensional array. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) * * @example * ```ts @@ -3085,7 +3117,7 @@ export const generatorsFuncs = { /** * generates years. - * @param arraySize - number of elements in each one-dimensional array. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) * * @example * ```ts @@ -3103,7 +3135,7 @@ export const generatorsFuncs = { /** * generates json objects with fixed structure. - * @param arraySize - number of elements in each one-dimensional array. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) * * json structure can equal this: * ``` @@ -3148,7 +3180,7 @@ export const generatorsFuncs = { * interval example: "1 years 12 days 5 minutes" * * @param isUnique - property that controls if generated values gonna be unique or not. - * @param arraySize - number of elements in each one-dimensional array. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) * @example * ```ts * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ @@ -3166,7 +3198,7 @@ export const generatorsFuncs = { /** * generates random strings. * @param isUnique - property that controls if generated values gonna be unique or not. - * @param arraySize - number of elements in each one-dimensional array. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) * * @example * ```ts @@ -3185,7 +3217,7 @@ export const generatorsFuncs = { /** * generates v4 UUID strings if arraySize is not specified, or v4 UUID 1D arrays if it is. * - * @param arraySize - number of elements in each one-dimensional array. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) * * @example * ```ts @@ -3205,7 +3237,7 @@ export const generatorsFuncs = { /** * generates person's first names. * @param isUnique - property that controls if generated values gonna be unique or not. - * @param arraySize - number of elements in each one-dimensional array. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) * * @example * ```ts @@ -3224,7 +3256,7 @@ export const generatorsFuncs = { /** * generates person's last names. * @param isUnique - property that controls if generated values gonna be unique or not. - * @param arraySize - number of elements in each one-dimensional array. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) * * @example * ```ts @@ -3243,7 +3275,7 @@ export const generatorsFuncs = { /** * generates person's full names. * @param isUnique - property that controls if generated values gonna be unique or not. - * @param arraySize - number of elements in each one-dimensional array. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) * * @example * ```ts @@ -3261,7 +3293,7 @@ export const generatorsFuncs = { /** * generates unique emails. - * @param arraySize - number of elements in each one-dimensional array. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) * * @example * ```ts @@ -3278,7 +3310,7 @@ export const generatorsFuncs = { /** * generates unique phone numbers. - * @param arraySize - number of elements in each one-dimensional array. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) * * @param template - phone number template, where all '#' symbols will be substituted with generated digits. * @param prefixes - array of any string you want to be your phone number prefixes.(not compatible with template property) @@ -3319,7 +3351,7 @@ export const generatorsFuncs = { /** * generates country's names. * @param isUnique - property that controls if generated values gonna be unique or not. - * @param arraySize - number of elements in each one-dimensional array. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) * * @example * ```ts @@ -3338,7 +3370,7 @@ export const generatorsFuncs = { /** * generates city's names. * @param isUnique - property that controls if generated values gonna be unique or not. - * @param arraySize - number of elements in each one-dimensional array. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) * * @example * ```ts @@ -3357,7 +3389,7 @@ export const generatorsFuncs = { /** * generates street address. * @param isUnique - property that controls if generated values gonna be unique or not. - * @param arraySize - number of elements in each one-dimensional array. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) * * @example * ```ts @@ -3375,7 +3407,7 @@ export const generatorsFuncs = { /** * generates job titles. - * @param arraySize - number of elements in each one-dimensional array. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) * * @example * ```ts @@ -3394,7 +3426,7 @@ export const generatorsFuncs = { * generates postal codes. * * @param isUnique - property that controls if generated values gonna be unique or not. - * @param arraySize - number of elements in each one-dimensional array. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) * * @example * ```ts @@ -3412,7 +3444,7 @@ export const generatorsFuncs = { /** * generates states of America. - * @param arraySize - number of elements in each one-dimensional array. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) * * @example * ```ts @@ -3431,7 +3463,7 @@ export const generatorsFuncs = { * generates company's names. * * @param isUnique - property that controls if generated values gonna be unique or not. - * @param arraySize - number of elements in each one-dimensional array. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) * * @example * ```ts @@ -3451,7 +3483,7 @@ export const generatorsFuncs = { * generates 'lorem ipsum' text sentences. * * @param sentencesCount - number of sentences you want to generate as one generated value(string). - * @param arraySize - number of elements in each one-dimensional array. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) * * @example * ```ts @@ -3474,7 +3506,7 @@ export const generatorsFuncs = { * @param maxXValue - upper bound of range for x coordinate. * @param minYValue - lower bound of range for y coordinate. * @param maxYValue - upper bound of range for y coordinate. - * @param arraySize - number of elements in each one-dimensional array. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) * * @example * ```ts @@ -3508,7 +3540,7 @@ export const generatorsFuncs = { * @param maxBValue - upper bound of range for y parameter. * @param minCValue - lower bound of range for y parameter. * @param maxCValue - upper bound of range for y parameter. - * @param arraySize - number of elements in each one-dimensional array. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) * * @example * ```ts diff --git a/drizzle-seed/src/services/SeedService.ts b/drizzle-seed/src/services/SeedService.ts index 4ee43a921..579d070a6 100644 --- a/drizzle-seed/src/services/SeedService.ts +++ b/drizzle-seed/src/services/SeedService.ts @@ -252,7 +252,6 @@ class SeedService { } if (columnPossibleGenerator.generator === undefined) { - console.log(col); throw new Error( `column with type ${col.columnType} is not supported for now.`, ); @@ -260,6 +259,7 @@ class SeedService { columnPossibleGenerator.generator.isUnique = col.isUnique; columnPossibleGenerator.generator.dataType = col.dataType; + columnPossibleGenerator.generator.length = col.typeParams.length; tablePossibleGenerators.columnsPossibleGenerators.push( columnPossibleGenerator, @@ -432,263 +432,235 @@ class SeedService { table: Table, col: Column, ) => { - let generator: AbstractGenerator | undefined; + const pickGenerator = (table: Table, col: Column) => { + // ARRAY + if (col.columnType.match(/\[\w*]/g) !== null && col.baseColumn !== undefined) { + const baseColumnGen = this.pickGeneratorForPostgresColumn(table, col.baseColumn!) as AbstractGenerator; + if (baseColumnGen === undefined) { + throw new Error(`column with type ${col.baseColumn!.columnType} is not supported for now.`); + } + const generator = new GenerateArray({ baseColumnGen, size: col.size }); - // INT ------------------------------------------------------------------------------------------------------------ - if ( - (col.columnType.includes('serial') - || col.columnType === 'integer' - || col.columnType === 'smallint' - || col.columnType.includes('bigint')) - && table.primaryKeys.includes(col.name) - ) { - generator = new GenerateIntPrimaryKey({}); + return generator; + } - generator.isUnique = col.isUnique; - generator.dataType = col.dataType; - return generator; - } + // INT ------------------------------------------------------------------------------------------------------------ + if ( + (col.columnType.includes('serial') + || col.columnType === 'integer' + || col.columnType === 'smallint' + || col.columnType.includes('bigint')) + && table.primaryKeys.includes(col.name) + ) { + const generator = new GenerateIntPrimaryKey({}); - let minValue: number | bigint | undefined; - let maxValue: number | bigint | undefined; - if (col.columnType.includes('serial')) { - minValue = 1; - if (col.columnType === 'smallserial') { - // 2^16 / 2 - 1, 2 bytes - maxValue = 32767; - } else if (col.columnType === 'serial') { - // 2^32 / 2 - 1, 4 bytes - maxValue = 2147483647; - } else if (col.columnType === 'bigserial') { - // 2^64 / 2 - 1, 8 bytes - minValue = BigInt(1); - maxValue = BigInt('9223372036854775807'); + return generator; } - } else if (col.columnType.includes('int')) { - if (col.columnType === 'smallint') { - // 2^16 / 2 - 1, 2 bytes - minValue = -32768; - maxValue = 32767; - } else if (col.columnType === 'integer') { - // 2^32 / 2 - 1, 4 bytes - minValue = -2147483648; - maxValue = 2147483647; - } else if (col.columnType.includes('bigint')) { - if (col.dataType === 'bigint') { + + let minValue: number | bigint | undefined; + let maxValue: number | bigint | undefined; + if (col.columnType.includes('serial')) { + minValue = 1; + if (col.columnType === 'smallserial') { + // 2^16 / 2 - 1, 2 bytes + maxValue = 32767; + } else if (col.columnType === 'serial') { + // 2^32 / 2 - 1, 4 bytes + maxValue = 2147483647; + } else if (col.columnType === 'bigserial') { // 2^64 / 2 - 1, 8 bytes - minValue = BigInt('-9223372036854775808'); + minValue = BigInt(1); maxValue = BigInt('9223372036854775807'); - } else if (col.dataType === 'number') { - // if you’re expecting values above 2^31 but below 2^53 - minValue = -9007199254740991; - maxValue = 9007199254740991; + } + } else if (col.columnType.includes('int')) { + if (col.columnType === 'smallint') { + // 2^16 / 2 - 1, 2 bytes + minValue = -32768; + maxValue = 32767; + } else if (col.columnType === 'integer') { + // 2^32 / 2 - 1, 4 bytes + minValue = -2147483648; + maxValue = 2147483647; + } else if (col.columnType.includes('bigint')) { + if (col.dataType === 'bigint') { + // 2^64 / 2 - 1, 8 bytes + minValue = BigInt('-9223372036854775808'); + maxValue = BigInt('9223372036854775807'); + } else if (col.dataType === 'number') { + // if you’re expecting values above 2^31 but below 2^53 + minValue = -9007199254740991; + maxValue = 9007199254740991; + } } } - } - if ( - col.columnType.includes('int') - && !col.columnType.includes('interval') - && !col.columnType.includes('point') - ) { - generator = new GenerateInt({ - minValue, - maxValue, - }); + if ( + col.columnType.includes('int') + && !col.columnType.includes('interval') + && !col.columnType.includes('point') + ) { + const generator = new GenerateInt({ + minValue, + maxValue, + }); - generator.isUnique = col.isUnique; - generator.dataType = col.dataType; - return generator; - } + return generator; + } - if (col.columnType.includes('serial')) { - generator = new GenerateIntPrimaryKey({}); + if (col.columnType.includes('serial')) { + const generator = new GenerateIntPrimaryKey({}); - (generator as GenerateIntPrimaryKey).maxValue = maxValue; - generator.isUnique = col.isUnique; - generator.dataType = col.dataType; - return generator; - } + (generator as GenerateIntPrimaryKey).maxValue = maxValue; - // NUMBER(real, double, decimal, numeric) - if ( - col.columnType === 'real' - || col.columnType === 'doubleprecision' - || col.columnType === 'decimal' - || col.columnType === 'numeric' - ) { - generator = new GenerateNumber({}); + return generator; + } - generator.isUnique = col.isUnique; - generator.dataType = col.dataType; - return generator; - } + // NUMBER(real, double, decimal, numeric) + if ( + col.columnType === 'real' + || col.columnType === 'double precision' + || col.columnType === 'decimal' + || col.columnType === 'numeric' + ) { + const generator = new GenerateNumber({}); - // STRING - if ( - (col.columnType === 'text' - || col.columnType === 'varchar' - || col.columnType === 'char') - && table.primaryKeys.includes(col.name) - ) { - generator = new GenerateUniqueString({}); + return generator; + } - generator.isUnique = col.isUnique; - generator.dataType = col.dataType; - return generator; - } + // STRING + if ( + (col.columnType === 'text' + || col.columnType.startsWith('varchar') + || col.columnType.startsWith('char')) + && table.primaryKeys.includes(col.name) + ) { + const generator = new GenerateUniqueString({}); - if ( - (col.columnType === 'text' - || col.columnType === 'varchar' - || col.columnType === 'char') - && col.name.toLowerCase().includes('name') - ) { - generator = new GenerateFirstName({}); + return generator; + } - generator.isUnique = col.isUnique; - generator.dataType = col.dataType; - return generator; - } + if ( + (col.columnType === 'text' + || col.columnType.startsWith('varchar') + || col.columnType.startsWith('char')) + && col.name.toLowerCase().includes('name') + ) { + const generator = new GenerateFirstName({}); - if ( - (col.columnType === 'text' - || col.columnType === 'varchar' - || col.columnType === 'char') - && col.name.toLowerCase().includes('email') - ) { - generator = new GenerateEmail({}); + return generator; + } - generator.isUnique = col.isUnique; - generator.dataType = col.dataType; - return generator; - } + if ( + (col.columnType === 'text' + || col.columnType.startsWith('varchar') + || col.columnType.startsWith('char')) + && col.name.toLowerCase().includes('email') + ) { + const generator = new GenerateEmail({}); - if ( - col.columnType === 'text' - || col.columnType === 'varchar' - || col.columnType === 'char' - ) { - // console.log(col, table) - generator = new GenerateString({}); + return generator; + } - generator.isUnique = col.isUnique; - generator.dataType = col.dataType; - return generator; - } + if ( + col.columnType === 'text' + || col.columnType.startsWith('varchar') + || col.columnType.startsWith('char') + ) { + // console.log(col, table) + const generator = new GenerateString({}); - // UUID - if (col.columnType === 'uuid') { - generator = new GenerateUUID({}); + return generator; + } - generator.isUnique = col.isUnique; - generator.dataType = col.dataType; - return generator; - } + // UUID + if (col.columnType === 'uuid') { + const generator = new GenerateUUID({}); - // BOOLEAN - if (col.columnType === 'boolean') { - generator = new GenerateBoolean({}); + return generator; + } - generator.isUnique = col.isUnique; - generator.dataType = col.dataType; - return generator; - } + // BOOLEAN + if (col.columnType === 'boolean') { + const generator = new GenerateBoolean({}); - // DATE, TIME, TIMESTAMP - if (col.columnType.includes('date')) { - generator = new GenerateDate({}); + return generator; + } - generator.isUnique = col.isUnique; - generator.dataType = col.dataType; - return generator; - } + // DATE, TIME, TIMESTAMP + if (col.columnType.includes('date')) { + const generator = new GenerateDate({}); - if (col.columnType === 'time') { - generator = new GenerateTime({}); + return generator; + } - generator.isUnique = col.isUnique; - generator.dataType = col.dataType; - return generator; - } + if (col.columnType === 'time') { + const generator = new GenerateTime({}); - if (col.columnType.includes('timestamp')) { - generator = new GenerateTimestamp({}); + return generator; + } - generator.isUnique = col.isUnique; - generator.dataType = col.dataType; - return generator; - } + if (col.columnType.includes('timestamp')) { + const generator = new GenerateTimestamp({}); - // JSON, JSONB - if (col.columnType === 'json' || col.columnType === 'jsonb') { - generator = new GenerateJson({}); + return generator; + } - generator.isUnique = col.isUnique; - generator.dataType = col.dataType; - return generator; - } + // JSON, JSONB + if (col.columnType === 'json' || col.columnType === 'jsonb') { + const generator = new GenerateJson({}); - // if (col.columnType === "jsonb") { - // const generator = new GenerateJsonb({}); - // return generator; - // } + return generator; + } - // ENUM - if (col.enumValues !== undefined) { - generator = new GenerateEnum({ - enumValues: col.enumValues, - }); + // if (col.columnType === "jsonb") { + // const generator = new GenerateJsonb({}); + // return generator; + // } - generator.isUnique = col.isUnique; - generator.dataType = col.dataType; - return generator; - } + // ENUM + if (col.enumValues !== undefined) { + const generator = new GenerateEnum({ + enumValues: col.enumValues, + }); - // INTERVAL - if (col.columnType === 'interval') { - generator = new GenerateInterval({}); + return generator; + } - generator.isUnique = col.isUnique; - generator.dataType = col.dataType; - return generator; - } + // INTERVAL + if (col.columnType === 'interval') { + const generator = new GenerateInterval({}); - // POINT, LINE - if (col.columnType.includes('point')) { - generator = new GeneratePoint({}); + return generator; + } - generator.isUnique = col.isUnique; - generator.dataType = col.dataType; - return generator; - } + // POINT, LINE + if (col.columnType.includes('point')) { + const generator = new GeneratePoint({}); - if (col.columnType.includes('line')) { - generator = new GenerateLine({}); + return generator; + } - generator.isUnique = col.isUnique; - generator.dataType = col.dataType; - return generator; - } + if (col.columnType.includes('line')) { + const generator = new GenerateLine({}); + + return generator; + } - // ARRAY - if (col.columnType.includes('array') && col.baseColumn !== undefined) { - const baseColumnGen = this.pickGeneratorForPostgresColumn(table, col.baseColumn!) as AbstractGenerator; - if (baseColumnGen === undefined) { - throw new Error(`column with type ${col.baseColumn!.columnType} is not supported for now.`); + if (col.hasDefault && col.default !== undefined) { + const generator = new GenerateDefault({ + defaultValue: col.default, + }); + return generator; } - generator = new GenerateArray({ baseColumnGen, size: col.size }); + return; + }; + + const generator = pickGenerator(table, col); + if (generator !== undefined) { generator.isUnique = col.isUnique; generator.dataType = col.dataType; - return generator; - } - - if (col.hasDefault && col.default !== undefined) { - generator = new GenerateDefault({ - defaultValue: col.default, - }); - return generator; + generator.length = col.typeParams.length; } return generator; diff --git a/drizzle-seed/src/types/tables.ts b/drizzle-seed/src/types/tables.ts index 8473179ed..dc28c748d 100644 --- a/drizzle-seed/src/types/tables.ts +++ b/drizzle-seed/src/types/tables.ts @@ -4,6 +4,12 @@ export type Column = { name: string; dataType: string; columnType: string; + typeParams: { + precision?: number; + scale?: number; + length?: number; + dimensions?: number; + }; size?: number; default?: any; hasDefault: boolean; From 05d907bf3f409f4425abe528b6ec889b5f03300a Mon Sep 17 00:00:00 2001 From: OleksiiKH0240 Date: Mon, 16 Dec 2024 13:09:43 +0200 Subject: [PATCH 07/32] added checks to string-like generators, updated sqlite and mysql seeding using sql column type --- drizzle-seed/src/index.ts | 35 +++- .../src/services/GeneratorsWrappers.ts | 196 ++++++++++++++++-- drizzle-seed/src/services/SeedService.ts | 19 +- 3 files changed, 222 insertions(+), 28 deletions(-) diff --git a/drizzle-seed/src/index.ts b/drizzle-seed/src/index.ts index a60987867..fde4442a1 100644 --- a/drizzle-seed/src/index.ts +++ b/drizzle-seed/src/index.ts @@ -898,12 +898,41 @@ const getMySqlInfo = (schema: { [key: string]: MySqlTable }) => { } tableRelations[dbToTsTableNamesMap[tableConfig.name] as string]!.push(...newRelations); + const getTypeParams = (sqlType: string) => { + // get type params and set only type + const typeParams: Column['typeParams'] = {}; + + if ( + sqlType.startsWith('decimal') + || sqlType.startsWith('real') + || sqlType.startsWith('double') + || sqlType.startsWith('float') + ) { + const match = sqlType.match(/\((\d+),(\d+)\)/); + if (match) { + typeParams['precision'] = Number(match[1]); + typeParams['scale'] = Number(match[2]); + } + } else if ( + sqlType.startsWith('varchar') + || sqlType.startsWith('binary') + || sqlType.startsWith('varbinary') + ) { + const match = sqlType.match(/\((\d+)\)/); + if (match) { + typeParams['length'] = Number(match[1]); + } + } + + return typeParams; + }; + tables.push({ name: dbToTsTableNamesMap[tableConfig.name] as string, columns: tableConfig.columns.map((column) => ({ name: dbToTsColumnNamesMap[column.name] as string, - columnType: column.columnType.replace('MySql', '').toLowerCase(), - typeParams: {}, + columnType: column.getSQLType(), + typeParams: getTypeParams(column.getSQLType()), dataType: column.dataType, hasDefault: column.hasDefault, default: column.default, @@ -1092,7 +1121,7 @@ const getSqliteInfo = (schema: { [key: string]: SQLiteTable }) => { name: dbToTsTableNamesMap[tableConfig.name] as string, columns: tableConfig.columns.map((column) => ({ name: dbToTsColumnNamesMap[column.name] as string, - columnType: column.columnType.replace('SQLite', '').toLowerCase(), + columnType: column.getSQLType(), typeParams: {}, dataType: column.dataType, hasDefault: column.hasDefault, diff --git a/drizzle-seed/src/services/GeneratorsWrappers.ts b/drizzle-seed/src/services/GeneratorsWrappers.ts index a1a751f9a..374fa6447 100644 --- a/drizzle-seed/src/services/GeneratorsWrappers.ts +++ b/drizzle-seed/src/services/GeneratorsWrappers.ts @@ -1,17 +1,17 @@ import { entityKind } from 'drizzle-orm'; import prand from 'pure-rand'; -import adjectives from '../datasets/adjectives.ts'; -import cityNames from '../datasets/cityNames.ts'; -import companyNameSuffixes from '../datasets/companyNameSuffixes.ts'; -import countries from '../datasets/countries.ts'; -import emailDomains from '../datasets/emailDomains.ts'; -import firstNames from '../datasets/firstNames.ts'; -import jobsTitles from '../datasets/jobsTitles.ts'; -import lastNames from '../datasets/lastNames.ts'; -import loremIpsumSentences from '../datasets/loremIpsumSentences.ts'; +import adjectives, { maxStringLength as maxAdjectiveLength } from '../datasets/adjectives.ts'; +import cityNames, { maxStringLength as maxCityNameLength } from '../datasets/cityNames.ts'; +import companyNameSuffixes, { maxStringLength as maxCompanyNameSuffixLength } from '../datasets/companyNameSuffixes.ts'; +import countries, { maxStringLength as maxCountryLength } from '../datasets/countries.ts'; +import emailDomains, { maxStringLength as maxEmailDomainLength } from '../datasets/emailDomains.ts'; +import firstNames, { maxStringLength as maxFirstNameLength } from '../datasets/firstNames.ts'; +import jobsTitles, { maxStringLength as maxJobTitleLength } from '../datasets/jobsTitles.ts'; +import lastNames, { maxStringLength as maxLastNameLength } from '../datasets/lastNames.ts'; +import loremIpsumSentences, { maxStringLength as maxLoremIpsumLength } from '../datasets/loremIpsumSentences.ts'; import phonesInfo from '../datasets/phonesInfo.ts'; -import states from '../datasets/states.ts'; -import streetSuffix from '../datasets/streetSuffix.ts'; +import states, { maxStringLength as maxStateLength } from '../datasets/states.ts'; +import streetSuffix, { maxStringLength as maxStreetSuffixLength } from '../datasets/streetSuffix.ts'; import { fastCartesianProduct, fillTemplate, getWeightedIndices, isObject } from './utils.ts'; export abstract class AbstractGenerator { @@ -24,7 +24,7 @@ export abstract class AbstractGenerator { public timeSpent?: number; public arraySize?: number; public baseColumnDataType?: string; - public length?: number; + public stringLength?: number; constructor(public params: T) {} @@ -1350,8 +1350,8 @@ export class GenerateString extends AbstractGenerator<{ let minStringLength = 8; let maxStringLength = 20; - if (this.length !== undefined) { - maxStringLength = this.length; + if (this.stringLength !== undefined) { + maxStringLength = this.stringLength; if (maxStringLength === 1) minStringLength = maxStringLength; if (maxStringLength < minStringLength) minStringLength = 1; } @@ -1406,8 +1406,8 @@ export class GenerateUniqueString extends AbstractGenerator<{ isUnique?: boolean let minStringLength = 8; let maxStringLength = 20; // TODO: revise later - if (this.length !== undefined) { - maxStringLength = this.length; + if (this.stringLength !== undefined) { + maxStringLength = this.stringLength; if (maxStringLength === 1 || maxStringLength < minStringLength) minStringLength = maxStringLength; } @@ -1515,6 +1515,12 @@ export class GenerateFirstName extends AbstractGenerator<{ const rng = prand.xoroshiro128plus(seed); + if (this.stringLength !== undefined && this.stringLength < maxFirstNameLength) { + throw new Error( + `You can't use first name generator with a db column length restriction of ${this.stringLength}. Set the maximum string length to at least ${maxFirstNameLength}.`, + ); + } + this.state = { rng }; } @@ -1547,6 +1553,13 @@ export class GenerateUniqueFirstName extends AbstractGenerator<{ if (count > firstNames.length) { throw new Error('count exceeds max number of unique first names.'); } + + if (this.stringLength !== undefined && this.stringLength < maxFirstNameLength) { + throw new Error( + `You can't use first name generator with a db column length restriction of ${this.stringLength}. Set the maximum string length to at least ${maxFirstNameLength}.`, + ); + } + const genIndicesObj = new GenerateUniqueInt({ minValue: 0, maxValue: firstNames.length - 1 }); genIndicesObj.init({ count, seed }); @@ -1582,6 +1595,12 @@ export class GenerateLastName extends AbstractGenerator<{ const rng = prand.xoroshiro128plus(seed); + if (this.stringLength !== undefined && this.stringLength < maxLastNameLength) { + throw new Error( + `You can't use last name generator with a db column length restriction of ${this.stringLength}. Set the maximum string length to at least ${maxLastNameLength}.`, + ); + } + this.state = { rng }; } generate() { @@ -1609,6 +1628,12 @@ export class GenerateUniqueLastName extends AbstractGenerator<{ isUnique?: boole throw new Error('count exceeds max number of unique last names.'); } + if (this.stringLength !== undefined && this.stringLength < maxLastNameLength) { + throw new Error( + `You can't use last name generator with a db column length restriction of ${this.stringLength}. Set the maximum string length to at least ${maxLastNameLength}.`, + ); + } + const genIndicesObj = new GenerateUniqueInt({ minValue: 0, maxValue: lastNames.length - 1 }); genIndicesObj.init({ count, seed }); @@ -1643,6 +1668,14 @@ export class GenerateFullName extends AbstractGenerator<{ const rng = prand.xoroshiro128plus(seed); + if (this.stringLength !== undefined && this.stringLength < (maxFirstNameLength + maxLastNameLength + 1)) { + throw new Error( + `You can't use full name generator with a db column length restriction of ${this.stringLength}. Set the maximum string length to at least ${ + maxFirstNameLength + maxLastNameLength + 1 + }.`, + ); + } + this.state = { rng }; } @@ -1686,6 +1719,15 @@ export class GenerateUniqueFullName extends AbstractGenerator<{ `count exceeds max number of unique full names(${maxUniqueFullNamesNumber}).`, ); } + + if (this.stringLength !== undefined && this.stringLength < (maxFirstNameLength + maxLastNameLength + 1)) { + throw new Error( + `You can't use full name generator with a db column length restriction of ${this.stringLength}. Set the maximum string length to at least ${ + maxFirstNameLength + maxLastNameLength + 1 + }.`, + ); + } + const rng = prand.xoroshiro128plus(seed); const fullnameSet = new Set(); @@ -1747,6 +1789,13 @@ export class GenerateEmail extends AbstractGenerator<{ ); } + const maxEmailLength = maxAdjectiveLength + maxFirstNameLength + maxEmailDomainLength + 2; + if (this.stringLength !== undefined && this.stringLength < maxEmailLength) { + throw new Error( + `You can't use email generator with a db column length restriction of ${this.stringLength}. Set the maximum string length to at least ${maxEmailLength}.`, + ); + } + const arraysToGenerateFrom = [adjectivesArray, namesArray, domainsArray]; const genIndicesObj = new GenerateUniqueInt({ minValue: 0, @@ -1805,6 +1854,13 @@ export class GeneratePhoneNumber extends AbstractGenerator<{ const rng = prand.xoroshiro128plus(seed); if (template !== undefined) { + if (this.stringLength !== undefined && this.stringLength < template.length) { + throw new Error( + `Length of phone number template is shorter than db column length restriction: ${this.stringLength}. + Set the maximum string length to at least ${template.length}.`, + ); + } + const iterArray = [...template.matchAll(/#/g)]; const placeholdersCount = iterArray.length; @@ -1860,6 +1916,17 @@ export class GeneratePhoneNumber extends AbstractGenerator<{ } } + const maxPrefixLength = Math.max(...prefixesArray.map((prefix) => prefix.length)); + const maxGeneratedDigits = Math.max(...generatedDigitsNumbers); + + if (this.stringLength !== undefined && this.stringLength < (maxPrefixLength + maxGeneratedDigits)) { + throw new Error( + `You can't use phone number generator with a db column length restriction of ${this.stringLength}. Set the maximum string length to at least ${ + maxPrefixLength + maxGeneratedDigits + }.`, + ); + } + if (new Set(prefixesArray).size !== prefixesArray.length) { throw new Error('prefixes are not unique.'); } @@ -1939,7 +2006,7 @@ export class GeneratePhoneNumber extends AbstractGenerator<{ numberBody = '0'.repeat(digitsNumberDiff) + numberBody; } - phoneNumber = (prefix.includes('+') ? '' : '+') + prefix + '' + numberBody; + phoneNumber = prefix + '' + numberBody; return phoneNumber; } else { @@ -1972,6 +2039,12 @@ export class GenerateCountry extends AbstractGenerator<{ const rng = prand.xoroshiro128plus(seed); + if (this.stringLength !== undefined && this.stringLength < maxCountryLength) { + throw new Error( + `You can't use country generator with a db column length restriction of ${this.stringLength}. Set the maximum string length to at least ${maxCountryLength}.`, + ); + } + this.state = { rng }; } @@ -2002,6 +2075,12 @@ export class GenerateUniqueCountry extends AbstractGenerator<{ isUnique?: boolea throw new Error('count exceeds max number of unique countries.'); } + if (this.stringLength !== undefined && this.stringLength < maxCountryLength) { + throw new Error( + `You can't use country generator with a db column length restriction of ${this.stringLength}. Set the maximum string length to at least ${maxCountryLength}.`, + ); + } + const genIndicesObj = new GenerateUniqueInt({ minValue: 0, maxValue: countries.length - 1 }); genIndicesObj.init({ count, seed }); @@ -2034,6 +2113,12 @@ export class GenerateJobTitle extends AbstractGenerator<{ const rng = prand.xoroshiro128plus(seed); + if (this.stringLength !== undefined && this.stringLength < maxJobTitleLength) { + throw new Error( + `You can't use job title generator with a db column length restriction of ${this.stringLength}. Set the maximum string length to at least ${maxJobTitleLength}.`, + ); + } + this.state = { rng }; } @@ -2066,6 +2151,14 @@ export class GenerateStreetAdddress extends AbstractGenerator<{ const rng = prand.xoroshiro128plus(seed); const possStreetNames = [firstNames, lastNames]; + + const maxStreetAddressLength = 4 + Math.max(maxFirstNameLength, maxLastNameLength) + 1 + maxStreetSuffixLength; + if (this.stringLength !== undefined && this.stringLength < maxStreetAddressLength) { + throw new Error( + `You can't use street address generator with a db column length restriction of ${this.stringLength}. Set the maximum string length to at least ${maxStreetAddressLength}.`, + ); + } + this.state = { rng, possStreetNames }; } @@ -2116,6 +2209,13 @@ export class GenerateUniqueStreetAdddress extends AbstractGenerator<{ isUnique?: ); } + const maxStreetAddressLength = 4 + Math.max(maxFirstNameLength, maxLastNameLength) + 1 + maxStreetSuffixLength; + if (this.stringLength !== undefined && this.stringLength < maxStreetAddressLength) { + throw new Error( + `You can't use street address generator with a db column length restriction of ${this.stringLength}. Set the maximum string length to at least ${maxStreetAddressLength}.`, + ); + } + const rng = prand.xoroshiro128plus(seed); // ["1", "2", ..., "999"] @@ -2193,6 +2293,12 @@ export class GenerateCity extends AbstractGenerator<{ const rng = prand.xoroshiro128plus(seed); + if (this.stringLength !== undefined && this.stringLength < maxCityNameLength) { + throw new Error( + `You can't use city generator with a db column length restriction of ${this.stringLength}. Set the maximum string length to at least ${maxCityNameLength}.`, + ); + } + this.state = { rng }; } @@ -2221,6 +2327,12 @@ export class GenerateUniqueCity extends AbstractGenerator<{ isUnique?: boolean } throw new Error('count exceeds max number of unique cities.'); } + if (this.stringLength !== undefined && this.stringLength < maxCityNameLength) { + throw new Error( + `You can't use city generator with a db column length restriction of ${this.stringLength}. Set the maximum string length to at least ${maxCityNameLength}.`, + ); + } + const genIndicesObj = new GenerateUniqueInt({ minValue: 0, maxValue: cityNames.length - 1 }); genIndicesObj.init({ count, seed }); @@ -2257,6 +2369,13 @@ export class GeneratePostcode extends AbstractGenerator<{ const rng = prand.xoroshiro128plus(seed); const templates = ['#####', '#####-####']; + const maxPostcodeLength = Math.max(...templates.map((template) => template.length)); + if (this.stringLength !== undefined && this.stringLength < maxPostcodeLength) { + throw new Error( + `You can't use postcode generator with a db column length restriction of ${this.stringLength}. Set the maximum string length to at least ${maxPostcodeLength}.`, + ); + } + this.state = { rng, templates }; } @@ -2330,6 +2449,13 @@ export class GenerateUniquePostcode extends AbstractGenerator<{ isUnique?: boole }, ]; + const maxPostcodeLength = Math.max(...templates.map((template) => template.template.length)); + if (this.stringLength !== undefined && this.stringLength < maxPostcodeLength) { + throw new Error( + `You can't use postcode generator with a db column length restriction of ${this.stringLength}. Set the maximum string length to at least ${maxPostcodeLength}.`, + ); + } + for (const templateObj of templates) { templateObj.indicesGen.skipCheck = true; templateObj.indicesGen.init({ count, seed }); @@ -2381,6 +2507,12 @@ export class GenerateState extends AbstractGenerator<{ const rng = prand.xoroshiro128plus(seed); + if (this.stringLength !== undefined && this.stringLength < maxStateLength) { + throw new Error( + `You can't use state generator with a db column length restriction of ${this.stringLength}. Set the maximum string length to at least ${maxStateLength}.`, + ); + } + this.state = { rng }; } @@ -2419,6 +2551,17 @@ export class GenerateCompanyName extends AbstractGenerator<{ { template: '#, # and #', placeholdersCount: 3 }, ]; + // max( { template: '#', placeholdersCount: 1 }, { template: '#, # and #', placeholdersCount: 3 } ) + const maxCompanyNameLength = Math.max( + maxLastNameLength + maxCompanyNameSuffixLength + 1, + 3 * maxLastNameLength + 7, + ); + if (this.stringLength !== undefined && this.stringLength < maxCompanyNameLength) { + throw new Error( + `You can't use company name generator with a db column length restriction of ${this.stringLength}. Set the maximum string length to at least ${maxCompanyNameLength}.`, + ); + } + this.state = { rng, templates }; } @@ -2482,6 +2625,17 @@ export class GenerateUniqueCompanyName extends AbstractGenerator<{ isUnique?: bo ); } + // max( { template: '#', placeholdersCount: 1 }, { template: '#, # and #', placeholdersCount: 3 } ) + const maxCompanyNameLength = Math.max( + maxLastNameLength + maxCompanyNameSuffixLength + 1, + 3 * maxLastNameLength + 7, + ); + if (this.stringLength !== undefined && this.stringLength < maxCompanyNameLength) { + throw new Error( + `You can't use company name generator with a db column length restriction of ${this.stringLength}. Set the maximum string length to at least ${maxCompanyNameLength}.`, + ); + } + const rng = prand.xoroshiro128plus(seed); // when count reach maxUniqueCompanyNameNumber template will be deleted from array const templates = [ @@ -2570,6 +2724,14 @@ export class GenerateLoremIpsum extends AbstractGenerator<{ const rng = prand.xoroshiro128plus(seed); if (this.params.sentencesCount === undefined) this.params.sentencesCount = 1; + const maxLoremIpsumSentencesLength = maxLoremIpsumLength * this.params.sentencesCount + this.params.sentencesCount + - 1; + if (this.stringLength !== undefined && this.stringLength < maxLoremIpsumSentencesLength) { + throw new Error( + `You can't use lorem ipsum generator with a db column length restriction of ${this.stringLength}. Set the maximum string length to at least ${maxLoremIpsumSentencesLength}.`, + ); + } + this.state = { rng }; } diff --git a/drizzle-seed/src/services/SeedService.ts b/drizzle-seed/src/services/SeedService.ts index 854fbe437..66617cb66 100644 --- a/drizzle-seed/src/services/SeedService.ts +++ b/drizzle-seed/src/services/SeedService.ts @@ -259,7 +259,7 @@ export class SeedService { columnPossibleGenerator.generator.isUnique = col.isUnique; columnPossibleGenerator.generator.dataType = col.dataType; - columnPossibleGenerator.generator.length = col.typeParams.length; + columnPossibleGenerator.generator.stringLength = col.typeParams.length; tablePossibleGenerators.columnsPossibleGenerators.push( columnPossibleGenerator, @@ -660,7 +660,7 @@ export class SeedService { if (generator !== undefined) { generator.isUnique = col.isUnique; generator.dataType = col.dataType; - generator.length = col.typeParams.length; + generator.stringLength = col.typeParams.length; } return generator; @@ -850,7 +850,7 @@ export class SeedService { ) => { // int section --------------------------------------------------------------------------------------- if ( - (col.columnType === 'integer' || col.columnType === 'numeric') + ((col.columnType === 'integer' && col.dataType === 'number') || col.columnType === 'numeric') && table.primaryKeys.includes(col.name) ) { const generator = new GenerateIntPrimaryKey({}); @@ -858,15 +858,15 @@ export class SeedService { } if ( - col.columnType === 'integer' + (col.columnType === 'integer' && col.dataType === 'number') || col.columnType === 'numeric' - || col.columnType === 'bigint' + || (col.dataType === 'bigint' && col.columnType === 'blob') ) { const generator = new GenerateInt({}); return generator; } - if (col.columnType === 'boolean') { + if (col.dataType === 'boolean' && col.columnType === 'integer') { const generator = new GenerateBoolean({}); return generator; } @@ -918,12 +918,15 @@ export class SeedService { return generator; } - if (col.columnType === 'textjson' || col.columnType === 'blobjson') { + if ( + (col.columnType === 'text' && col.dataType === 'json') + || (col.columnType === 'blob' && col.dataType === 'json') + ) { const generator = new GenerateJson({}); return generator; } - if (col.columnType === 'timestamp' || col.columnType === 'timestamp_ms') { + if ((col.columnType === 'integer' && col.dataType === 'date')) { const generator = new GenerateTimestamp({}); return generator; } From f6eaa62b7c29e88ab19e45dc581776c7247852eb Mon Sep 17 00:00:00 2001 From: OleksiiKH0240 Date: Mon, 16 Dec 2024 14:51:09 +0200 Subject: [PATCH 08/32] added array support for studio --- drizzle-seed/src/services/SeedService.ts | 58 ++++++++++++++++++------ 1 file changed, 43 insertions(+), 15 deletions(-) diff --git a/drizzle-seed/src/services/SeedService.ts b/drizzle-seed/src/services/SeedService.ts index 66617cb66..02cf5af2c 100644 --- a/drizzle-seed/src/services/SeedService.ts +++ b/drizzle-seed/src/services/SeedService.ts @@ -205,8 +205,12 @@ export class SeedService { const genObj = refinements[table.name]!.columns[col.name]!; // TODO: for now only GenerateValuesFromArray support notNull property genObj.notNull = col.notNull; - if (col.dataType === 'array') { - if (col.baseColumn?.dataType === 'array' && col.baseColumn?.columnType === 'array') { + if (col.columnType.match(/\[\w*]/g) !== null) { + if ( + (col.baseColumn?.dataType === 'array' && col.baseColumn.columnType.match(/\[\w*]/g) !== null) + // studio case + || (col.typeParams.dimensions !== undefined && col.typeParams.dimensions > 1) + ) { throw new Error("for now you can't specify generators for columns of dimensition greater than 1."); } @@ -444,6 +448,29 @@ export class SeedService { return generator; } + // ARRAY for studio + if (col.columnType.match(/\[\w*]/g) !== null) { + // remove dimensions from type + const baseColumnType = col.columnType.replace(/\[\w*]/g, ''); + const baseColumn: Column = { + ...col, + }; + baseColumn.columnType = baseColumnType; + + const baseColumnGen = this.pickGeneratorForPostgresColumn(table, baseColumn) as AbstractGenerator; + if (baseColumnGen === undefined) { + throw new Error(`column with type ${col.baseColumn!.columnType} is not supported for now.`); + } + + let generator = new GenerateArray({ baseColumnGen }); + + for (let i = 0; i < col.typeParams.dimensions! - 1; i++) { + generator = new GenerateArray({ baseColumnGen: generator }); + } + + return generator; + } + // INT ------------------------------------------------------------------------------------------------------------ if ( (col.columnType.includes('serial') @@ -486,7 +513,8 @@ export class SeedService { // 2^64 / 2 - 1, 8 bytes minValue = BigInt('-9223372036854775808'); maxValue = BigInt('9223372036854775807'); - } else if (col.dataType === 'number') { + } else { + // if (col.dataType === 'number') // if you’re expecting values above 2^31 but below 2^53 minValue = -9007199254740991; maxValue = 9007199254740991; @@ -850,15 +878,25 @@ export class SeedService { ) => { // int section --------------------------------------------------------------------------------------- if ( - ((col.columnType === 'integer' && col.dataType === 'number') || col.columnType === 'numeric') + (col.columnType === 'integer' || col.columnType === 'numeric') && table.primaryKeys.includes(col.name) ) { const generator = new GenerateIntPrimaryKey({}); return generator; } + if (col.columnType === 'integer' && col.dataType === 'boolean') { + const generator = new GenerateBoolean({}); + return generator; + } + + if ((col.columnType === 'integer' && col.dataType === 'date')) { + const generator = new GenerateTimestamp({}); + return generator; + } + if ( - (col.columnType === 'integer' && col.dataType === 'number') + col.columnType === 'integer' || col.columnType === 'numeric' || (col.dataType === 'bigint' && col.columnType === 'blob') ) { @@ -866,11 +904,6 @@ export class SeedService { return generator; } - if (col.dataType === 'boolean' && col.columnType === 'integer') { - const generator = new GenerateBoolean({}); - return generator; - } - // number section ------------------------------------------------------------------------------------ if (col.columnType === 'real' || col.columnType === 'numeric') { const generator = new GenerateNumber({}); @@ -926,11 +959,6 @@ export class SeedService { return generator; } - if ((col.columnType === 'integer' && col.dataType === 'date')) { - const generator = new GenerateTimestamp({}); - return generator; - } - if (col.hasDefault && col.default !== undefined) { const generator = new GenerateDefault({ defaultValue: col.default, From 1c8cbad2203a5131154435e8cd545fcd34387f25 Mon Sep 17 00:00:00 2001 From: OleksiiKH0240 Date: Tue, 17 Dec 2024 12:42:10 +0200 Subject: [PATCH 09/32] updated postgres interval generators to work with fields, added tests --- .../src/services/GeneratorsWrappers.ts | 194 ++++++++++++------ drizzle-seed/src/services/SeedService.ts | 11 +- .../mysql_all_data_types.test.ts | 2 +- .../tests/pg/allDataTypesTest/pgSchema.ts | 16 ++ .../pg_all_data_types.test.ts | 30 +++ 5 files changed, 192 insertions(+), 61 deletions(-) diff --git a/drizzle-seed/src/services/GeneratorsWrappers.ts b/drizzle-seed/src/services/GeneratorsWrappers.ts index 374fa6447..4ab06ec3f 100644 --- a/drizzle-seed/src/services/GeneratorsWrappers.ts +++ b/drizzle-seed/src/services/GeneratorsWrappers.ts @@ -1222,19 +1222,76 @@ export class GenerateEnum extends AbstractGenerator<{ enumValues: (string | numb } export class GenerateInterval extends AbstractGenerator<{ + fields?: + | 'year' + | 'month' + | 'day' + | 'hour' + | 'minute' + | 'second' + | 'year to month' + | 'day to hour' + | 'day to minute' + | 'day to second' + | 'hour to minute' + | 'hour to second' + | 'minute to second'; isUnique?: boolean; arraySize?: number; }> { static override readonly [entityKind]: string = 'GenerateInterval'; - private state: { rng: prand.RandomGenerator } | undefined; + private state: { + rng: prand.RandomGenerator; + fieldsToGenerate: string[]; + } | undefined; override uniqueVersionOfGen = GenerateUniqueInterval; + private config: { [key: string]: { from: number; to: number } } = { + year: { + from: 0, + to: 5, + }, + month: { + from: 0, + to: 11, + }, + day: { + from: 1, + to: 29, + }, + hour: { + from: 0, + to: 23, + }, + minute: { + from: 0, + to: 59, + }, + second: { + from: 0, + to: 59, + }, + }; override init({ count, seed }: { count: number; seed: number }) { super.init({ count, seed }); + const allFields = ['year', 'month', 'day', 'hour', 'minute', 'second']; + let fieldsToGenerate: string[] = allFields; + + if (this.params.fields !== undefined && this.params.fields?.includes(' to ')) { + const tokens = this.params.fields.split(' to '); + const endIdx = allFields.indexOf(tokens[1]!); + fieldsToGenerate = allFields.slice(0, endIdx + 1); + } else if (this.params.fields !== undefined) { + const endIdx = allFields.indexOf(this.params.fields); + fieldsToGenerate = allFields.slice(0, endIdx + 1); + } + + this.config[fieldsToGenerate[Math.floor(fieldsToGenerate.length / 2)]!]!.from = 1; + const rng = prand.xoroshiro128plus(seed); - this.state = { rng }; + this.state = { rng, fieldsToGenerate }; } generate() { @@ -1242,55 +1299,98 @@ export class GenerateInterval extends AbstractGenerator<{ throw new Error('state is not defined.'); } - let yearsNumb: number, - monthsNumb: number, - daysNumb: number, - hoursNumb: number, - minutesNumb: number, - secondsNumb: number; - - let interval = ''; - - [yearsNumb, this.state.rng] = prand.uniformIntDistribution(0, 5, this.state.rng); - [monthsNumb, this.state.rng] = prand.uniformIntDistribution(0, 12, this.state.rng); - - [daysNumb, this.state.rng] = prand.uniformIntDistribution(1, 29, this.state.rng); - - [hoursNumb, this.state.rng] = prand.uniformIntDistribution(0, 24, this.state.rng); - - [minutesNumb, this.state.rng] = prand.uniformIntDistribution(0, 60, this.state.rng); + let interval = '', numb: number; - [secondsNumb, this.state.rng] = prand.uniformIntDistribution(0, 60, this.state.rng); - - interval = `${yearsNumb === 0 ? '' : `${yearsNumb} years `}` - + `${monthsNumb === 0 ? '' : `${monthsNumb} months `}` - + `${daysNumb === 0 ? '' : `${daysNumb} days `}` - + `${hoursNumb === 0 ? '' : `${hoursNumb} hours `}` - + `${minutesNumb === 0 ? '' : `${minutesNumb} minutes `}` - + `${secondsNumb === 0 ? '' : `${secondsNumb} seconds`}`; + for (const field of this.state.fieldsToGenerate) { + const from = this.config[field]!.from, to = this.config[field]!.to; + [numb, this.state.rng] = prand.uniformIntDistribution(from, to, this.state.rng); + interval += `${numb === 0 ? '' : `${numb} ${field} `}`; + } return interval; } } -export class GenerateUniqueInterval extends AbstractGenerator<{ isUnique?: boolean }> { +export class GenerateUniqueInterval extends AbstractGenerator<{ + fields?: + | 'year' + | 'month' + | 'day' + | 'hour' + | 'minute' + | 'second' + | 'year to month' + | 'day to hour' + | 'day to minute' + | 'day to second' + | 'hour to minute' + | 'hour to second' + | 'minute to second'; + isUnique?: boolean; +}> { static override readonly [entityKind]: string = 'GenerateUniqueInterval'; private state: { rng: prand.RandomGenerator; + fieldsToGenerate: string[]; intervalSet: Set; } | undefined; public override isUnique = true; + private config: { [key: string]: { from: number; to: number } } = { + year: { + from: 0, + to: 5, + }, + month: { + from: 0, + to: 11, + }, + day: { + from: 1, + to: 29, + }, + hour: { + from: 0, + to: 23, + }, + minute: { + from: 0, + to: 59, + }, + second: { + from: 0, + to: 59, + }, + }; override init({ count, seed }: { count: number; seed: number }) { - const maxUniqueIntervalsNumber = 6 * 13 * 29 * 25 * 61 * 61; + const allFields = ['year', 'month', 'day', 'hour', 'minute', 'second']; + let fieldsToGenerate: string[] = allFields; + + if (this.params.fields !== undefined && this.params.fields?.includes(' to ')) { + const tokens = this.params.fields.split(' to '); + const endIdx = allFields.indexOf(tokens[1]!); + fieldsToGenerate = allFields.slice(0, endIdx + 1); + } else if (this.params.fields !== undefined) { + const endIdx = allFields.indexOf(this.params.fields); + fieldsToGenerate = allFields.slice(0, endIdx + 1); + } + + this.config[fieldsToGenerate[Math.floor(fieldsToGenerate.length / 2)]!]!.from = 1; + + let maxUniqueIntervalsNumber = 1; + for (const field of fieldsToGenerate) { + const from = this.config[field]!.from, to = this.config[field]!.to; + maxUniqueIntervalsNumber *= from - to + 1; + } + if (count > maxUniqueIntervalsNumber) { throw new RangeError(`count exceeds max number of unique intervals(${maxUniqueIntervalsNumber})`); } const rng = prand.xoroshiro128plus(seed); const intervalSet = new Set(); - this.state = { rng, intervalSet }; + this.state = { rng, fieldsToGenerate, intervalSet }; } generate() { @@ -1298,34 +1398,12 @@ export class GenerateUniqueInterval extends AbstractGenerator<{ isUnique?: boole throw new Error('state is not defined.'); } - let yearsNumb: number, - monthsNumb: number, - daysNumb: number, - hoursNumb: number, - minutesNumb: number, - secondsNumb: number; - - let interval = ''; + let interval = '', numb: number; - for (;;) { - [yearsNumb, this.state.rng] = prand.uniformIntDistribution(0, 5, this.state.rng); - [monthsNumb, this.state.rng] = prand.uniformIntDistribution(0, 12, this.state.rng); - [daysNumb, this.state.rng] = prand.uniformIntDistribution(1, 29, this.state.rng); - [hoursNumb, this.state.rng] = prand.uniformIntDistribution(0, 24, this.state.rng); - [minutesNumb, this.state.rng] = prand.uniformIntDistribution(0, 60, this.state.rng); - [secondsNumb, this.state.rng] = prand.uniformIntDistribution(0, 60, this.state.rng); - - interval = `${yearsNumb === 0 ? '' : `${yearsNumb} years `}` - + `${monthsNumb === 0 ? '' : `${monthsNumb} months `}` - + `${daysNumb === 0 ? '' : `${daysNumb} days `}` - + `${hoursNumb === 0 ? '' : `${hoursNumb} hours `}` - + `${minutesNumb === 0 ? '' : `${minutesNumb} minutes `}` - + `${secondsNumb === 0 ? '' : `${secondsNumb} seconds`}`; - - if (!this.state.intervalSet.has(interval)) { - this.state.intervalSet.add(interval); - break; - } + for (const field of this.state.fieldsToGenerate) { + const from = this.config[field]!.from, to = this.config[field]!.to; + [numb, this.state.rng] = prand.uniformIntDistribution(from, to, this.state.rng); + interval += `${numb === 0 ? '' : `${numb} ${field} `}`; } return interval; @@ -1531,7 +1609,6 @@ export class GenerateFirstName extends AbstractGenerator<{ // logic for this generator // names dataset contains about 30000 unique names. - // TODO: generate names accordingly to max column length let idx: number; [idx, this.state.rng] = prand.uniformIntDistribution(0, firstNames.length - 1, this.state.rng); @@ -3343,6 +3420,7 @@ export const generatorsFuncs = { * * @param isUnique - property that controls if generated values gonna be unique or not. * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * @param fields - range of values you want to see in your intervals. * @example * ```ts * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ diff --git a/drizzle-seed/src/services/SeedService.ts b/drizzle-seed/src/services/SeedService.ts index 02cf5af2c..62283c817 100644 --- a/drizzle-seed/src/services/SeedService.ts +++ b/drizzle-seed/src/services/SeedService.ts @@ -655,8 +655,15 @@ export class SeedService { } // INTERVAL - if (col.columnType === 'interval') { - const generator = new GenerateInterval({}); + if (col.columnType.startsWith('interval')) { + if (col.columnType === 'interval') { + const generator = new GenerateInterval({}); + + return generator; + } + + const fields = col.columnType.replace('interval ', '') as GenerateInterval['params']['fields']; + const generator = new GenerateInterval({ fields }); return generator; } diff --git a/drizzle-seed/tests/mysql/allDataTypesTest/mysql_all_data_types.test.ts b/drizzle-seed/tests/mysql/allDataTypesTest/mysql_all_data_types.test.ts index 5a9ba9908..f39a55fef 100644 --- a/drizzle-seed/tests/mysql/allDataTypesTest/mysql_all_data_types.test.ts +++ b/drizzle-seed/tests/mysql/allDataTypesTest/mysql_all_data_types.test.ts @@ -11,7 +11,7 @@ import { seed } from '../../../src/index.ts'; import * as schema from './mysqlSchema.ts'; let mysqlContainer: Docker.Container; -let client: Connection; +let client: Connection | undefined; let db: MySql2Database; async function createDockerDB(): Promise { diff --git a/drizzle-seed/tests/pg/allDataTypesTest/pgSchema.ts b/drizzle-seed/tests/pg/allDataTypesTest/pgSchema.ts index 75ba20a43..16a55baf4 100644 --- a/drizzle-seed/tests/pg/allDataTypesTest/pgSchema.ts +++ b/drizzle-seed/tests/pg/allDataTypesTest/pgSchema.ts @@ -97,3 +97,19 @@ export const ndArrays = schema.table('nd_arrays', { integer3DArray: integer('integer_3d_array').array(3).array(4).array(5), integer4DArray: integer('integer_4d_array').array(3).array(4).array(5).array(6), }); + +export const intervals = schema.table('intervals', { + intervalYear: interval({ fields: 'year' }), + intervalYearToMonth: interval({ fields: 'year to month' }), + intervalMonth: interval({ fields: 'month' }), + intervalDay: interval({ fields: 'day' }), + intervalDayToHour: interval({ fields: 'day to hour' }), + intervalDayToMinute: interval({ fields: 'day to minute' }), + intervalDayToSecond: interval({ fields: 'day to second' }), + intervalHour: interval({ fields: 'hour' }), + intervalHourToMinute: interval({ fields: 'hour to minute' }), + intervalHourToSecond: interval({ fields: 'hour to second' }), + intervalMinute: interval({ fields: 'minute' }), + intervalMinuteToSecond: interval({ fields: 'minute to second' }), + intervalSecond: interval({ fields: 'second' }), +}); diff --git a/drizzle-seed/tests/pg/allDataTypesTest/pg_all_data_types.test.ts b/drizzle-seed/tests/pg/allDataTypesTest/pg_all_data_types.test.ts index 7dfbc089b..62d0895c0 100644 --- a/drizzle-seed/tests/pg/allDataTypesTest/pg_all_data_types.test.ts +++ b/drizzle-seed/tests/pg/allDataTypesTest/pg_all_data_types.test.ts @@ -105,6 +105,26 @@ beforeAll(async () => { ); `, ); + + await db.execute( + sql` + CREATE TABLE IF NOT EXISTS "seeder_lib_pg"."intervals" ( + "intervalYear" interval year, + "intervalYearToMonth" interval year to month, + "intervalMonth" interval month, + "intervalDay" interval day, + "intervalDayToHour" interval day to hour, + "intervalDayToMinute" interval day to minute, + "intervalDayToSecond" interval day to second, + "intervalHour" interval hour, + "intervalHourToMinute" interval hour to minute, + "intervalHourToSecond" interval hour to second, + "intervalMinute" interval minute, + "intervalMinuteToSecond" interval minute to second, + "intervalSecond" interval second + ); + `, + ); }); afterAll(async () => { @@ -157,3 +177,13 @@ test('nd arrays', async () => { expect(predicate0 && predicate1 && predicate2 && predicate3 && predicate4).toBe(true); }); + +test('intervals test', async () => { + await seed(db, { intervals: schema.intervals }, { count: 1000 }); + + const intervals = await db.select().from(schema.intervals); + // every value in each rows does not equal undefined. + const predicate = intervals.every((row) => Object.values(row).every((val) => val !== undefined && val !== null)); + + expect(predicate).toBe(true); +}); From 42d7de9eee27adf5a4b60ca5461ac99b739af1e6 Mon Sep 17 00:00:00 2001 From: Mario564 Date: Tue, 17 Dec 2024 10:50:41 -0800 Subject: [PATCH 10/32] Add SS support in drizzle-zod --- .../src/singlestore-core/columns/all.ts | 5 +- .../src/singlestore-core/columns/char.ts | 71 +-- .../src/singlestore-core/columns/text.ts | 2 +- .../src/singlestore-core/columns/varchar.ts | 71 +-- drizzle-zod/src/column.ts | 84 ++- drizzle-zod/tests/singlestore.test.ts | 491 ++++++++++++++++++ 6 files changed, 649 insertions(+), 75 deletions(-) create mode 100644 drizzle-zod/tests/singlestore.test.ts diff --git a/drizzle-orm/src/singlestore-core/columns/all.ts b/drizzle-orm/src/singlestore-core/columns/all.ts index 66d289e3f..450417060 100644 --- a/drizzle-orm/src/singlestore-core/columns/all.ts +++ b/drizzle-orm/src/singlestore-core/columns/all.ts @@ -15,7 +15,7 @@ import { mediumint } from './mediumint.ts'; import { real } from './real.ts'; import { serial } from './serial.ts'; import { smallint } from './smallint.ts'; -import { text } from './text.ts'; +import { longtext, mediumtext, text, tinytext } from './text.ts'; import { time } from './time.ts'; import { timestamp } from './timestamp.ts'; import { tinyint } from './tinyint.ts'; @@ -42,7 +42,10 @@ export function getSingleStoreColumnBuilders() { real, serial, smallint, + longtext, + mediumtext, text, + tinytext, time, timestamp, tinyint, diff --git a/drizzle-orm/src/singlestore-core/columns/char.ts b/drizzle-orm/src/singlestore-core/columns/char.ts index 512460f92..903946688 100644 --- a/drizzle-orm/src/singlestore-core/columns/char.ts +++ b/drizzle-orm/src/singlestore-core/columns/char.ts @@ -5,26 +5,31 @@ import type { AnySingleStoreTable } from '~/singlestore-core/table.ts'; import { getColumnNameAndConfig, type Writable } from '~/utils.ts'; import { SingleStoreColumn, SingleStoreColumnBuilder } from './common.ts'; -export type SingleStoreCharBuilderInitial = - SingleStoreCharBuilder<{ - name: TName; - dataType: 'string'; - columnType: 'SingleStoreChar'; - data: TEnum[number]; - driverParam: number | string; - enumValues: TEnum; - generated: undefined; - }>; +export type SingleStoreCharBuilderInitial< + TName extends string, + TEnum extends [string, ...string[]], + TLength extends number | undefined, +> = SingleStoreCharBuilder<{ + name: TName; + dataType: 'string'; + columnType: 'SingleStoreChar'; + data: TEnum[number]; + driverParam: number | string; + enumValues: TEnum; + generated: undefined; + length: TLength; +}>; -export class SingleStoreCharBuilder> - extends SingleStoreColumnBuilder< - T, - SingleStoreCharConfig - > -{ +export class SingleStoreCharBuilder< + T extends ColumnBuilderBaseConfig<'string', 'SingleStoreChar'> & { length?: number | undefined }, +> extends SingleStoreColumnBuilder< + T, + SingleStoreCharConfig, + { length: T['length'] } +> { static override readonly [entityKind]: string = 'SingleStoreCharBuilder'; - constructor(name: T['name'], config: SingleStoreCharConfig) { + constructor(name: T['name'], config: SingleStoreCharConfig) { super(name, 'string', 'SingleStoreChar'); this.config.length = config.length; this.config.enum = config.enum; @@ -33,20 +38,20 @@ export class SingleStoreCharBuilder( table: AnySingleStoreTable<{ name: TTableName }>, - ): SingleStoreChar & { enumValues: T['enumValues'] }> { - return new SingleStoreChar & { enumValues: T['enumValues'] }>( + ): SingleStoreChar & { length: T['length']; enumValues: T['enumValues'] }> { + return new SingleStoreChar & { length: T['length']; enumValues: T['enumValues'] }>( table, this.config as ColumnBuilderRuntimeConfig, ); } } -export class SingleStoreChar> - extends SingleStoreColumn> +export class SingleStoreChar & { length?: number | undefined }> + extends SingleStoreColumn, { length: T['length'] }> { static override readonly [entityKind]: string = 'SingleStoreChar'; - readonly length: number | undefined = this.config.length; + readonly length: T['length'] = this.config.length; override readonly enumValues = this.config.enum; getSQLType(): string { @@ -56,19 +61,25 @@ export class SingleStoreChar { - length?: number; enum?: TEnum; + length?: TLength; } -export function char(): SingleStoreCharBuilderInitial<'', [string, ...string[]]>; -export function char>( - config?: SingleStoreCharConfig>, -): SingleStoreCharBuilderInitial<'', Writable>; -export function char>( +export function char(): SingleStoreCharBuilderInitial<'', [string, ...string[]], undefined>; +export function char, L extends number | undefined>( + config?: SingleStoreCharConfig, L>, +): SingleStoreCharBuilderInitial<'', Writable, L>; +export function char< + TName extends string, + U extends string, + T extends Readonly<[U, ...U[]]>, + L extends number | undefined, +>( name: TName, - config?: SingleStoreCharConfig>, -): SingleStoreCharBuilderInitial>; + config?: SingleStoreCharConfig, L>, +): SingleStoreCharBuilderInitial, L>; export function char(a?: string | SingleStoreCharConfig, b: SingleStoreCharConfig = {}): any { const { name, config } = getColumnNameAndConfig(a, b); return new SingleStoreCharBuilder(name, config as any); diff --git a/drizzle-orm/src/singlestore-core/columns/text.ts b/drizzle-orm/src/singlestore-core/columns/text.ts index 425da550f..d4d46ab2c 100644 --- a/drizzle-orm/src/singlestore-core/columns/text.ts +++ b/drizzle-orm/src/singlestore-core/columns/text.ts @@ -48,7 +48,7 @@ export class SingleStoreText = - SingleStoreVarCharBuilder< - { - name: TName; - dataType: 'string'; - columnType: 'SingleStoreVarChar'; - data: TEnum[number]; - driverParam: number | string; - enumValues: TEnum; - generated: undefined; - } - >; +export type SingleStoreVarCharBuilderInitial< + TName extends string, + TEnum extends [string, ...string[]], + TLength extends number | undefined, +> = SingleStoreVarCharBuilder< + { + name: TName; + dataType: 'string'; + columnType: 'SingleStoreVarChar'; + data: TEnum[number]; + driverParam: number | string; + enumValues: TEnum; + generated: undefined; + length: TLength; + } +>; -export class SingleStoreVarCharBuilder> - extends SingleStoreColumnBuilder> -{ +export class SingleStoreVarCharBuilder< + T extends ColumnBuilderBaseConfig<'string', 'SingleStoreVarChar'> & { length?: number | undefined }, +> extends SingleStoreColumnBuilder, { length: T['length'] }> { static override readonly [entityKind]: string = 'SingleStoreVarCharBuilder'; /** @internal */ - constructor(name: T['name'], config: SingleStoreVarCharConfig) { + constructor(name: T['name'], config: SingleStoreVarCharConfig) { super(name, 'string', 'SingleStoreVarChar'); this.config.length = config.length; this.config.enum = config.enum; @@ -33,21 +37,22 @@ export class SingleStoreVarCharBuilder( table: AnySingleStoreTable<{ name: TTableName }>, - ): SingleStoreVarChar & { enumValues: T['enumValues'] }> { - return new SingleStoreVarChar & { enumValues: T['enumValues'] }>( + ): SingleStoreVarChar & { length: T['length']; enumValues: T['enumValues'] }> { + return new SingleStoreVarChar< + MakeColumnConfig & { length: T['length']; enumValues: T['enumValues'] } + >( table, this.config as ColumnBuilderRuntimeConfig, ); } } -export class SingleStoreVarChar> - extends SingleStoreColumn> -{ +export class SingleStoreVarChar< + T extends ColumnBaseConfig<'string', 'SingleStoreVarChar'> & { length?: number | undefined }, +> extends SingleStoreColumn, { length: T['length'] }> { static override readonly [entityKind]: string = 'SingleStoreVarChar'; - readonly length: number | undefined = this.config.length; - + readonly length: T['length'] = this.config.length; override readonly enumValues = this.config.enum; getSQLType(): string { @@ -57,18 +62,24 @@ export class SingleStoreVarChar { - length: number; enum?: TEnum; + length?: TLength; } -export function varchar>( - config: SingleStoreVarCharConfig>, -): SingleStoreVarCharBuilderInitial<'', Writable>; -export function varchar>( +export function varchar, L extends number | undefined>( + config: SingleStoreVarCharConfig, L>, +): SingleStoreVarCharBuilderInitial<'', Writable, L>; +export function varchar< + TName extends string, + U extends string, + T extends Readonly<[U, ...U[]]>, + L extends number | undefined, +>( name: TName, - config: SingleStoreVarCharConfig>, -): SingleStoreVarCharBuilderInitial>; + config: SingleStoreVarCharConfig, L>, +): SingleStoreVarCharBuilderInitial, L>; export function varchar(a?: string | SingleStoreVarCharConfig, b?: SingleStoreVarCharConfig): any { const { name, config } = getColumnNameAndConfig(a, b); return new SingleStoreVarCharBuilder(name, config as any); diff --git a/drizzle-zod/src/column.ts b/drizzle-zod/src/column.ts index d6fff3445..23bc3c142 100644 --- a/drizzle-zod/src/column.ts +++ b/drizzle-zod/src/column.ts @@ -37,6 +37,21 @@ import type { PgVarchar, PgVector, } from 'drizzle-orm/pg-core'; +import type { + SingleStoreBigInt53, + SingleStoreChar, + SingleStoreDouble, + SingleStoreFloat, + SingleStoreInt, + SingleStoreMediumInt, + SingleStoreReal, + SingleStoreSerial, + SingleStoreSmallInt, + SingleStoreText, + SingleStoreTinyInt, + SingleStoreVarChar, + SingleStoreYear, +} from 'drizzle-orm/singlestore-core'; import type { SQLiteInteger, SQLiteReal, SQLiteText } from 'drizzle-orm/sqlite-core'; import { z } from 'zod'; import type { z as zod } from 'zod'; @@ -114,57 +129,92 @@ function numberColumnToSchema(column: Column, z: typeof zod): z.ZodTypeAny { let max!: number; let integer = false; - if (isColumnType>(column, ['MySqlTinyInt'])) { + if (isColumnType | SingleStoreTinyInt>(column, ['MySqlTinyInt', 'SingleStoreTinyInt'])) { min = unsigned ? 0 : CONSTANTS.INT8_MIN; max = unsigned ? CONSTANTS.INT8_UNSIGNED_MAX : CONSTANTS.INT8_MAX; integer = true; } else if ( - isColumnType | PgSmallSerial | MySqlSmallInt>(column, [ + isColumnType | PgSmallSerial | MySqlSmallInt | SingleStoreSmallInt>(column, [ 'PgSmallInt', 'PgSmallSerial', 'MySqlSmallInt', + 'SingleStoreSmallInt', ]) ) { min = unsigned ? 0 : CONSTANTS.INT16_MIN; max = unsigned ? CONSTANTS.INT16_UNSIGNED_MAX : CONSTANTS.INT16_MAX; integer = true; } else if ( - isColumnType | MySqlFloat | MySqlMediumInt>(column, [ + isColumnType< + PgReal | MySqlFloat | MySqlMediumInt | SingleStoreMediumInt | SingleStoreFloat + >(column, [ 'PgReal', 'MySqlFloat', 'MySqlMediumInt', + 'SingleStoreMediumInt', + 'SingleStoreFloat', ]) ) { min = unsigned ? 0 : CONSTANTS.INT24_MIN; max = unsigned ? CONSTANTS.INT24_UNSIGNED_MAX : CONSTANTS.INT24_MAX; - integer = isColumnType(column, ['MySqlMediumInt']); + integer = isColumnType(column, ['MySqlMediumInt', 'SingleStoreMediumInt']); } else if ( - isColumnType | PgSerial | MySqlInt>(column, ['PgInteger', 'PgSerial', 'MySqlInt']) + isColumnType | PgSerial | MySqlInt | SingleStoreInt>(column, [ + 'PgInteger', + 'PgSerial', + 'MySqlInt', + 'SingleStoreInt', + ]) ) { min = unsigned ? 0 : CONSTANTS.INT32_MIN; max = unsigned ? CONSTANTS.INT32_UNSIGNED_MAX : CONSTANTS.INT32_MAX; integer = true; } else if ( - isColumnType | MySqlReal | MySqlDouble | SQLiteReal>(column, [ + isColumnType< + | PgDoublePrecision + | MySqlReal + | MySqlDouble + | SingleStoreReal + | SingleStoreDouble + | SQLiteReal + >(column, [ 'PgDoublePrecision', 'MySqlReal', 'MySqlDouble', + 'SingleStoreReal', + 'SingleStoreDouble', 'SQLiteReal', ]) ) { min = unsigned ? 0 : CONSTANTS.INT48_MIN; max = unsigned ? CONSTANTS.INT48_UNSIGNED_MAX : CONSTANTS.INT48_MAX; } else if ( - isColumnType | PgBigSerial53 | MySqlBigInt53 | MySqlSerial | SQLiteInteger>( + isColumnType< + | PgBigInt53 + | PgBigSerial53 + | MySqlBigInt53 + | MySqlSerial + | SingleStoreBigInt53 + | SingleStoreSerial + | SQLiteInteger + >( column, - ['PgBigInt53', 'PgBigSerial53', 'MySqlBigInt53', 'MySqlSerial', 'SQLiteInteger'], + [ + 'PgBigInt53', + 'PgBigSerial53', + 'MySqlBigInt53', + 'MySqlSerial', + 'SingleStoreBigInt53', + 'SingleStoreSerial', + 'SQLiteInteger', + ], ) ) { - unsigned = unsigned || isColumnType(column, ['MySqlSerial']); + unsigned = unsigned || isColumnType(column, ['MySqlSerial', 'SingleStoreSerial']); min = unsigned ? 0 : Number.MIN_SAFE_INTEGER; max = Number.MAX_SAFE_INTEGER; integer = true; - } else if (isColumnType>(column, ['MySqlYear'])) { + } else if (isColumnType | SingleStoreYear>(column, ['MySqlYear', 'SingleStoreYear'])) { min = 1901; max = 2155; integer = true; @@ -196,9 +246,11 @@ function stringColumnToSchema(column: Column, z: typeof zod): z.ZodTypeAny { if (isColumnType | SQLiteText>(column, ['PgVarchar', 'SQLiteText'])) { max = column.length; - } else if (isColumnType>(column, ['MySqlVarChar'])) { + } else if ( + isColumnType | SingleStoreVarChar>(column, ['MySqlVarChar', 'SingleStoreVarChar']) + ) { max = column.length ?? CONSTANTS.INT16_UNSIGNED_MAX; - } else if (isColumnType>(column, ['MySqlText'])) { + } else if (isColumnType | SingleStoreText>(column, ['MySqlText', 'SingleStoreText'])) { if (column.textType === 'longtext') { max = CONSTANTS.INT32_UNSIGNED_MAX; } else if (column.textType === 'mediumtext') { @@ -210,7 +262,13 @@ function stringColumnToSchema(column: Column, z: typeof zod): z.ZodTypeAny { } } - if (isColumnType | MySqlChar>(column, ['PgChar', 'MySqlChar'])) { + if ( + isColumnType | MySqlChar | SingleStoreChar>(column, [ + 'PgChar', + 'MySqlChar', + 'SingleStoreChar', + ]) + ) { max = column.length; fixed = true; } diff --git a/drizzle-zod/tests/singlestore.test.ts b/drizzle-zod/tests/singlestore.test.ts new file mode 100644 index 000000000..b91c74be8 --- /dev/null +++ b/drizzle-zod/tests/singlestore.test.ts @@ -0,0 +1,491 @@ +import { type Equal } from 'drizzle-orm'; +import { customType, int, serial, singlestoreSchema, singlestoreTable, text } from 'drizzle-orm/singlestore-core'; +import { test } from 'vitest'; +import { z } from 'zod'; +import { jsonSchema } from '~/column.ts'; +import { CONSTANTS } from '~/constants.ts'; +import { createInsertSchema, createSelectSchema, createUpdateSchema } from '../src'; +import { Expect, expectSchemaShape } from './utils.ts'; + +const intSchema = z.number().min(CONSTANTS.INT32_MIN).max(CONSTANTS.INT32_MAX).int(); +const serialNumberModeSchema = z.number().min(0).max(Number.MAX_SAFE_INTEGER).int(); +const textSchema = z.string().max(CONSTANTS.INT16_UNSIGNED_MAX); + +test('table - select', (t) => { + const table = singlestoreTable('test', { + id: serial().primaryKey(), + name: text().notNull(), + }); + + const result = createSelectSchema(table); + const expected = z.object({ id: serialNumberModeSchema, name: textSchema }); + expectSchemaShape(t, expected).from(result); + Expect>(); +}); + +test('table in schema - select', (tc) => { + const schema = singlestoreSchema('test'); + const table = schema.table('test', { + id: serial().primaryKey(), + name: text().notNull(), + }); + + const result = createSelectSchema(table); + const expected = z.object({ id: serialNumberModeSchema, name: textSchema }); + expectSchemaShape(tc, expected).from(result); + Expect>(); +}); + +test('table - insert', (t) => { + const table = singlestoreTable('test', { + id: serial().primaryKey(), + name: text().notNull(), + age: int(), + }); + + const result = createInsertSchema(table); + const expected = z.object({ + id: serialNumberModeSchema.optional(), + name: textSchema, + age: intSchema.nullable().optional(), + }); + expectSchemaShape(t, expected).from(result); + Expect>(); +}); + +test('table - update', (t) => { + const table = singlestoreTable('test', { + id: serial().primaryKey(), + name: text().notNull(), + age: int(), + }); + + const result = createUpdateSchema(table); + const expected = z.object({ + id: serialNumberModeSchema.optional(), + name: textSchema.optional(), + age: intSchema.nullable().optional(), + }); + expectSchemaShape(t, expected).from(result); + Expect>(); +}); + +// TODO: SingleStore doesn't support views yet. Add these tests when they're added + +// test('view qb - select', (t) => { +// const table = singlestoreTable('test', { +// id: serial().primaryKey(), +// name: text().notNull(), +// }); +// const view = mysqlView('test').as((qb) => qb.select({ id: table.id, age: sql``.as('age') }).from(table)); + +// const result = createSelectSchema(view); +// const expected = z.object({ id: serialNumberModeSchema, age: z.any() }); +// expectSchemaShape(t, expected).from(result); +// Expect>(); +// }); + +// test('view columns - select', (t) => { +// const view = mysqlView('test', { +// id: serial().primaryKey(), +// name: text().notNull(), +// }).as(sql``); + +// const result = createSelectSchema(view); +// const expected = z.object({ id: serialNumberModeSchema, name: textSchema }); +// expectSchemaShape(t, expected).from(result); +// Expect>(); +// }); + +// test('view with nested fields - select', (t) => { +// const table = singlestoreTable('test', { +// id: serial().primaryKey(), +// name: text().notNull(), +// }); +// const view = mysqlView('test').as((qb) => +// qb.select({ +// id: table.id, +// nested: { +// name: table.name, +// age: sql``.as('age'), +// }, +// table, +// }).from(table) +// ); + +// const result = createSelectSchema(view); +// const expected = z.object({ +// id: serialNumberModeSchema, +// nested: z.object({ name: textSchema, age: z.any() }), +// table: z.object({ id: serialNumberModeSchema, name: textSchema }), +// }); +// expectSchemaShape(t, expected).from(result); +// Expect>(); +// }); + +test('nullability - select', (t) => { + const table = singlestoreTable('test', { + c1: int(), + c2: int().notNull(), + c3: int().default(1), + c4: int().notNull().default(1), + }); + + const result = createSelectSchema(table); + const expected = z.object({ + c1: intSchema.nullable(), + c2: intSchema, + c3: intSchema.nullable(), + c4: intSchema, + }); + expectSchemaShape(t, expected).from(result); + Expect>(); +}); + +test('nullability - insert', (t) => { + const table = singlestoreTable('test', { + c1: int(), + c2: int().notNull(), + c3: int().default(1), + c4: int().notNull().default(1), + c5: int().generatedAlwaysAs(1), + }); + + const result = createInsertSchema(table); + const expected = z.object({ + c1: intSchema.nullable().optional(), + c2: intSchema, + c3: intSchema.nullable().optional(), + c4: intSchema.optional(), + }); + expectSchemaShape(t, expected).from(result); + Expect>(); +}); + +test('nullability - update', (t) => { + const table = singlestoreTable('test', { + c1: int(), + c2: int().notNull(), + c3: int().default(1), + c4: int().notNull().default(1), + c5: int().generatedAlwaysAs(1), + }); + + const result = createUpdateSchema(table); + const expected = z.object({ + c1: intSchema.nullable().optional(), + c2: intSchema.optional(), + c3: intSchema.nullable().optional(), + c4: intSchema.optional(), + }); + expectSchemaShape(t, expected).from(result); + Expect>(); +}); + +test('refine table - select', (t) => { + const table = singlestoreTable('test', { + c1: int(), + c2: int().notNull(), + c3: int().notNull(), + }); + + const result = createSelectSchema(table, { + c2: (schema) => schema.max(1000), + c3: z.string().transform(Number), + }); + const expected = z.object({ + c1: intSchema.nullable(), + c2: intSchema.max(1000), + c3: z.string().transform(Number), + }); + + expectSchemaShape(t, expected).from(result); + Expect>(); +}); + +test('refine table - select with custom data type', (t) => { + const customText = customType({ dataType: () => 'text' }); + const table = singlestoreTable('test', { + c1: int(), + c2: int().notNull(), + c3: int().notNull(), + c4: customText(), + }); + + const customTextSchema = z.string().min(1).max(100); + const result = createSelectSchema(table, { + c2: (schema) => schema.max(1000), + c3: z.string().transform(Number), + c4: customTextSchema, + }); + const expected = z.object({ + c1: intSchema.nullable(), + c2: intSchema.max(1000), + c3: z.string().transform(Number), + c4: customTextSchema, + }); + + expectSchemaShape(t, expected).from(result); + Expect>(); +}); + +test('refine table - insert', (t) => { + const table = singlestoreTable('test', { + c1: int(), + c2: int().notNull(), + c3: int().notNull(), + c4: int().generatedAlwaysAs(1), + }); + + const result = createInsertSchema(table, { + c2: (schema) => schema.max(1000), + c3: z.string().transform(Number), + }); + const expected = z.object({ + c1: intSchema.nullable().optional(), + c2: intSchema.max(1000), + c3: z.string().transform(Number), + }); + expectSchemaShape(t, expected).from(result); + Expect>(); +}); + +test('refine table - update', (t) => { + const table = singlestoreTable('test', { + c1: int(), + c2: int().notNull(), + c3: int().notNull(), + c4: int().generatedAlwaysAs(1), + }); + + const result = createUpdateSchema(table, { + c2: (schema) => schema.max(1000), + c3: z.string().transform(Number), + }); + const expected = z.object({ + c1: intSchema.nullable().optional(), + c2: intSchema.max(1000).optional(), + c3: z.string().transform(Number), + }); + expectSchemaShape(t, expected).from(result); + Expect>(); +}); + +// test('refine view - select', (t) => { +// const table = singlestoreTable('test', { +// c1: int(), +// c2: int(), +// c3: int(), +// c4: int(), +// c5: int(), +// c6: int(), +// }); +// const view = mysqlView('test').as((qb) => +// qb.select({ +// c1: table.c1, +// c2: table.c2, +// c3: table.c3, +// nested: { +// c4: table.c4, +// c5: table.c5, +// c6: table.c6, +// }, +// table, +// }).from(table) +// ); + +// const result = createSelectSchema(view, { +// c2: (schema) => schema.max(1000), +// c3: z.string().transform(Number), +// nested: { +// c5: (schema) => schema.max(1000), +// c6: z.string().transform(Number), +// }, +// table: { +// c2: (schema) => schema.max(1000), +// c3: z.string().transform(Number), +// }, +// }); +// const expected = z.object({ +// c1: intSchema.nullable(), +// c2: intSchema.max(1000).nullable(), +// c3: z.string().transform(Number), +// nested: z.object({ +// c4: intSchema.nullable(), +// c5: intSchema.max(1000).nullable(), +// c6: z.string().transform(Number), +// }), +// table: z.object({ +// c1: intSchema.nullable(), +// c2: intSchema.max(1000).nullable(), +// c3: z.string().transform(Number), +// c4: intSchema.nullable(), +// c5: intSchema.nullable(), +// c6: intSchema.nullable(), +// }), +// }); +// expectSchemaShape(t, expected).from(result); +// Expect>(); +// }); + +test('all data types', (t) => { + const table = singlestoreTable('test', ({ + bigint, + binary, + boolean, + char, + date, + datetime, + decimal, + double, + float, + int, + json, + mediumint, + singlestoreEnum, + real, + serial, + smallint, + text, + time, + timestamp, + tinyint, + varchar, + varbinary, + year, + longtext, + mediumtext, + tinytext, + }) => ({ + bigint1: bigint({ mode: 'number' }).notNull(), + bigint2: bigint({ mode: 'bigint' }).notNull(), + bigint3: bigint({ unsigned: true, mode: 'number' }).notNull(), + bigint4: bigint({ unsigned: true, mode: 'bigint' }).notNull(), + binary: binary({ length: 10 }).notNull(), + boolean: boolean().notNull(), + char1: char({ length: 10 }).notNull(), + char2: char({ length: 1, enum: ['a', 'b', 'c'] }).notNull(), + date1: date({ mode: 'date' }).notNull(), + date2: date({ mode: 'string' }).notNull(), + datetime1: datetime({ mode: 'date' }).notNull(), + datetime2: datetime({ mode: 'string' }).notNull(), + decimal1: decimal().notNull(), + decimal2: decimal({ unsigned: true }).notNull(), + double1: double().notNull(), + double2: double({ unsigned: true }).notNull(), + float1: float().notNull(), + float2: float({ unsigned: true }).notNull(), + int1: int().notNull(), + int2: int({ unsigned: true }).notNull(), + json: json().notNull(), + mediumint1: mediumint().notNull(), + mediumint2: mediumint({ unsigned: true }).notNull(), + enum: singlestoreEnum('enum', ['a', 'b', 'c']).notNull(), + real: real().notNull(), + serial: serial().notNull(), + smallint1: smallint().notNull(), + smallint2: smallint({ unsigned: true }).notNull(), + text1: text().notNull(), + text2: text({ enum: ['a', 'b', 'c'] }).notNull(), + time: time().notNull(), + timestamp1: timestamp({ mode: 'date' }).notNull(), + timestamp2: timestamp({ mode: 'string' }).notNull(), + tinyint1: tinyint().notNull(), + tinyint2: tinyint({ unsigned: true }).notNull(), + varchar1: varchar({ length: 10 }).notNull(), + varchar2: varchar({ length: 1, enum: ['a', 'b', 'c'] }).notNull(), + varbinary: varbinary({ length: 10 }).notNull(), + year: year().notNull(), + longtext1: longtext().notNull(), + longtext2: longtext({ enum: ['a', 'b', 'c'] }).notNull(), + mediumtext1: mediumtext().notNull(), + mediumtext2: mediumtext({ enum: ['a', 'b', 'c'] }).notNull(), + tinytext1: tinytext().notNull(), + tinytext2: tinytext({ enum: ['a', 'b', 'c'] }).notNull(), + })); + + const result = createSelectSchema(table); + const expected = z.object({ + bigint1: z.number().min(Number.MIN_SAFE_INTEGER).max(Number.MAX_SAFE_INTEGER).int(), + bigint2: z.bigint().min(CONSTANTS.INT64_MIN).max(CONSTANTS.INT64_MAX), + bigint3: z.number().min(0).max(Number.MAX_SAFE_INTEGER).int(), + bigint4: z.bigint().min(0n).max(CONSTANTS.INT64_UNSIGNED_MAX), + binary: z.string(), + boolean: z.boolean(), + char1: z.string().length(10), + char2: z.enum(['a', 'b', 'c']), + date1: z.date(), + date2: z.string(), + datetime1: z.date(), + datetime2: z.string(), + decimal1: z.string(), + decimal2: z.string(), + double1: z.number().min(CONSTANTS.INT48_MIN).max(CONSTANTS.INT48_MAX), + double2: z.number().min(0).max(CONSTANTS.INT48_UNSIGNED_MAX), + float1: z.number().min(CONSTANTS.INT24_MIN).max(CONSTANTS.INT24_MAX), + float2: z.number().min(0).max(CONSTANTS.INT24_UNSIGNED_MAX), + int1: z.number().min(CONSTANTS.INT32_MIN).max(CONSTANTS.INT32_MAX).int(), + int2: z.number().min(0).max(CONSTANTS.INT32_UNSIGNED_MAX).int(), + json: jsonSchema, + mediumint1: z.number().min(CONSTANTS.INT24_MIN).max(CONSTANTS.INT24_MAX).int(), + mediumint2: z.number().min(0).max(CONSTANTS.INT24_UNSIGNED_MAX).int(), + enum: z.enum(['a', 'b', 'c']), + real: z.number().min(CONSTANTS.INT48_MIN).max(CONSTANTS.INT48_MAX), + serial: z.number().min(0).max(Number.MAX_SAFE_INTEGER).int(), + smallint1: z.number().min(CONSTANTS.INT16_MIN).max(CONSTANTS.INT16_MAX).int(), + smallint2: z.number().min(0).max(CONSTANTS.INT16_UNSIGNED_MAX).int(), + text1: z.string().max(CONSTANTS.INT16_UNSIGNED_MAX), + text2: z.enum(['a', 'b', 'c']), + time: z.string(), + timestamp1: z.date(), + timestamp2: z.string(), + tinyint1: z.number().min(CONSTANTS.INT8_MIN).max(CONSTANTS.INT8_MAX).int(), + tinyint2: z.number().min(0).max(CONSTANTS.INT8_UNSIGNED_MAX).int(), + varchar1: z.string().max(10), + varchar2: z.enum(['a', 'b', 'c']), + varbinary: z.string(), + year: z.number().min(1901).max(2155).int(), + longtext1: z.string().max(CONSTANTS.INT32_UNSIGNED_MAX), + longtext2: z.enum(['a', 'b', 'c']), + mediumtext1: z.string().max(CONSTANTS.INT24_UNSIGNED_MAX), + mediumtext2: z.enum(['a', 'b', 'c']), + tinytext1: z.string().max(CONSTANTS.INT8_UNSIGNED_MAX), + tinytext2: z.enum(['a', 'b', 'c']), + }); + expectSchemaShape(t, expected).from(result); + Expect>(); +}); + +/* Disallow unknown keys in table refinement - select */ { + const table = singlestoreTable('test', { id: int() }); + // @ts-expect-error + createSelectSchema(table, { unknown: z.string() }); +} + +/* Disallow unknown keys in table refinement - insert */ { + const table = singlestoreTable('test', { id: int() }); + // @ts-expect-error + createInsertSchema(table, { unknown: z.string() }); +} + +/* Disallow unknown keys in table refinement - update */ { + const table = singlestoreTable('test', { id: int() }); + // @ts-expect-error + createUpdateSchema(table, { unknown: z.string() }); +} + +// /* Disallow unknown keys in view qb - select */ { +// const table = singlestoreTable('test', { id: int() }); +// const view = mysqlView('test').as((qb) => qb.select().from(table)); +// const nestedSelect = mysqlView('test').as((qb) => qb.select({ table }).from(table)); +// // @ts-expect-error +// createSelectSchema(view, { unknown: z.string() }); +// // @ts-expect-error +// createSelectSchema(nestedSelect, { table: { unknown: z.string() } }); +// } + +// /* Disallow unknown keys in view columns - select */ { +// const view = mysqlView('test', { id: int() }).as(sql``); +// // @ts-expect-error +// createSelectSchema(view, { unknown: z.string() }); +// } From 6f8a6ccfbe0af44a3a8b50717016be873de95ce1 Mon Sep 17 00:00:00 2001 From: Mario564 Date: Tue, 17 Dec 2024 12:45:58 -0800 Subject: [PATCH 11/32] Fix infinitely recursive type in drizzle-valibot --- drizzle-valibot/src/column.types.ts | 98 +++++++------------- drizzle-valibot/src/schema.types.internal.ts | 3 +- 2 files changed, 32 insertions(+), 69 deletions(-) diff --git a/drizzle-valibot/src/column.types.ts b/drizzle-valibot/src/column.types.ts index e6cd797ed..b9567b12d 100644 --- a/drizzle-valibot/src/column.types.ts +++ b/drizzle-valibot/src/column.types.ts @@ -30,21 +30,23 @@ export type ExtractAdditionalProperties = { fixedLength: TColumn['_']['columnType'] extends 'PgChar' | 'MySqlChar' | 'PgHalfVector' | 'PgVector' | 'PgArray' ? true : false; - arrayPipelines: []; }; -type RemovePipeIfNoElements> = T extends - infer TPiped extends { pipe: [any, ...any[]] } ? TPiped['pipe'][1] extends undefined ? T['pipe'][0] : TPiped - : never; +type GetLengthAction, TType extends string | ArrayLike> = + T['fixedLength'] extends true ? v.LengthAction + : v.MaxLengthAction; -type BuildArraySchema< - TWrapped extends v.GenericSchema, - TPipelines extends any[][], -> = TPipelines extends [infer TFirst extends any[], ...infer TRest extends any[][]] - ? BuildArraySchema, ...TFirst]>>, TRest> - : TPipelines extends [infer TFirst extends any[]] - ? BuildArraySchema, ...TFirst]>>, []> - : TWrapped; +type GetArraySchema = v.ArraySchema< + GetValibotType< + T['_']['data'], + T['_']['dataType'], + T['_']['columnType'], + GetEnumValuesFromColumn, + GetBaseColumn, + ExtractAdditionalProperties + >, + undefined +>; export type GetValibotType< TData, @@ -53,51 +55,22 @@ export type GetValibotType< TEnumValues extends [string, ...string[]] | undefined, TBaseColumn extends Column | undefined, TAdditionalProperties extends Record, -> = TColumnType extends 'PgHalfVector' | 'PgVector' ? RemovePipeIfNoElements< - v.SchemaWithPipe< - RemoveNeverElements<[ - v.ArraySchema, undefined>, - TAdditionalProperties['max'] extends number - ? TAdditionalProperties['fixedLength'] extends true ? v.LengthAction - : v.MaxLengthAction - : never, - ]> +> = TColumnType extends 'PgHalfVector' | 'PgVector' ? TAdditionalProperties['max'] extends number ? v.SchemaWithPipe< + [v.ArraySchema, undefined>, GetLengthAction] > - > + : v.ArraySchema, undefined> : TColumnType extends 'PgUUID' ? v.SchemaWithPipe<[v.StringSchema, v.UuidAction]> - // PG array handling start - // Nesting `GetValibotType` within `v.ArraySchema` will cause infinite recursion - // The workaround is to accumulate all the array validations (done via `arrayPipelines` in `TAdditionalProperties`) and then build the schema afterwards - : TAdditionalProperties['arrayFinished'] extends true ? GetValibotType< - TData, - TDataType, - TColumnType, - TEnumValues, - TBaseColumn, - Omit - > extends infer TSchema extends v.GenericSchema ? BuildArraySchema - : never - : TBaseColumn extends Column ? GetValibotType< - TBaseColumn['_']['data'], - TBaseColumn['_']['dataType'], - TBaseColumn['_']['columnType'], - GetEnumValuesFromColumn, - GetBaseColumn, - Omit, 'arrayPipelines'> & { - arrayPipelines: [ - RemoveNeverElements<[ - TAdditionalProperties['max'] extends number - ? TAdditionalProperties['fixedLength'] extends true - ? v.LengthAction[], number, undefined> - : v.MaxLengthAction[], number, undefined> - : never, - ]>, - ...TAdditionalProperties['arrayPipelines'], - ]; - arrayFinished: GetBaseColumn extends undefined ? true : false; - } + : TColumnType extends 'PgBinaryVector' ? v.SchemaWithPipe< + RemoveNeverElements<[ + v.StringSchema, + v.RegexAction, + TAdditionalProperties['max'] extends number ? GetLengthAction : never, + ]> > - // PG array handling end + : TBaseColumn extends Column ? TAdditionalProperties['max'] extends number ? v.SchemaWithPipe< + [GetArraySchema, GetLengthAction] + > + : GetArraySchema : ArrayHasAtLeastOneValue extends true ? v.EnumSchema>, undefined> : TData extends infer TTuple extends [any, ...any[]] ? v.TupleSchema< @@ -147,19 +120,10 @@ export type GetValibotType< v.MaxValueAction, ]> : TData extends boolean ? v.BooleanSchema - : TData extends string ? RemovePipeIfNoElements< - v.SchemaWithPipe< - RemoveNeverElements<[ - v.StringSchema, - TColumnType extends 'PgBinaryVector' ? v.RegexAction - : never, - TAdditionalProperties['max'] extends number - ? TAdditionalProperties['fixedLength'] extends true ? v.LengthAction - : v.MaxLengthAction - : never, - ]> - > - > + : TData extends string + ? TAdditionalProperties['max'] extends number + ? v.SchemaWithPipe<[v.StringSchema, GetLengthAction]> + : v.StringSchema : v.AnySchema; type HandleSelectColumn< diff --git a/drizzle-valibot/src/schema.types.internal.ts b/drizzle-valibot/src/schema.types.internal.ts index 57dcedc7c..e39c34d8a 100644 --- a/drizzle-valibot/src/schema.types.internal.ts +++ b/drizzle-valibot/src/schema.types.internal.ts @@ -67,8 +67,7 @@ export type BuildSchema< TType extends 'select' | 'insert' | 'update', TColumns extends Record, TRefinements extends Record | undefined, -> // @ts-ignore false-positive - = v.ObjectSchema< +> = v.ObjectSchema< Simplify< RemoveNever< { From 1b05f63f6ce744d779acec6d008cedd1914bd1ad Mon Sep 17 00:00:00 2001 From: OleksiiKH0240 Date: Wed, 18 Dec 2024 14:04:38 +0200 Subject: [PATCH 12/32] fixes, partial implementation of generator versioning --- drizzle-seed/src/index.ts | 26 +- drizzle-seed/src/services/GeneratorFuncs.ts | 737 +++++++++++++++++ .../GeneratorVersions/GeneratorsV1.ts | 15 + .../{GeneratorsWrappers.ts => Generators.ts} | 739 +----------------- drizzle-seed/src/services/SeedService.ts | 228 +++--- drizzle-seed/src/types/seedService.ts | 2 +- .../tests/benchmarks/generatorsBenchmark.ts | 2 +- 7 files changed, 912 insertions(+), 837 deletions(-) create mode 100644 drizzle-seed/src/services/GeneratorFuncs.ts create mode 100644 drizzle-seed/src/services/GeneratorVersions/GeneratorsV1.ts rename drizzle-seed/src/services/{GeneratorsWrappers.ts => Generators.ts} (81%) diff --git a/drizzle-seed/src/index.ts b/drizzle-seed/src/index.ts index fde4442a1..248e7ead5 100644 --- a/drizzle-seed/src/index.ts +++ b/drizzle-seed/src/index.ts @@ -9,8 +9,8 @@ import { getTableConfig as getPgTableConfig, PgDatabase, PgTable } from 'drizzle import type { SQLiteColumn } from 'drizzle-orm/sqlite-core'; import { BaseSQLiteDatabase, getTableConfig as getSqliteTableConfig, SQLiteTable } from 'drizzle-orm/sqlite-core'; -import type { AbstractGenerator } from './services/GeneratorsWrappers.ts'; -import { generatorsFuncs } from './services/GeneratorsWrappers.ts'; +import { generatorsFuncs } from './services/GeneratorFuncs.ts'; +import type { AbstractGenerator } from './services/Generators.ts'; import { SeedService } from './services/SeedService.ts'; import type { DrizzleStudioObjectType, DrizzleStudioRelationType } from './types/drizzleStudio.ts'; import type { RefinementsType } from './types/seedService.ts'; @@ -138,7 +138,7 @@ class SeedPromise< constructor( private db: DB, private schema: SCHEMA, - private options?: { count?: number; seed?: number }, + private options?: { count?: number; seed?: number; version?: number }, ) {} then( @@ -259,7 +259,6 @@ export async function seedForDrizzleStudio( sqlDialect, tables, isCyclicRelations, - {}, // TODO: fix later refinements, options, ); @@ -337,7 +336,7 @@ export function seed< | SQLiteTable | any; }, ->(db: DB, schema: SCHEMA, options?: { count?: number; seed?: number }) { +>(db: DB, schema: SCHEMA, options?: { count?: number; seed?: number; version?: number }) { return new SeedPromise(db, schema, options); } @@ -352,7 +351,7 @@ const seedFunc = async ( | SQLiteTable | any; }, - options: { count?: number; seed?: number } = {}, + options: { count?: number; seed?: number; version?: number } = {}, refinements?: RefinementsType, ) => { if (is(db, PgDatabase)) { @@ -490,17 +489,16 @@ const filterPgTables = (schema: { const seedPostgres = async ( db: PgDatabase, schema: { [key: string]: PgTable }, - options: { count?: number; seed?: number } = {}, + options: { count?: number; seed?: number; version?: number } = {}, refinements?: RefinementsType, ) => { const seedService = new SeedService(); - const { tables, relations, tableRelations } = getPostgresInfo(schema); + const { tables, relations } = getPostgresInfo(schema); const generatedTablesGenerators = seedService.generatePossibleGenerators( 'postgresql', tables, relations, - tableRelations, refinements, options, ); @@ -784,10 +782,10 @@ const filterMySqlTables = (schema: { const seedMySql = async ( db: MySqlDatabase, schema: { [key: string]: MySqlTable }, - options: { count?: number; seed?: number } = {}, + options: { count?: number; seed?: number; version?: number } = {}, refinements?: RefinementsType, ) => { - const { tables, relations, tableRelations } = getMySqlInfo(schema); + const { tables, relations } = getMySqlInfo(schema); const seedService = new SeedService(); @@ -795,7 +793,6 @@ const seedMySql = async ( 'mysql', tables, relations, - tableRelations, refinements, options, ); @@ -1003,10 +1000,10 @@ const filterSqliteTables = (schema: { const seedSqlite = async ( db: BaseSQLiteDatabase, schema: { [key: string]: SQLiteTable }, - options: { count?: number; seed?: number } = {}, + options: { count?: number; seed?: number; version?: number } = {}, refinements?: RefinementsType, ) => { - const { tables, relations, tableRelations } = getSqliteInfo(schema); + const { tables, relations } = getSqliteInfo(schema); const seedService = new SeedService(); @@ -1014,7 +1011,6 @@ const seedSqlite = async ( 'sqlite', tables, relations, - tableRelations, refinements, options, ); diff --git a/drizzle-seed/src/services/GeneratorFuncs.ts b/drizzle-seed/src/services/GeneratorFuncs.ts new file mode 100644 index 000000000..08f36b1dc --- /dev/null +++ b/drizzle-seed/src/services/GeneratorFuncs.ts @@ -0,0 +1,737 @@ +import type { AbstractGenerator } from './Generators.ts'; +import { + GenerateBoolean, + GenerateCity, + GenerateCompanyName, + GenerateCountry, + GenerateDate, + GenerateDatetime, + GenerateDefault, + GenerateEmail, + GenerateFirstName, + GenerateFullName, + GenerateInt, + GenerateInterval, + GenerateIntPrimaryKey, + GenerateJobTitle, + GenerateJson, + GenerateLastName, + GenerateLine, + GenerateLoremIpsum, + GenerateNumber, + GeneratePhoneNumber, + GeneratePoint, + GeneratePostcode, + GenerateState, + GenerateStreetAdddress, + GenerateString, + GenerateTime, + GenerateTimestamp, + GenerateUUID, + GenerateValuesFromArray, + GenerateYear, + WeightedRandomGenerator, +} from './Generators.ts'; + +function createGenerator, T>( + generatorConstructor: new(params: T) => GeneratorType, +) { + return ( + ...args: GeneratorType extends GenerateValuesFromArray | GenerateDefault | WeightedRandomGenerator ? [T] + : ([] | [T]) + ): GeneratorType => { + let params = args[0]; + if (params === undefined) params = {} as T; + return new generatorConstructor(params); + }; +} + +export const generatorsFuncs = { + /** + * generates same given value each time the generator is called. + * @param defaultValue - value you want to generate + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * posts: { + * columns: { + * content: funcs.default({ defaultValue: "post content" }), + * }, + * }, + * })); + * ``` + */ + default: createGenerator(GenerateDefault), + + /** + * generates values from given array + * @param values - array of values you want to generate. can be array of weighted values. + * @param isUnique - property that controls if generated values gonna be unique or not. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * posts: { + * columns: { + * title: funcs.valuesFromArray({ + * values: ["Title1", "Title2", "Title3", "Title4", "Title5"], + * isUnique: true + * }), + * }, + * }, + * })); + * + * ``` + * weighted values example + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * posts: { + * columns: { + * title: funcs.valuesFromArray({ + * values: [ + * { weight: 0.35, values: ["Title1", "Title2"] }, + * { weight: 0.5, values: ["Title3", "Title4"] }, + * { weight: 0.15, values: ["Title5"] }, + * ], + * isUnique: false + * }), + * }, + * }, + * })); + * + * ``` + */ + valuesFromArray: createGenerator(GenerateValuesFromArray), + + /** + * generates sequential integers starting with 1. + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * posts: { + * columns: { + * id: funcs.intPrimaryKey(), + * }, + * }, + * })); + * + * ``` + */ + intPrimaryKey: createGenerator(GenerateIntPrimaryKey), + + /** + * generates numbers with floating point in given range. + * @param minValue - lower border of range. + * @param maxValue - upper border of range. + * @param precision - precision of generated number: + * precision equals 10 means that values will be accurate to one tenth (1.2, 34.6); + * precision equals 100 means that values will be accurate to one hundredth (1.23, 34.67). + * @param isUnique - property that controls if generated values gonna be unique or not. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * products: { + * columns: { + * unitPrice: funcs.number({ minValue: 10, maxValue: 120, precision: 100, isUnique: false }), + * }, + * }, + * })); + * + * ``` + */ + number: createGenerator(GenerateNumber), + // uniqueNumber: createGenerator(GenerateUniqueNumber), + + /** + * generates integers within given range. + * @param minValue - lower border of range. + * @param maxValue - upper border of range. + * @param isUnique - property that controls if generated values gonna be unique or not. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * products: { + * columns: { + * unitsInStock: funcs.number({ minValue: 0, maxValue: 100, isUnique: false }), + * }, + * }, + * })); + * + * ``` + */ + int: createGenerator(GenerateInt), + // uniqueInt: createGenerator(GenerateUniqueInt), + + /** + * generates boolean values(true or false) + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * users: { + * columns: { + * isAvailable: funcs.boolean() + * }, + * }, + * })); + * + * ``` + */ + boolean: createGenerator(GenerateBoolean), + + /** + * generates date within given range. + * @param minDate - lower border of range. + * @param maxDate - upper border of range. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * users: { + * columns: { + * birthDate: funcs.date({ minDate: "1990-01-01", maxDate: "2010-12-31" }) + * }, + * }, + * })); + * + * ``` + */ + date: createGenerator(GenerateDate), + + /** + * generates time in 24 hours style. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * users: { + * columns: { + * birthTime: funcs.time() + * }, + * }, + * })); + * + * ``` + */ + time: createGenerator(GenerateTime), + + /** + * generates timestamps. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * orders: { + * columns: { + * shippedDate: funcs.timestamp() + * }, + * }, + * })); + * + * ``` + */ + timestamp: createGenerator(GenerateTimestamp), + + /** + * generates datetime objects. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * orders: { + * columns: { + * shippedDate: funcs.datetime() + * }, + * }, + * })); + * + * ``` + */ + datetime: createGenerator(GenerateDatetime), + + /** + * generates years. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * users: { + * columns: { + * birthYear: funcs.year() + * }, + * }, + * })); + * + * ``` + */ + year: createGenerator(GenerateYear), + + /** + * generates json objects with fixed structure. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * json structure can equal this: + * ``` + * { + * email, + * name, + * isGraduated, + * hasJob, + * salary, + * startedWorking, + * visitedCountries, + * } + * ``` + * or this + * ``` + * { + * email, + * name, + * isGraduated, + * hasJob, + * visitedCountries, + * } + * ``` + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * users: { + * columns: { + * metadata: funcs.json() + * }, + * }, + * })); + * ``` + */ + json: createGenerator(GenerateJson), + // jsonb: createGenerator(GenerateJsonb), + + /** + * generates time intervals. + * + * interval example: "1 years 12 days 5 minutes" + * + * @param isUnique - property that controls if generated values gonna be unique or not. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * @param fields - range of values you want to see in your intervals. + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * users: { + * columns: { + * timeSpentOnWebsite: funcs.interval() + * }, + * }, + * })); + * ``` + */ + interval: createGenerator(GenerateInterval), + // uniqueInterval: createGenerator(GenerateUniqueInterval), + + /** + * generates random strings. + * @param isUnique - property that controls if generated values gonna be unique or not. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * users: { + * columns: { + * hashedPassword: funcs.string({isUnique: false}) + * }, + * }, + * })); + * ``` + */ + string: createGenerator(GenerateString), + // uniqueString: createGenerator(GenerateUniqueString), + + /** + * generates v4 UUID strings if arraySize is not specified, or v4 UUID 1D arrays if it is. + * + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * users: { + * columns: { + * uuid: funcs.uuid({ + * arraySize: 4 + * }) + * }, + * }, + * })); + * ``` + */ + uuid: createGenerator(GenerateUUID), + + /** + * generates person's first names. + * @param isUnique - property that controls if generated values gonna be unique or not. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * users: { + * columns: { + * firstName: funcs.firstName({isUnique: true}) + * }, + * }, + * })); + * ``` + */ + firstName: createGenerator(GenerateFirstName), + // uniqueFirstName: createGenerator(GenerateUniqueName), + + /** + * generates person's last names. + * @param isUnique - property that controls if generated values gonna be unique or not. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * users: { + * columns: { + * lastName: funcs.lastName({isUnique: false}) + * }, + * }, + * })); + * ``` + */ + lastName: createGenerator(GenerateLastName), + // uniqueLastName: createGenerator(GenerateUniqueSurname), + + /** + * generates person's full names. + * @param isUnique - property that controls if generated values gonna be unique or not. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * users: { + * columns: { + * fullName: funcs.fullName({isUnique: true}) + * }, + * }, + * })); + * ``` + */ + fullName: createGenerator(GenerateFullName), + // uniqueFullName: createGenerator(GenerateUniqueFullName), + + /** + * generates unique emails. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * users: { + * columns: { + * email: funcs.email() + * }, + * }, + * })); + * ``` + */ + email: createGenerator(GenerateEmail), + + /** + * generates unique phone numbers. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @param template - phone number template, where all '#' symbols will be substituted with generated digits. + * @param prefixes - array of any string you want to be your phone number prefixes.(not compatible with template property) + * @param generatedDigitsNumbers - number of digits that will be added at the end of prefixes.(not compatible with template property) + * @example + * ```ts + * //generate phone number using template property + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * users: { + * columns: { + * phoneNumber: funcs.phoneNumber({template: "+(380) ###-####"}) + * }, + * }, + * })); + * + * //generate phone number using prefixes and generatedDigitsNumbers properties + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * users: { + * columns: { + * phoneNumber: funcs.phoneNumber({prefixes: [ "+380 99", "+380 67" ], generatedDigitsNumbers: 7}) + * }, + * }, + * })); + * + * //generate phone number using prefixes and generatedDigitsNumbers properties but with different generatedDigitsNumbers for prefixes + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * users: { + * columns: { + * phoneNumber: funcs.phoneNumber({prefixes: [ "+380 99", "+380 67", "+1" ], generatedDigitsNumbers: [7, 7, 10]}) + * }, + * }, + * })); + * + * ``` + */ + phoneNumber: createGenerator(GeneratePhoneNumber), + + /** + * generates country's names. + * @param isUnique - property that controls if generated values gonna be unique or not. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * users: { + * columns: { + * country: funcs.country({isUnique: false}) + * }, + * }, + * })); + * ``` + */ + country: createGenerator(GenerateCountry), + // uniqueCountry: createGenerator(GenerateUniqueCountry), + + /** + * generates city's names. + * @param isUnique - property that controls if generated values gonna be unique or not. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * users: { + * columns: { + * city: funcs.city({isUnique: false}) + * }, + * }, + * })); + * ``` + */ + city: createGenerator(GenerateCity), + // uniqueCity: createGenerator(GenerateUniqueCityName), + + /** + * generates street address. + * @param isUnique - property that controls if generated values gonna be unique or not. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * users: { + * columns: { + * streetAddress: funcs.streetAddress({isUnique: true}) + * }, + * }, + * })); + * ``` + */ + streetAddress: createGenerator(GenerateStreetAdddress), + // uniqueStreetAddress: createGenerator(GenerateUniqueStreetAdddress), + + /** + * generates job titles. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * users: { + * columns: { + * jobTitle: funcs.jobTitle() + * }, + * }, + * })); + * ``` + */ + jobTitle: createGenerator(GenerateJobTitle), + + /** + * generates postal codes. + * + * @param isUnique - property that controls if generated values gonna be unique or not. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * users: { + * columns: { + * postcode: funcs.postcode({isUnique: true}) + * }, + * }, + * })); + * ``` + */ + postcode: createGenerator(GeneratePostcode), + // uniquePostcoe: createGenerator(GenerateUniquePostcode), + + /** + * generates states of America. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * users: { + * columns: { + * state: funcs.state() + * }, + * }, + * })); + * ``` + */ + state: createGenerator(GenerateState), + + /** + * generates company's names. + * + * @param isUnique - property that controls if generated values gonna be unique or not. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * users: { + * columns: { + * company: funcs.companyName({isUnique: true}) + * }, + * }, + * })); + * ``` + */ + companyName: createGenerator(GenerateCompanyName), + // uniqueCompanyName: createGenerator(GenerateUniqueCompanyName), + + /** + * generates 'lorem ipsum' text sentences. + * + * @param sentencesCount - number of sentences you want to generate as one generated value(string). + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * posts: { + * columns: { + * content: funcs.loremIpsum({sentencesCount: 2}) + * }, + * }, + * })); + * ``` + */ + loremIpsum: createGenerator(GenerateLoremIpsum), + + /** + * generates 2D points within specified ranges for x and y coordinates. + * + * @param isUnique - property that controls if generated values gonna be unique or not. + * @param minXValue - lower bound of range for x coordinate. + * @param maxXValue - upper bound of range for x coordinate. + * @param minYValue - lower bound of range for y coordinate. + * @param maxYValue - upper bound of range for y coordinate. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * triangles: { + * columns: { + * pointCoords: funcs.point({ + * isUnique: true, + * minXValue: -5, maxXValue:20, + * minYValue: 0, maxYValue: 30 + * }) + * }, + * }, + * })); + * ``` + */ + point: createGenerator(GeneratePoint), + // uniquePoint: createGenerator(GenerateUniquePoint), + + /** + * generates 2D lines within specified ranges for a, b and c parameters of line. + * + * ``` + * line equation: a*x + b*y + c = 0 + * ``` + * + * @param isUnique - property that controls if generated values gonna be unique or not. + * @param minAValue - lower bound of range for a parameter. + * @param maxAValue - upper bound of range for x parameter. + * @param minBValue - lower bound of range for y parameter. + * @param maxBValue - upper bound of range for y parameter. + * @param minCValue - lower bound of range for y parameter. + * @param maxCValue - upper bound of range for y parameter. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * lines: { + * columns: { + * lineParams: funcs.point({ + * isUnique: true, + * minAValue: -5, maxAValue:20, + * minBValue: 0, maxBValue: 30, + * minCValue: 0, maxCValue: 10 + * }) + * }, + * }, + * })); + * ``` + */ + line: createGenerator(GenerateLine), + // uniqueLine: createGenerator(GenerateUniqueLine), + + /** + * gives you the opportunity to call different generators with different probabilities to generate values for one column. + * @param params - array of generators with probabilities you would like to call them to generate values. + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * posts: { + * columns: { + * content: funcs.weightedRandom([ + * { + * weight: 0.6, + * value: funcs.loremIpsum({ sentencesCount: 3 }), + * }, + * { + * weight: 0.4, + * value: funcs.default({ defaultValue: "TODO" }), + * }, + * ]), + * }, + * }, + * })); + * ``` + */ + weightedRandom: createGenerator(WeightedRandomGenerator), +}; diff --git a/drizzle-seed/src/services/GeneratorVersions/GeneratorsV1.ts b/drizzle-seed/src/services/GeneratorVersions/GeneratorsV1.ts new file mode 100644 index 000000000..e4fed2926 --- /dev/null +++ b/drizzle-seed/src/services/GeneratorVersions/GeneratorsV1.ts @@ -0,0 +1,15 @@ +import { entityKind } from 'drizzle-orm'; +import { GenerateInterval, GenerateUniqueInterval } from '../Generators.ts'; + +export const version = 1; + +export class GenerateIntervalV1 extends GenerateInterval { + static override readonly [entityKind]: string = 'GenerateInterval'; + static override readonly ['version']: number = 1; + override uniqueVersionOfGen = GenerateUniqueIntervalV1; +} + +export class GenerateUniqueIntervalV1 extends GenerateUniqueInterval { + static override readonly [entityKind]: string = 'GenerateUniqueInterval'; + static override readonly ['version']: number = 1; +} diff --git a/drizzle-seed/src/services/GeneratorsWrappers.ts b/drizzle-seed/src/services/Generators.ts similarity index 81% rename from drizzle-seed/src/services/GeneratorsWrappers.ts rename to drizzle-seed/src/services/Generators.ts index 4ab06ec3f..f5fc44aab 100644 --- a/drizzle-seed/src/services/GeneratorsWrappers.ts +++ b/drizzle-seed/src/services/Generators.ts @@ -14,8 +14,11 @@ import states, { maxStringLength as maxStateLength } from '../datasets/states.ts import streetSuffix, { maxStringLength as maxStreetSuffixLength } from '../datasets/streetSuffix.ts'; import { fastCartesianProduct, fillTemplate, getWeightedIndices, isObject } from './utils.ts'; +export const version = 2; + export abstract class AbstractGenerator { static readonly [entityKind]: string = 'AbstractGenerator'; + static readonly ['version']: number = 2; public isUnique = false; public notNull = false; @@ -90,19 +93,6 @@ export abstract class AbstractGenerator { } } -function createGenerator, T>( - generatorConstructor: new(params: T) => GeneratorType, -) { - return ( - ...args: GeneratorType extends GenerateValuesFromArray | GenerateDefault | WeightedRandomGenerator ? [T] - : ([] | [T]) - ): GeneratorType => { - let params = args[0]; - if (params === undefined) params = {} as T; - return new generatorConstructor(params); - }; -} - // Generators Classes ----------------------------------------------------------------------------------------------------------------------- export class GenerateArray extends AbstractGenerator<{ baseColumnGen: AbstractGenerator; size?: number }> { static override readonly [entityKind]: string = 'GenerateArray'; @@ -1253,7 +1243,7 @@ export class GenerateInterval extends AbstractGenerator<{ }, month: { from: 0, - to: 11, + to: 12, }, day: { from: 1, @@ -1261,15 +1251,15 @@ export class GenerateInterval extends AbstractGenerator<{ }, hour: { from: 0, - to: 23, + to: 24, }, minute: { from: 0, - to: 59, + to: 60, }, second: { from: 0, - to: 59, + to: 60, }, }; @@ -1288,8 +1278,6 @@ export class GenerateInterval extends AbstractGenerator<{ fieldsToGenerate = allFields.slice(0, endIdx + 1); } - this.config[fieldsToGenerate[Math.floor(fieldsToGenerate.length / 2)]!]!.from = 1; - const rng = prand.xoroshiro128plus(seed); this.state = { rng, fieldsToGenerate }; } @@ -1304,7 +1292,7 @@ export class GenerateInterval extends AbstractGenerator<{ for (const field of this.state.fieldsToGenerate) { const from = this.config[field]!.from, to = this.config[field]!.to; [numb, this.state.rng] = prand.uniformIntDistribution(from, to, this.state.rng); - interval += `${numb === 0 ? '' : `${numb} ${field} `}`; + interval += `${numb} ${field} `; } return interval; @@ -1376,8 +1364,6 @@ export class GenerateUniqueInterval extends AbstractGenerator<{ fieldsToGenerate = allFields.slice(0, endIdx + 1); } - this.config[fieldsToGenerate[Math.floor(fieldsToGenerate.length / 2)]!]!.from = 1; - let maxUniqueIntervalsNumber = 1; for (const field of fieldsToGenerate) { const from = this.config[field]!.from, to = this.config[field]!.to; @@ -1398,12 +1384,21 @@ export class GenerateUniqueInterval extends AbstractGenerator<{ throw new Error('state is not defined.'); } - let interval = '', numb: number; + let interval, numb: number; - for (const field of this.state.fieldsToGenerate) { - const from = this.config[field]!.from, to = this.config[field]!.to; - [numb, this.state.rng] = prand.uniformIntDistribution(from, to, this.state.rng); - interval += `${numb === 0 ? '' : `${numb} ${field} `}`; + for (;;) { + interval = ''; + + for (const field of this.state.fieldsToGenerate) { + const from = this.config[field]!.from, to = this.config[field]!.to; + [numb, this.state.rng] = prand.uniformIntDistribution(from, to, this.state.rng); + interval += `${numb} ${field} `; + } + + if (!this.state.intervalSet.has(interval)) { + this.state.intervalSet.add(interval); + break; + } } return interval; @@ -3137,693 +3132,3 @@ export class GenerateUniqueLine extends AbstractGenerator<{ } } } - -export const generatorsFuncs = { - /** - * generates same given value each time the generator is called. - * @param defaultValue - value you want to generate - * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * posts: { - * columns: { - * content: funcs.default({ defaultValue: "post content" }), - * }, - * }, - * })); - * ``` - */ - default: createGenerator(GenerateDefault), - - /** - * generates values from given array - * @param values - array of values you want to generate. can be array of weighted values. - * @param isUnique - property that controls if generated values gonna be unique or not. - * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * posts: { - * columns: { - * title: funcs.valuesFromArray({ - * values: ["Title1", "Title2", "Title3", "Title4", "Title5"], - * isUnique: true - * }), - * }, - * }, - * })); - * - * ``` - * weighted values example - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * posts: { - * columns: { - * title: funcs.valuesFromArray({ - * values: [ - * { weight: 0.35, values: ["Title1", "Title2"] }, - * { weight: 0.5, values: ["Title3", "Title4"] }, - * { weight: 0.15, values: ["Title5"] }, - * ], - * isUnique: false - * }), - * }, - * }, - * })); - * - * ``` - */ - valuesFromArray: createGenerator(GenerateValuesFromArray), - - /** - * generates sequential integers starting with 1. - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * posts: { - * columns: { - * id: funcs.intPrimaryKey(), - * }, - * }, - * })); - * - * ``` - */ - intPrimaryKey: createGenerator(GenerateIntPrimaryKey), - - /** - * generates numbers with floating point in given range. - * @param minValue - lower border of range. - * @param maxValue - upper border of range. - * @param precision - precision of generated number: - * precision equals 10 means that values will be accurate to one tenth (1.2, 34.6); - * precision equals 100 means that values will be accurate to one hundredth (1.23, 34.67). - * @param isUnique - property that controls if generated values gonna be unique or not. - * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * products: { - * columns: { - * unitPrice: funcs.number({ minValue: 10, maxValue: 120, precision: 100, isUnique: false }), - * }, - * }, - * })); - * - * ``` - */ - number: createGenerator(GenerateNumber), - // uniqueNumber: createGenerator(GenerateUniqueNumber), - - /** - * generates integers within given range. - * @param minValue - lower border of range. - * @param maxValue - upper border of range. - * @param isUnique - property that controls if generated values gonna be unique or not. - * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * products: { - * columns: { - * unitsInStock: funcs.number({ minValue: 0, maxValue: 100, isUnique: false }), - * }, - * }, - * })); - * - * ``` - */ - int: createGenerator(GenerateInt), - // uniqueInt: createGenerator(GenerateUniqueInt), - - /** - * generates boolean values(true or false) - * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * users: { - * columns: { - * isAvailable: funcs.boolean() - * }, - * }, - * })); - * - * ``` - */ - boolean: createGenerator(GenerateBoolean), - - /** - * generates date within given range. - * @param minDate - lower border of range. - * @param maxDate - upper border of range. - * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * users: { - * columns: { - * birthDate: funcs.date({ minDate: "1990-01-01", maxDate: "2010-12-31" }) - * }, - * }, - * })); - * - * ``` - */ - date: createGenerator(GenerateDate), - - /** - * generates time in 24 hours style. - * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * users: { - * columns: { - * birthTime: funcs.time() - * }, - * }, - * })); - * - * ``` - */ - time: createGenerator(GenerateTime), - - /** - * generates timestamps. - * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * orders: { - * columns: { - * shippedDate: funcs.timestamp() - * }, - * }, - * })); - * - * ``` - */ - timestamp: createGenerator(GenerateTimestamp), - - /** - * generates datetime objects. - * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * orders: { - * columns: { - * shippedDate: funcs.datetime() - * }, - * }, - * })); - * - * ``` - */ - datetime: createGenerator(GenerateDatetime), - - /** - * generates years. - * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * users: { - * columns: { - * birthYear: funcs.year() - * }, - * }, - * })); - * - * ``` - */ - year: createGenerator(GenerateYear), - - /** - * generates json objects with fixed structure. - * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) - * - * json structure can equal this: - * ``` - * { - * email, - * name, - * isGraduated, - * hasJob, - * salary, - * startedWorking, - * visitedCountries, - * } - * ``` - * or this - * ``` - * { - * email, - * name, - * isGraduated, - * hasJob, - * visitedCountries, - * } - * ``` - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * users: { - * columns: { - * metadata: funcs.json() - * }, - * }, - * })); - * ``` - */ - json: createGenerator(GenerateJson), - // jsonb: createGenerator(GenerateJsonb), - - /** - * generates time intervals. - * - * interval example: "1 years 12 days 5 minutes" - * - * @param isUnique - property that controls if generated values gonna be unique or not. - * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) - * @param fields - range of values you want to see in your intervals. - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * users: { - * columns: { - * timeSpentOnWebsite: funcs.interval() - * }, - * }, - * })); - * ``` - */ - interval: createGenerator(GenerateInterval), - // uniqueInterval: createGenerator(GenerateUniqueInterval), - - /** - * generates random strings. - * @param isUnique - property that controls if generated values gonna be unique or not. - * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * users: { - * columns: { - * hashedPassword: funcs.string({isUnique: false}) - * }, - * }, - * })); - * ``` - */ - string: createGenerator(GenerateString), - // uniqueString: createGenerator(GenerateUniqueString), - - /** - * generates v4 UUID strings if arraySize is not specified, or v4 UUID 1D arrays if it is. - * - * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * users: { - * columns: { - * uuid: funcs.uuid({ - * arraySize: 4 - * }) - * }, - * }, - * })); - * ``` - */ - uuid: createGenerator(GenerateUUID), - - /** - * generates person's first names. - * @param isUnique - property that controls if generated values gonna be unique or not. - * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * users: { - * columns: { - * firstName: funcs.firstName({isUnique: true}) - * }, - * }, - * })); - * ``` - */ - firstName: createGenerator(GenerateFirstName), - // uniqueFirstName: createGenerator(GenerateUniqueName), - - /** - * generates person's last names. - * @param isUnique - property that controls if generated values gonna be unique or not. - * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * users: { - * columns: { - * lastName: funcs.lastName({isUnique: false}) - * }, - * }, - * })); - * ``` - */ - lastName: createGenerator(GenerateLastName), - // uniqueLastName: createGenerator(GenerateUniqueSurname), - - /** - * generates person's full names. - * @param isUnique - property that controls if generated values gonna be unique or not. - * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * users: { - * columns: { - * fullName: funcs.fullName({isUnique: true}) - * }, - * }, - * })); - * ``` - */ - fullName: createGenerator(GenerateFullName), - // uniqueFullName: createGenerator(GenerateUniqueFullName), - - /** - * generates unique emails. - * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * users: { - * columns: { - * email: funcs.email() - * }, - * }, - * })); - * ``` - */ - email: createGenerator(GenerateEmail), - - /** - * generates unique phone numbers. - * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) - * - * @param template - phone number template, where all '#' symbols will be substituted with generated digits. - * @param prefixes - array of any string you want to be your phone number prefixes.(not compatible with template property) - * @param generatedDigitsNumbers - number of digits that will be added at the end of prefixes.(not compatible with template property) - * @example - * ```ts - * //generate phone number using template property - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * users: { - * columns: { - * phoneNumber: funcs.phoneNumber({template: "+(380) ###-####"}) - * }, - * }, - * })); - * - * //generate phone number using prefixes and generatedDigitsNumbers properties - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * users: { - * columns: { - * phoneNumber: funcs.phoneNumber({prefixes: [ "+380 99", "+380 67" ], generatedDigitsNumbers: 7}) - * }, - * }, - * })); - * - * //generate phone number using prefixes and generatedDigitsNumbers properties but with different generatedDigitsNumbers for prefixes - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * users: { - * columns: { - * phoneNumber: funcs.phoneNumber({prefixes: [ "+380 99", "+380 67", "+1" ], generatedDigitsNumbers: [7, 7, 10]}) - * }, - * }, - * })); - * - * ``` - */ - phoneNumber: createGenerator(GeneratePhoneNumber), - - /** - * generates country's names. - * @param isUnique - property that controls if generated values gonna be unique or not. - * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * users: { - * columns: { - * country: funcs.country({isUnique: false}) - * }, - * }, - * })); - * ``` - */ - country: createGenerator(GenerateCountry), - // uniqueCountry: createGenerator(GenerateUniqueCountry), - - /** - * generates city's names. - * @param isUnique - property that controls if generated values gonna be unique or not. - * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * users: { - * columns: { - * city: funcs.city({isUnique: false}) - * }, - * }, - * })); - * ``` - */ - city: createGenerator(GenerateCity), - // uniqueCity: createGenerator(GenerateUniqueCityName), - - /** - * generates street address. - * @param isUnique - property that controls if generated values gonna be unique or not. - * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * users: { - * columns: { - * streetAddress: funcs.streetAddress({isUnique: true}) - * }, - * }, - * })); - * ``` - */ - streetAddress: createGenerator(GenerateStreetAdddress), - // uniqueStreetAddress: createGenerator(GenerateUniqueStreetAdddress), - - /** - * generates job titles. - * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * users: { - * columns: { - * jobTitle: funcs.jobTitle() - * }, - * }, - * })); - * ``` - */ - jobTitle: createGenerator(GenerateJobTitle), - - /** - * generates postal codes. - * - * @param isUnique - property that controls if generated values gonna be unique or not. - * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * users: { - * columns: { - * postcode: funcs.postcode({isUnique: true}) - * }, - * }, - * })); - * ``` - */ - postcode: createGenerator(GeneratePostcode), - // uniquePostcoe: createGenerator(GenerateUniquePostcode), - - /** - * generates states of America. - * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * users: { - * columns: { - * state: funcs.state() - * }, - * }, - * })); - * ``` - */ - state: createGenerator(GenerateState), - - /** - * generates company's names. - * - * @param isUnique - property that controls if generated values gonna be unique or not. - * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * users: { - * columns: { - * company: funcs.companyName({isUnique: true}) - * }, - * }, - * })); - * ``` - */ - companyName: createGenerator(GenerateCompanyName), - // uniqueCompanyName: createGenerator(GenerateUniqueCompanyName), - - /** - * generates 'lorem ipsum' text sentences. - * - * @param sentencesCount - number of sentences you want to generate as one generated value(string). - * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * posts: { - * columns: { - * content: funcs.loremIpsum({sentencesCount: 2}) - * }, - * }, - * })); - * ``` - */ - loremIpsum: createGenerator(GenerateLoremIpsum), - - /** - * generates 2D points within specified ranges for x and y coordinates. - * - * @param isUnique - property that controls if generated values gonna be unique or not. - * @param minXValue - lower bound of range for x coordinate. - * @param maxXValue - upper bound of range for x coordinate. - * @param minYValue - lower bound of range for y coordinate. - * @param maxYValue - upper bound of range for y coordinate. - * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * triangles: { - * columns: { - * pointCoords: funcs.point({ - * isUnique: true, - * minXValue: -5, maxXValue:20, - * minYValue: 0, maxYValue: 30 - * }) - * }, - * }, - * })); - * ``` - */ - point: createGenerator(GeneratePoint), - // uniquePoint: createGenerator(GenerateUniquePoint), - - /** - * generates 2D lines within specified ranges for a, b and c parameters of line. - * - * ``` - * line equation: a*x + b*y + c = 0 - * ``` - * - * @param isUnique - property that controls if generated values gonna be unique or not. - * @param minAValue - lower bound of range for a parameter. - * @param maxAValue - upper bound of range for x parameter. - * @param minBValue - lower bound of range for y parameter. - * @param maxBValue - upper bound of range for y parameter. - * @param minCValue - lower bound of range for y parameter. - * @param maxCValue - upper bound of range for y parameter. - * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * lines: { - * columns: { - * lineParams: funcs.point({ - * isUnique: true, - * minAValue: -5, maxAValue:20, - * minBValue: 0, maxBValue: 30, - * minCValue: 0, maxCValue: 10 - * }) - * }, - * }, - * })); - * ``` - */ - line: createGenerator(GenerateLine), - // uniqueLine: createGenerator(GenerateUniqueLine), - - /** - * gives you the opportunity to call different generators with different probabilities to generate values for one column. - * @param params - array of generators with probabilities you would like to call them to generate values. - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * posts: { - * columns: { - * content: funcs.weightedRandom([ - * { - * weight: 0.6, - * value: funcs.loremIpsum({ sentencesCount: 3 }), - * }, - * { - * weight: 0.4, - * value: funcs.default({ defaultValue: "TODO" }), - * }, - * ]), - * }, - * }, - * })); - * ``` - */ - weightedRandom: createGenerator(WeightedRandomGenerator), -}; diff --git a/drizzle-seed/src/services/SeedService.ts b/drizzle-seed/src/services/SeedService.ts index 62283c817..6982ac980 100644 --- a/drizzle-seed/src/services/SeedService.ts +++ b/drizzle-seed/src/services/SeedService.ts @@ -11,35 +11,11 @@ import type { RefinementsType, TableGeneratorsType, } from '../types/seedService.ts'; -import type { Column, Prettify, Relation, RelationWithReferences, Table } from '../types/tables.ts'; -import type { AbstractGenerator } from './GeneratorsWrappers.ts'; -import { - GenerateArray, - GenerateBoolean, - GenerateDate, - GenerateDatetime, - GenerateDefault, - GenerateEmail, - GenerateEnum, - GenerateFirstName, - GenerateInt, - GenerateInterval, - GenerateIntPrimaryKey, - GenerateJson, - GenerateLine, - GenerateNumber, - GeneratePoint, - GenerateSelfRelationsValuesFromArray, - GenerateString, - GenerateTime, - GenerateTimestamp, - GenerateUniqueString, - GenerateUUID, - GenerateValuesFromArray, - GenerateWeightedCount, - GenerateYear, - HollowGenerator, -} from './GeneratorsWrappers.ts'; +import type { Column, Prettify, Relation, Table } from '../types/tables.ts'; +import type { AbstractGenerator } from './Generators.ts'; + +import * as Generators from './Generators.ts'; +import * as GeneratorsV1 from './GeneratorVersions/GeneratorsV1.ts'; import { equalSets, generateHashFromString } from './utils.ts'; export class SeedService { @@ -57,13 +33,13 @@ export class SeedService { connectionType: 'postgresql' | 'mysql' | 'sqlite', tables: Table[], relations: (Relation & { isCyclic: boolean })[], - tableRelations: { [tableName: string]: RelationWithReferences[] }, refinements?: RefinementsType, - options?: { count?: number; seed?: number }, + options?: { count?: number; seed?: number; version?: number }, ) => { let columnPossibleGenerator: Prettify; let tablePossibleGenerators: Prettify; const customSeed = options?.seed === undefined ? 0 : options.seed; + const version = options?.version === undefined ? Generators.version : options.version; // sorting table in order which they will be filled up (tables with foreign keys case) // relations = relations.filter(rel => rel.type === "one"); @@ -221,35 +197,42 @@ export class SeedService { columnPossibleGenerator.wasRefined = true; } else if (Object.hasOwn(foreignKeyColumns, col.name)) { // TODO: I might need to assign repeatedValuesCount to column there instead of doing so in generateTablesValues - const cyclicRelation = tableRelations[table.name]!.find((rel) => - rel.isCyclic === true + const cyclicRelation = relations.find((rel) => + rel.table === table.name + && rel.isCyclic === true && rel.columns.includes(col.name) ); + // const cyclicRelation = tableRelations[table.name]!.find((rel) => + // rel.isCyclic === true + // && rel.columns.includes(col.name) + // ); + if (cyclicRelation !== undefined) { columnPossibleGenerator.isCyclic = true; } const predicate = cyclicRelation !== undefined && col.notNull === false; if (predicate === true) { - columnPossibleGenerator.generator = new GenerateDefault({ defaultValue: null }); + columnPossibleGenerator.generator = new Generators.GenerateDefault({ defaultValue: null }); columnPossibleGenerator.wasDefinedBefore = true; } - columnPossibleGenerator.generator = new HollowGenerator({}); + columnPossibleGenerator.generator = new Generators.HollowGenerator({}); } // TODO: rewrite pickGeneratorFor... using new col properties: isUnique and notNull else if (connectionType === 'postgresql') { - columnPossibleGenerator.generator = this.pickGeneratorForPostgresColumn( + columnPossibleGenerator.generator = this.selectGeneratorForPostgresColumn( table, col, + version, ); } else if (connectionType === 'mysql') { - columnPossibleGenerator.generator = this.pickGeneratorForMysqlColumn( + columnPossibleGenerator.generator = this.selectGeneratorForMysqlColumn( table, col, ); } else if (connectionType === 'sqlite') { - columnPossibleGenerator.generator = this.pickGeneratorForSqlite( + columnPossibleGenerator.generator = this.selectGeneratorForSqlite( table, col, ); @@ -420,7 +403,7 @@ export class SeedService { count: number, seed: number, ) => { - const gen = new GenerateWeightedCount({}); + const gen = new Generators.GenerateWeightedCount({}); gen.init({ count: weightedCount, seed }); let weightedWithCount = 0; for (let i = 0; i < count; i++) { @@ -430,20 +413,59 @@ export class SeedService { return weightedWithCount; }; - // TODO: revise serial part generators + selectGeneratorOfVersion = (version: number, generatorEntityKind: string) => { + const GeneratorVersions = [Generators, GeneratorsV1]; + + type GeneratorConstructorT = new(params: any) => AbstractGenerator; + let generatorConstructor: GeneratorConstructorT | undefined; + for (const gens of GeneratorVersions) { + const { version: gensVersion, ...filteredGens } = gens; + if (version > gensVersion) break; + + for (const gen of Object.values(filteredGens)) { + const abstractGen = gen as typeof AbstractGenerator; + if (abstractGen[entityKind] === generatorEntityKind) { + generatorConstructor = abstractGen as unknown as GeneratorConstructorT; + + if (abstractGen.version === version) { + return { generatorConstructor }; + } + } + } + } + + if (generatorConstructor === undefined) { + throw new Error(`Can't select version for ${generatorEntityKind} generator`); + } + + return { generatorConstructor }; + }; - pickGeneratorForPostgresColumn = ( + // TODO: revise serial part generators + selectGeneratorForPostgresColumn = ( table: Table, col: Column, + version: number, ) => { const pickGenerator = (table: Table, col: Column) => { // ARRAY if (col.columnType.match(/\[\w*]/g) !== null && col.baseColumn !== undefined) { - const baseColumnGen = this.pickGeneratorForPostgresColumn(table, col.baseColumn!) as AbstractGenerator; + const baseColumnGen = this.selectGeneratorForPostgresColumn( + table, + col.baseColumn!, + version, + ) as AbstractGenerator; if (baseColumnGen === undefined) { throw new Error(`column with type ${col.baseColumn!.columnType} is not supported for now.`); } - const generator = new GenerateArray({ baseColumnGen, size: col.size }); + + const { generatorConstructor } = this.selectGeneratorOfVersion( + version, + Generators.GenerateArray[entityKind], + ); + + const generator = new generatorConstructor!({ baseColumnGen, size: col.size }); + // const generator = new Generators.GenerateArray({ baseColumnGen, size: col.size }); return generator; } @@ -457,15 +479,15 @@ export class SeedService { }; baseColumn.columnType = baseColumnType; - const baseColumnGen = this.pickGeneratorForPostgresColumn(table, baseColumn) as AbstractGenerator; + const baseColumnGen = this.selectGeneratorForPostgresColumn(table, baseColumn, version) as AbstractGenerator; if (baseColumnGen === undefined) { throw new Error(`column with type ${col.baseColumn!.columnType} is not supported for now.`); } - let generator = new GenerateArray({ baseColumnGen }); + let generator = new Generators.GenerateArray({ baseColumnGen }); for (let i = 0; i < col.typeParams.dimensions! - 1; i++) { - generator = new GenerateArray({ baseColumnGen: generator }); + generator = new Generators.GenerateArray({ baseColumnGen: generator }); } return generator; @@ -479,7 +501,7 @@ export class SeedService { || col.columnType.includes('bigint')) && table.primaryKeys.includes(col.name) ) { - const generator = new GenerateIntPrimaryKey({}); + const generator = new Generators.GenerateIntPrimaryKey({}); return generator; } @@ -527,7 +549,7 @@ export class SeedService { && !col.columnType.includes('interval') && !col.columnType.includes('point') ) { - const generator = new GenerateInt({ + const generator = new Generators.GenerateInt({ minValue, maxValue, }); @@ -536,9 +558,9 @@ export class SeedService { } if (col.columnType.includes('serial')) { - const generator = new GenerateIntPrimaryKey({}); + const generator = new Generators.GenerateIntPrimaryKey({}); - (generator as GenerateIntPrimaryKey).maxValue = maxValue; + (generator as Generators.GenerateIntPrimaryKey).maxValue = maxValue; return generator; } @@ -550,7 +572,7 @@ export class SeedService { || col.columnType === 'decimal' || col.columnType === 'numeric' ) { - const generator = new GenerateNumber({}); + const generator = new Generators.GenerateNumber({}); return generator; } @@ -562,7 +584,7 @@ export class SeedService { || col.columnType.startsWith('char')) && table.primaryKeys.includes(col.name) ) { - const generator = new GenerateUniqueString({}); + const generator = new Generators.GenerateUniqueString({}); return generator; } @@ -573,7 +595,7 @@ export class SeedService { || col.columnType.startsWith('char')) && col.name.toLowerCase().includes('name') ) { - const generator = new GenerateFirstName({}); + const generator = new Generators.GenerateFirstName({}); return generator; } @@ -584,7 +606,7 @@ export class SeedService { || col.columnType.startsWith('char')) && col.name.toLowerCase().includes('email') ) { - const generator = new GenerateEmail({}); + const generator = new Generators.GenerateEmail({}); return generator; } @@ -595,47 +617,47 @@ export class SeedService { || col.columnType.startsWith('char') ) { // console.log(col, table) - const generator = new GenerateString({}); + const generator = new Generators.GenerateString({}); return generator; } // UUID if (col.columnType === 'uuid') { - const generator = new GenerateUUID({}); + const generator = new Generators.GenerateUUID({}); return generator; } // BOOLEAN if (col.columnType === 'boolean') { - const generator = new GenerateBoolean({}); + const generator = new Generators.GenerateBoolean({}); return generator; } // DATE, TIME, TIMESTAMP if (col.columnType.includes('date')) { - const generator = new GenerateDate({}); + const generator = new Generators.GenerateDate({}); return generator; } if (col.columnType === 'time') { - const generator = new GenerateTime({}); + const generator = new Generators.GenerateTime({}); return generator; } if (col.columnType.includes('timestamp')) { - const generator = new GenerateTimestamp({}); + const generator = new Generators.GenerateTimestamp({}); return generator; } // JSON, JSONB if (col.columnType === 'json' || col.columnType === 'jsonb') { - const generator = new GenerateJson({}); + const generator = new Generators.GenerateJson({}); return generator; } @@ -647,7 +669,7 @@ export class SeedService { // ENUM if (col.enumValues !== undefined) { - const generator = new GenerateEnum({ + const generator = new Generators.GenerateEnum({ enumValues: col.enumValues, }); @@ -657,32 +679,32 @@ export class SeedService { // INTERVAL if (col.columnType.startsWith('interval')) { if (col.columnType === 'interval') { - const generator = new GenerateInterval({}); + const generator = new Generators.GenerateInterval({}); return generator; } - const fields = col.columnType.replace('interval ', '') as GenerateInterval['params']['fields']; - const generator = new GenerateInterval({ fields }); + const fields = col.columnType.replace('interval ', '') as Generators.GenerateInterval['params']['fields']; + const generator = new Generators.GenerateInterval({ fields }); return generator; } // POINT, LINE if (col.columnType.includes('point')) { - const generator = new GeneratePoint({}); + const generator = new Generators.GeneratePoint({}); return generator; } if (col.columnType.includes('line')) { - const generator = new GenerateLine({}); + const generator = new Generators.GenerateLine({}); return generator; } if (col.hasDefault && col.default !== undefined) { - const generator = new GenerateDefault({ + const generator = new Generators.GenerateDefault({ defaultValue: col.default, }); return generator; @@ -701,7 +723,7 @@ export class SeedService { return generator; }; - pickGeneratorForMysqlColumn = ( + selectGeneratorForMysqlColumn = ( table: Table, col: Column, ) => { @@ -711,7 +733,7 @@ export class SeedService { (col.columnType.includes('serial') || col.columnType.includes('int')) && table.primaryKeys.includes(col.name) ) { - const generator = new GenerateIntPrimaryKey({}); + const generator = new Generators.GenerateIntPrimaryKey({}); return generator; } @@ -746,7 +768,7 @@ export class SeedService { } if (col.columnType.includes('int')) { - const generator = new GenerateInt({ + const generator = new Generators.GenerateInt({ minValue, maxValue, }); @@ -754,7 +776,7 @@ export class SeedService { } if (col.columnType.includes('serial')) { - const generator = new GenerateIntPrimaryKey({}); + const generator = new Generators.GenerateIntPrimaryKey({}); generator.maxValue = maxValue; return generator; } @@ -766,7 +788,7 @@ export class SeedService { || col.columnType === 'decimal' || col.columnType === 'float' ) { - const generator = new GenerateNumber({}); + const generator = new Generators.GenerateNumber({}); return generator; } @@ -780,7 +802,7 @@ export class SeedService { || col.columnType.includes('varbinary')) && table.primaryKeys.includes(col.name) ) { - const generator = new GenerateUniqueString({}); + const generator = new Generators.GenerateUniqueString({}); return generator; } @@ -793,7 +815,7 @@ export class SeedService { || col.columnType.includes('varbinary')) && col.name.toLowerCase().includes('name') ) { - const generator = new GenerateFirstName({}); + const generator = new Generators.GenerateFirstName({}); return generator; } @@ -806,7 +828,7 @@ export class SeedService { || col.columnType.includes('varbinary')) && col.name.toLowerCase().includes('email') ) { - const generator = new GenerateEmail({}); + const generator = new Generators.GenerateEmail({}); return generator; } @@ -819,58 +841,58 @@ export class SeedService { || col.columnType.includes('varbinary') ) { // console.log(col, table); - const generator = new GenerateString({}); + const generator = new Generators.GenerateString({}); return generator; } // BOOLEAN if (col.columnType === 'boolean') { - const generator = new GenerateBoolean({}); + const generator = new Generators.GenerateBoolean({}); return generator; } // DATE, TIME, TIMESTAMP, DATETIME, YEAR if (col.columnType.includes('datetime')) { - const generator = new GenerateDatetime({}); + const generator = new Generators.GenerateDatetime({}); return generator; } if (col.columnType.includes('date')) { - const generator = new GenerateDate({}); + const generator = new Generators.GenerateDate({}); return generator; } if (col.columnType === 'time') { - const generator = new GenerateTime({}); + const generator = new Generators.GenerateTime({}); return generator; } if (col.columnType.includes('timestamp')) { - const generator = new GenerateTimestamp({}); + const generator = new Generators.GenerateTimestamp({}); return generator; } if (col.columnType === 'year') { - const generator = new GenerateYear({}); + const generator = new Generators.GenerateYear({}); return generator; } // JSON if (col.columnType === 'json') { - const generator = new GenerateJson({}); + const generator = new Generators.GenerateJson({}); return generator; } // ENUM if (col.enumValues !== undefined) { - const generator = new GenerateEnum({ + const generator = new Generators.GenerateEnum({ enumValues: col.enumValues, }); return generator; } if (col.hasDefault && col.default !== undefined) { - const generator = new GenerateDefault({ + const generator = new Generators.GenerateDefault({ defaultValue: col.default, }); return generator; @@ -879,7 +901,7 @@ export class SeedService { return; }; - pickGeneratorForSqlite = ( + selectGeneratorForSqlite = ( table: Table, col: Column, ) => { @@ -888,17 +910,17 @@ export class SeedService { (col.columnType === 'integer' || col.columnType === 'numeric') && table.primaryKeys.includes(col.name) ) { - const generator = new GenerateIntPrimaryKey({}); + const generator = new Generators.GenerateIntPrimaryKey({}); return generator; } if (col.columnType === 'integer' && col.dataType === 'boolean') { - const generator = new GenerateBoolean({}); + const generator = new Generators.GenerateBoolean({}); return generator; } if ((col.columnType === 'integer' && col.dataType === 'date')) { - const generator = new GenerateTimestamp({}); + const generator = new Generators.GenerateTimestamp({}); return generator; } @@ -907,13 +929,13 @@ export class SeedService { || col.columnType === 'numeric' || (col.dataType === 'bigint' && col.columnType === 'blob') ) { - const generator = new GenerateInt({}); + const generator = new Generators.GenerateInt({}); return generator; } // number section ------------------------------------------------------------------------------------ if (col.columnType === 'real' || col.columnType === 'numeric') { - const generator = new GenerateNumber({}); + const generator = new Generators.GenerateNumber({}); return generator; } @@ -924,7 +946,7 @@ export class SeedService { || col.columnType === 'blob') && table.primaryKeys.includes(col.name) ) { - const generator = new GenerateUniqueString({}); + const generator = new Generators.GenerateUniqueString({}); return generator; } @@ -934,7 +956,7 @@ export class SeedService { || col.columnType === 'blob') && col.name.toLowerCase().includes('name') ) { - const generator = new GenerateFirstName({}); + const generator = new Generators.GenerateFirstName({}); return generator; } @@ -944,7 +966,7 @@ export class SeedService { || col.columnType === 'blob') && col.name.toLowerCase().includes('email') ) { - const generator = new GenerateEmail({}); + const generator = new Generators.GenerateEmail({}); return generator; } @@ -954,7 +976,7 @@ export class SeedService { || col.columnType === 'blob' || col.columnType === 'blobbuffer' ) { - const generator = new GenerateString({}); + const generator = new Generators.GenerateString({}); return generator; } @@ -962,12 +984,12 @@ export class SeedService { (col.columnType === 'text' && col.dataType === 'json') || (col.columnType === 'blob' && col.dataType === 'json') ) { - const generator = new GenerateJson({}); + const generator = new Generators.GenerateJson({}); return generator; } if (col.hasDefault && col.default !== undefined) { - const generator = new GenerateDefault({ + const generator = new Generators.GenerateDefault({ defaultValue: col.default, }); return generator; @@ -1134,7 +1156,7 @@ export class SeedService { }))!.map((rows) => rows[refColName]) as (string | number | boolean)[]; hasSelfRelation = true; - genObj = new GenerateSelfRelationsValuesFromArray({ + genObj = new Generators.GenerateSelfRelationsValuesFromArray({ values: refColumnValues, }); } else if ( @@ -1154,10 +1176,10 @@ export class SeedService { weightedCountSeed = table.withFromTable[rel.refTable]!.weightedCountSeed; } - genObj = new GenerateValuesFromArray({ values: refColumnValues }); - (genObj as GenerateValuesFromArray).notNull = tableGenerators[rel.columns[colIdx]!]!.notNull; - (genObj as GenerateValuesFromArray).weightedCountSeed = weightedCountSeed; - (genObj as GenerateValuesFromArray).maxRepeatedValuesCount = repeatedValuesCount; + genObj = new Generators.GenerateValuesFromArray({ values: refColumnValues }); + (genObj as Generators.GenerateValuesFromArray).notNull = tableGenerators[rel.columns[colIdx]!]!.notNull; + (genObj as Generators.GenerateValuesFromArray).weightedCountSeed = weightedCountSeed; + (genObj as Generators.GenerateValuesFromArray).maxRepeatedValuesCount = repeatedValuesCount; } // console.log(rel.columns[colIdx], tableGenerators) diff --git a/drizzle-seed/src/types/seedService.ts b/drizzle-seed/src/types/seedService.ts index 0b5237468..1ae06f44c 100644 --- a/drizzle-seed/src/types/seedService.ts +++ b/drizzle-seed/src/types/seedService.ts @@ -1,4 +1,4 @@ -import type { AbstractGenerator } from '../services/GeneratorsWrappers.ts'; +import type { AbstractGenerator } from '../services/Generators.ts'; import type { Prettify } from './tables.ts'; export type TableGeneratorsType = { diff --git a/drizzle-seed/tests/benchmarks/generatorsBenchmark.ts b/drizzle-seed/tests/benchmarks/generatorsBenchmark.ts index 9d79ebaa8..3473e555a 100644 --- a/drizzle-seed/tests/benchmarks/generatorsBenchmark.ts +++ b/drizzle-seed/tests/benchmarks/generatorsBenchmark.ts @@ -40,7 +40,7 @@ import { GenerateValuesFromArray, GenerateYear, WeightedRandomGenerator, -} from '../../src/services/GeneratorsWrappers.ts'; +} from '../../src/services/Generators.ts'; const benchmark = ({ generatorName, generator, count = 100000, seed = 1 }: { generatorName: string; From 06a936837e1f3d61cf0f89002579209faf5570e0 Mon Sep 17 00:00:00 2001 From: OleksiiKH0240 Date: Wed, 18 Dec 2024 16:20:32 +0200 Subject: [PATCH 13/32] added generator versioning with version v1 for string and interval generators --- .../GeneratorVersions/GeneratorsV1.ts | 152 ++++- drizzle-seed/src/services/Generators.ts | 2 +- drizzle-seed/src/services/SeedService.ts | 525 ++++++++++-------- 3 files changed, 428 insertions(+), 251 deletions(-) diff --git a/drizzle-seed/src/services/GeneratorVersions/GeneratorsV1.ts b/drizzle-seed/src/services/GeneratorVersions/GeneratorsV1.ts index e4fed2926..81655a941 100644 --- a/drizzle-seed/src/services/GeneratorVersions/GeneratorsV1.ts +++ b/drizzle-seed/src/services/GeneratorVersions/GeneratorsV1.ts @@ -1,5 +1,6 @@ import { entityKind } from 'drizzle-orm'; -import { GenerateInterval, GenerateUniqueInterval } from '../Generators.ts'; +import prand from 'pure-rand'; +import { AbstractGenerator, GenerateInterval } from '../Generators.ts'; export const version = 1; @@ -9,7 +10,152 @@ export class GenerateIntervalV1 extends GenerateInterval { override uniqueVersionOfGen = GenerateUniqueIntervalV1; } -export class GenerateUniqueIntervalV1 extends GenerateUniqueInterval { +export class GenerateUniqueIntervalV1 extends AbstractGenerator<{ + isUnique?: boolean; +}> { static override readonly [entityKind]: string = 'GenerateUniqueInterval'; - static override readonly ['version']: number = 1; + + private state: { + rng: prand.RandomGenerator; + intervalSet: Set; + } | undefined; + public override isUnique = true; + + override init({ count, seed }: { count: number; seed: number }) { + const maxUniqueIntervalsNumber = 6 * 13 * 29 * 25 * 61 * 61; + if (count > maxUniqueIntervalsNumber) { + throw new RangeError(`count exceeds max number of unique intervals(${maxUniqueIntervalsNumber})`); + } + + const rng = prand.xoroshiro128plus(seed); + const intervalSet = new Set(); + this.state = { rng, intervalSet }; + } + + generate() { + if (this.state === undefined) { + throw new Error('state is not defined.'); + } + + let yearsNumb: number, + monthsNumb: number, + daysNumb: number, + hoursNumb: number, + minutesNumb: number, + secondsNumb: number; + + let interval = ''; + + for (;;) { + [yearsNumb, this.state.rng] = prand.uniformIntDistribution(0, 5, this.state.rng); + [monthsNumb, this.state.rng] = prand.uniformIntDistribution(0, 12, this.state.rng); + [daysNumb, this.state.rng] = prand.uniformIntDistribution(1, 29, this.state.rng); + [hoursNumb, this.state.rng] = prand.uniformIntDistribution(0, 24, this.state.rng); + [minutesNumb, this.state.rng] = prand.uniformIntDistribution(0, 60, this.state.rng); + [secondsNumb, this.state.rng] = prand.uniformIntDistribution(0, 60, this.state.rng); + + interval = `${yearsNumb === 0 ? '' : `${yearsNumb} years `}` + + `${monthsNumb === 0 ? '' : `${monthsNumb} months `}` + + `${daysNumb === 0 ? '' : `${daysNumb} days `}` + + `${hoursNumb === 0 ? '' : `${hoursNumb} hours `}` + + `${minutesNumb === 0 ? '' : `${minutesNumb} minutes `}` + + `${secondsNumb === 0 ? '' : `${secondsNumb} seconds`}`; + + if (!this.state.intervalSet.has(interval)) { + this.state.intervalSet.add(interval); + break; + } + } + + return interval; + } +} + +export class GenerateStringV1 extends AbstractGenerator<{ + isUnique?: boolean; + arraySize?: number; +}> { + static override readonly [entityKind]: string = 'GenerateString'; + + private state: { rng: prand.RandomGenerator } | undefined; + override uniqueVersionOfGen = GenerateUniqueStringV1; + + override init({ count, seed }: { count: number; seed: number }) { + super.init({ count, seed }); + + const rng = prand.xoroshiro128plus(seed); + this.state = { rng }; + } + + generate() { + if (this.state === undefined) { + throw new Error('state is not defined.'); + } + + const minStringLength = 7; + const maxStringLength = 20; + const stringChars = '1234567890abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ'; + let idx: number, + strLength: number, + currStr: string; + + currStr = ''; + [strLength, this.state.rng] = prand.uniformIntDistribution( + minStringLength, + maxStringLength, + this.state.rng, + ); + for (let j = 0; j < strLength; j++) { + [idx, this.state.rng] = prand.uniformIntDistribution( + 0, + stringChars.length - 1, + this.state.rng, + ); + currStr += stringChars[idx]; + } + return currStr; + } +} + +export class GenerateUniqueStringV1 extends AbstractGenerator<{ isUnique?: boolean }> { + static override readonly [entityKind]: string = 'GenerateUniqueString'; + + private state: { rng: prand.RandomGenerator } | undefined; + public override isUnique = true; + + override init({ seed }: { seed: number }) { + const rng = prand.xoroshiro128plus(seed); + this.state = { rng }; + } + + generate({ i }: { i: number }) { + if (this.state === undefined) { + throw new Error('state is not defined.'); + } + + const minStringLength = 7; + const maxStringLength = 20; + const stringChars = '1234567890abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ'; + let idx: number, + strLength: number; + let currStr: string; + + currStr = ''; + const uniqueStr = i.toString(16); + [strLength, this.state.rng] = prand.uniformIntDistribution( + minStringLength, + maxStringLength - uniqueStr.length, + this.state.rng, + ); + for (let j = 0; j < strLength - uniqueStr.length; j++) { + [idx, this.state.rng] = prand.uniformIntDistribution( + 0, + stringChars.length - 1, + this.state.rng, + ); + currStr += stringChars[idx]; + } + + return currStr.slice(0, 4) + uniqueStr + currStr.slice(4); + } } diff --git a/drizzle-seed/src/services/Generators.ts b/drizzle-seed/src/services/Generators.ts index f5fc44aab..fa01a1e4f 100644 --- a/drizzle-seed/src/services/Generators.ts +++ b/drizzle-seed/src/services/Generators.ts @@ -1235,7 +1235,7 @@ export class GenerateInterval extends AbstractGenerator<{ rng: prand.RandomGenerator; fieldsToGenerate: string[]; } | undefined; - override uniqueVersionOfGen = GenerateUniqueInterval; + override uniqueVersionOfGen: new(params: any) => AbstractGenerator = GenerateUniqueInterval; private config: { [key: string]: { from: number; to: number } } = { year: { from: 0, diff --git a/drizzle-seed/src/services/SeedService.ts b/drizzle-seed/src/services/SeedService.ts index 6982ac980..6ceae5167 100644 --- a/drizzle-seed/src/services/SeedService.ts +++ b/drizzle-seed/src/services/SeedService.ts @@ -178,7 +178,12 @@ export class SeedService { && refinements[table.name]!.columns !== undefined && refinements[table.name]!.columns[col.name] !== undefined ) { - const genObj = refinements[table.name]!.columns[col.name]!; + let genObj = refinements[table.name]!.columns[col.name]!; + + const genObjEntityKind = genObj.getEntityKind(); + const generatorConstructor = this.selectGeneratorOfVersion(version, genObjEntityKind); + genObj = new generatorConstructor(genObj.params); + // TODO: for now only GenerateValuesFromArray support notNull property genObj.notNull = col.notNull; if (col.columnType.match(/\[\w*]/g) !== null) { @@ -203,11 +208,6 @@ export class SeedService { && rel.columns.includes(col.name) ); - // const cyclicRelation = tableRelations[table.name]!.find((rel) => - // rel.isCyclic === true - // && rel.columns.includes(col.name) - // ); - if (cyclicRelation !== undefined) { columnPossibleGenerator.isCyclic = true; } @@ -230,11 +230,13 @@ export class SeedService { columnPossibleGenerator.generator = this.selectGeneratorForMysqlColumn( table, col, + version, ); } else if (connectionType === 'sqlite') { columnPossibleGenerator.generator = this.selectGeneratorForSqlite( table, col, + version, ); } @@ -428,17 +430,17 @@ export class SeedService { generatorConstructor = abstractGen as unknown as GeneratorConstructorT; if (abstractGen.version === version) { - return { generatorConstructor }; + return generatorConstructor; } } } } if (generatorConstructor === undefined) { - throw new Error(`Can't select version for ${generatorEntityKind} generator`); + throw new Error(`Can't select ${generatorEntityKind} generator for ${version} version.`); } - return { generatorConstructor }; + return generatorConstructor; }; // TODO: revise serial part generators @@ -459,13 +461,13 @@ export class SeedService { throw new Error(`column with type ${col.baseColumn!.columnType} is not supported for now.`); } - const { generatorConstructor } = this.selectGeneratorOfVersion( - version, - Generators.GenerateArray[entityKind], - ); + // const { generatorConstructor } = this.selectGeneratorOfVersion( + // version, + // Generators.GenerateArray[entityKind], + // ); - const generator = new generatorConstructor!({ baseColumnGen, size: col.size }); - // const generator = new Generators.GenerateArray({ baseColumnGen, size: col.size }); + // const generator = new generatorConstructor!({ baseColumnGen, size: col.size }); + const generator = new Generators.GenerateArray({ baseColumnGen, size: col.size }); return generator; } @@ -616,7 +618,6 @@ export class SeedService { || col.columnType.startsWith('varchar') || col.columnType.startsWith('char') ) { - // console.log(col, table) const generator = new Generators.GenerateString({}); return generator; @@ -713,8 +714,15 @@ export class SeedService { return; }; - const generator = pickGenerator(table, col); + let generator = pickGenerator(table, col) as AbstractGenerator || undefined; if (generator !== undefined) { + const generatorConstructor = this.selectGeneratorOfVersion( + version, + generator.getEntityKind(), + ); + + generator = new generatorConstructor(generator.params); + generator.isUnique = col.isUnique; generator.dataType = col.dataType; generator.stringLength = col.typeParams.length; @@ -726,276 +734,304 @@ export class SeedService { selectGeneratorForMysqlColumn = ( table: Table, col: Column, + version: number, ) => { - // console.log(col); - // INT ------------------------------------------------------------------------------------------------------------ - if ( - (col.columnType.includes('serial') || col.columnType.includes('int')) - && table.primaryKeys.includes(col.name) - ) { - const generator = new Generators.GenerateIntPrimaryKey({}); - return generator; - } + const pickGenerator = (table: Table, col: Column) => { + // INT ------------------------------------------------------------------------------------------------------------ + if ( + (col.columnType.includes('serial') || col.columnType.includes('int')) + && table.primaryKeys.includes(col.name) + ) { + const generator = new Generators.GenerateIntPrimaryKey({}); + return generator; + } - let minValue: number | bigint | undefined; - let maxValue: number | bigint | undefined; - if (col.columnType === 'serial') { - // 2^64 % 2 - 1, 8 bytes - minValue = BigInt(0); - maxValue = BigInt('9223372036854775807'); - } else if (col.columnType.includes('int')) { - if (col.columnType === 'tinyint') { - // 2^8 / 2 - 1, 1 bytes - minValue = -128; - maxValue = 127; - } else if (col.columnType === 'smallint') { - // 2^16 / 2 - 1, 2 bytes - minValue = -32768; - maxValue = 32767; - } else if (col.columnType === 'mediumint') { - // 2^16 / 2 - 1, 2 bytes - minValue = -8388608; - maxValue = 8388607; - } else if (col.columnType === 'int') { - // 2^32 / 2 - 1, 4 bytes - minValue = -2147483648; - maxValue = 2147483647; - } else if (col.columnType === 'bigint') { - // 2^64 / 2 - 1, 8 bytes - minValue = BigInt('-9223372036854775808'); + let minValue: number | bigint | undefined; + let maxValue: number | bigint | undefined; + if (col.columnType === 'serial') { + // 2^64 % 2 - 1, 8 bytes + minValue = BigInt(0); maxValue = BigInt('9223372036854775807'); + } else if (col.columnType.includes('int')) { + if (col.columnType === 'tinyint') { + // 2^8 / 2 - 1, 1 bytes + minValue = -128; + maxValue = 127; + } else if (col.columnType === 'smallint') { + // 2^16 / 2 - 1, 2 bytes + minValue = -32768; + maxValue = 32767; + } else if (col.columnType === 'mediumint') { + // 2^16 / 2 - 1, 2 bytes + minValue = -8388608; + maxValue = 8388607; + } else if (col.columnType === 'int') { + // 2^32 / 2 - 1, 4 bytes + minValue = -2147483648; + maxValue = 2147483647; + } else if (col.columnType === 'bigint') { + // 2^64 / 2 - 1, 8 bytes + minValue = BigInt('-9223372036854775808'); + maxValue = BigInt('9223372036854775807'); + } } - } - if (col.columnType.includes('int')) { - const generator = new Generators.GenerateInt({ - minValue, - maxValue, - }); - return generator; - } + if (col.columnType.includes('int')) { + const generator = new Generators.GenerateInt({ + minValue, + maxValue, + }); + return generator; + } - if (col.columnType.includes('serial')) { - const generator = new Generators.GenerateIntPrimaryKey({}); - generator.maxValue = maxValue; - return generator; - } + if (col.columnType.includes('serial')) { + const generator = new Generators.GenerateIntPrimaryKey({}); + generator.maxValue = maxValue; + return generator; + } - // NUMBER(real, double, decimal, float) - if ( - col.columnType === 'real' - || col.columnType === 'double' - || col.columnType === 'decimal' - || col.columnType === 'float' - ) { - const generator = new Generators.GenerateNumber({}); - return generator; - } + // NUMBER(real, double, decimal, float) + if ( + col.columnType === 'real' + || col.columnType === 'double' + || col.columnType === 'decimal' + || col.columnType === 'float' + ) { + const generator = new Generators.GenerateNumber({}); + return generator; + } - // STRING - if ( - (col.columnType === 'text' - || col.columnType === 'blob' - || col.columnType.includes('char') - || col.columnType.includes('varchar') - || col.columnType.includes('binary') - || col.columnType.includes('varbinary')) - && table.primaryKeys.includes(col.name) - ) { - const generator = new Generators.GenerateUniqueString({}); - return generator; - } + // STRING + if ( + (col.columnType === 'text' + || col.columnType === 'blob' + || col.columnType.includes('char') + || col.columnType.includes('varchar') + || col.columnType.includes('binary') + || col.columnType.includes('varbinary')) + && table.primaryKeys.includes(col.name) + ) { + const generator = new Generators.GenerateUniqueString({}); + return generator; + } - if ( - (col.columnType === 'text' - || col.columnType === 'blob' - || col.columnType.includes('char') - || col.columnType.includes('varchar') - || col.columnType.includes('binary') - || col.columnType.includes('varbinary')) - && col.name.toLowerCase().includes('name') - ) { - const generator = new Generators.GenerateFirstName({}); - return generator; - } + if ( + (col.columnType === 'text' + || col.columnType === 'blob' + || col.columnType.includes('char') + || col.columnType.includes('varchar') + || col.columnType.includes('binary') + || col.columnType.includes('varbinary')) + && col.name.toLowerCase().includes('name') + ) { + const generator = new Generators.GenerateFirstName({}); + return generator; + } - if ( - (col.columnType === 'text' + if ( + (col.columnType === 'text' + || col.columnType === 'blob' + || col.columnType.includes('char') + || col.columnType.includes('varchar') + || col.columnType.includes('binary') + || col.columnType.includes('varbinary')) + && col.name.toLowerCase().includes('email') + ) { + const generator = new Generators.GenerateEmail({}); + return generator; + } + + if ( + col.columnType === 'text' || col.columnType === 'blob' || col.columnType.includes('char') || col.columnType.includes('varchar') || col.columnType.includes('binary') - || col.columnType.includes('varbinary')) - && col.name.toLowerCase().includes('email') - ) { - const generator = new Generators.GenerateEmail({}); - return generator; - } + || col.columnType.includes('varbinary') + ) { + const generator = new Generators.GenerateString({}); + return generator; + } - if ( - col.columnType === 'text' - || col.columnType === 'blob' - || col.columnType.includes('char') - || col.columnType.includes('varchar') - || col.columnType.includes('binary') - || col.columnType.includes('varbinary') - ) { - // console.log(col, table); - const generator = new Generators.GenerateString({}); - return generator; - } + // BOOLEAN + if (col.columnType === 'boolean') { + const generator = new Generators.GenerateBoolean({}); + return generator; + } - // BOOLEAN - if (col.columnType === 'boolean') { - const generator = new Generators.GenerateBoolean({}); - return generator; - } + // DATE, TIME, TIMESTAMP, DATETIME, YEAR + if (col.columnType.includes('datetime')) { + const generator = new Generators.GenerateDatetime({}); + return generator; + } - // DATE, TIME, TIMESTAMP, DATETIME, YEAR - if (col.columnType.includes('datetime')) { - const generator = new Generators.GenerateDatetime({}); - return generator; - } + if (col.columnType.includes('date')) { + const generator = new Generators.GenerateDate({}); + return generator; + } - if (col.columnType.includes('date')) { - const generator = new Generators.GenerateDate({}); - return generator; - } + if (col.columnType === 'time') { + const generator = new Generators.GenerateTime({}); + return generator; + } - if (col.columnType === 'time') { - const generator = new Generators.GenerateTime({}); - return generator; - } + if (col.columnType.includes('timestamp')) { + const generator = new Generators.GenerateTimestamp({}); + return generator; + } - if (col.columnType.includes('timestamp')) { - const generator = new Generators.GenerateTimestamp({}); - return generator; - } + if (col.columnType === 'year') { + const generator = new Generators.GenerateYear({}); + return generator; + } - if (col.columnType === 'year') { - const generator = new Generators.GenerateYear({}); - return generator; - } + // JSON + if (col.columnType === 'json') { + const generator = new Generators.GenerateJson({}); + return generator; + } - // JSON - if (col.columnType === 'json') { - const generator = new Generators.GenerateJson({}); - return generator; - } + // ENUM + if (col.enumValues !== undefined) { + const generator = new Generators.GenerateEnum({ + enumValues: col.enumValues, + }); + return generator; + } - // ENUM - if (col.enumValues !== undefined) { - const generator = new Generators.GenerateEnum({ - enumValues: col.enumValues, - }); - return generator; - } + if (col.hasDefault && col.default !== undefined) { + const generator = new Generators.GenerateDefault({ + defaultValue: col.default, + }); + return generator; + } - if (col.hasDefault && col.default !== undefined) { - const generator = new Generators.GenerateDefault({ - defaultValue: col.default, - }); - return generator; + return; + }; + + let generator = pickGenerator(table, col) as AbstractGenerator || undefined; + if (generator !== undefined) { + const generatorConstructor = this.selectGeneratorOfVersion( + version, + generator.getEntityKind(), + ); + + generator = new generatorConstructor(generator.params); } - return; + return generator; }; selectGeneratorForSqlite = ( table: Table, col: Column, + version: number, ) => { - // int section --------------------------------------------------------------------------------------- - if ( - (col.columnType === 'integer' || col.columnType === 'numeric') - && table.primaryKeys.includes(col.name) - ) { - const generator = new Generators.GenerateIntPrimaryKey({}); - return generator; - } + const pickGenerator = (table: Table, col: Column) => { + // int section --------------------------------------------------------------------------------------- + if ( + (col.columnType === 'integer' || col.columnType === 'numeric') + && table.primaryKeys.includes(col.name) + ) { + const generator = new Generators.GenerateIntPrimaryKey({}); + return generator; + } - if (col.columnType === 'integer' && col.dataType === 'boolean') { - const generator = new Generators.GenerateBoolean({}); - return generator; - } + if (col.columnType === 'integer' && col.dataType === 'boolean') { + const generator = new Generators.GenerateBoolean({}); + return generator; + } - if ((col.columnType === 'integer' && col.dataType === 'date')) { - const generator = new Generators.GenerateTimestamp({}); - return generator; - } + if ((col.columnType === 'integer' && col.dataType === 'date')) { + const generator = new Generators.GenerateTimestamp({}); + return generator; + } - if ( - col.columnType === 'integer' - || col.columnType === 'numeric' - || (col.dataType === 'bigint' && col.columnType === 'blob') - ) { - const generator = new Generators.GenerateInt({}); - return generator; - } + if ( + col.columnType === 'integer' + || col.columnType === 'numeric' + || (col.dataType === 'bigint' && col.columnType === 'blob') + ) { + const generator = new Generators.GenerateInt({}); + return generator; + } - // number section ------------------------------------------------------------------------------------ - if (col.columnType === 'real' || col.columnType === 'numeric') { - const generator = new Generators.GenerateNumber({}); - return generator; - } + // number section ------------------------------------------------------------------------------------ + if (col.columnType === 'real' || col.columnType === 'numeric') { + const generator = new Generators.GenerateNumber({}); + return generator; + } - // string section ------------------------------------------------------------------------------------ - if ( - (col.columnType === 'text' - || col.columnType === 'numeric' - || col.columnType === 'blob') - && table.primaryKeys.includes(col.name) - ) { - const generator = new Generators.GenerateUniqueString({}); - return generator; - } + // string section ------------------------------------------------------------------------------------ + if ( + (col.columnType === 'text' + || col.columnType === 'numeric' + || col.columnType === 'blob') + && table.primaryKeys.includes(col.name) + ) { + const generator = new Generators.GenerateUniqueString({}); + return generator; + } - if ( - (col.columnType === 'text' - || col.columnType === 'numeric' - || col.columnType === 'blob') - && col.name.toLowerCase().includes('name') - ) { - const generator = new Generators.GenerateFirstName({}); - return generator; - } + if ( + (col.columnType === 'text' + || col.columnType === 'numeric' + || col.columnType === 'blob') + && col.name.toLowerCase().includes('name') + ) { + const generator = new Generators.GenerateFirstName({}); + return generator; + } - if ( - (col.columnType === 'text' + if ( + (col.columnType === 'text' + || col.columnType === 'numeric' + || col.columnType === 'blob') + && col.name.toLowerCase().includes('email') + ) { + const generator = new Generators.GenerateEmail({}); + return generator; + } + + if ( + col.columnType === 'text' || col.columnType === 'numeric' - || col.columnType === 'blob') - && col.name.toLowerCase().includes('email') - ) { - const generator = new Generators.GenerateEmail({}); - return generator; - } + || col.columnType === 'blob' + || col.columnType === 'blobbuffer' + ) { + const generator = new Generators.GenerateString({}); + return generator; + } - if ( - col.columnType === 'text' - || col.columnType === 'numeric' - || col.columnType === 'blob' - || col.columnType === 'blobbuffer' - ) { - const generator = new Generators.GenerateString({}); - return generator; - } + if ( + (col.columnType === 'text' && col.dataType === 'json') + || (col.columnType === 'blob' && col.dataType === 'json') + ) { + const generator = new Generators.GenerateJson({}); + return generator; + } - if ( - (col.columnType === 'text' && col.dataType === 'json') - || (col.columnType === 'blob' && col.dataType === 'json') - ) { - const generator = new Generators.GenerateJson({}); - return generator; - } + if (col.hasDefault && col.default !== undefined) { + const generator = new Generators.GenerateDefault({ + defaultValue: col.default, + }); + return generator; + } - if (col.hasDefault && col.default !== undefined) { - const generator = new Generators.GenerateDefault({ - defaultValue: col.default, - }); - return generator; + return; + }; + + let generator = pickGenerator(table, col) as AbstractGenerator || undefined; + if (generator !== undefined) { + const generatorConstructor = this.selectGeneratorOfVersion( + version, + generator.getEntityKind(), + ); + + generator = new generatorConstructor(generator.params); } - return; + return generator; }; filterCyclicTables = (tablesGenerators: ReturnType) => { @@ -1056,9 +1092,6 @@ export class SeedService { tablesUniqueNotNullColumn?: { [tableName: string]: { uniqueNotNullColName: string } }; }, ) => { - // console.time( - // "generateTablesValues-----------------------------------------------------" - // ); const customSeed = options?.seed === undefined ? 0 : options.seed; let tableCount: number | undefined; let columnsGenerators: Prettify[]; @@ -1074,7 +1107,6 @@ export class SeedService { }[] = options?.tablesValues === undefined ? [] : options.tablesValues; let pRNGSeed: number; - // relations = relations.filter(rel => rel.type === "one"); let filteredRelations: typeof relations; let preserveData: boolean, insertDataInDb: boolean = true, updateDataInDb: boolean = false; @@ -1182,7 +1214,6 @@ export class SeedService { (genObj as Generators.GenerateValuesFromArray).maxRepeatedValuesCount = repeatedValuesCount; } - // console.log(rel.columns[colIdx], tableGenerators) if (genObj !== undefined) { tableGenerators[rel.columns[colIdx]!]!.generator = genObj; } From 78f9c1c7ef3d55825c9fe1b32e51d490207ce596 Mon Sep 17 00:00:00 2001 From: Mario564 Date: Wed, 18 Dec 2024 09:43:51 -0800 Subject: [PATCH 14/32] Add SS support to drizzle-valibot --- drizzle-valibot/src/column.ts | 84 +++- drizzle-valibot/src/column.types.ts | 18 +- drizzle-valibot/tests/singlestore.test.ts | 500 ++++++++++++++++++++++ 3 files changed, 584 insertions(+), 18 deletions(-) create mode 100644 drizzle-valibot/tests/singlestore.test.ts diff --git a/drizzle-valibot/src/column.ts b/drizzle-valibot/src/column.ts index 5d7a1784e..cf90751ef 100644 --- a/drizzle-valibot/src/column.ts +++ b/drizzle-valibot/src/column.ts @@ -37,6 +37,21 @@ import type { PgVarchar, PgVector, } from 'drizzle-orm/pg-core'; +import { + SingleStoreBigInt53, + SingleStoreChar, + SingleStoreDouble, + SingleStoreFloat, + SingleStoreInt, + SingleStoreMediumInt, + SingleStoreReal, + SingleStoreSerial, + SingleStoreSmallInt, + SingleStoreText, + SingleStoreTinyInt, + SingleStoreVarChar, + SingleStoreYear, +} from 'drizzle-orm/singlestore-core'; import type { SQLiteInteger, SQLiteReal, SQLiteText } from 'drizzle-orm/sqlite-core'; import * as v from 'valibot'; import { CONSTANTS } from './constants.ts'; @@ -116,57 +131,92 @@ function numberColumnToSchema(column: Column): v.GenericSchema { let max!: number; let integer = false; - if (isColumnType>(column, ['MySqlTinyInt'])) { + if (isColumnType | SingleStoreTinyInt>(column, ['MySqlTinyInt', 'SingleStoreTinyInt'])) { min = unsigned ? 0 : CONSTANTS.INT8_MIN; max = unsigned ? CONSTANTS.INT8_UNSIGNED_MAX : CONSTANTS.INT8_MAX; integer = true; } else if ( - isColumnType | PgSmallSerial | MySqlSmallInt>(column, [ + isColumnType | PgSmallSerial | MySqlSmallInt | SingleStoreSmallInt>(column, [ 'PgSmallInt', 'PgSmallSerial', 'MySqlSmallInt', + 'SingleStoreSmallInt', ]) ) { min = unsigned ? 0 : CONSTANTS.INT16_MIN; max = unsigned ? CONSTANTS.INT16_UNSIGNED_MAX : CONSTANTS.INT16_MAX; integer = true; } else if ( - isColumnType | MySqlFloat | MySqlMediumInt>(column, [ + isColumnType< + PgReal | MySqlFloat | MySqlMediumInt | SingleStoreFloat | SingleStoreMediumInt + >(column, [ 'PgReal', 'MySqlFloat', 'MySqlMediumInt', + 'SingleStoreFloat', + 'SingleStoreMediumInt', ]) ) { min = unsigned ? 0 : CONSTANTS.INT24_MIN; max = unsigned ? CONSTANTS.INT24_UNSIGNED_MAX : CONSTANTS.INT24_MAX; - integer = isColumnType(column, ['MySqlMediumInt']); + integer = isColumnType(column, ['MySqlMediumInt', 'SingleStoreMediumInt']); } else if ( - isColumnType | PgSerial | MySqlInt>(column, ['PgInteger', 'PgSerial', 'MySqlInt']) + isColumnType | PgSerial | MySqlInt | SingleStoreInt>(column, [ + 'PgInteger', + 'PgSerial', + 'MySqlInt', + 'SingleStoreInt', + ]) ) { min = unsigned ? 0 : CONSTANTS.INT32_MIN; max = unsigned ? CONSTANTS.INT32_UNSIGNED_MAX : CONSTANTS.INT32_MAX; integer = true; } else if ( - isColumnType | MySqlReal | MySqlDouble | SQLiteReal>(column, [ + isColumnType< + | PgDoublePrecision + | MySqlReal + | MySqlDouble + | SingleStoreReal + | SingleStoreDouble + | SQLiteReal + >(column, [ 'PgDoublePrecision', 'MySqlReal', 'MySqlDouble', + 'SingleStoreReal', + 'SingleStoreDouble', 'SQLiteReal', ]) ) { min = unsigned ? 0 : CONSTANTS.INT48_MIN; max = unsigned ? CONSTANTS.INT48_UNSIGNED_MAX : CONSTANTS.INT48_MAX; } else if ( - isColumnType | PgBigSerial53 | MySqlBigInt53 | MySqlSerial | SQLiteInteger>( + isColumnType< + | PgBigInt53 + | PgBigSerial53 + | MySqlBigInt53 + | MySqlSerial + | SingleStoreBigInt53 + | SingleStoreSerial + | SQLiteInteger + >( column, - ['PgBigInt53', 'PgBigSerial53', 'MySqlBigInt53', 'MySqlSerial', 'SQLiteInteger'], + [ + 'PgBigInt53', + 'PgBigSerial53', + 'MySqlBigInt53', + 'MySqlSerial', + 'SingleStoreBigInt53', + 'SingleStoreSerial', + 'SQLiteInteger', + ], ) ) { - unsigned = unsigned || isColumnType(column, ['MySqlSerial']); + unsigned = unsigned || isColumnType(column, ['MySqlSerial', 'SingleStoreSerial']); min = unsigned ? 0 : Number.MIN_SAFE_INTEGER; max = Number.MAX_SAFE_INTEGER; integer = true; - } else if (isColumnType>(column, ['MySqlYear'])) { + } else if (isColumnType | SingleStoreYear>(column, ['MySqlYear', 'SingleStoreYear'])) { min = 1901; max = 2155; integer = true; @@ -201,9 +251,11 @@ function stringColumnToSchema(column: Column): v.GenericSchema { if (isColumnType | SQLiteText>(column, ['PgVarchar', 'SQLiteText'])) { max = column.length; - } else if (isColumnType>(column, ['MySqlVarChar'])) { + } else if ( + isColumnType | SingleStoreVarChar>(column, ['MySqlVarChar', 'SingleStoreVarChar']) + ) { max = column.length ?? CONSTANTS.INT16_UNSIGNED_MAX; - } else if (isColumnType>(column, ['MySqlText'])) { + } else if (isColumnType | SingleStoreText>(column, ['MySqlText', 'SingleStoreText'])) { if (column.textType === 'longtext') { max = CONSTANTS.INT32_UNSIGNED_MAX; } else if (column.textType === 'mediumtext') { @@ -215,7 +267,13 @@ function stringColumnToSchema(column: Column): v.GenericSchema { } } - if (isColumnType | MySqlChar>(column, ['PgChar', 'MySqlChar'])) { + if ( + isColumnType | MySqlChar | SingleStoreChar>(column, [ + 'PgChar', + 'MySqlChar', + 'SingleStoreChar', + ]) + ) { max = column.length; fixed = true; } diff --git a/drizzle-valibot/src/column.types.ts b/drizzle-valibot/src/column.types.ts index b9567b12d..2b30cb60a 100644 --- a/drizzle-valibot/src/column.types.ts +++ b/drizzle-valibot/src/column.types.ts @@ -20,15 +20,16 @@ export type GetBaseColumn = TColumn['_'] extends { baseC export type EnumValuesToEnum = { readonly [K in TEnumValues[number]]: K }; export type ExtractAdditionalProperties = { - max: TColumn['_']['columnType'] extends 'PgVarchar' | 'SQLiteText' | 'PgChar' | 'MySqlChar' + max: TColumn['_']['columnType'] extends 'PgVarchar' | 'SQLiteText' | 'PgChar' | 'MySqlChar' | 'SingleStoreChar' ? Assume['length'] - : TColumn['_']['columnType'] extends 'MySqlText' | 'MySqlVarChar' ? number + : TColumn['_']['columnType'] extends 'MySqlText' | 'MySqlVarChar' | 'SingleStoreText' | 'SingleStoreVarChar' + ? number : TColumn['_']['columnType'] extends 'PgBinaryVector' | 'PgHalfVector' | 'PgVector' ? Assume['dimensions'] : TColumn['_']['columnType'] extends 'PgArray' ? Assume['size'] : undefined; - fixedLength: TColumn['_']['columnType'] extends 'PgChar' | 'MySqlChar' | 'PgHalfVector' | 'PgVector' | 'PgArray' - ? true + fixedLength: TColumn['_']['columnType'] extends + 'PgChar' | 'PgHalfVector' | 'PgVector' | 'PgArray' | 'MySqlChar' | 'SingleStoreChar' ? true : false; }; @@ -98,19 +99,26 @@ export type GetValibotType< v.MaxValueAction, TColumnType extends | 'MySqlTinyInt' + | 'SingleStoreTinyInt' | 'PgSmallInt' | 'PgSmallSerial' | 'MySqlSmallInt' | 'MySqlMediumInt' + | 'SingleStoreSmallInt' + | 'SingleStoreMediumInt' | 'PgInteger' | 'PgSerial' | 'MySqlInt' + | 'SingleStoreInt' | 'PgBigInt53' | 'PgBigSerial53' | 'MySqlBigInt53' | 'MySqlSerial' + | 'SingleStoreBigInt53' + | 'SingleStoreSerial' | 'SQLiteInteger' - | 'MySqlYear' ? v.IntegerAction + | 'MySqlYear' + | 'SingleStoreYear' ? v.IntegerAction : never, ]> > diff --git a/drizzle-valibot/tests/singlestore.test.ts b/drizzle-valibot/tests/singlestore.test.ts new file mode 100644 index 000000000..0827ba7a1 --- /dev/null +++ b/drizzle-valibot/tests/singlestore.test.ts @@ -0,0 +1,500 @@ +import { type Equal } from 'drizzle-orm'; +import { customType, int, serial, singlestoreSchema, singlestoreTable, text } from 'drizzle-orm/singlestore-core'; +import * as v from 'valibot'; +import { test } from 'vitest'; +import { jsonSchema } from '~/column.ts'; +import { CONSTANTS } from '~/constants.ts'; +import { createInsertSchema, createSelectSchema, createUpdateSchema } from '../src'; +import { Expect, expectSchemaShape } from './utils.ts'; + +const intSchema = v.pipe( + v.number(), + v.minValue(CONSTANTS.INT32_MIN as number), + v.maxValue(CONSTANTS.INT32_MAX as number), + v.integer(), +); +const serialNumberModeSchema = v.pipe( + v.number(), + v.minValue(0 as number), + v.maxValue(Number.MAX_SAFE_INTEGER as number), + v.integer(), +); +const textSchema = v.pipe(v.string(), v.maxLength(CONSTANTS.INT16_UNSIGNED_MAX as number)); + +test('table - select', (t) => { + const table = singlestoreTable('test', { + id: serial().primaryKey(), + name: text().notNull(), + }); + + const result = createSelectSchema(table); + const expected = v.object({ id: serialNumberModeSchema, name: textSchema }); + expectSchemaShape(t, expected).from(result); + Expect>(); +}); + +test('table in schema - select', (tc) => { + const schema = singlestoreSchema('test'); + const table = schema.table('test', { + id: serial().primaryKey(), + name: text().notNull(), + }); + + const result = createSelectSchema(table); + const expected = v.object({ id: serialNumberModeSchema, name: textSchema }); + expectSchemaShape(tc, expected).from(result); + Expect>(); +}); + +test('table - insert', (t) => { + const table = singlestoreTable('test', { + id: serial().primaryKey(), + name: text().notNull(), + age: int(), + }); + + const result = createInsertSchema(table); + const expected = v.object({ + id: v.optional(serialNumberModeSchema), + name: textSchema, + age: v.optional(v.nullable(intSchema)), + }); + expectSchemaShape(t, expected).from(result); + Expect>(); +}); + +test('table - update', (t) => { + const table = singlestoreTable('test', { + id: serial().primaryKey(), + name: text().notNull(), + age: int(), + }); + + const result = createUpdateSchema(table); + const expected = v.object({ + id: v.optional(serialNumberModeSchema), + name: v.optional(textSchema), + age: v.optional(v.nullable(intSchema)), + }); + expectSchemaShape(t, expected).from(result); + Expect>(); +}); + +// TODO: SingleStore doesn't support views yet. Add these tests when they're added + +// test('view qb - select', (t) => { +// const table = singlestoreTable('test', { +// id: serial().primaryKey(), +// name: text().notNull(), +// }); +// const view = mysqlView('test').as((qb) => qb.select({ id: table.id, age: sql``.as('age') }).from(table)); + +// const result = createSelectSchema(view); +// const expected = v.object({ id: serialNumberModeSchema, age: v.any() }); +// expectSchemaShape(t, expected).from(result); +// Expect>(); +// }); + +// test('view columns - select', (t) => { +// const view = mysqlView('test', { +// id: serial().primaryKey(), +// name: text().notNull(), +// }).as(sql``); + +// const result = createSelectSchema(view); +// const expected = v.object({ id: serialNumberModeSchema, name: textSchema }); +// expectSchemaShape(t, expected).from(result); +// Expect>(); +// }); + +// test('view with nested fields - select', (t) => { +// const table = singlestoreTable('test', { +// id: serial().primaryKey(), +// name: text().notNull(), +// }); +// const view = mysqlView('test').as((qb) => +// qb.select({ +// id: table.id, +// nested: { +// name: table.name, +// age: sql``.as('age'), +// }, +// table, +// }).from(table) +// ); + +// const result = createSelectSchema(view); +// const expected = v.object({ +// id: serialNumberModeSchema, +// nested: v.object({ name: textSchema, age: v.any() }), +// table: v.object({ id: serialNumberModeSchema, name: textSchema }), +// }); +// expectSchemaShape(t, expected).from(result); +// Expect>(); +// }); + +test('nullability - select', (t) => { + const table = singlestoreTable('test', { + c1: int(), + c2: int().notNull(), + c3: int().default(1), + c4: int().notNull().default(1), + }); + + const result = createSelectSchema(table); + const expected = v.object({ + c1: v.nullable(intSchema), + c2: intSchema, + c3: v.nullable(intSchema), + c4: intSchema, + }); + expectSchemaShape(t, expected).from(result); + Expect>(); +}); + +test('nullability - insert', (t) => { + const table = singlestoreTable('test', { + c1: int(), + c2: int().notNull(), + c3: int().default(1), + c4: int().notNull().default(1), + c5: int().generatedAlwaysAs(1), + }); + + const result = createInsertSchema(table); + const expected = v.object({ + c1: v.optional(v.nullable(intSchema)), + c2: intSchema, + c3: v.optional(v.nullable(intSchema)), + c4: v.optional(intSchema), + }); + expectSchemaShape(t, expected).from(result); + Expect>(); +}); + +test('nullability - update', (t) => { + const table = singlestoreTable('test', { + c1: int(), + c2: int().notNull(), + c3: int().default(1), + c4: int().notNull().default(1), + c5: int().generatedAlwaysAs(1), + }); + + const result = createUpdateSchema(table); + const expected = v.object({ + c1: v.optional(v.nullable(intSchema)), + c2: v.optional(intSchema), + c3: v.optional(v.nullable(intSchema)), + c4: v.optional(intSchema), + }); + expectSchemaShape(t, expected).from(result); + Expect>(); +}); + +test('refine table - select', (t) => { + const table = singlestoreTable('test', { + c1: int(), + c2: int().notNull(), + c3: int().notNull(), + }); + + const result = createSelectSchema(table, { + c2: (schema) => v.pipe(schema, v.maxValue(1000)), + c3: v.pipe(v.string(), v.transform(Number)), + }); + const expected = v.object({ + c1: v.nullable(intSchema), + c2: v.pipe(intSchema, v.maxValue(1000)), + c3: v.pipe(v.string(), v.transform(Number)), + }); + expectSchemaShape(t, expected).from(result); + Expect>(); +}); + +test('refine table - select with custom data type', (t) => { + const customText = customType({ dataType: () => 'text' }); + const table = singlestoreTable('test', { + c1: int(), + c2: int().notNull(), + c3: int().notNull(), + c4: customText(), + }); + + const customTextSchema = v.pipe(v.string(), v.minLength(1), v.maxLength(100)); + const result = createSelectSchema(table, { + c2: (schema) => v.pipe(schema, v.maxValue(1000)), + c3: v.pipe(v.string(), v.transform(Number)), + c4: customTextSchema, + }); + const expected = v.object({ + c1: v.nullable(intSchema), + c2: v.pipe(intSchema, v.maxValue(1000)), + c3: v.pipe(v.string(), v.transform(Number)), + c4: customTextSchema, + }); + + expectSchemaShape(t, expected).from(result); + Expect>(); +}); + +test('refine table - insert', (t) => { + const table = singlestoreTable('test', { + c1: int(), + c2: int().notNull(), + c3: int().notNull(), + c4: int().generatedAlwaysAs(1), + }); + + const result = createInsertSchema(table, { + c2: (schema) => v.pipe(schema, v.maxValue(1000)), + c3: v.pipe(v.string(), v.transform(Number)), + }); + const expected = v.object({ + c1: v.optional(v.nullable(intSchema)), + c2: v.pipe(intSchema, v.maxValue(1000)), + c3: v.pipe(v.string(), v.transform(Number)), + }); + expectSchemaShape(t, expected).from(result); + Expect>(); +}); + +test('refine table - update', (t) => { + const table = singlestoreTable('test', { + c1: int(), + c2: int().notNull(), + c3: int().notNull(), + c4: int().generatedAlwaysAs(1), + }); + + const result = createUpdateSchema(table, { + c2: (schema) => v.pipe(schema, v.maxValue(1000)), + c3: v.pipe(v.string(), v.transform(Number)), + }); + const expected = v.object({ + c1: v.optional(v.nullable(intSchema)), + c2: v.optional(v.pipe(intSchema, v.maxValue(1000))), + c3: v.pipe(v.string(), v.transform(Number)), + }); + expectSchemaShape(t, expected).from(result); + Expect>(); +}); + +// test('refine view - select', (t) => { +// const table = singlestoreTable('test', { +// c1: int(), +// c2: int(), +// c3: int(), +// c4: int(), +// c5: int(), +// c6: int(), +// }); +// const view = mysqlView('test').as((qb) => +// qb.select({ +// c1: table.c1, +// c2: table.c2, +// c3: table.c3, +// nested: { +// c4: table.c4, +// c5: table.c5, +// c6: table.c6, +// }, +// table, +// }).from(table) +// ); + +// const result = createSelectSchema(view, { +// c2: (schema) => v.pipe(schema, v.maxValue(1000)), +// c3: v.pipe(v.string(), v.transform(Number)), +// nested: { +// c5: (schema) => v.pipe(schema, v.maxValue(1000)), +// c6: v.pipe(v.string(), v.transform(Number)), +// }, +// table: { +// c2: (schema) => v.pipe(schema, v.maxValue(1000)), +// c3: v.pipe(v.string(), v.transform(Number)), +// }, +// }); +// const expected = v.object({ +// c1: v.nullable(intSchema), +// c2: v.nullable(v.pipe(intSchema, v.maxValue(1000))), +// c3: v.pipe(v.string(), v.transform(Number)), +// nested: v.object({ +// c4: v.nullable(intSchema), +// c5: v.nullable(v.pipe(intSchema, v.maxValue(1000))), +// c6: v.pipe(v.string(), v.transform(Number)), +// }), +// table: v.object({ +// c1: v.nullable(intSchema), +// c2: v.nullable(v.pipe(intSchema, v.maxValue(1000))), +// c3: v.pipe(v.string(), v.transform(Number)), +// c4: v.nullable(intSchema), +// c5: v.nullable(intSchema), +// c6: v.nullable(intSchema), +// }), +// }); +// expectSchemaShape(t, expected).from(result); +// Expect>(); +// }); + +test('all data types', (t) => { + const table = singlestoreTable('test', ({ + bigint, + binary, + boolean, + char, + date, + datetime, + decimal, + double, + float, + int, + json, + mediumint, + singlestoreEnum, + real, + serial, + smallint, + text, + time, + timestamp, + tinyint, + varchar, + varbinary, + year, + longtext, + mediumtext, + tinytext, + }) => ({ + bigint1: bigint({ mode: 'number' }).notNull(), + bigint2: bigint({ mode: 'bigint' }).notNull(), + bigint3: bigint({ unsigned: true, mode: 'number' }).notNull(), + bigint4: bigint({ unsigned: true, mode: 'bigint' }).notNull(), + binary: binary({ length: 10 }).notNull(), + boolean: boolean().notNull(), + char1: char({ length: 10 }).notNull(), + char2: char({ length: 1, enum: ['a', 'b', 'c'] }).notNull(), + date1: date({ mode: 'date' }).notNull(), + date2: date({ mode: 'string' }).notNull(), + datetime1: datetime({ mode: 'date' }).notNull(), + datetime2: datetime({ mode: 'string' }).notNull(), + decimal1: decimal().notNull(), + decimal2: decimal({ unsigned: true }).notNull(), + double1: double().notNull(), + double2: double({ unsigned: true }).notNull(), + float1: float().notNull(), + float2: float({ unsigned: true }).notNull(), + int1: int().notNull(), + int2: int({ unsigned: true }).notNull(), + json: json().notNull(), + mediumint1: mediumint().notNull(), + mediumint2: mediumint({ unsigned: true }).notNull(), + enum: singlestoreEnum('enum', ['a', 'b', 'c']).notNull(), + real: real().notNull(), + serial: serial().notNull(), + smallint1: smallint().notNull(), + smallint2: smallint({ unsigned: true }).notNull(), + text1: text().notNull(), + text2: text({ enum: ['a', 'b', 'c'] }).notNull(), + time: time().notNull(), + timestamp1: timestamp({ mode: 'date' }).notNull(), + timestamp2: timestamp({ mode: 'string' }).notNull(), + tinyint1: tinyint().notNull(), + tinyint2: tinyint({ unsigned: true }).notNull(), + varchar1: varchar({ length: 10 }).notNull(), + varchar2: varchar({ length: 1, enum: ['a', 'b', 'c'] }).notNull(), + varbinary: varbinary({ length: 10 }).notNull(), + year: year().notNull(), + longtext1: longtext().notNull(), + longtext2: longtext({ enum: ['a', 'b', 'c'] }).notNull(), + mediumtext1: mediumtext().notNull(), + mediumtext2: mediumtext({ enum: ['a', 'b', 'c'] }).notNull(), + tinytext1: tinytext().notNull(), + tinytext2: tinytext({ enum: ['a', 'b', 'c'] }).notNull(), + })); + + const result = createSelectSchema(table); + const expected = v.object({ + bigint1: v.pipe(v.number(), v.minValue(Number.MIN_SAFE_INTEGER), v.maxValue(Number.MAX_SAFE_INTEGER), v.integer()), + bigint2: v.pipe(v.bigint(), v.minValue(CONSTANTS.INT64_MIN), v.maxValue(CONSTANTS.INT64_MAX)), + bigint3: v.pipe(v.number(), v.minValue(0 as number), v.maxValue(Number.MAX_SAFE_INTEGER), v.integer()), + bigint4: v.pipe(v.bigint(), v.minValue(0n as bigint), v.maxValue(CONSTANTS.INT64_UNSIGNED_MAX)), + binary: v.string(), + boolean: v.boolean(), + char1: v.pipe(v.string(), v.length(10 as number)), + char2: v.enum({ a: 'a', b: 'b', c: 'c' }), + date1: v.date(), + date2: v.string(), + datetime1: v.date(), + datetime2: v.string(), + decimal1: v.string(), + decimal2: v.string(), + double1: v.pipe(v.number(), v.minValue(CONSTANTS.INT48_MIN), v.maxValue(CONSTANTS.INT48_MAX)), + double2: v.pipe(v.number(), v.minValue(0 as number), v.maxValue(CONSTANTS.INT48_UNSIGNED_MAX)), + float1: v.pipe(v.number(), v.minValue(CONSTANTS.INT24_MIN), v.maxValue(CONSTANTS.INT24_MAX)), + float2: v.pipe(v.number(), v.minValue(0 as number), v.maxValue(CONSTANTS.INT24_UNSIGNED_MAX)), + int1: v.pipe(v.number(), v.minValue(CONSTANTS.INT32_MIN), v.maxValue(CONSTANTS.INT32_MAX), v.integer()), + int2: v.pipe(v.number(), v.minValue(0 as number), v.maxValue(CONSTANTS.INT32_UNSIGNED_MAX), v.integer()), + json: jsonSchema, + mediumint1: v.pipe(v.number(), v.minValue(CONSTANTS.INT24_MIN), v.maxValue(CONSTANTS.INT24_MAX), v.integer()), + mediumint2: v.pipe(v.number(), v.minValue(0 as number), v.maxValue(CONSTANTS.INT24_UNSIGNED_MAX), v.integer()), + enum: v.enum({ a: 'a', b: 'b', c: 'c' }), + real: v.pipe(v.number(), v.minValue(CONSTANTS.INT48_MIN), v.maxValue(CONSTANTS.INT48_MAX)), + serial: v.pipe(v.number(), v.minValue(0 as number), v.maxValue(Number.MAX_SAFE_INTEGER), v.integer()), + smallint1: v.pipe(v.number(), v.minValue(CONSTANTS.INT16_MIN), v.maxValue(CONSTANTS.INT16_MAX), v.integer()), + smallint2: v.pipe(v.number(), v.minValue(0 as number), v.maxValue(CONSTANTS.INT16_UNSIGNED_MAX), v.integer()), + text1: v.pipe(v.string(), v.maxLength(CONSTANTS.INT16_UNSIGNED_MAX)), + text2: v.enum({ a: 'a', b: 'b', c: 'c' }), + time: v.string(), + timestamp1: v.date(), + timestamp2: v.string(), + tinyint1: v.pipe(v.number(), v.minValue(CONSTANTS.INT8_MIN), v.maxValue(CONSTANTS.INT8_MAX), v.integer()), + tinyint2: v.pipe(v.number(), v.minValue(0 as number), v.maxValue(CONSTANTS.INT8_UNSIGNED_MAX), v.integer()), + varchar1: v.pipe(v.string(), v.maxLength(10 as number)), + varchar2: v.enum({ a: 'a', b: 'b', c: 'c' }), + varbinary: v.string(), + year: v.pipe(v.number(), v.minValue(1901 as number), v.maxValue(2155 as number), v.integer()), + longtext1: v.pipe(v.string(), v.maxLength(CONSTANTS.INT32_UNSIGNED_MAX)), + longtext2: v.enum({ a: 'a', b: 'b', c: 'c' }), + mediumtext1: v.pipe(v.string(), v.maxLength(CONSTANTS.INT24_UNSIGNED_MAX)), + mediumtext2: v.enum({ a: 'a', b: 'b', c: 'c' }), + tinytext1: v.pipe(v.string(), v.maxLength(CONSTANTS.INT8_UNSIGNED_MAX)), + tinytext2: v.enum({ a: 'a', b: 'b', c: 'c' }), + }); + expectSchemaShape(t, expected).from(result); + Expect>(); +}); + +/* Disallow unknown keys in table refinement - select */ { + const table = singlestoreTable('test', { id: int() }); + // @ts-expect-error + createSelectSchema(table, { unknown: v.string() }); +} + +/* Disallow unknown keys in table refinement - insert */ { + const table = singlestoreTable('test', { id: int() }); + // @ts-expect-error + createInsertSchema(table, { unknown: v.string() }); +} + +/* Disallow unknown keys in table refinement - update */ { + const table = singlestoreTable('test', { id: int() }); + // @ts-expect-error + createUpdateSchema(table, { unknown: v.string() }); +} + +// /* Disallow unknown keys in view qb - select */ { +// const table = singlestoreTable('test', { id: int() }); +// const view = mysqlView('test').as((qb) => qb.select().from(table)); +// const nestedSelect = mysqlView('test').as((qb) => qb.select({ table }).from(table)); +// // @ts-expect-error +// createSelectSchema(view, { unknown: v.string() }); +// // @ts-expect-error +// createSelectSchema(nestedSelect, { table: { unknown: v.string() } }); +// } + +// /* Disallow unknown keys in view columns - select */ { +// const view = mysqlView('test', { id: int() }).as(sql``); +// // @ts-expect-error +// createSelectSchema(view, { unknown: v.string() }); +// } From 455725c635960ba9757870c2522dd0e8e9ce0388 Mon Sep 17 00:00:00 2001 From: OleksiiKH0240 Date: Wed, 18 Dec 2024 19:46:41 +0200 Subject: [PATCH 15/32] updated changelogs --- changelogs/drizzle-seed/0.1.4.md | 106 +++++++++++++++++++++++++++++++ 1 file changed, 106 insertions(+) create mode 100644 changelogs/drizzle-seed/0.1.4.md diff --git a/changelogs/drizzle-seed/0.1.4.md b/changelogs/drizzle-seed/0.1.4.md new file mode 100644 index 000000000..84c3821af --- /dev/null +++ b/changelogs/drizzle-seed/0.1.4.md @@ -0,0 +1,106 @@ +## Improvements + +- A new parameter, `version`, was added to the `seed` function options, which will control generator versioning. + +```ts +await seed(db, schema, { version: 1 }); +``` + +Generator versions will change within a single major version of drizzle-seed. + +The old version of the generator will become available if the generator has been changed and the output of the new generator does not match the output of the old generator. + +#### For example: + +`FirstNameGen` was changed, and old version, `V1`, of this generator became available. + +Later, `LastNameGen` was changed in the next minor version of drizzle-seed, making `V2` version of this generator available. + +| | `V1` | `V2` | `V3(latest)` | +| :--------------: | :--------------: | :-------------: | :--------------: | +| **FirstNameGen** | `FirstNameGenV1` | `same as V3` | `FirstNameGenV3` | +| **LastNameGen** | `same as V2` | `LastNameGenV2` | `LastNameGenV3` | + +If you omit version, `V3` version of generators will be used. + +If you specify version of 2, `FirstNameGen` will use its `V3` version and `LastNameGen` will use its `V2` version. + +If you specify version of 1, `FirstNameGen` will use its `V1` version and `LastNameGen` will use its `V2` version. + +## + +- added `fields` as new parameter in `interval` generator + +`interval` generator generates intervals based on the following principle: + +fields to the right of the last specified field are zeroed out, while fields to the left remain valid. + +Thus, for example, there is no difference between the `year to month` fields and the `month` fields, because fields to the right of `month` (`day`, `hour`, `minute`, `second`) are zeroed out, while fields to the left (`year`) remain valid. + +Example + +```ts +import { pgTable, interval } from "drizzle-orm/pg-core"; +import { drizzle } from "drizzle-orm/node-postgres"; +import { seed } from "drizzle-seed"; + +const intervals = pgTable("intervals", { + interval: interval(), +}); + +async function main() { + const db = drizzle(process.env.DATABASE_URL!); + + await seed(db, { intervals }, { count: 1000 }).refine((funcs) => ({ + intervals: { + columns: { + interval: funcs.interval({ + fields: "day to hour", + }), + }, + }, + })); +} + +main(); +``` + +You can also specify fields in a table and seed them automatically. + +```ts +import { pgTable, interval } from "drizzle-orm/pg-core"; +import { drizzle } from "drizzle-orm/node-postgres"; +import { seed } from "drizzle-seed"; + +const intervals = pgTable("intervals", { + interval: interval({ fields: "day to hour" }), +}); + +async function main() { + const db = drizzle(process.env.DATABASE_URL!); + + await seed(db, { intervals }, { count: 1000 }); +} + +main(); +``` + +## Breaking changes + +- Unique `interval` generator was changed, so `1` version of this generator become available. **The latest version is `2`.** + +**Cause:** + +**Bug in generator:** +Old version of generator could generate intervals like: `1 minute 60 second`, `2 minute 0 second` and treat them as different intervals. + +However, after inserting the `1 minute 60 second` interval, PostgreSQL database will convert it to `2 minute 0 second`. As a result, subsequent insertion of the `2 minute 0 second` interval into a unique column will cause an error. + +## + +- Both non-unique and unique `string` generators were changed, making version `1` of these generators available. **The latest version is `2`.** + +**Cause:** + +**Generating strings based on text-like column length:** +Now (in version 2), the maximum length of a string depends on the length of the text column (e.g., `varchar(20)`). From c5efbadeab13d3a20df0037319a27f088dfbc12c Mon Sep 17 00:00:00 2001 From: Mario564 Date: Wed, 18 Dec 2024 09:52:41 -0800 Subject: [PATCH 16/32] Add SS support to drizzle-typebox --- drizzle-typebox/src/column.ts | 84 +++- drizzle-typebox/src/column.types.ts | 9 +- drizzle-typebox/tests/singlestore.test.ts | 497 ++++++++++++++++++++++ drizzle-valibot/src/column.ts | 2 +- 4 files changed, 577 insertions(+), 15 deletions(-) create mode 100644 drizzle-typebox/tests/singlestore.test.ts diff --git a/drizzle-typebox/src/column.ts b/drizzle-typebox/src/column.ts index 510ba7bbe..9bef765bf 100644 --- a/drizzle-typebox/src/column.ts +++ b/drizzle-typebox/src/column.ts @@ -39,6 +39,21 @@ import type { PgVarchar, PgVector, } from 'drizzle-orm/pg-core'; +import { + type SingleStoreBigInt53, + SingleStoreChar, + type SingleStoreDouble, + type SingleStoreFloat, + type SingleStoreInt, + type SingleStoreMediumInt, + type SingleStoreReal, + type SingleStoreSerial, + type SingleStoreSmallInt, + SingleStoreText, + type SingleStoreTinyInt, + SingleStoreVarChar, + SingleStoreYear, +} from 'drizzle-orm/singlestore-core'; import type { SQLiteInteger, SQLiteReal, SQLiteText } from 'drizzle-orm/sqlite-core'; import { CONSTANTS } from './constants.ts'; import { isColumnType, isWithEnum } from './utils.ts'; @@ -133,57 +148,92 @@ function numberColumnToSchema(column: Column, t: typeof typebox): TSchema { let max!: number; let integer = false; - if (isColumnType>(column, ['MySqlTinyInt'])) { + if (isColumnType | SingleStoreTinyInt>(column, ['MySqlTinyInt', 'SingleStoreTinyInt'])) { min = unsigned ? 0 : CONSTANTS.INT8_MIN; max = unsigned ? CONSTANTS.INT8_UNSIGNED_MAX : CONSTANTS.INT8_MAX; integer = true; } else if ( - isColumnType | PgSmallSerial | MySqlSmallInt>(column, [ + isColumnType | PgSmallSerial | MySqlSmallInt | SingleStoreSmallInt>(column, [ 'PgSmallInt', 'PgSmallSerial', 'MySqlSmallInt', + 'SingleStoreSmallInt', ]) ) { min = unsigned ? 0 : CONSTANTS.INT16_MIN; max = unsigned ? CONSTANTS.INT16_UNSIGNED_MAX : CONSTANTS.INT16_MAX; integer = true; } else if ( - isColumnType | MySqlFloat | MySqlMediumInt>(column, [ + isColumnType< + PgReal | MySqlFloat | MySqlMediumInt | SingleStoreFloat | SingleStoreMediumInt + >(column, [ 'PgReal', 'MySqlFloat', 'MySqlMediumInt', + 'SingleStoreFloat', + 'SingleStoreMediumInt', ]) ) { min = unsigned ? 0 : CONSTANTS.INT24_MIN; max = unsigned ? CONSTANTS.INT24_UNSIGNED_MAX : CONSTANTS.INT24_MAX; - integer = isColumnType(column, ['MySqlMediumInt']); + integer = isColumnType(column, ['MySqlMediumInt', 'SingleStoreMediumInt']); } else if ( - isColumnType | PgSerial | MySqlInt>(column, ['PgInteger', 'PgSerial', 'MySqlInt']) + isColumnType | PgSerial | MySqlInt | SingleStoreInt>(column, [ + 'PgInteger', + 'PgSerial', + 'MySqlInt', + 'SingleStoreInt', + ]) ) { min = unsigned ? 0 : CONSTANTS.INT32_MIN; max = unsigned ? CONSTANTS.INT32_UNSIGNED_MAX : CONSTANTS.INT32_MAX; integer = true; } else if ( - isColumnType | MySqlReal | MySqlDouble | SQLiteReal>(column, [ + isColumnType< + | PgDoublePrecision + | MySqlReal + | MySqlDouble + | SingleStoreReal + | SingleStoreDouble + | SQLiteReal + >(column, [ 'PgDoublePrecision', 'MySqlReal', 'MySqlDouble', + 'SingleStoreReal', + 'SingleStoreDouble', 'SQLiteReal', ]) ) { min = unsigned ? 0 : CONSTANTS.INT48_MIN; max = unsigned ? CONSTANTS.INT48_UNSIGNED_MAX : CONSTANTS.INT48_MAX; } else if ( - isColumnType | PgBigSerial53 | MySqlBigInt53 | MySqlSerial | SQLiteInteger>( + isColumnType< + | PgBigInt53 + | PgBigSerial53 + | MySqlBigInt53 + | MySqlSerial + | SingleStoreBigInt53 + | SingleStoreSerial + | SQLiteInteger + >( column, - ['PgBigInt53', 'PgBigSerial53', 'MySqlBigInt53', 'MySqlSerial', 'SQLiteInteger'], + [ + 'PgBigInt53', + 'PgBigSerial53', + 'MySqlBigInt53', + 'MySqlSerial', + 'SingleStoreBigInt53', + 'SingleStoreSerial', + 'SQLiteInteger', + ], ) ) { - unsigned = unsigned || isColumnType(column, ['MySqlSerial']); + unsigned = unsigned || isColumnType(column, ['MySqlSerial', 'SingleStoreSerial']); min = unsigned ? 0 : Number.MIN_SAFE_INTEGER; max = Number.MAX_SAFE_INTEGER; integer = true; - } else if (isColumnType>(column, ['MySqlYear'])) { + } else if (isColumnType | SingleStoreYear>(column, ['MySqlYear', 'SingleStoreYear'])) { min = 1901; max = 2155; integer = true; @@ -226,9 +276,11 @@ function stringColumnToSchema(column: Column, t: typeof typebox): TSchema { if (isColumnType | SQLiteText>(column, ['PgVarchar', 'SQLiteText'])) { max = column.length; - } else if (isColumnType>(column, ['MySqlVarChar'])) { + } else if ( + isColumnType | SingleStoreVarChar>(column, ['MySqlVarChar', 'SingleStoreVarChar']) + ) { max = column.length ?? CONSTANTS.INT16_UNSIGNED_MAX; - } else if (isColumnType>(column, ['MySqlText'])) { + } else if (isColumnType | SingleStoreText>(column, ['MySqlText', 'SingleStoreText'])) { if (column.textType === 'longtext') { max = CONSTANTS.INT32_UNSIGNED_MAX; } else if (column.textType === 'mediumtext') { @@ -240,7 +292,13 @@ function stringColumnToSchema(column: Column, t: typeof typebox): TSchema { } } - if (isColumnType | MySqlChar>(column, ['PgChar', 'MySqlChar'])) { + if ( + isColumnType | MySqlChar | SingleStoreChar>(column, [ + 'PgChar', + 'MySqlChar', + 'SingleStoreChar', + ]) + ) { max = column.length; fixed = true; } diff --git a/drizzle-typebox/src/column.types.ts b/drizzle-typebox/src/column.types.ts index 7010f234e..2644946c1 100644 --- a/drizzle-typebox/src/column.types.ts +++ b/drizzle-typebox/src/column.types.ts @@ -21,19 +21,26 @@ export type GetTypeboxType< TBaseColumn extends Column | undefined, > = TColumnType extends | 'MySqlTinyInt' + | 'SingleStoreTinyInt' | 'PgSmallInt' | 'PgSmallSerial' | 'MySqlSmallInt' | 'MySqlMediumInt' + | 'SingleStoreSmallInt' + | 'SingleStoreMediumInt' | 'PgInteger' | 'PgSerial' | 'MySqlInt' + | 'SingleStoreInt' | 'PgBigInt53' | 'PgBigSerial53' | 'MySqlBigInt53' | 'MySqlSerial' + | 'SingleStoreBigInt53' + | 'SingleStoreSerial' | 'SQLiteInteger' - | 'MySqlYear' ? t.TInteger + | 'MySqlYear' + | 'SingleStoreYear' ? t.TInteger : TColumnType extends 'PgBinaryVector' ? t.TRegExp : TBaseColumn extends Column ? t.TArray< GetTypeboxType< diff --git a/drizzle-typebox/tests/singlestore.test.ts b/drizzle-typebox/tests/singlestore.test.ts new file mode 100644 index 000000000..f643ab3b7 --- /dev/null +++ b/drizzle-typebox/tests/singlestore.test.ts @@ -0,0 +1,497 @@ +import { Type as t } from '@sinclair/typebox'; +import { type Equal, sql } from 'drizzle-orm'; +import { customType, int, serial, singlestoreSchema, singlestoreTable, text } from 'drizzle-orm/singlestore-core'; +import { test } from 'vitest'; +import { jsonSchema } from '~/column.ts'; +import { CONSTANTS } from '~/constants.ts'; +import { createInsertSchema, createSelectSchema, createUpdateSchema } from '../src'; +import { Expect, expectSchemaShape } from './utils.ts'; + +const intSchema = t.Integer({ + minimum: CONSTANTS.INT32_MIN, + maximum: CONSTANTS.INT32_MAX, +}); +const serialNumberModeSchema = t.Integer({ + minimum: 0, + maximum: Number.MAX_SAFE_INTEGER, +}); +const textSchema = t.String({ maxLength: CONSTANTS.INT16_UNSIGNED_MAX }); + +test('table - select', (tc) => { + const table = singlestoreTable('test', { + id: serial().primaryKey(), + name: text().notNull(), + }); + + const result = createSelectSchema(table); + const expected = t.Object({ id: serialNumberModeSchema, name: textSchema }); + expectSchemaShape(tc, expected).from(result); + Expect>(); +}); + +test('table in schema - select', (tc) => { + const schema = singlestoreSchema('test'); + const table = schema.table('test', { + id: serial().primaryKey(), + name: text().notNull(), + }); + + const result = createSelectSchema(table); + const expected = t.Object({ id: serialNumberModeSchema, name: textSchema }); + expectSchemaShape(tc, expected).from(result); + Expect>(); +}); + +test('table - insert', (tc) => { + const table = singlestoreTable('test', { + id: serial().primaryKey(), + name: text().notNull(), + age: int(), + }); + + const result = createInsertSchema(table); + const expected = t.Object({ + id: t.Optional(serialNumberModeSchema), + name: textSchema, + age: t.Optional(t.Union([intSchema, t.Null()])), + }); + expectSchemaShape(tc, expected).from(result); + Expect>(); +}); + +test('table - update', (tc) => { + const table = singlestoreTable('test', { + id: serial().primaryKey(), + name: text().notNull(), + age: int(), + }); + + const result = createUpdateSchema(table); + const expected = t.Object({ + id: t.Optional(serialNumberModeSchema), + name: t.Optional(textSchema), + age: t.Optional(t.Union([intSchema, t.Null()])), + }); + expectSchemaShape(tc, expected).from(result); + Expect>(); +}); + +// TODO: SingleStore doesn't support views yet. Add these tests when they're added + +// test('view qb - select', (tc) => { +// const table = singlestoreTable('test', { +// id: serial().primaryKey(), +// name: text().notNull(), +// }); +// const view = mysqlView('test').as((qb) => qb.select({ id: table.id, age: sql``.as('age') }).from(table)); + +// const result = createSelectSchema(view); +// const expected = t.Object({ id: serialNumberModeSchema, age: t.Any() }); +// expectSchemaShape(tc, expected).from(result); +// Expect>(); +// }); + +// test('view columns - select', (tc) => { +// const view = mysqlView('test', { +// id: serial().primaryKey(), +// name: text().notNull(), +// }).as(sql``); + +// const result = createSelectSchema(view); +// const expected = t.Object({ id: serialNumberModeSchema, name: textSchema }); +// expectSchemaShape(tc, expected).from(result); +// Expect>(); +// }); + +// test('view with nested fields - select', (tc) => { +// const table = singlestoreTable('test', { +// id: serial().primaryKey(), +// name: text().notNull(), +// }); +// const view = mysqlView('test').as((qb) => +// qb.select({ +// id: table.id, +// nested: { +// name: table.name, +// age: sql``.as('age'), +// }, +// table, +// }).from(table) +// ); + +// const result = createSelectSchema(view); +// const expected = t.Object({ +// id: serialNumberModeSchema, +// nested: t.Object({ name: textSchema, age: t.Any() }), +// table: t.Object({ id: serialNumberModeSchema, name: textSchema }), +// }); +// expectSchemaShape(tc, expected).from(result); +// Expect>(); +// }); + +test('nullability - select', (tc) => { + const table = singlestoreTable('test', { + c1: int(), + c2: int().notNull(), + c3: int().default(1), + c4: int().notNull().default(1), + }); + + const result = createSelectSchema(table); + const expected = t.Object({ + c1: t.Union([intSchema, t.Null()]), + c2: intSchema, + c3: t.Union([intSchema, t.Null()]), + c4: intSchema, + }); + expectSchemaShape(tc, expected).from(result); + Expect>(); +}); + +test('nullability - insert', (tc) => { + const table = singlestoreTable('test', { + c1: int(), + c2: int().notNull(), + c3: int().default(1), + c4: int().notNull().default(1), + c5: int().generatedAlwaysAs(1), + }); + + const result = createInsertSchema(table); + const expected = t.Object({ + c1: t.Optional(t.Union([intSchema, t.Null()])), + c2: intSchema, + c3: t.Optional(t.Union([intSchema, t.Null()])), + c4: t.Optional(intSchema), + }); + expectSchemaShape(tc, expected).from(result); + Expect>(); +}); + +test('nullability - update', (tc) => { + const table = singlestoreTable('test', { + c1: int(), + c2: int().notNull(), + c3: int().default(1), + c4: int().notNull().default(1), + c5: int().generatedAlwaysAs(1), + }); + + const result = createUpdateSchema(table); + const expected = t.Object({ + c1: t.Optional(t.Union([intSchema, t.Null()])), + c2: t.Optional(intSchema), + c3: t.Optional(t.Union([intSchema, t.Null()])), + c4: t.Optional(intSchema), + }); + expectSchemaShape(tc, expected).from(result); + Expect>(); +}); + +test('refine table - select', (tc) => { + const table = singlestoreTable('test', { + c1: int(), + c2: int().notNull(), + c3: int().notNull(), + }); + + const result = createSelectSchema(table, { + c2: (schema) => t.Integer({ minimum: schema.minimum, maximum: 1000 }), + c3: t.Integer({ minimum: 1, maximum: 10 }), + }); + const expected = t.Object({ + c1: t.Union([intSchema, t.Null()]), + c2: t.Integer({ minimum: CONSTANTS.INT32_MIN, maximum: 1000 }), + c3: t.Integer({ minimum: 1, maximum: 10 }), + }); + + expectSchemaShape(tc, expected).from(result); + Expect>(); +}); + +test('refine table - select with custom data type', (tc) => { + const customText = customType({ dataType: () => 'text' }); + const table = singlestoreTable('test', { + c1: int(), + c2: int().notNull(), + c3: int().notNull(), + c4: customText(), + }); + + const customTextSchema = t.String({ minLength: 1, maxLength: 100 }); + const result = createSelectSchema(table, { + c2: (schema) => t.Integer({ minimum: schema.minimum, maximum: 1000 }), + c3: t.Integer({ minimum: 1, maximum: 10 }), + c4: customTextSchema, + }); + const expected = t.Object({ + c1: t.Union([intSchema, t.Null()]), + c2: t.Integer({ minimum: CONSTANTS.INT32_MIN, maximum: 1000 }), + c3: t.Integer({ minimum: 1, maximum: 10 }), + c4: customTextSchema, + }); + + expectSchemaShape(tc, expected).from(result); + Expect>(); +}); + +test('refine table - insert', (tc) => { + const table = singlestoreTable('test', { + c1: int(), + c2: int().notNull(), + c3: int().notNull(), + c4: int().generatedAlwaysAs(1), + }); + + const result = createInsertSchema(table, { + c2: (schema) => t.Integer({ minimum: schema.minimum, maximum: 1000 }), + c3: t.Integer({ minimum: 1, maximum: 10 }), + }); + const expected = t.Object({ + c1: t.Optional(t.Union([intSchema, t.Null()])), + c2: t.Integer({ minimum: CONSTANTS.INT32_MIN, maximum: 1000 }), + c3: t.Integer({ minimum: 1, maximum: 10 }), + }); + expectSchemaShape(tc, expected).from(result); + Expect>(); +}); + +test('refine table - update', (tc) => { + const table = singlestoreTable('test', { + c1: int(), + c2: int().notNull(), + c3: int().notNull(), + c4: int().generatedAlwaysAs(1), + }); + + const result = createUpdateSchema(table, { + c2: (schema) => t.Integer({ minimum: schema.minimum, maximum: 1000 }), + c3: t.Integer({ minimum: 1, maximum: 10 }), + }); + const expected = t.Object({ + c1: t.Optional(t.Union([intSchema, t.Null()])), + c2: t.Optional(t.Integer({ minimum: CONSTANTS.INT32_MIN, maximum: 1000 })), + c3: t.Integer({ minimum: 1, maximum: 10 }), + }); + expectSchemaShape(tc, expected).from(result); + Expect>(); +}); + +// test('refine view - select', (tc) => { +// const table = singlestoreTable('test', { +// c1: int(), +// c2: int(), +// c3: int(), +// c4: int(), +// c5: int(), +// c6: int(), +// }); +// const view = mysqlView('test').as((qb) => +// qb.select({ +// c1: table.c1, +// c2: table.c2, +// c3: table.c3, +// nested: { +// c4: table.c4, +// c5: table.c5, +// c6: table.c6, +// }, +// table, +// }).from(table) +// ); + +// const result = createSelectSchema(view, { +// c2: (schema) => t.Integer({ minimum: schema.minimum, maximum: 1000 }), +// c3: t.Integer({ minimum: 1, maximum: 10 }), +// nested: { +// c5: (schema) => t.Integer({ minimum: schema.minimum, maximum: 1000 }), +// c6: t.Integer({ minimum: 1, maximum: 10 }), +// }, +// table: { +// c2: (schema) => t.Integer({ minimum: schema.minimum, maximum: 1000 }), +// c3: t.Integer({ minimum: 1, maximum: 10 }), +// }, +// }); +// const expected = t.Object({ +// c1: t.Union([intSchema, t.Null()]), +// c2: t.Union([t.Integer({ minimum: CONSTANTS.INT32_MIN, maximum: 1000 }), t.Null()]), +// c3: t.Integer({ minimum: 1, maximum: 10 }), +// nested: t.Object({ +// c4: t.Union([intSchema, t.Null()]), +// c5: t.Union([t.Integer({ minimum: CONSTANTS.INT32_MIN, maximum: 1000 }), t.Null()]), +// c6: t.Integer({ minimum: 1, maximum: 10 }), +// }), +// table: t.Object({ +// c1: t.Union([intSchema, t.Null()]), +// c2: t.Union([t.Integer({ minimum: CONSTANTS.INT32_MIN, maximum: 1000 }), t.Null()]), +// c3: t.Integer({ minimum: 1, maximum: 10 }), +// c4: t.Union([intSchema, t.Null()]), +// c5: t.Union([intSchema, t.Null()]), +// c6: t.Union([intSchema, t.Null()]), +// }), +// }); +// expectSchemaShape(tc, expected).from(result); +// Expect>(); +// }); + +test('all data types', (tc) => { + const table = singlestoreTable('test', ({ + bigint, + binary, + boolean, + char, + date, + datetime, + decimal, + double, + float, + int, + json, + mediumint, + singlestoreEnum, + real, + serial, + smallint, + text, + time, + timestamp, + tinyint, + varchar, + varbinary, + year, + longtext, + mediumtext, + tinytext, + }) => ({ + bigint1: bigint({ mode: 'number' }).notNull(), + bigint2: bigint({ mode: 'bigint' }).notNull(), + bigint3: bigint({ unsigned: true, mode: 'number' }).notNull(), + bigint4: bigint({ unsigned: true, mode: 'bigint' }).notNull(), + binary: binary({ length: 10 }).notNull(), + boolean: boolean().notNull(), + char1: char({ length: 10 }).notNull(), + char2: char({ length: 1, enum: ['a', 'b', 'c'] }).notNull(), + date1: date({ mode: 'date' }).notNull(), + date2: date({ mode: 'string' }).notNull(), + datetime1: datetime({ mode: 'date' }).notNull(), + datetime2: datetime({ mode: 'string' }).notNull(), + decimal1: decimal().notNull(), + decimal2: decimal({ unsigned: true }).notNull(), + double1: double().notNull(), + double2: double({ unsigned: true }).notNull(), + float1: float().notNull(), + float2: float({ unsigned: true }).notNull(), + int1: int().notNull(), + int2: int({ unsigned: true }).notNull(), + json: json().notNull(), + mediumint1: mediumint().notNull(), + mediumint2: mediumint({ unsigned: true }).notNull(), + enum: singlestoreEnum('enum', ['a', 'b', 'c']).notNull(), + real: real().notNull(), + serial: serial().notNull(), + smallint1: smallint().notNull(), + smallint2: smallint({ unsigned: true }).notNull(), + text1: text().notNull(), + text2: text({ enum: ['a', 'b', 'c'] }).notNull(), + time: time().notNull(), + timestamp1: timestamp({ mode: 'date' }).notNull(), + timestamp2: timestamp({ mode: 'string' }).notNull(), + tinyint1: tinyint().notNull(), + tinyint2: tinyint({ unsigned: true }).notNull(), + varchar1: varchar({ length: 10 }).notNull(), + varchar2: varchar({ length: 1, enum: ['a', 'b', 'c'] }).notNull(), + varbinary: varbinary({ length: 10 }).notNull(), + year: year().notNull(), + longtext1: longtext().notNull(), + longtext2: longtext({ enum: ['a', 'b', 'c'] }).notNull(), + mediumtext1: mediumtext().notNull(), + mediumtext2: mediumtext({ enum: ['a', 'b', 'c'] }).notNull(), + tinytext1: tinytext().notNull(), + tinytext2: tinytext({ enum: ['a', 'b', 'c'] }).notNull(), + })); + + const result = createSelectSchema(table); + const expected = t.Object({ + bigint1: t.Integer({ minimum: Number.MIN_SAFE_INTEGER, maximum: Number.MAX_SAFE_INTEGER }), + bigint2: t.BigInt({ minimum: CONSTANTS.INT64_MIN, maximum: CONSTANTS.INT64_MAX }), + bigint3: t.Integer({ minimum: 0, maximum: Number.MAX_SAFE_INTEGER }), + bigint4: t.BigInt({ minimum: 0n, maximum: CONSTANTS.INT64_UNSIGNED_MAX }), + binary: t.String(), + boolean: t.Boolean(), + char1: t.String({ minLength: 10, maxLength: 10 }), + char2: t.Enum({ a: 'a', b: 'b', c: 'c' }), + date1: t.Date(), + date2: t.String(), + datetime1: t.Date(), + datetime2: t.String(), + decimal1: t.String(), + decimal2: t.String(), + double1: t.Number({ minimum: CONSTANTS.INT48_MIN, maximum: CONSTANTS.INT48_MAX }), + double2: t.Number({ minimum: 0, maximum: CONSTANTS.INT48_UNSIGNED_MAX }), + float1: t.Number({ minimum: CONSTANTS.INT24_MIN, maximum: CONSTANTS.INT24_MAX }), + float2: t.Number({ minimum: 0, maximum: CONSTANTS.INT24_UNSIGNED_MAX }), + int1: t.Integer({ minimum: CONSTANTS.INT32_MIN, maximum: CONSTANTS.INT32_MAX }), + int2: t.Integer({ minimum: 0, maximum: CONSTANTS.INT32_UNSIGNED_MAX }), + json: jsonSchema, + mediumint1: t.Integer({ minimum: CONSTANTS.INT24_MIN, maximum: CONSTANTS.INT24_MAX }), + mediumint2: t.Integer({ minimum: 0, maximum: CONSTANTS.INT24_UNSIGNED_MAX }), + enum: t.Enum({ a: 'a', b: 'b', c: 'c' }), + real: t.Number({ minimum: CONSTANTS.INT48_MIN, maximum: CONSTANTS.INT48_MAX }), + serial: t.Integer({ minimum: 0, maximum: Number.MAX_SAFE_INTEGER }), + smallint1: t.Integer({ minimum: CONSTANTS.INT16_MIN, maximum: CONSTANTS.INT16_MAX }), + smallint2: t.Integer({ minimum: 0, maximum: CONSTANTS.INT16_UNSIGNED_MAX }), + text1: t.String({ maxLength: CONSTANTS.INT16_UNSIGNED_MAX }), + text2: t.Enum({ a: 'a', b: 'b', c: 'c' }), + time: t.String(), + timestamp1: t.Date(), + timestamp2: t.String(), + tinyint1: t.Integer({ minimum: CONSTANTS.INT8_MIN, maximum: CONSTANTS.INT8_MAX }), + tinyint2: t.Integer({ minimum: 0, maximum: CONSTANTS.INT8_UNSIGNED_MAX }), + varchar1: t.String({ maxLength: 10 }), + varchar2: t.Enum({ a: 'a', b: 'b', c: 'c' }), + varbinary: t.String(), + year: t.Integer({ minimum: 1901, maximum: 2155 }), + longtext1: t.String({ maxLength: CONSTANTS.INT32_UNSIGNED_MAX }), + longtext2: t.Enum({ a: 'a', b: 'b', c: 'c' }), + mediumtext1: t.String({ maxLength: CONSTANTS.INT24_UNSIGNED_MAX }), + mediumtext2: t.Enum({ a: 'a', b: 'b', c: 'c' }), + tinytext1: t.String({ maxLength: CONSTANTS.INT8_UNSIGNED_MAX }), + tinytext2: t.Enum({ a: 'a', b: 'b', c: 'c' }), + }); + expectSchemaShape(tc, expected).from(result); + Expect>(); +}); + +/* Disallow unknown keys in table refinement - select */ { + const table = singlestoreTable('test', { id: int() }); + // @ts-expect-error + createSelectSchema(table, { unknown: t.String() }); +} + +/* Disallow unknown keys in table refinement - insert */ { + const table = singlestoreTable('test', { id: int() }); + // @ts-expect-error + createInsertSchema(table, { unknown: t.String() }); +} + +/* Disallow unknown keys in table refinement - update */ { + const table = singlestoreTable('test', { id: int() }); + // @ts-expect-error + createUpdateSchema(table, { unknown: t.String() }); +} + +// /* Disallow unknown keys in view qb - select */ { +// const table = singlestoreTable('test', { id: int() }); +// const view = mysqlView('test').as((qb) => qb.select().from(table)); +// const nestedSelect = mysqlView('test').as((qb) => qb.select({ table }).from(table)); +// // @ts-expect-error +// createSelectSchema(view, { unknown: t.String() }); +// // @ts-expect-error +// createSelectSchema(nestedSelect, { table: { unknown: t.String() } }); +// } + +// /* Disallow unknown keys in view columns - select */ { +// const view = mysqlView('test', { id: int() }).as(sql``); +// // @ts-expect-error +// createSelectSchema(view, { unknown: t.String() }); +// } diff --git a/drizzle-valibot/src/column.ts b/drizzle-valibot/src/column.ts index cf90751ef..040dbac21 100644 --- a/drizzle-valibot/src/column.ts +++ b/drizzle-valibot/src/column.ts @@ -37,7 +37,7 @@ import type { PgVarchar, PgVector, } from 'drizzle-orm/pg-core'; -import { +import type { SingleStoreBigInt53, SingleStoreChar, SingleStoreDouble, From cb89775679a335e8700019e0e4878b98479e0e1e Mon Sep 17 00:00:00 2001 From: OleksiiKH0240 Date: Thu, 19 Dec 2024 14:56:51 +0200 Subject: [PATCH 17/32] updated generator versioning --- drizzle-seed/src/services/GeneratorFuncs.ts | 4 +- .../GeneratorVersions/GeneratorsV1.ts | 161 ------------ .../GeneratorVersions/GeneratorsV2.ts | 240 ++++++++++++++++++ drizzle-seed/src/services/Generators.ts | 148 +++-------- drizzle-seed/src/services/SeedService.ts | 16 +- 5 files changed, 288 insertions(+), 281 deletions(-) delete mode 100644 drizzle-seed/src/services/GeneratorVersions/GeneratorsV1.ts create mode 100644 drizzle-seed/src/services/GeneratorVersions/GeneratorsV2.ts diff --git a/drizzle-seed/src/services/GeneratorFuncs.ts b/drizzle-seed/src/services/GeneratorFuncs.ts index 08f36b1dc..65e31fc09 100644 --- a/drizzle-seed/src/services/GeneratorFuncs.ts +++ b/drizzle-seed/src/services/GeneratorFuncs.ts @@ -24,7 +24,6 @@ import { GeneratePostcode, GenerateState, GenerateStreetAdddress, - GenerateString, GenerateTime, GenerateTimestamp, GenerateUUID, @@ -32,6 +31,7 @@ import { GenerateYear, WeightedRandomGenerator, } from './Generators.ts'; +import { GenerateStringV2 } from './GeneratorVersions/GeneratorsV2.ts'; function createGenerator, T>( generatorConstructor: new(params: T) => GeneratorType, @@ -359,7 +359,7 @@ export const generatorsFuncs = { * })); * ``` */ - string: createGenerator(GenerateString), + string: createGenerator(GenerateStringV2), // uniqueString: createGenerator(GenerateUniqueString), /** diff --git a/drizzle-seed/src/services/GeneratorVersions/GeneratorsV1.ts b/drizzle-seed/src/services/GeneratorVersions/GeneratorsV1.ts deleted file mode 100644 index 81655a941..000000000 --- a/drizzle-seed/src/services/GeneratorVersions/GeneratorsV1.ts +++ /dev/null @@ -1,161 +0,0 @@ -import { entityKind } from 'drizzle-orm'; -import prand from 'pure-rand'; -import { AbstractGenerator, GenerateInterval } from '../Generators.ts'; - -export const version = 1; - -export class GenerateIntervalV1 extends GenerateInterval { - static override readonly [entityKind]: string = 'GenerateInterval'; - static override readonly ['version']: number = 1; - override uniqueVersionOfGen = GenerateUniqueIntervalV1; -} - -export class GenerateUniqueIntervalV1 extends AbstractGenerator<{ - isUnique?: boolean; -}> { - static override readonly [entityKind]: string = 'GenerateUniqueInterval'; - - private state: { - rng: prand.RandomGenerator; - intervalSet: Set; - } | undefined; - public override isUnique = true; - - override init({ count, seed }: { count: number; seed: number }) { - const maxUniqueIntervalsNumber = 6 * 13 * 29 * 25 * 61 * 61; - if (count > maxUniqueIntervalsNumber) { - throw new RangeError(`count exceeds max number of unique intervals(${maxUniqueIntervalsNumber})`); - } - - const rng = prand.xoroshiro128plus(seed); - const intervalSet = new Set(); - this.state = { rng, intervalSet }; - } - - generate() { - if (this.state === undefined) { - throw new Error('state is not defined.'); - } - - let yearsNumb: number, - monthsNumb: number, - daysNumb: number, - hoursNumb: number, - minutesNumb: number, - secondsNumb: number; - - let interval = ''; - - for (;;) { - [yearsNumb, this.state.rng] = prand.uniformIntDistribution(0, 5, this.state.rng); - [monthsNumb, this.state.rng] = prand.uniformIntDistribution(0, 12, this.state.rng); - [daysNumb, this.state.rng] = prand.uniformIntDistribution(1, 29, this.state.rng); - [hoursNumb, this.state.rng] = prand.uniformIntDistribution(0, 24, this.state.rng); - [minutesNumb, this.state.rng] = prand.uniformIntDistribution(0, 60, this.state.rng); - [secondsNumb, this.state.rng] = prand.uniformIntDistribution(0, 60, this.state.rng); - - interval = `${yearsNumb === 0 ? '' : `${yearsNumb} years `}` - + `${monthsNumb === 0 ? '' : `${monthsNumb} months `}` - + `${daysNumb === 0 ? '' : `${daysNumb} days `}` - + `${hoursNumb === 0 ? '' : `${hoursNumb} hours `}` - + `${minutesNumb === 0 ? '' : `${minutesNumb} minutes `}` - + `${secondsNumb === 0 ? '' : `${secondsNumb} seconds`}`; - - if (!this.state.intervalSet.has(interval)) { - this.state.intervalSet.add(interval); - break; - } - } - - return interval; - } -} - -export class GenerateStringV1 extends AbstractGenerator<{ - isUnique?: boolean; - arraySize?: number; -}> { - static override readonly [entityKind]: string = 'GenerateString'; - - private state: { rng: prand.RandomGenerator } | undefined; - override uniqueVersionOfGen = GenerateUniqueStringV1; - - override init({ count, seed }: { count: number; seed: number }) { - super.init({ count, seed }); - - const rng = prand.xoroshiro128plus(seed); - this.state = { rng }; - } - - generate() { - if (this.state === undefined) { - throw new Error('state is not defined.'); - } - - const minStringLength = 7; - const maxStringLength = 20; - const stringChars = '1234567890abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ'; - let idx: number, - strLength: number, - currStr: string; - - currStr = ''; - [strLength, this.state.rng] = prand.uniformIntDistribution( - minStringLength, - maxStringLength, - this.state.rng, - ); - for (let j = 0; j < strLength; j++) { - [idx, this.state.rng] = prand.uniformIntDistribution( - 0, - stringChars.length - 1, - this.state.rng, - ); - currStr += stringChars[idx]; - } - return currStr; - } -} - -export class GenerateUniqueStringV1 extends AbstractGenerator<{ isUnique?: boolean }> { - static override readonly [entityKind]: string = 'GenerateUniqueString'; - - private state: { rng: prand.RandomGenerator } | undefined; - public override isUnique = true; - - override init({ seed }: { seed: number }) { - const rng = prand.xoroshiro128plus(seed); - this.state = { rng }; - } - - generate({ i }: { i: number }) { - if (this.state === undefined) { - throw new Error('state is not defined.'); - } - - const minStringLength = 7; - const maxStringLength = 20; - const stringChars = '1234567890abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ'; - let idx: number, - strLength: number; - let currStr: string; - - currStr = ''; - const uniqueStr = i.toString(16); - [strLength, this.state.rng] = prand.uniformIntDistribution( - minStringLength, - maxStringLength - uniqueStr.length, - this.state.rng, - ); - for (let j = 0; j < strLength - uniqueStr.length; j++) { - [idx, this.state.rng] = prand.uniformIntDistribution( - 0, - stringChars.length - 1, - this.state.rng, - ); - currStr += stringChars[idx]; - } - - return currStr.slice(0, 4) + uniqueStr + currStr.slice(4); - } -} diff --git a/drizzle-seed/src/services/GeneratorVersions/GeneratorsV2.ts b/drizzle-seed/src/services/GeneratorVersions/GeneratorsV2.ts new file mode 100644 index 000000000..daedcdc98 --- /dev/null +++ b/drizzle-seed/src/services/GeneratorVersions/GeneratorsV2.ts @@ -0,0 +1,240 @@ +import { entityKind } from 'drizzle-orm'; +import prand from 'pure-rand'; +import { AbstractGenerator, GenerateInterval } from '../Generators.ts'; + +export const version = 2; + +export class GenerateIntervalV2 extends GenerateInterval { + static override readonly [entityKind]: string = 'GenerateInterval'; + static override readonly ['version']: number = 2; + override uniqueVersionOfGen = GenerateUniqueIntervalV2; +} + +export class GenerateUniqueIntervalV2 extends AbstractGenerator<{ + fields?: + | 'year' + | 'month' + | 'day' + | 'hour' + | 'minute' + | 'second' + | 'year to month' + | 'day to hour' + | 'day to minute' + | 'day to second' + | 'hour to minute' + | 'hour to second' + | 'minute to second'; + isUnique?: boolean; +}> { + static override readonly [entityKind]: string = 'GenerateUniqueInterval'; + static override readonly ['version']: number = 2; + + private state: { + rng: prand.RandomGenerator; + fieldsToGenerate: string[]; + intervalSet: Set; + } | undefined; + public override isUnique = true; + private config: { [key: string]: { from: number; to: number } } = { + year: { + from: 0, + to: 5, + }, + month: { + from: 0, + to: 11, + }, + day: { + from: 0, + to: 29, + }, + hour: { + from: 0, + to: 23, + }, + minute: { + from: 0, + to: 59, + }, + second: { + from: 0, + to: 59, + }, + }; + + override init({ count, seed }: { count: number; seed: number }) { + const allFields = ['year', 'month', 'day', 'hour', 'minute', 'second']; + let fieldsToGenerate: string[] = allFields; + + if (this.params.fields !== undefined && this.params.fields?.includes(' to ')) { + const tokens = this.params.fields.split(' to '); + const endIdx = allFields.indexOf(tokens[1]!); + fieldsToGenerate = allFields.slice(0, endIdx + 1); + } else if (this.params.fields !== undefined) { + const endIdx = allFields.indexOf(this.params.fields); + fieldsToGenerate = allFields.slice(0, endIdx + 1); + } + + let maxUniqueIntervalsNumber = 1; + for (const field of fieldsToGenerate) { + const from = this.config[field]!.from, to = this.config[field]!.to; + maxUniqueIntervalsNumber *= from - to + 1; + } + + if (count > maxUniqueIntervalsNumber) { + throw new RangeError(`count exceeds max number of unique intervals(${maxUniqueIntervalsNumber})`); + } + + const rng = prand.xoroshiro128plus(seed); + const intervalSet = new Set(); + this.state = { rng, fieldsToGenerate, intervalSet }; + } + + generate() { + if (this.state === undefined) { + throw new Error('state is not defined.'); + } + + let interval, numb: number; + + for (;;) { + interval = ''; + + for (const field of this.state.fieldsToGenerate) { + const from = this.config[field]!.from, to = this.config[field]!.to; + [numb, this.state.rng] = prand.uniformIntDistribution(from, to, this.state.rng); + interval += `${numb} ${field} `; + } + + if (!this.state.intervalSet.has(interval)) { + this.state.intervalSet.add(interval); + break; + } + } + + return interval; + } +} + +export class GenerateStringV2 extends AbstractGenerator<{ + isUnique?: boolean; + arraySize?: number; +}> { + static override readonly [entityKind]: string = 'GenerateString'; + static override readonly ['version']: number = 2; + + private state: { + rng: prand.RandomGenerator; + minStringLength: number; + maxStringLength: number; + } | undefined; + override uniqueVersionOfGen = GenerateUniqueStringV2; + + override init({ count, seed }: { count: number; seed: number }) { + super.init({ count, seed }); + + let minStringLength = 8; + let maxStringLength = 20; + if (this.stringLength !== undefined) { + maxStringLength = this.stringLength; + if (maxStringLength === 1) minStringLength = maxStringLength; + if (maxStringLength < minStringLength) minStringLength = 1; + } + + const rng = prand.xoroshiro128plus(seed); + this.state = { rng, minStringLength, maxStringLength }; + } + + generate() { + if (this.state === undefined) { + throw new Error('state is not defined.'); + } + + const minStringLength = this.state.minStringLength, + maxStringLength = this.state.maxStringLength; + const stringChars = '1234567890abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ'; + let idx: number, + strLength: number, + currStr: string; + + currStr = ''; + [strLength, this.state.rng] = prand.uniformIntDistribution( + minStringLength, + maxStringLength, + this.state.rng, + ); + for (let j = 0; j < strLength; j++) { + [idx, this.state.rng] = prand.uniformIntDistribution( + 0, + stringChars.length - 1, + this.state.rng, + ); + currStr += stringChars[idx]; + } + return currStr; + } +} + +export class GenerateUniqueStringV2 extends AbstractGenerator<{ isUnique?: boolean }> { + static override readonly [entityKind]: string = 'GenerateUniqueString'; + static override readonly ['version']: number = 2; + + private state: { + rng: prand.RandomGenerator; + minStringLength: number; + maxStringLength: number; + } | undefined; + public override isUnique = true; + + override init({ seed, count }: { seed: number; count: number }) { + const rng = prand.xoroshiro128plus(seed); + + let minStringLength = 8; + let maxStringLength = 20; + // TODO: revise later + if (this.stringLength !== undefined) { + maxStringLength = this.stringLength; + if (maxStringLength === 1 || maxStringLength < minStringLength) minStringLength = maxStringLength; + } + + if (maxStringLength < count.toString(16).length) { + throw new Error( + `You can't generate ${count} unique strings, with a maximum string length of ${maxStringLength}.`, + ); + } + + this.state = { rng, minStringLength, maxStringLength }; + } + + generate({ i }: { i: number }) { + if (this.state === undefined) { + throw new Error('state is not defined.'); + } + + const minStringLength = this.state.minStringLength, + maxStringLength = this.state.maxStringLength; + const stringChars = '1234567890abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ'; + let idx: number, + strLength: number; + let currStr: string; + + currStr = ''; + const uniqueStr = i.toString(16); + [strLength, this.state.rng] = prand.uniformIntDistribution( + minStringLength, + maxStringLength - uniqueStr.length, + this.state.rng, + ); + for (let j = 0; j < strLength - uniqueStr.length; j++) { + [idx, this.state.rng] = prand.uniformIntDistribution( + 0, + stringChars.length - 1, + this.state.rng, + ); + currStr += stringChars[idx]; + } + + return uniqueStr + currStr; + } +} diff --git a/drizzle-seed/src/services/Generators.ts b/drizzle-seed/src/services/Generators.ts index fa01a1e4f..eb5e256e4 100644 --- a/drizzle-seed/src/services/Generators.ts +++ b/drizzle-seed/src/services/Generators.ts @@ -14,7 +14,8 @@ import states, { maxStringLength as maxStateLength } from '../datasets/states.ts import streetSuffix, { maxStringLength as maxStreetSuffixLength } from '../datasets/streetSuffix.ts'; import { fastCartesianProduct, fillTemplate, getWeightedIndices, isObject } from './utils.ts'; -export const version = 2; +export const latestVersion = 2; +export const version = 1; export abstract class AbstractGenerator { static readonly [entityKind]: string = 'AbstractGenerator'; @@ -1299,84 +1300,27 @@ export class GenerateInterval extends AbstractGenerator<{ } } +// has a newer version export class GenerateUniqueInterval extends AbstractGenerator<{ - fields?: - | 'year' - | 'month' - | 'day' - | 'hour' - | 'minute' - | 'second' - | 'year to month' - | 'day to hour' - | 'day to minute' - | 'day to second' - | 'hour to minute' - | 'hour to second' - | 'minute to second'; isUnique?: boolean; }> { static override readonly [entityKind]: string = 'GenerateUniqueInterval'; private state: { rng: prand.RandomGenerator; - fieldsToGenerate: string[]; intervalSet: Set; } | undefined; public override isUnique = true; - private config: { [key: string]: { from: number; to: number } } = { - year: { - from: 0, - to: 5, - }, - month: { - from: 0, - to: 11, - }, - day: { - from: 1, - to: 29, - }, - hour: { - from: 0, - to: 23, - }, - minute: { - from: 0, - to: 59, - }, - second: { - from: 0, - to: 59, - }, - }; override init({ count, seed }: { count: number; seed: number }) { - const allFields = ['year', 'month', 'day', 'hour', 'minute', 'second']; - let fieldsToGenerate: string[] = allFields; - - if (this.params.fields !== undefined && this.params.fields?.includes(' to ')) { - const tokens = this.params.fields.split(' to '); - const endIdx = allFields.indexOf(tokens[1]!); - fieldsToGenerate = allFields.slice(0, endIdx + 1); - } else if (this.params.fields !== undefined) { - const endIdx = allFields.indexOf(this.params.fields); - fieldsToGenerate = allFields.slice(0, endIdx + 1); - } - - let maxUniqueIntervalsNumber = 1; - for (const field of fieldsToGenerate) { - const from = this.config[field]!.from, to = this.config[field]!.to; - maxUniqueIntervalsNumber *= from - to + 1; - } - + const maxUniqueIntervalsNumber = 6 * 13 * 29 * 25 * 61 * 61; if (count > maxUniqueIntervalsNumber) { throw new RangeError(`count exceeds max number of unique intervals(${maxUniqueIntervalsNumber})`); } const rng = prand.xoroshiro128plus(seed); const intervalSet = new Set(); - this.state = { rng, fieldsToGenerate, intervalSet }; + this.state = { rng, intervalSet }; } generate() { @@ -1384,16 +1328,29 @@ export class GenerateUniqueInterval extends AbstractGenerator<{ throw new Error('state is not defined.'); } - let interval, numb: number; + let yearsNumb: number, + monthsNumb: number, + daysNumb: number, + hoursNumb: number, + minutesNumb: number, + secondsNumb: number; - for (;;) { - interval = ''; + let interval = ''; - for (const field of this.state.fieldsToGenerate) { - const from = this.config[field]!.from, to = this.config[field]!.to; - [numb, this.state.rng] = prand.uniformIntDistribution(from, to, this.state.rng); - interval += `${numb} ${field} `; - } + for (;;) { + [yearsNumb, this.state.rng] = prand.uniformIntDistribution(0, 5, this.state.rng); + [monthsNumb, this.state.rng] = prand.uniformIntDistribution(0, 12, this.state.rng); + [daysNumb, this.state.rng] = prand.uniformIntDistribution(1, 29, this.state.rng); + [hoursNumb, this.state.rng] = prand.uniformIntDistribution(0, 24, this.state.rng); + [minutesNumb, this.state.rng] = prand.uniformIntDistribution(0, 60, this.state.rng); + [secondsNumb, this.state.rng] = prand.uniformIntDistribution(0, 60, this.state.rng); + + interval = `${yearsNumb === 0 ? '' : `${yearsNumb} years `}` + + `${monthsNumb === 0 ? '' : `${monthsNumb} months `}` + + `${daysNumb === 0 ? '' : `${daysNumb} days `}` + + `${hoursNumb === 0 ? '' : `${hoursNumb} hours `}` + + `${minutesNumb === 0 ? '' : `${minutesNumb} minutes `}` + + `${secondsNumb === 0 ? '' : `${secondsNumb} seconds`}`; if (!this.state.intervalSet.has(interval)) { this.state.intervalSet.add(interval); @@ -1405,32 +1362,21 @@ export class GenerateUniqueInterval extends AbstractGenerator<{ } } +// has a newer version export class GenerateString extends AbstractGenerator<{ isUnique?: boolean; arraySize?: number; }> { static override readonly [entityKind]: string = 'GenerateString'; - private state: { - rng: prand.RandomGenerator; - minStringLength: number; - maxStringLength: number; - } | undefined; + private state: { rng: prand.RandomGenerator } | undefined; override uniqueVersionOfGen = GenerateUniqueString; override init({ count, seed }: { count: number; seed: number }) { super.init({ count, seed }); - let minStringLength = 8; - let maxStringLength = 20; - if (this.stringLength !== undefined) { - maxStringLength = this.stringLength; - if (maxStringLength === 1) minStringLength = maxStringLength; - if (maxStringLength < minStringLength) minStringLength = 1; - } - const rng = prand.xoroshiro128plus(seed); - this.state = { rng, minStringLength, maxStringLength }; + this.state = { rng }; } generate() { @@ -1438,8 +1384,8 @@ export class GenerateString extends AbstractGenerator<{ throw new Error('state is not defined.'); } - const minStringLength = this.state.minStringLength, - maxStringLength = this.state.maxStringLength; + const minStringLength = 7; + const maxStringLength = 20; const stringChars = '1234567890abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ'; let idx: number, strLength: number, @@ -1463,34 +1409,16 @@ export class GenerateString extends AbstractGenerator<{ } } +// has a newer version export class GenerateUniqueString extends AbstractGenerator<{ isUnique?: boolean }> { static override readonly [entityKind]: string = 'GenerateUniqueString'; - private state: { - rng: prand.RandomGenerator; - minStringLength: number; - maxStringLength: number; - } | undefined; + private state: { rng: prand.RandomGenerator } | undefined; public override isUnique = true; - override init({ seed, count }: { seed: number; count: number }) { + override init({ seed }: { seed: number }) { const rng = prand.xoroshiro128plus(seed); - - let minStringLength = 8; - let maxStringLength = 20; - // TODO: revise later - if (this.stringLength !== undefined) { - maxStringLength = this.stringLength; - if (maxStringLength === 1 || maxStringLength < minStringLength) minStringLength = maxStringLength; - } - - if (maxStringLength < count.toString(16).length) { - throw new Error( - `You can't generate ${count} unique strings, with a maximum string length of ${maxStringLength}.`, - ); - } - - this.state = { rng, minStringLength, maxStringLength }; + this.state = { rng }; } generate({ i }: { i: number }) { @@ -1498,8 +1426,8 @@ export class GenerateUniqueString extends AbstractGenerator<{ isUnique?: boolean throw new Error('state is not defined.'); } - const minStringLength = this.state.minStringLength, - maxStringLength = this.state.maxStringLength; + const minStringLength = 7; + const maxStringLength = 20; const stringChars = '1234567890abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ'; let idx: number, strLength: number; @@ -1521,7 +1449,7 @@ export class GenerateUniqueString extends AbstractGenerator<{ isUnique?: boolean currStr += stringChars[idx]; } - return uniqueStr + currStr; + return currStr.slice(0, 4) + uniqueStr + currStr.slice(4); } } diff --git a/drizzle-seed/src/services/SeedService.ts b/drizzle-seed/src/services/SeedService.ts index 6ceae5167..b55736e4d 100644 --- a/drizzle-seed/src/services/SeedService.ts +++ b/drizzle-seed/src/services/SeedService.ts @@ -15,7 +15,7 @@ import type { Column, Prettify, Relation, Table } from '../types/tables.ts'; import type { AbstractGenerator } from './Generators.ts'; import * as Generators from './Generators.ts'; -import * as GeneratorsV1 from './GeneratorVersions/GeneratorsV1.ts'; +import * as GeneratorsV2 from './GeneratorVersions/GeneratorsV2.ts'; import { equalSets, generateHashFromString } from './utils.ts'; export class SeedService { @@ -39,10 +39,12 @@ export class SeedService { let columnPossibleGenerator: Prettify; let tablePossibleGenerators: Prettify; const customSeed = options?.seed === undefined ? 0 : options.seed; - const version = options?.version === undefined ? Generators.version : options.version; + const version = options?.version === undefined ? Generators.latestVersion : options.version; + if (version < Generators.version || version > Generators.latestVersion) { + throw new Error(`Version should be in range [${Generators.version}, ${Generators.latestVersion}].`); + } // sorting table in order which they will be filled up (tables with foreign keys case) - // relations = relations.filter(rel => rel.type === "one"); const { tablesInOutRelations } = this.getInfoFromRelations(relations); const orderedTablesNames = this.getOrderedTablesList(tablesInOutRelations); tables = tables.sort((table1, table2) => { @@ -416,22 +418,20 @@ export class SeedService { }; selectGeneratorOfVersion = (version: number, generatorEntityKind: string) => { - const GeneratorVersions = [Generators, GeneratorsV1]; + const GeneratorVersions = [GeneratorsV2, Generators]; type GeneratorConstructorT = new(params: any) => AbstractGenerator; let generatorConstructor: GeneratorConstructorT | undefined; for (const gens of GeneratorVersions) { const { version: gensVersion, ...filteredGens } = gens; - if (version > gensVersion) break; + if (gensVersion > version) continue; for (const gen of Object.values(filteredGens)) { const abstractGen = gen as typeof AbstractGenerator; if (abstractGen[entityKind] === generatorEntityKind) { generatorConstructor = abstractGen as unknown as GeneratorConstructorT; - if (abstractGen.version === version) { - return generatorConstructor; - } + return generatorConstructor; } } } From db33c875c09f955403e11a2568654053ca6fe74d Mon Sep 17 00:00:00 2001 From: OleksiiKH0240 Date: Thu, 19 Dec 2024 15:09:56 +0200 Subject: [PATCH 18/32] updated changelogs --- changelogs/drizzle-seed/0.1.4.md | 20 ++++++++++---------- 1 file changed, 10 insertions(+), 10 deletions(-) diff --git a/changelogs/drizzle-seed/0.1.4.md b/changelogs/drizzle-seed/0.1.4.md index 84c3821af..9d3c0a5e2 100644 --- a/changelogs/drizzle-seed/0.1.4.md +++ b/changelogs/drizzle-seed/0.1.4.md @@ -8,24 +8,24 @@ await seed(db, schema, { version: 1 }); Generator versions will change within a single major version of drizzle-seed. -The old version of the generator will become available if the generator has been changed and the output of the new generator does not match the output of the old generator. +The new version of the generator will become available if the generator has been changed and the output of the new generator does not match the output of the old generator. #### For example: -`FirstNameGen` was changed, and old version, `V1`, of this generator became available. +`LastNameGen` was changed, and new version, `V2`, of this generator became available. -Later, `LastNameGen` was changed in the next minor version of drizzle-seed, making `V2` version of this generator available. +Later, `FirstNameGen` was changed in the next minor version of drizzle-seed, making `V3` version of this generator available. | | `V1` | `V2` | `V3(latest)` | | :--------------: | :--------------: | :-------------: | :--------------: | -| **FirstNameGen** | `FirstNameGenV1` | `same as V3` | `FirstNameGenV3` | -| **LastNameGen** | `same as V2` | `LastNameGenV2` | `LastNameGenV3` | +| **LastNameGen** | `LastNameGenV1` | `LastNameGenV2` | | +| **FirstNameGen** | `FirstNameGenV1` | | `FirstNameGenV3` | -If you omit version, `V3` version of generators will be used. +If you omit version, latest version of generators(`V3`) will be used: `FirstNameGen` will use its `V3` version, and `LastNameGen` will use its `V2` version. -If you specify version of 2, `FirstNameGen` will use its `V3` version and `LastNameGen` will use its `V2` version. +If you specify version of 2, `FirstNameGen` will use its `V1` version and `LastNameGen` will use its `V2` version. -If you specify version of 1, `FirstNameGen` will use its `V1` version and `LastNameGen` will use its `V2` version. +If you specify version of 1, `FirstNameGen` will use its `V1` version and `LastNameGen` will use its `V1` version. ## @@ -87,7 +87,7 @@ main(); ## Breaking changes -- Unique `interval` generator was changed, so `1` version of this generator become available. **The latest version is `2`.** +- Unique `interval` generator was changed, so `2` version of this generator become available. **The latest version is `2`.** **Cause:** @@ -98,7 +98,7 @@ However, after inserting the `1 minute 60 second` interval, PostgreSQL database ## -- Both non-unique and unique `string` generators were changed, making version `1` of these generators available. **The latest version is `2`.** +- Both non-unique and unique `string` generators were changed, making version `2` of these generators available. **The latest version is `2`.** **Cause:** From 88bfe72fbf3db73bdc57cd0e4531a8063ceaff7d Mon Sep 17 00:00:00 2001 From: AndriiSherman Date: Fri, 20 Dec 2024 11:17:38 +0200 Subject: [PATCH 19/32] Review --- .../drizzle-seed/{0.1.4.md => 0.2.0.md} | 43 ++++++++++----- drizzle-seed/src/services/GeneratorFuncs.ts | 53 ++++++++++++++++++- drizzle-seed/src/services/Generators.ts | 5 +- drizzle-seed/src/services/SeedService.ts | 22 ++++---- drizzle-seed/src/services/apiVersion.ts | 1 + .../GeneratorsV2.ts => versioning/v2.ts} | 10 ++-- 6 files changed, 99 insertions(+), 35 deletions(-) rename changelogs/drizzle-seed/{0.1.4.md => 0.2.0.md} (63%) create mode 100644 drizzle-seed/src/services/apiVersion.ts rename drizzle-seed/src/services/{GeneratorVersions/GeneratorsV2.ts => versioning/v2.ts} (96%) diff --git a/changelogs/drizzle-seed/0.1.4.md b/changelogs/drizzle-seed/0.2.0.md similarity index 63% rename from changelogs/drizzle-seed/0.1.4.md rename to changelogs/drizzle-seed/0.2.0.md index 9d3c0a5e2..be19a8949 100644 --- a/changelogs/drizzle-seed/0.1.4.md +++ b/changelogs/drizzle-seed/0.2.0.md @@ -1,33 +1,50 @@ -## Improvements +## API updates -- A new parameter, `version`, was added to the `seed` function options, which will control generator versioning. +We are introducing a new parameter, `version`, to the `seed` function options. This parameter, which controls generator versioning, has been added to make it easier to update deterministic generators in the future. Since values should remain consistent after each regeneration, it is crucial to provide a well-designed API for gradual updates ```ts -await seed(db, schema, { version: 1 }); +await seed(db, schema, { version: 2 }); ``` -Generator versions will change within a single major version of drizzle-seed. +#### Example: -The new version of the generator will become available if the generator has been changed and the output of the new generator does not match the output of the old generator. +> This is not an actual API change; it is just an example of how we will proceed with `drizzle-seed` versioning -#### For example: +For example, `lastName` generator was changed, and new version, `V2`, of this generator became available. -`LastNameGen` was changed, and new version, `V2`, of this generator became available. - -Later, `FirstNameGen` was changed in the next minor version of drizzle-seed, making `V3` version of this generator available. +Later, `firstName` generator was changed, making `V3` version of this generator available. | | `V1` | `V2` | `V3(latest)` | | :--------------: | :--------------: | :-------------: | :--------------: | | **LastNameGen** | `LastNameGenV1` | `LastNameGenV2` | | | **FirstNameGen** | `FirstNameGenV1` | | `FirstNameGenV3` | -If you omit version, latest version of generators(`V3`) will be used: `FirstNameGen` will use its `V3` version, and `LastNameGen` will use its `V2` version. -If you specify version of 2, `FirstNameGen` will use its `V1` version and `LastNameGen` will use its `V2` version. +##### Use the `firstName` generator of version 3 and the `lastName` generator of version 2 +```ts +await seed(db, schema); +``` -If you specify version of 1, `FirstNameGen` will use its `V1` version and `LastNameGen` will use its `V1` version. +If you are not ready to use latest generator version right away, you can specify max version to use -## +##### Use the `firstName` generator of version 1 and the `lastName` generator of version 2 +```ts +await seed(db, schema, { version: '2' }); +``` + +##### Use the `firstName` generator of version 1 and the `lastName` generator of version 1. +```ts +await seed(db, schema, { version: '1' }); +``` + +Each update with breaking changes for generators will be documented on our docs and in release notes, explaining which version you should use, if you are not ready to upgrade the way generators works + +## Breaking changes + +### `interval` unique generator was changed and upgraded to v2 +### `string` generators were changed and upgraded to v2 + +## New Features - added `fields` as new parameter in `interval` generator diff --git a/drizzle-seed/src/services/GeneratorFuncs.ts b/drizzle-seed/src/services/GeneratorFuncs.ts index 65e31fc09..2810134f0 100644 --- a/drizzle-seed/src/services/GeneratorFuncs.ts +++ b/drizzle-seed/src/services/GeneratorFuncs.ts @@ -31,7 +31,7 @@ import { GenerateYear, WeightedRandomGenerator, } from './Generators.ts'; -import { GenerateStringV2 } from './GeneratorVersions/GeneratorsV2.ts'; +import { GenerateStringV2 } from './versioning/v2.ts'; function createGenerator, T>( generatorConstructor: new(params: T) => GeneratorType, @@ -735,3 +735,54 @@ export const generatorsFuncs = { */ weightedRandom: createGenerator(WeightedRandomGenerator), }; + +export const generatorsList: (new(params: any) => AbstractGenerator)[] = [ + GenerateBoolean, + GenerateCity, + GenerateCompanyName, + GenerateCountry, + GenerateDate, + GenerateDatetime, + GenerateDefault, + GenerateEmail, + GenerateFirstName, + GenerateFullName, + GenerateInt, + GenerateInterval, + GenerateIntPrimaryKey, + GenerateJobTitle, + GenerateJson, + GenerateLastName, + GenerateLine, + GenerateLoremIpsum, + GenerateNumber, + GeneratePhoneNumber, + GeneratePoint, + GeneratePostcode, + GenerateState, + GenerateStreetAdddress, + GenerateTime, + GenerateTimestamp, + GenerateUUID, + GenerateValuesFromArray, + GenerateYear, + WeightedRandomGenerator, +]; + +// +const generatorsObj = { + [entityKind]: [ + GenerateStreetAdddress, + GenerateUUID, + ], + [entityKind]: [ + GenerateVarchar, + GenerateVarcharV2, + GenerateVarcharV3, + ], + [entityKind]: [ + GenerateVarchar, + GenerateVarcharV2, + GenerateVarcharV3, + ], +}; diff --git a/drizzle-seed/src/services/Generators.ts b/drizzle-seed/src/services/Generators.ts index eb5e256e4..629aff5a3 100644 --- a/drizzle-seed/src/services/Generators.ts +++ b/drizzle-seed/src/services/Generators.ts @@ -14,12 +14,9 @@ import states, { maxStringLength as maxStateLength } from '../datasets/states.ts import streetSuffix, { maxStringLength as maxStreetSuffixLength } from '../datasets/streetSuffix.ts'; import { fastCartesianProduct, fillTemplate, getWeightedIndices, isObject } from './utils.ts'; -export const latestVersion = 2; -export const version = 1; - export abstract class AbstractGenerator { static readonly [entityKind]: string = 'AbstractGenerator'; - static readonly ['version']: number = 2; + static readonly version: number = 1; public isUnique = false; public notNull = false; diff --git a/drizzle-seed/src/services/SeedService.ts b/drizzle-seed/src/services/SeedService.ts index b55736e4d..4ba30ed11 100644 --- a/drizzle-seed/src/services/SeedService.ts +++ b/drizzle-seed/src/services/SeedService.ts @@ -14,9 +14,10 @@ import type { import type { Column, Prettify, Relation, Table } from '../types/tables.ts'; import type { AbstractGenerator } from './Generators.ts'; -import * as Generators from './Generators.ts'; -import * as GeneratorsV2 from './GeneratorVersions/GeneratorsV2.ts'; +import { latestVersion } from './apiVersion.ts'; +// import * as Generators from './Generators.ts'; import { equalSets, generateHashFromString } from './utils.ts'; +import * as GeneratorsV2 from './versioning/v2.ts'; export class SeedService { static readonly [entityKind]: string = 'SeedService'; @@ -39,9 +40,9 @@ export class SeedService { let columnPossibleGenerator: Prettify; let tablePossibleGenerators: Prettify; const customSeed = options?.seed === undefined ? 0 : options.seed; - const version = options?.version === undefined ? Generators.latestVersion : options.version; - if (version < Generators.version || version > Generators.latestVersion) { - throw new Error(`Version should be in range [${Generators.version}, ${Generators.latestVersion}].`); + const version = options?.version === undefined ? latestVersion : options.version; + if (version < 1 || version > latestVersion) { + throw new Error(`Version should be in range [1, ${latestVersion}].`); } // sorting table in order which they will be filled up (tables with foreign keys case) @@ -248,6 +249,8 @@ export class SeedService { ); } + // check if new version exists + columnPossibleGenerator.generator.isUnique = col.isUnique; columnPossibleGenerator.generator.dataType = col.dataType; columnPossibleGenerator.generator.stringLength = col.typeParams.length; @@ -418,15 +421,12 @@ export class SeedService { }; selectGeneratorOfVersion = (version: number, generatorEntityKind: string) => { - const GeneratorVersions = [GeneratorsV2, Generators]; - type GeneratorConstructorT = new(params: any) => AbstractGenerator; + const GeneratorVersions = [...Object.values(GeneratorsV2), ...Object.values(Generators)]; + let generatorConstructor: GeneratorConstructorT | undefined; for (const gens of GeneratorVersions) { - const { version: gensVersion, ...filteredGens } = gens; - if (gensVersion > version) continue; - - for (const gen of Object.values(filteredGens)) { + for (const gen of Object.values(gens)) { const abstractGen = gen as typeof AbstractGenerator; if (abstractGen[entityKind] === generatorEntityKind) { generatorConstructor = abstractGen as unknown as GeneratorConstructorT; diff --git a/drizzle-seed/src/services/apiVersion.ts b/drizzle-seed/src/services/apiVersion.ts new file mode 100644 index 000000000..6cda0267e --- /dev/null +++ b/drizzle-seed/src/services/apiVersion.ts @@ -0,0 +1 @@ +export const latestVersion = 2; diff --git a/drizzle-seed/src/services/GeneratorVersions/GeneratorsV2.ts b/drizzle-seed/src/services/versioning/v2.ts similarity index 96% rename from drizzle-seed/src/services/GeneratorVersions/GeneratorsV2.ts rename to drizzle-seed/src/services/versioning/v2.ts index daedcdc98..96451b2f8 100644 --- a/drizzle-seed/src/services/GeneratorVersions/GeneratorsV2.ts +++ b/drizzle-seed/src/services/versioning/v2.ts @@ -2,11 +2,9 @@ import { entityKind } from 'drizzle-orm'; import prand from 'pure-rand'; import { AbstractGenerator, GenerateInterval } from '../Generators.ts'; -export const version = 2; - export class GenerateIntervalV2 extends GenerateInterval { static override readonly [entityKind]: string = 'GenerateInterval'; - static override readonly ['version']: number = 2; + override readonly version: number = 2; override uniqueVersionOfGen = GenerateUniqueIntervalV2; } @@ -28,7 +26,7 @@ export class GenerateUniqueIntervalV2 extends AbstractGenerator<{ isUnique?: boolean; }> { static override readonly [entityKind]: string = 'GenerateUniqueInterval'; - static override readonly ['version']: number = 2; + override readonly version: number = 2; private state: { rng: prand.RandomGenerator; @@ -122,7 +120,7 @@ export class GenerateStringV2 extends AbstractGenerator<{ arraySize?: number; }> { static override readonly [entityKind]: string = 'GenerateString'; - static override readonly ['version']: number = 2; + override readonly version: number = 2; private state: { rng: prand.RandomGenerator; @@ -178,7 +176,7 @@ export class GenerateStringV2 extends AbstractGenerator<{ export class GenerateUniqueStringV2 extends AbstractGenerator<{ isUnique?: boolean }> { static override readonly [entityKind]: string = 'GenerateUniqueString'; - static override readonly ['version']: number = 2; + override readonly version: number = 2; private state: { rng: prand.RandomGenerator; From e11941e086a79b9e1441e6eb5d5bbedc02300c52 Mon Sep 17 00:00:00 2001 From: OleksiiKH0240 Date: Fri, 20 Dec 2024 11:44:53 +0200 Subject: [PATCH 20/32] changes in generator selection --- drizzle-seed/src/index.ts | 11 +- drizzle-seed/src/services/SeedService.ts | 126 ++++++++++++++--------- 2 files changed, 86 insertions(+), 51 deletions(-) diff --git a/drizzle-seed/src/index.ts b/drizzle-seed/src/index.ts index 248e7ead5..50e40def2 100644 --- a/drizzle-seed/src/index.ts +++ b/drizzle-seed/src/index.ts @@ -570,7 +570,10 @@ const getPostgresInfo = (schema: { [key: string]: PgTable }) => { // might be empty list const newRelations = tableConfig.foreignKeys.map((fk) => { const table = dbToTsTableNamesMap[tableConfig.name] as string; - const refTable = dbToTsTableNamesMap[getTableName(fk.reference().foreignTable)] as string; + const refTableName0 = fk.reference(); + const refTableName1 = refTableName0.foreignTable; + const refTableName2 = getTableName(refTableName1); + const refTable = dbToTsTableNamesMap[refTableName2] as string; const dbToTsColumnNamesMapForRefTable = getDbToTsColumnNamesMap( fk.reference().foreignTable, @@ -625,7 +628,7 @@ const getPostgresInfo = (schema: { [key: string]: PgTable }) => { }; const getTypeParams = (sqlType: string) => { - // get type params and set only type + // get type params const typeParams: Column['typeParams'] = {}; // handle dimensions @@ -642,7 +645,7 @@ const getPostgresInfo = (schema: { [key: string]: PgTable }) => { || sqlType.startsWith('double precision') || sqlType.startsWith('real') ) { - const match = sqlType.match(/\((\d+),(\d+)\)/); + const match = sqlType.match(/\((\d+), *(\d+)\)/); if (match) { typeParams['precision'] = Number(match[1]); typeParams['scale'] = Number(match[2]); @@ -905,7 +908,7 @@ const getMySqlInfo = (schema: { [key: string]: MySqlTable }) => { || sqlType.startsWith('double') || sqlType.startsWith('float') ) { - const match = sqlType.match(/\((\d+),(\d+)\)/); + const match = sqlType.match(/\((\d+), *(\d+)\)/); if (match) { typeParams['precision'] = Number(match[1]); typeParams['scale'] = Number(match[2]); diff --git a/drizzle-seed/src/services/SeedService.ts b/drizzle-seed/src/services/SeedService.ts index b55736e4d..329cc6289 100644 --- a/drizzle-seed/src/services/SeedService.ts +++ b/drizzle-seed/src/services/SeedService.ts @@ -461,12 +461,6 @@ export class SeedService { throw new Error(`column with type ${col.baseColumn!.columnType} is not supported for now.`); } - // const { generatorConstructor } = this.selectGeneratorOfVersion( - // version, - // Generators.GenerateArray[entityKind], - // ); - - // const generator = new generatorConstructor!({ baseColumnGen, size: col.size }); const generator = new Generators.GenerateArray({ baseColumnGen, size: col.size }); return generator; @@ -569,11 +563,23 @@ export class SeedService { // NUMBER(real, double, decimal, numeric) if ( - col.columnType === 'real' - || col.columnType === 'double precision' - || col.columnType === 'decimal' - || col.columnType === 'numeric' + col.columnType.startsWith('real') + || col.columnType.startsWith('double precision') + || col.columnType.startsWith('decimal') + || col.columnType.startsWith('numeric') ) { + if (col.typeParams.precision !== undefined) { + const precision = col.typeParams.precision; + const scale = col.typeParams.scale === undefined ? 0 : col.typeParams.scale; + + const maxAbsoluteValue = Math.pow(10, precision - scale) - Math.pow(10, -scale); + const generator = new Generators.GenerateNumber({ + minValue: -maxAbsoluteValue, + maxValue: maxAbsoluteValue, + precision: Math.pow(10, scale), + }); + return generator; + } const generator = new Generators.GenerateNumber({}); return generator; @@ -792,11 +798,25 @@ export class SeedService { // NUMBER(real, double, decimal, float) if ( - col.columnType === 'real' - || col.columnType === 'double' - || col.columnType === 'decimal' - || col.columnType === 'float' + col.columnType.startsWith('real') + || col.columnType.startsWith('double') + || col.columnType.startsWith('decimal') + || col.columnType.startsWith('float') + || col.columnType.startsWith('numeric') ) { + if (col.typeParams.precision !== undefined) { + const precision = col.typeParams.precision; + const scale = col.typeParams.scale === undefined ? 0 : col.typeParams.scale; + + const maxAbsoluteValue = Math.pow(10, precision - scale) - Math.pow(10, -scale); + const generator = new Generators.GenerateNumber({ + minValue: -maxAbsoluteValue, + maxValue: maxAbsoluteValue, + precision: Math.pow(10, scale), + }); + return generator; + } + const generator = new Generators.GenerateNumber({}); return generator; } @@ -805,10 +825,10 @@ export class SeedService { if ( (col.columnType === 'text' || col.columnType === 'blob' - || col.columnType.includes('char') - || col.columnType.includes('varchar') - || col.columnType.includes('binary') - || col.columnType.includes('varbinary')) + || col.columnType.startsWith('char') + || col.columnType.startsWith('varchar') + || col.columnType.startsWith('binary') + || col.columnType.startsWith('varbinary')) && table.primaryKeys.includes(col.name) ) { const generator = new Generators.GenerateUniqueString({}); @@ -818,10 +838,10 @@ export class SeedService { if ( (col.columnType === 'text' || col.columnType === 'blob' - || col.columnType.includes('char') - || col.columnType.includes('varchar') - || col.columnType.includes('binary') - || col.columnType.includes('varbinary')) + || col.columnType.startsWith('char') + || col.columnType.startsWith('varchar') + || col.columnType.startsWith('binary') + || col.columnType.startsWith('varbinary')) && col.name.toLowerCase().includes('name') ) { const generator = new Generators.GenerateFirstName({}); @@ -831,10 +851,10 @@ export class SeedService { if ( (col.columnType === 'text' || col.columnType === 'blob' - || col.columnType.includes('char') - || col.columnType.includes('varchar') - || col.columnType.includes('binary') - || col.columnType.includes('varbinary')) + || col.columnType.startsWith('char') + || col.columnType.startsWith('varchar') + || col.columnType.startsWith('binary') + || col.columnType.startsWith('varbinary')) && col.name.toLowerCase().includes('email') ) { const generator = new Generators.GenerateEmail({}); @@ -844,10 +864,10 @@ export class SeedService { if ( col.columnType === 'text' || col.columnType === 'blob' - || col.columnType.includes('char') - || col.columnType.includes('varchar') - || col.columnType.includes('binary') - || col.columnType.includes('varbinary') + || col.columnType.startsWith('char') + || col.columnType.startsWith('varchar') + || col.columnType.startsWith('binary') + || col.columnType.startsWith('varbinary') ) { const generator = new Generators.GenerateString({}); return generator; @@ -949,7 +969,6 @@ export class SeedService { if ( col.columnType === 'integer' - || col.columnType === 'numeric' || (col.dataType === 'bigint' && col.columnType === 'blob') ) { const generator = new Generators.GenerateInt({}); @@ -957,16 +976,29 @@ export class SeedService { } // number section ------------------------------------------------------------------------------------ - if (col.columnType === 'real' || col.columnType === 'numeric') { + if (col.columnType.startsWith('real') || col.columnType.startsWith('numeric')) { + if (col.typeParams.precision !== undefined) { + const precision = col.typeParams.precision; + const scale = col.typeParams.scale === undefined ? 0 : col.typeParams.scale; + + const maxAbsoluteValue = Math.pow(10, precision - scale) - Math.pow(10, -scale); + const generator = new Generators.GenerateNumber({ + minValue: -maxAbsoluteValue, + maxValue: maxAbsoluteValue, + precision: Math.pow(10, scale), + }); + return generator; + } + const generator = new Generators.GenerateNumber({}); return generator; } // string section ------------------------------------------------------------------------------------ if ( - (col.columnType === 'text' - || col.columnType === 'numeric' - || col.columnType === 'blob') + (col.columnType.startsWith('text') + || col.columnType.startsWith('numeric') + || col.columnType.startsWith('blob')) && table.primaryKeys.includes(col.name) ) { const generator = new Generators.GenerateUniqueString({}); @@ -974,9 +1006,9 @@ export class SeedService { } if ( - (col.columnType === 'text' - || col.columnType === 'numeric' - || col.columnType === 'blob') + (col.columnType.startsWith('text') + || col.columnType.startsWith('numeric') + || col.columnType.startsWith('blob')) && col.name.toLowerCase().includes('name') ) { const generator = new Generators.GenerateFirstName({}); @@ -984,9 +1016,9 @@ export class SeedService { } if ( - (col.columnType === 'text' - || col.columnType === 'numeric' - || col.columnType === 'blob') + (col.columnType.startsWith('text') + || col.columnType.startsWith('numeric') + || col.columnType.startsWith('blob')) && col.name.toLowerCase().includes('email') ) { const generator = new Generators.GenerateEmail({}); @@ -994,18 +1026,18 @@ export class SeedService { } if ( - col.columnType === 'text' - || col.columnType === 'numeric' - || col.columnType === 'blob' - || col.columnType === 'blobbuffer' + col.columnType.startsWith('text') + || col.columnType.startsWith('numeric') + || col.columnType.startsWith('blob') + || col.columnType.startsWith('blobbuffer') ) { const generator = new Generators.GenerateString({}); return generator; } if ( - (col.columnType === 'text' && col.dataType === 'json') - || (col.columnType === 'blob' && col.dataType === 'json') + (col.columnType.startsWith('text') && col.dataType === 'json') + || (col.columnType.startsWith('blob') && col.dataType === 'json') ) { const generator = new Generators.GenerateJson({}); return generator; From ec5d35eb459e0330894074889e0fdcd4da091158 Mon Sep 17 00:00:00 2001 From: OleksiiKH0240 Date: Fri, 20 Dec 2024 20:59:10 +0200 Subject: [PATCH 21/32] updated generator versioning --- drizzle-seed/src/index.ts | 5 +- drizzle-seed/src/services/GeneratorFuncs.ts | 227 +++++++++++--- drizzle-seed/src/services/Generators.ts | 238 ++++++++------ drizzle-seed/src/services/SeedService.ts | 295 +++++++++--------- drizzle-seed/src/services/versioning/v2.ts | 22 +- .../tests/benchmarks/generatorsBenchmark.ts | 8 +- .../tests/pg/allDataTypesTest/pgSchema.ts | 46 +-- 7 files changed, 514 insertions(+), 327 deletions(-) diff --git a/drizzle-seed/src/index.ts b/drizzle-seed/src/index.ts index 50e40def2..48aad2973 100644 --- a/drizzle-seed/src/index.ts +++ b/drizzle-seed/src/index.ts @@ -1,4 +1,5 @@ -import { entityKind, getTableName, is, sql } from 'drizzle-orm'; +/* eslint-disable drizzle-internal/require-entity-kind */ +import { getTableName, is, sql } from 'drizzle-orm'; import type { MySqlColumn, MySqlSchema } from 'drizzle-orm/mysql-core'; import { getTableConfig as getMysqlTableConfig, MySqlDatabase, MySqlTable } from 'drizzle-orm/mysql-core'; @@ -131,7 +132,7 @@ class SeedPromise< [key: string]: PgTable | PgSchema | MySqlTable | MySqlSchema | SQLiteTable; }, > implements Promise { - static readonly [entityKind]: string = 'SeedPromise'; + static readonly entityKind: string = 'SeedPromise'; [Symbol.toStringTag] = 'SeedPromise'; diff --git a/drizzle-seed/src/services/GeneratorFuncs.ts b/drizzle-seed/src/services/GeneratorFuncs.ts index 2810134f0..be715a33b 100644 --- a/drizzle-seed/src/services/GeneratorFuncs.ts +++ b/drizzle-seed/src/services/GeneratorFuncs.ts @@ -1,5 +1,6 @@ import type { AbstractGenerator } from './Generators.ts'; import { + GenerateArray, GenerateBoolean, GenerateCity, GenerateCompanyName, @@ -8,6 +9,7 @@ import { GenerateDatetime, GenerateDefault, GenerateEmail, + GenerateEnum, GenerateFirstName, GenerateFullName, GenerateInt, @@ -22,16 +24,34 @@ import { GeneratePhoneNumber, GeneratePoint, GeneratePostcode, + GenerateSelfRelationsValuesFromArray, GenerateState, - GenerateStreetAdddress, + GenerateStreetAddress, + GenerateString, GenerateTime, GenerateTimestamp, + GenerateUniqueCity, + GenerateUniqueCompanyName, + GenerateUniqueCountry, + GenerateUniqueFirstName, + GenerateUniqueFullName, + GenerateUniqueInt, + GenerateUniqueInterval, + GenerateUniqueLastName, + GenerateUniqueLine, + GenerateUniqueNumber, + GenerateUniquePoint, + GenerateUniquePostcode, + GenerateUniqueStreetAddress, + GenerateUniqueString, GenerateUUID, GenerateValuesFromArray, + GenerateWeightedCount, GenerateYear, + HollowGenerator, WeightedRandomGenerator, } from './Generators.ts'; -import { GenerateStringV2 } from './versioning/v2.ts'; +import { GenerateStringV2, GenerateUniqueIntervalV2, GenerateUniqueStringV2 } from './versioning/v2.ts'; function createGenerator, T>( generatorConstructor: new(params: T) => GeneratorType, @@ -359,7 +379,7 @@ export const generatorsFuncs = { * })); * ``` */ - string: createGenerator(GenerateStringV2), + string: createGenerator(GenerateString), // uniqueString: createGenerator(GenerateUniqueString), /** @@ -550,8 +570,8 @@ export const generatorsFuncs = { * })); * ``` */ - streetAddress: createGenerator(GenerateStreetAdddress), - // uniqueStreetAddress: createGenerator(GenerateUniqueStreetAdddress), + streetAddress: createGenerator(GenerateStreetAddress), + // uniqueStreetAddress: createGenerator(GenerateUniqueStreetAddress), /** * generates job titles. @@ -736,53 +756,162 @@ export const generatorsFuncs = { weightedRandom: createGenerator(WeightedRandomGenerator), }; -export const generatorsList: (new(params: any) => AbstractGenerator)[] = [ - GenerateBoolean, - GenerateCity, - GenerateCompanyName, - GenerateCountry, - GenerateDate, - GenerateDatetime, - GenerateDefault, - GenerateEmail, - GenerateFirstName, - GenerateFullName, - GenerateInt, - GenerateInterval, - GenerateIntPrimaryKey, - GenerateJobTitle, - GenerateJson, - GenerateLastName, - GenerateLine, - GenerateLoremIpsum, - GenerateNumber, - GeneratePhoneNumber, - GeneratePoint, - GeneratePostcode, - GenerateState, - GenerateStreetAdddress, - GenerateTime, - GenerateTimestamp, - GenerateUUID, - GenerateValuesFromArray, - GenerateYear, - WeightedRandomGenerator, -]; +// TODO: revise +// so far, version changes don’t affect generator parameters. +export const generatorsFuncsV2 = { ...generatorsFuncs }; -// -const generatorsObj = { - [entityKind]: [ - GenerateStreetAdddress, +export const generatorsMap = { + HollowGenerator: [ + HollowGenerator, + ], + GenerateDefault: [ + GenerateDefault, + ], + GenerateValuesFromArray: [ + GenerateValuesFromArray, + ], + GenerateSelfRelationsValuesFromArray: [ + GenerateSelfRelationsValuesFromArray, + ], + GenerateIntPrimaryKey: [ + GenerateIntPrimaryKey, + ], + GenerateNumber: [ + GenerateNumber, + ], + GenerateUniqueNumber: [ + GenerateUniqueNumber, + ], + GenerateInt: [ + GenerateInt, + ], + GenerateUniqueInt: [ + GenerateUniqueInt, + ], + GenerateBoolean: [ + GenerateBoolean, + ], + GenerateDate: [ + GenerateDate, + ], + GenerateTime: [ + GenerateTime, + ], + GenerateTimestamp: [ + GenerateTimestamp, + ], + GenerateDatetime: [ + GenerateDatetime, + ], + GenerateYear: [ + GenerateYear, + ], + GenerateJson: [ + GenerateJson, + ], + GenerateEnum: [ + GenerateEnum, + ], + GenerateInterval: [ + GenerateInterval, + ], + GenerateUniqueInterval: [ + GenerateUniqueInterval, + GenerateUniqueIntervalV2, + ], + GenerateString: [ + GenerateString, + GenerateStringV2, + ], + GenerateUniqueString: [ + GenerateUniqueString, + GenerateUniqueStringV2, + ], + GenerateUUID: [ GenerateUUID, ], - [entityKind]: [ - GenerateVarchar, - GenerateVarcharV2, - GenerateVarcharV3, + GenerateFirstName: [ + GenerateFirstName, + ], + GenerateUniqueFirstName: [ + GenerateUniqueFirstName, + ], + GenerateLastName: [ + GenerateLastName, + ], + GenerateUniqueLastName: [ + GenerateUniqueLastName, + ], + GenerateFullName: [ + GenerateFullName, + ], + GenerateUniqueFullName: [ + GenerateUniqueFullName, + ], + GenerateEmail: [ + GenerateEmail, + ], + GeneratePhoneNumber: [ + GeneratePhoneNumber, + ], + GenerateCountry: [ + GenerateCountry, + ], + GenerateUniqueCountry: [ + GenerateUniqueCountry, + ], + GenerateCity: [ + GenerateCity, + ], + GenerateUniqueCity: [ + GenerateUniqueCity, + ], + GenerateStreetAddress: [ + GenerateStreetAddress, + ], + GenerateUniqueStreetAddress: [ + GenerateUniqueStreetAddress, + ], + GenerateJobTitle: [ + GenerateJobTitle, + ], + GeneratePostcode: [ + GeneratePostcode, + ], + GenerateUniquePostcode: [ + GenerateUniquePostcode, + ], + GenerateState: [ + GenerateState, + ], + GenerateCompanyName: [ + GenerateCompanyName, + ], + GenerateUniqueCompanyName: [ + GenerateUniqueCompanyName, + ], + GenerateLoremIpsum: [ + GenerateLoremIpsum, + ], + GeneratePoint: [ + GeneratePoint, + ], + GenerateUniquePoint: [ + GenerateUniquePoint, + ], + GenerateLine: [ + GenerateLine, + ], + GenerateUniqueLine: [ + GenerateUniqueLine, + ], + WeightedRandomGenerator: [ + WeightedRandomGenerator, + ], + GenerateArray: [ + GenerateArray, ], - [entityKind]: [ - GenerateVarchar, - GenerateVarcharV2, - GenerateVarcharV3, + GenerateWeightedCount: [ + GenerateWeightedCount, ], }; diff --git a/drizzle-seed/src/services/Generators.ts b/drizzle-seed/src/services/Generators.ts index 629aff5a3..fff543739 100644 --- a/drizzle-seed/src/services/Generators.ts +++ b/drizzle-seed/src/services/Generators.ts @@ -1,4 +1,4 @@ -import { entityKind } from 'drizzle-orm'; +/* eslint-disable drizzle-internal/require-entity-kind */ import prand from 'pure-rand'; import adjectives, { maxStringLength as maxAdjectiveLength } from '../datasets/adjectives.ts'; import cityNames, { maxStringLength as maxCityNameLength } from '../datasets/cityNames.ts'; @@ -15,22 +15,39 @@ import streetSuffix, { maxStringLength as maxStreetSuffixLength } from '../datas import { fastCartesianProduct, fillTemplate, getWeightedIndices, isObject } from './utils.ts'; export abstract class AbstractGenerator { - static readonly [entityKind]: string = 'AbstractGenerator'; + static readonly entityKind: string = 'AbstractGenerator'; static readonly version: number = 1; public isUnique = false; public notNull = false; + + // param for generators which have a unique version of themselves public uniqueVersionOfGen?: new(params: T) => AbstractGenerator; + public dataType?: string; public timeSpent?: number; + + // public arraySize?: number; public baseColumnDataType?: string; + + // param for text-like generators public stringLength?: number; + // params for GenerateValuesFromArray + public weightedCountSeed?: number | undefined; + public maxRepeatedValuesCount?: number | { weight: number; count: number | number[] }[] | undefined; + + // param for GenerateIntP + constructor(public params: T) {} init(params: { count: number | { weight: number; count: number | number[] }[]; seed: number }): void; init() { + this.updateParams(); + } + + updateParams() { if ((this.params as any).arraySize !== undefined) { this.arraySize = (this.params as any).arraySize; } @@ -48,10 +65,11 @@ export abstract class AbstractGenerator { getEntityKind(): string { const constructor = this.constructor as typeof AbstractGenerator; - return constructor[entityKind]; + return constructor.entityKind; } - replaceIfUnique({ count, seed }: { count: number; seed: number }) { + replaceIfUnique() { + this.updateParams(); if ( this.uniqueVersionOfGen !== undefined && this.isUnique === true @@ -59,10 +77,7 @@ export abstract class AbstractGenerator { const uniqueGen = new this.uniqueVersionOfGen({ ...this.params, }); - uniqueGen.init({ - count, - seed, - }); + uniqueGen.isUnique = this.isUnique; uniqueGen.dataType = this.dataType; @@ -71,9 +86,10 @@ export abstract class AbstractGenerator { return; } - replaceIfArray({ count, seed }: { count: number; seed: number }) { + replaceIfArray() { + this.updateParams(); if (!(this.getEntityKind() === 'GenerateArray') && this.arraySize !== undefined) { - const uniqueGen = this.replaceIfUnique({ count, seed }); + const uniqueGen = this.replaceIfUnique(); const baseColumnGen = uniqueGen === undefined ? this : uniqueGen; baseColumnGen.dataType = this.baseColumnDataType; const arrayGen = new GenerateArray( @@ -82,7 +98,6 @@ export abstract class AbstractGenerator { size: this.arraySize, }, ); - arrayGen.init({ count, seed }); return arrayGen; } @@ -93,7 +108,7 @@ export abstract class AbstractGenerator { // Generators Classes ----------------------------------------------------------------------------------------------------------------------- export class GenerateArray extends AbstractGenerator<{ baseColumnGen: AbstractGenerator; size?: number }> { - static override readonly [entityKind]: string = 'GenerateArray'; + static override readonly entityKind: string = 'GenerateArray'; public override arraySize = 10; override init({ count, seed }: { count: number; seed: number }) { @@ -113,7 +128,7 @@ export class GenerateArray extends AbstractGenerator<{ baseColumnGen: AbstractGe } export class GenerateWeightedCount extends AbstractGenerator<{}> { - static override readonly [entityKind]: string = 'GenerateWeightedCount'; + static override readonly entityKind: string = 'GenerateWeightedCount'; private state: { rng: prand.RandomGenerator; @@ -151,7 +166,7 @@ export class GenerateWeightedCount extends AbstractGenerator<{}> { } export class HollowGenerator extends AbstractGenerator<{}> { - static override readonly [entityKind]: string = 'HollowGenerator'; + static override readonly entityKind: string = 'HollowGenerator'; override init() {} @@ -162,7 +177,7 @@ export class GenerateDefault extends AbstractGenerator<{ defaultValue: unknown; arraySize?: number; }> { - static override readonly [entityKind]: string = 'GenerateDefault'; + static override readonly entityKind: string = 'GenerateDefault'; generate() { return this.params.defaultValue; @@ -178,10 +193,8 @@ export class GenerateValuesFromArray extends AbstractGenerator< arraySize?: number; } > { - static override readonly [entityKind]: string = 'GenerateValuesFromArray'; + static override readonly entityKind: string = 'GenerateValuesFromArray'; - public weightedCountSeed: number | undefined = undefined; - public maxRepeatedValuesCount?: number | { weight: number; count: number | number[] }[] = undefined; private state: { rng: prand.RandomGenerator; values: @@ -389,7 +402,7 @@ export class GenerateValuesFromArray extends AbstractGenerator< } export class GenerateSelfRelationsValuesFromArray extends AbstractGenerator<{ values: (number | string | boolean)[] }> { - static override readonly [entityKind]: string = 'GenerateSelfRelationsValuesFromArray'; + static override readonly entityKind: string = 'GenerateSelfRelationsValuesFromArray'; private state: { rng: prand.RandomGenerator; @@ -427,7 +440,7 @@ export class GenerateSelfRelationsValuesFromArray extends AbstractGenerator<{ va } export class GenerateIntPrimaryKey extends AbstractGenerator<{}> { - static override readonly [entityKind]: string = 'GenerateIntPrimaryKey'; + static override readonly entityKind: string = 'GenerateIntPrimaryKey'; public maxValue?: number | bigint; @@ -455,7 +468,7 @@ export class GenerateNumber extends AbstractGenerator< arraySize?: number; } > { - static override readonly [entityKind]: string = 'GenerateNumber'; + static override readonly entityKind: string = 'GenerateNumber'; private state: { rng: prand.RandomGenerator; @@ -510,7 +523,7 @@ export class GenerateUniqueNumber extends AbstractGenerator< isUnique?: boolean; } > { - static override readonly [entityKind]: string = 'GenerateUniqueNumber'; + static override readonly entityKind: string = 'GenerateUniqueNumber'; private state: { genUniqueIntObj: GenerateUniqueInt; @@ -562,7 +575,7 @@ export class GenerateInt extends AbstractGenerator<{ isUnique?: boolean; arraySize?: number; }> { - static override readonly [entityKind]: string = 'GenerateInt'; + static override readonly entityKind: string = 'GenerateInt'; private state: { rng: prand.RandomGenerator; @@ -630,7 +643,7 @@ export class GenerateUniqueInt extends AbstractGenerator<{ maxValue?: number | bigint; isUnique?: boolean; }> { - static override readonly [entityKind]: string = 'GenerateUniqueInt'; + static override readonly entityKind: string = 'GenerateUniqueInt'; public genMaxRepeatedValuesCount: GenerateDefault | GenerateWeightedCount | undefined; public skipCheck?: boolean = false; @@ -798,7 +811,7 @@ export class GenerateUniqueInt extends AbstractGenerator<{ } export class GenerateBoolean extends AbstractGenerator<{ arraySize?: number }> { - static override readonly [entityKind]: string = 'GenerateBoolean'; + static override readonly entityKind: string = 'GenerateBoolean'; private state: { rng: prand.RandomGenerator; @@ -829,7 +842,7 @@ export class GenerateDate extends AbstractGenerator<{ maxDate?: string | Date; arraySize?: number; }> { - static override readonly [entityKind]: string = 'GenerateDate'; + static override readonly entityKind: string = 'GenerateDate'; private state: { rng: prand.RandomGenerator; @@ -891,7 +904,7 @@ export class GenerateDate extends AbstractGenerator<{ } } export class GenerateTime extends AbstractGenerator<{ arraySize?: number }> { - static override readonly [entityKind]: string = 'GenerateTime'; + static override readonly entityKind: string = 'GenerateTime'; private state: { rng: prand.RandomGenerator; @@ -927,7 +940,7 @@ export class GenerateTime extends AbstractGenerator<{ arraySize?: number }> { } } export class GenerateTimestampInt extends AbstractGenerator<{ unitOfTime?: 'seconds' | 'milliseconds' }> { - static override readonly [entityKind]: string = 'GenerateTimestampInt'; + static override readonly entityKind: string = 'GenerateTimestampInt'; private state: { generateTimestampObj: GenerateTimestamp; @@ -960,7 +973,7 @@ export class GenerateTimestampInt extends AbstractGenerator<{ unitOfTime?: 'seco } export class GenerateTimestamp extends AbstractGenerator<{ arraySize?: number }> { - static override readonly [entityKind]: string = 'GenerateTimestamp'; + static override readonly entityKind: string = 'GenerateTimestamp'; private state: { rng: prand.RandomGenerator; @@ -1004,7 +1017,7 @@ export class GenerateTimestamp extends AbstractGenerator<{ arraySize?: number }> } export class GenerateDatetime extends AbstractGenerator<{ arraySize?: number }> { - static override readonly [entityKind]: string = 'GenerateDatetime'; + static override readonly entityKind: string = 'GenerateDatetime'; private state: { rng: prand.RandomGenerator; @@ -1048,7 +1061,7 @@ export class GenerateDatetime extends AbstractGenerator<{ arraySize?: number }> } export class GenerateYear extends AbstractGenerator<{ arraySize?: number }> { - static override readonly [entityKind]: string = 'GenerateYear'; + static override readonly entityKind: string = 'GenerateYear'; private state: { rng: prand.RandomGenerator; @@ -1083,7 +1096,7 @@ export class GenerateYear extends AbstractGenerator<{ arraySize?: number }> { } export class GenerateJson extends AbstractGenerator<{ arraySize?: number }> { - static override readonly [entityKind]: string = 'GenerateJson'; + static override readonly entityKind: string = 'GenerateJson'; private state: { emailGeneratorObj: GenerateEmail; @@ -1187,7 +1200,7 @@ export class GenerateJson extends AbstractGenerator<{ arraySize?: number }> { } export class GenerateEnum extends AbstractGenerator<{ enumValues: (string | number | boolean)[] }> { - static override readonly [entityKind]: string = 'GenerateEnum'; + static override readonly entityKind: string = 'GenerateEnum'; private state: { enumValuesGenerator: GenerateValuesFromArray; @@ -1227,7 +1240,7 @@ export class GenerateInterval extends AbstractGenerator<{ isUnique?: boolean; arraySize?: number; }> { - static override readonly [entityKind]: string = 'GenerateInterval'; + static override readonly entityKind: string = 'GenerateInterval'; private state: { rng: prand.RandomGenerator; @@ -1299,25 +1312,83 @@ export class GenerateInterval extends AbstractGenerator<{ // has a newer version export class GenerateUniqueInterval extends AbstractGenerator<{ + fields?: + | 'year' + | 'month' + | 'day' + | 'hour' + | 'minute' + | 'second' + | 'year to month' + | 'day to hour' + | 'day to minute' + | 'day to second' + | 'hour to minute' + | 'hour to second' + | 'minute to second'; isUnique?: boolean; }> { - static override readonly [entityKind]: string = 'GenerateUniqueInterval'; + static override readonly 'entityKind': string = 'GenerateUniqueInterval'; private state: { rng: prand.RandomGenerator; + fieldsToGenerate: string[]; intervalSet: Set; } | undefined; public override isUnique = true; + private config: { [key: string]: { from: number; to: number } } = { + year: { + from: 0, + to: 5, + }, + month: { + from: 0, + to: 12, + }, + day: { + from: 1, + to: 29, + }, + hour: { + from: 0, + to: 24, + }, + minute: { + from: 0, + to: 60, + }, + second: { + from: 0, + to: 60, + }, + }; override init({ count, seed }: { count: number; seed: number }) { - const maxUniqueIntervalsNumber = 6 * 13 * 29 * 25 * 61 * 61; + const allFields = ['year', 'month', 'day', 'hour', 'minute', 'second']; + let fieldsToGenerate: string[] = allFields; + + if (this.params.fields !== undefined && this.params.fields?.includes(' to ')) { + const tokens = this.params.fields.split(' to '); + const endIdx = allFields.indexOf(tokens[1]!); + fieldsToGenerate = allFields.slice(0, endIdx + 1); + } else if (this.params.fields !== undefined) { + const endIdx = allFields.indexOf(this.params.fields); + fieldsToGenerate = allFields.slice(0, endIdx + 1); + } + + let maxUniqueIntervalsNumber = 1; + for (const field of fieldsToGenerate) { + const from = this.config[field]!.from, to = this.config[field]!.to; + maxUniqueIntervalsNumber *= from - to + 1; + } + if (count > maxUniqueIntervalsNumber) { throw new RangeError(`count exceeds max number of unique intervals(${maxUniqueIntervalsNumber})`); } const rng = prand.xoroshiro128plus(seed); const intervalSet = new Set(); - this.state = { rng, intervalSet }; + this.state = { rng, fieldsToGenerate, intervalSet }; } generate() { @@ -1325,29 +1396,16 @@ export class GenerateUniqueInterval extends AbstractGenerator<{ throw new Error('state is not defined.'); } - let yearsNumb: number, - monthsNumb: number, - daysNumb: number, - hoursNumb: number, - minutesNumb: number, - secondsNumb: number; - - let interval = ''; + let interval, numb: number; for (;;) { - [yearsNumb, this.state.rng] = prand.uniformIntDistribution(0, 5, this.state.rng); - [monthsNumb, this.state.rng] = prand.uniformIntDistribution(0, 12, this.state.rng); - [daysNumb, this.state.rng] = prand.uniformIntDistribution(1, 29, this.state.rng); - [hoursNumb, this.state.rng] = prand.uniformIntDistribution(0, 24, this.state.rng); - [minutesNumb, this.state.rng] = prand.uniformIntDistribution(0, 60, this.state.rng); - [secondsNumb, this.state.rng] = prand.uniformIntDistribution(0, 60, this.state.rng); - - interval = `${yearsNumb === 0 ? '' : `${yearsNumb} years `}` - + `${monthsNumb === 0 ? '' : `${monthsNumb} months `}` - + `${daysNumb === 0 ? '' : `${daysNumb} days `}` - + `${hoursNumb === 0 ? '' : `${hoursNumb} hours `}` - + `${minutesNumb === 0 ? '' : `${minutesNumb} minutes `}` - + `${secondsNumb === 0 ? '' : `${secondsNumb} seconds`}`; + interval = ''; + + for (const field of this.state.fieldsToGenerate) { + const from = this.config[field]!.from, to = this.config[field]!.to; + [numb, this.state.rng] = prand.uniformIntDistribution(from, to, this.state.rng); + interval += `${numb} ${field} `; + } if (!this.state.intervalSet.has(interval)) { this.state.intervalSet.add(interval); @@ -1364,7 +1422,7 @@ export class GenerateString extends AbstractGenerator<{ isUnique?: boolean; arraySize?: number; }> { - static override readonly [entityKind]: string = 'GenerateString'; + static override readonly entityKind: string = 'GenerateString'; private state: { rng: prand.RandomGenerator } | undefined; override uniqueVersionOfGen = GenerateUniqueString; @@ -1408,7 +1466,7 @@ export class GenerateString extends AbstractGenerator<{ // has a newer version export class GenerateUniqueString extends AbstractGenerator<{ isUnique?: boolean }> { - static override readonly [entityKind]: string = 'GenerateUniqueString'; + static override readonly entityKind: string = 'GenerateUniqueString'; private state: { rng: prand.RandomGenerator } | undefined; public override isUnique = true; @@ -1453,7 +1511,7 @@ export class GenerateUniqueString extends AbstractGenerator<{ isUnique?: boolean export class GenerateUUID extends AbstractGenerator<{ arraySize?: number; }> { - static override readonly [entityKind]: string = 'GenerateUUID'; + static override readonly entityKind: string = 'GenerateUUID'; public override isUnique = true; @@ -1500,7 +1558,7 @@ export class GenerateFirstName extends AbstractGenerator<{ isUnique?: boolean; arraySize?: number; }> { - static override readonly [entityKind]: string = 'GenerateFirstName'; + static override readonly entityKind: string = 'GenerateFirstName'; override timeSpent: number = 0; private state: { @@ -1539,7 +1597,7 @@ export class GenerateFirstName extends AbstractGenerator<{ export class GenerateUniqueFirstName extends AbstractGenerator<{ isUnique?: boolean; }> { - static override readonly [entityKind]: string = 'GenerateUniqueFirstName'; + static override readonly entityKind: string = 'GenerateUniqueFirstName'; private state: { genIndicesObj: GenerateUniqueInt; @@ -1580,7 +1638,7 @@ export class GenerateLastName extends AbstractGenerator<{ isUnique?: boolean; arraySize?: number; }> { - static override readonly [entityKind]: string = 'GenerateLastName'; + static override readonly entityKind: string = 'GenerateLastName'; private state: { rng: prand.RandomGenerator; @@ -1613,7 +1671,7 @@ export class GenerateLastName extends AbstractGenerator<{ } export class GenerateUniqueLastName extends AbstractGenerator<{ isUnique?: boolean }> { - static override readonly [entityKind]: string = 'GenerateUniqueLastName'; + static override readonly entityKind: string = 'GenerateUniqueLastName'; private state: { genIndicesObj: GenerateUniqueInt; @@ -1653,7 +1711,7 @@ export class GenerateFullName extends AbstractGenerator<{ isUnique?: boolean; arraySize?: number; }> { - static override readonly [entityKind]: string = 'GenerateFullName'; + static override readonly entityKind: string = 'GenerateFullName'; private state: { rng: prand.RandomGenerator; @@ -1698,7 +1756,7 @@ export class GenerateFullName extends AbstractGenerator<{ export class GenerateUniqueFullName extends AbstractGenerator<{ isUnique?: boolean; }> { - static override readonly [entityKind]: string = 'GenerateUniqueFullName'; + static override readonly entityKind: string = 'GenerateUniqueFullName'; private state: { fullnameSet: Set; @@ -1763,7 +1821,7 @@ export class GenerateUniqueFullName extends AbstractGenerator<{ export class GenerateEmail extends AbstractGenerator<{ arraySize?: number; }> { - static override readonly [entityKind]: string = 'GenerateEmail'; + static override readonly entityKind: string = 'GenerateEmail'; private state: { genIndicesObj: GenerateUniqueInt; @@ -1830,7 +1888,7 @@ export class GeneratePhoneNumber extends AbstractGenerator<{ generatedDigitsNumbers?: number | number[]; arraySize?: number; }> { - static override readonly [entityKind]: string = 'GeneratePhoneNumber'; + static override readonly entityKind: string = 'GeneratePhoneNumber'; private state: { rng: prand.RandomGenerator; @@ -2024,7 +2082,7 @@ export class GenerateCountry extends AbstractGenerator<{ isUnique?: boolean; arraySize?: number; }> { - static override readonly [entityKind]: string = 'GenerateCountry'; + static override readonly entityKind: string = 'GenerateCountry'; private state: { rng: prand.RandomGenerator; @@ -2060,7 +2118,7 @@ export class GenerateCountry extends AbstractGenerator<{ } export class GenerateUniqueCountry extends AbstractGenerator<{ isUnique?: boolean }> { - static override readonly [entityKind]: string = 'GenerateUniqueCountry'; + static override readonly entityKind: string = 'GenerateUniqueCountry'; private state: { genIndicesObj: GenerateUniqueInt; @@ -2099,7 +2157,7 @@ export class GenerateUniqueCountry extends AbstractGenerator<{ isUnique?: boolea export class GenerateJobTitle extends AbstractGenerator<{ arraySize?: number; }> { - static override readonly [entityKind]: string = 'GenerateJobTitle'; + static override readonly entityKind: string = 'GenerateJobTitle'; private state: { rng: prand.RandomGenerator; @@ -2131,17 +2189,17 @@ export class GenerateJobTitle extends AbstractGenerator<{ } } -export class GenerateStreetAdddress extends AbstractGenerator<{ +export class GenerateStreetAddress extends AbstractGenerator<{ isUnique?: boolean; arraySize?: number; }> { - static override readonly [entityKind]: string = 'GenerateStreetAdddress'; + static override readonly entityKind: string = 'GenerateStreetAddress'; private state: { rng: prand.RandomGenerator; possStreetNames: string[][]; } | undefined; - override uniqueVersionOfGen = GenerateUniqueStreetAdddress; + override uniqueVersionOfGen = GenerateUniqueStreetAddress; override init({ count, seed }: { count: number; seed: number }) { super.init({ count, seed }); @@ -2181,8 +2239,8 @@ export class GenerateStreetAdddress extends AbstractGenerator<{ } } -export class GenerateUniqueStreetAdddress extends AbstractGenerator<{ isUnique?: boolean }> { - static override readonly [entityKind]: string = 'GenerateUniqueStreetAdddress'; +export class GenerateUniqueStreetAddress extends AbstractGenerator<{ isUnique?: boolean }> { + static override readonly entityKind: string = 'GenerateUniqueStreetAddress'; private state: { rng: prand.RandomGenerator; @@ -2278,7 +2336,7 @@ export class GenerateCity extends AbstractGenerator<{ isUnique?: boolean; arraySize?: number; }> { - static override readonly [entityKind]: string = 'GenerateCity'; + static override readonly entityKind: string = 'GenerateCity'; private state: { rng: prand.RandomGenerator; @@ -2312,7 +2370,7 @@ export class GenerateCity extends AbstractGenerator<{ } export class GenerateUniqueCity extends AbstractGenerator<{ isUnique?: boolean }> { - static override readonly [entityKind]: string = 'GenerateUniqueCity'; + static override readonly entityKind: string = 'GenerateUniqueCity'; private state: { genIndicesObj: GenerateUniqueInt; @@ -2352,7 +2410,7 @@ export class GeneratePostcode extends AbstractGenerator<{ isUnique?: boolean; arraySize?: number; }> { - static override readonly [entityKind]: string = 'GeneratePostcode'; + static override readonly entityKind: string = 'GeneratePostcode'; private state: { rng: prand.RandomGenerator; @@ -2406,7 +2464,7 @@ export class GeneratePostcode extends AbstractGenerator<{ } export class GenerateUniquePostcode extends AbstractGenerator<{ isUnique?: boolean }> { - static override readonly [entityKind]: string = 'GenerateUniquePostcode'; + static override readonly entityKind: string = 'GenerateUniquePostcode'; private state: { rng: prand.RandomGenerator; @@ -2493,7 +2551,7 @@ export class GenerateUniquePostcode extends AbstractGenerator<{ isUnique?: boole export class GenerateState extends AbstractGenerator<{ arraySize?: number; }> { - static override readonly [entityKind]: string = 'GenerateState'; + static override readonly entityKind: string = 'GenerateState'; private state: { rng: prand.RandomGenerator; @@ -2529,7 +2587,7 @@ export class GenerateCompanyName extends AbstractGenerator<{ isUnique?: boolean; arraySize?: number; }> { - static override readonly [entityKind]: string = 'GenerateCompanyName'; + static override readonly entityKind: string = 'GenerateCompanyName'; private state: { rng: prand.RandomGenerator; @@ -2598,7 +2656,7 @@ export class GenerateCompanyName extends AbstractGenerator<{ } export class GenerateUniqueCompanyName extends AbstractGenerator<{ isUnique?: boolean }> { - static override readonly [entityKind]: string = 'GenerateUniqueCompanyName'; + static override readonly entityKind: string = 'GenerateUniqueCompanyName'; private state: { rng: prand.RandomGenerator; @@ -2709,7 +2767,7 @@ export class GenerateLoremIpsum extends AbstractGenerator<{ sentencesCount?: number; arraySize?: number; }> { - static override readonly [entityKind]: string = 'GenerateLoremIpsum'; + static override readonly entityKind: string = 'GenerateLoremIpsum'; private state: { rng: prand.RandomGenerator; @@ -2748,7 +2806,7 @@ export class GenerateLoremIpsum extends AbstractGenerator<{ } export class WeightedRandomGenerator extends AbstractGenerator<{ weight: number; value: AbstractGenerator }[]> { - static override readonly [entityKind]: string = 'WeightedRandomGenerator'; + static override readonly entityKind: string = 'WeightedRandomGenerator'; private state: { rng: prand.RandomGenerator; @@ -2818,7 +2876,7 @@ export class GeneratePoint extends AbstractGenerator<{ maxYValue?: number; arraySize?: number; }> { - static override readonly [entityKind]: string = 'GeneratePoint'; + static override readonly entityKind: string = 'GeneratePoint'; private state: { xCoordinateGen: GenerateNumber; @@ -2872,7 +2930,7 @@ export class GenerateUniquePoint extends AbstractGenerator<{ maxYValue?: number; isUnique?: boolean; }> { - static override readonly [entityKind]: string = 'GenerateUniquePoint'; + static override readonly entityKind: string = 'GenerateUniquePoint'; private state: { xCoordinateGen: GenerateUniqueNumber; @@ -2927,7 +2985,7 @@ export class GenerateLine extends AbstractGenerator<{ maxCValue?: number; arraySize?: number; }> { - static override readonly [entityKind]: string = 'GenerateLine'; + static override readonly entityKind: string = 'GenerateLine'; private state: { aCoefficientGen: GenerateNumber; @@ -2998,7 +3056,7 @@ export class GenerateUniqueLine extends AbstractGenerator<{ maxCValue?: number; isUnique?: boolean; }> { - static override readonly [entityKind]: string = 'GenerateUniqueLine'; + static override readonly entityKind: string = 'GenerateUniqueLine'; private state: { aCoefficientGen: GenerateUniqueNumber; diff --git a/drizzle-seed/src/services/SeedService.ts b/drizzle-seed/src/services/SeedService.ts index f9da6a50a..35cba442f 100644 --- a/drizzle-seed/src/services/SeedService.ts +++ b/drizzle-seed/src/services/SeedService.ts @@ -1,3 +1,4 @@ +/* eslint-disable drizzle-internal/require-entity-kind */ import { entityKind, eq, is } from 'drizzle-orm'; import type { MySqlTable, MySqlTableWithColumns } from 'drizzle-orm/mysql-core'; import { MySqlDatabase } from 'drizzle-orm/mysql-core'; @@ -12,15 +13,14 @@ import type { TableGeneratorsType, } from '../types/seedService.ts'; import type { Column, Prettify, Relation, Table } from '../types/tables.ts'; -import type { AbstractGenerator } from './Generators.ts'; +import { generatorsMap } from './GeneratorFuncs.ts'; +import type { AbstractGenerator, GenerateArray, GenerateInterval, GenerateWeightedCount } from './Generators.ts'; import { latestVersion } from './apiVersion.ts'; -// import * as Generators from './Generators.ts'; import { equalSets, generateHashFromString } from './utils.ts'; -import * as GeneratorsV2 from './versioning/v2.ts'; export class SeedService { - static readonly [entityKind]: string = 'SeedService'; + static readonly entityKind: string = 'SeedService'; private defaultCountForTable = 10; private postgresPgLiteMaxParametersNumber = 32740; @@ -29,6 +29,7 @@ export class SeedService { private mysqlMaxParametersNumber = 100000; // SQLITE_MAX_VARIABLE_NUMBER, which by default equals to 999 for SQLite versions prior to 3.32.0 (2020-05-22) or 32766 for SQLite versions after 3.32.0. private sqliteMaxParametersNumber = 32766; + private version?: number; generatePossibleGenerators = ( connectionType: 'postgresql' | 'mysql' | 'sqlite', @@ -40,8 +41,8 @@ export class SeedService { let columnPossibleGenerator: Prettify; let tablePossibleGenerators: Prettify; const customSeed = options?.seed === undefined ? 0 : options.seed; - const version = options?.version === undefined ? latestVersion : options.version; - if (version < 1 || version > latestVersion) { + this.version = options?.version === undefined ? latestVersion : options.version; + if (this.version < 1 || this.version > latestVersion) { throw new Error(`Version should be in range [1, ${latestVersion}].`); } @@ -181,21 +182,15 @@ export class SeedService { && refinements[table.name]!.columns !== undefined && refinements[table.name]!.columns[col.name] !== undefined ) { - let genObj = refinements[table.name]!.columns[col.name]!; + const genObj = refinements[table.name]!.columns[col.name]!; - const genObjEntityKind = genObj.getEntityKind(); - const generatorConstructor = this.selectGeneratorOfVersion(version, genObjEntityKind); - genObj = new generatorConstructor(genObj.params); - - // TODO: for now only GenerateValuesFromArray support notNull property - genObj.notNull = col.notNull; if (col.columnType.match(/\[\w*]/g) !== null) { if ( (col.baseColumn?.dataType === 'array' && col.baseColumn.columnType.match(/\[\w*]/g) !== null) // studio case || (col.typeParams.dimensions !== undefined && col.typeParams.dimensions > 1) ) { - throw new Error("for now you can't specify generators for columns of dimensition greater than 1."); + throw new Error("for now you can't specify generators for columns of dimension greater than 1."); } genObj.baseColumnDataType = col.baseColumn?.dataType; @@ -217,29 +212,26 @@ export class SeedService { const predicate = cyclicRelation !== undefined && col.notNull === false; if (predicate === true) { - columnPossibleGenerator.generator = new Generators.GenerateDefault({ defaultValue: null }); + columnPossibleGenerator.generator = new generatorsMap.GenerateDefault[0]!({ defaultValue: null }); columnPossibleGenerator.wasDefinedBefore = true; + } else { + columnPossibleGenerator.generator = new generatorsMap.HollowGenerator[0]!({}); } - - columnPossibleGenerator.generator = new Generators.HollowGenerator({}); } // TODO: rewrite pickGeneratorFor... using new col properties: isUnique and notNull else if (connectionType === 'postgresql') { columnPossibleGenerator.generator = this.selectGeneratorForPostgresColumn( table, col, - version, ); } else if (connectionType === 'mysql') { columnPossibleGenerator.generator = this.selectGeneratorForMysqlColumn( table, col, - version, ); } else if (connectionType === 'sqlite') { columnPossibleGenerator.generator = this.selectGeneratorForSqlite( table, col, - version, ); } @@ -249,9 +241,22 @@ export class SeedService { ); } - // check if new version exists + const arrayGen = columnPossibleGenerator.generator.replaceIfArray(); + if (arrayGen !== undefined) { + columnPossibleGenerator.generator = arrayGen; + } + + const uniqueGen = columnPossibleGenerator.generator.replaceIfUnique(); + if (uniqueGen !== undefined) { + columnPossibleGenerator.generator = uniqueGen; + } + + // selecting version of generator + columnPossibleGenerator.generator = this.selectVersionOfGenerator(columnPossibleGenerator.generator); columnPossibleGenerator.generator.isUnique = col.isUnique; + // TODO: for now only GenerateValuesFromArray support notNull property + columnPossibleGenerator.generator.notNull = col.notNull; columnPossibleGenerator.generator.dataType = col.dataType; columnPossibleGenerator.generator.stringLength = col.typeParams.length; @@ -264,6 +269,39 @@ export class SeedService { return tablesPossibleGenerators; }; + selectVersionOfGenerator = (generator: AbstractGenerator) => { + const entityKind = generator.getEntityKind(); + if (entityKind === 'GenerateArray') { + const oldBaseColumnGen = (generator as GenerateArray).params.baseColumnGen; + + const newBaseColumnGen = this.selectVersionOfGenerator(oldBaseColumnGen); + // newGenerator.baseColumnDataType = oldGenerator.baseColumnDataType; + + (generator as GenerateArray).params.baseColumnGen = newBaseColumnGen; + } + + let possibleGeneratorConstructors = + (generatorsMap as { [entityKind: string]: (typeof AbstractGenerator)[] })[entityKind]; + + possibleGeneratorConstructors = possibleGeneratorConstructors?.filter((possGenCon) => + possGenCon.version <= this.version! // sorting in ascending order by version + ).sort((a, b) => a.version - b.version); + const generatorConstructor = possibleGeneratorConstructors?.at(-1) as + | (new(params: any) => AbstractGenerator) + | undefined; + if (generatorConstructor === undefined) { + throw new Error(`Can't select ${entityKind} generator for ${this.version} version.`); + } + + const newGenerator = new generatorConstructor(generator.params); + newGenerator.baseColumnDataType = generator.baseColumnDataType; + newGenerator.isUnique = generator.isUnique; + newGenerator.dataType = generator.dataType; + newGenerator.stringLength = generator.stringLength; + + return newGenerator; + }; + cyclicTablesCompare = ( table1: Table, table2: Table, @@ -410,7 +448,9 @@ export class SeedService { count: number, seed: number, ) => { - const gen = new Generators.GenerateWeightedCount({}); + let gen = new generatorsMap.GenerateWeightedCount[0]!({}); + gen = this.selectVersionOfGenerator(gen) as GenerateWeightedCount; + // const gen = new GenerateWeightedCount({}); gen.init({ count: weightedCount, seed }); let weightedWithCount = 0; for (let i = 0; i < count; i++) { @@ -420,34 +460,10 @@ export class SeedService { return weightedWithCount; }; - selectGeneratorOfVersion = (version: number, generatorEntityKind: string) => { - type GeneratorConstructorT = new(params: any) => AbstractGenerator; - const GeneratorVersions = [...Object.values(GeneratorsV2), ...Object.values(Generators)]; - - let generatorConstructor: GeneratorConstructorT | undefined; - for (const gens of GeneratorVersions) { - for (const gen of Object.values(gens)) { - const abstractGen = gen as typeof AbstractGenerator; - if (abstractGen[entityKind] === generatorEntityKind) { - generatorConstructor = abstractGen as unknown as GeneratorConstructorT; - - return generatorConstructor; - } - } - } - - if (generatorConstructor === undefined) { - throw new Error(`Can't select ${generatorEntityKind} generator for ${version} version.`); - } - - return generatorConstructor; - }; - // TODO: revise serial part generators selectGeneratorForPostgresColumn = ( table: Table, col: Column, - version: number, ) => { const pickGenerator = (table: Table, col: Column) => { // ARRAY @@ -455,13 +471,22 @@ export class SeedService { const baseColumnGen = this.selectGeneratorForPostgresColumn( table, col.baseColumn!, - version, ) as AbstractGenerator; if (baseColumnGen === undefined) { throw new Error(`column with type ${col.baseColumn!.columnType} is not supported for now.`); } - const generator = new Generators.GenerateArray({ baseColumnGen, size: col.size }); + // const getBaseColumnDataType = (baseColumn: Column) => { + // if (baseColumn.baseColumn !== undefined) { + // return getBaseColumnDataType(baseColumn.baseColumn); + // } + + // return baseColumn.dataType; + // }; + // const baseColumnDataType = getBaseColumnDataType(col.baseColumn); + + const generator = new generatorsMap.GenerateArray[0]!({ baseColumnGen, size: col.size }); + // generator.baseColumnDataType = baseColumnDataType; return generator; } @@ -475,15 +500,15 @@ export class SeedService { }; baseColumn.columnType = baseColumnType; - const baseColumnGen = this.selectGeneratorForPostgresColumn(table, baseColumn, version) as AbstractGenerator; + const baseColumnGen = this.selectGeneratorForPostgresColumn(table, baseColumn) as AbstractGenerator; if (baseColumnGen === undefined) { throw new Error(`column with type ${col.baseColumn!.columnType} is not supported for now.`); } - let generator = new Generators.GenerateArray({ baseColumnGen }); + let generator = new generatorsMap.GenerateArray[0]!({ baseColumnGen }); for (let i = 0; i < col.typeParams.dimensions! - 1; i++) { - generator = new Generators.GenerateArray({ baseColumnGen: generator }); + generator = new generatorsMap.GenerateArray[0]!({ baseColumnGen: generator }); } return generator; @@ -497,7 +522,7 @@ export class SeedService { || col.columnType.includes('bigint')) && table.primaryKeys.includes(col.name) ) { - const generator = new Generators.GenerateIntPrimaryKey({}); + const generator = new generatorsMap.GenerateIntPrimaryKey[0]!({}); return generator; } @@ -545,7 +570,7 @@ export class SeedService { && !col.columnType.includes('interval') && !col.columnType.includes('point') ) { - const generator = new Generators.GenerateInt({ + const generator = new generatorsMap.GenerateInt[0]!({ minValue, maxValue, }); @@ -554,9 +579,9 @@ export class SeedService { } if (col.columnType.includes('serial')) { - const generator = new Generators.GenerateIntPrimaryKey({}); + const generator = new generatorsMap.GenerateIntPrimaryKey[0]!({}); - (generator as Generators.GenerateIntPrimaryKey).maxValue = maxValue; + generator.maxValue = maxValue; return generator; } @@ -573,14 +598,14 @@ export class SeedService { const scale = col.typeParams.scale === undefined ? 0 : col.typeParams.scale; const maxAbsoluteValue = Math.pow(10, precision - scale) - Math.pow(10, -scale); - const generator = new Generators.GenerateNumber({ + const generator = new generatorsMap.GenerateNumber[0]!({ minValue: -maxAbsoluteValue, maxValue: maxAbsoluteValue, precision: Math.pow(10, scale), }); return generator; } - const generator = new Generators.GenerateNumber({}); + const generator = new generatorsMap.GenerateNumber[0]!({}); return generator; } @@ -592,7 +617,7 @@ export class SeedService { || col.columnType.startsWith('char')) && table.primaryKeys.includes(col.name) ) { - const generator = new Generators.GenerateUniqueString({}); + const generator = new generatorsMap.GenerateUniqueString[0]!({}); return generator; } @@ -603,7 +628,7 @@ export class SeedService { || col.columnType.startsWith('char')) && col.name.toLowerCase().includes('name') ) { - const generator = new Generators.GenerateFirstName({}); + const generator = new generatorsMap.GenerateFirstName[0]!({}); return generator; } @@ -614,7 +639,7 @@ export class SeedService { || col.columnType.startsWith('char')) && col.name.toLowerCase().includes('email') ) { - const generator = new Generators.GenerateEmail({}); + const generator = new generatorsMap.GenerateEmail[0]!({}); return generator; } @@ -624,47 +649,47 @@ export class SeedService { || col.columnType.startsWith('varchar') || col.columnType.startsWith('char') ) { - const generator = new Generators.GenerateString({}); + const generator = new generatorsMap.GenerateString[0]!({}); return generator; } // UUID if (col.columnType === 'uuid') { - const generator = new Generators.GenerateUUID({}); + const generator = new generatorsMap.GenerateUUID[0]!({}); return generator; } // BOOLEAN if (col.columnType === 'boolean') { - const generator = new Generators.GenerateBoolean({}); + const generator = new generatorsMap.GenerateBoolean[0]!({}); return generator; } // DATE, TIME, TIMESTAMP if (col.columnType.includes('date')) { - const generator = new Generators.GenerateDate({}); + const generator = new generatorsMap.GenerateDate[0]!({}); return generator; } if (col.columnType === 'time') { - const generator = new Generators.GenerateTime({}); + const generator = new generatorsMap.GenerateTime[0]!({}); return generator; } if (col.columnType.includes('timestamp')) { - const generator = new Generators.GenerateTimestamp({}); + const generator = new generatorsMap.GenerateTimestamp[0]!({}); return generator; } // JSON, JSONB if (col.columnType === 'json' || col.columnType === 'jsonb') { - const generator = new Generators.GenerateJson({}); + const generator = new generatorsMap.GenerateJson[0]!({}); return generator; } @@ -676,7 +701,7 @@ export class SeedService { // ENUM if (col.enumValues !== undefined) { - const generator = new Generators.GenerateEnum({ + const generator = new generatorsMap.GenerateEnum[0]!({ enumValues: col.enumValues, }); @@ -686,32 +711,32 @@ export class SeedService { // INTERVAL if (col.columnType.startsWith('interval')) { if (col.columnType === 'interval') { - const generator = new Generators.GenerateInterval({}); + const generator = new generatorsMap.GenerateInterval[0]!({}); return generator; } - const fields = col.columnType.replace('interval ', '') as Generators.GenerateInterval['params']['fields']; - const generator = new Generators.GenerateInterval({ fields }); + const fields = col.columnType.replace('interval ', '') as GenerateInterval['params']['fields']; + const generator = new generatorsMap.GenerateInterval[0]!({ fields }); return generator; } // POINT, LINE if (col.columnType.includes('point')) { - const generator = new Generators.GeneratePoint({}); + const generator = new generatorsMap.GeneratePoint[0]!({}); return generator; } if (col.columnType.includes('line')) { - const generator = new Generators.GenerateLine({}); + const generator = new generatorsMap.GenerateLine[0]!({}); return generator; } if (col.hasDefault && col.default !== undefined) { - const generator = new Generators.GenerateDefault({ + const generator = new generatorsMap.GenerateDefault[0]!({ defaultValue: col.default, }); return generator; @@ -720,15 +745,8 @@ export class SeedService { return; }; - let generator = pickGenerator(table, col) as AbstractGenerator || undefined; + const generator = pickGenerator(table, col); if (generator !== undefined) { - const generatorConstructor = this.selectGeneratorOfVersion( - version, - generator.getEntityKind(), - ); - - generator = new generatorConstructor(generator.params); - generator.isUnique = col.isUnique; generator.dataType = col.dataType; generator.stringLength = col.typeParams.length; @@ -740,7 +758,6 @@ export class SeedService { selectGeneratorForMysqlColumn = ( table: Table, col: Column, - version: number, ) => { const pickGenerator = (table: Table, col: Column) => { // INT ------------------------------------------------------------------------------------------------------------ @@ -748,7 +765,7 @@ export class SeedService { (col.columnType.includes('serial') || col.columnType.includes('int')) && table.primaryKeys.includes(col.name) ) { - const generator = new Generators.GenerateIntPrimaryKey({}); + const generator = new generatorsMap.GenerateIntPrimaryKey[0]!({}); return generator; } @@ -783,7 +800,7 @@ export class SeedService { } if (col.columnType.includes('int')) { - const generator = new Generators.GenerateInt({ + const generator = new generatorsMap.GenerateInt[0]!({ minValue, maxValue, }); @@ -791,7 +808,7 @@ export class SeedService { } if (col.columnType.includes('serial')) { - const generator = new Generators.GenerateIntPrimaryKey({}); + const generator = new generatorsMap.GenerateIntPrimaryKey[0]!({}); generator.maxValue = maxValue; return generator; } @@ -809,7 +826,7 @@ export class SeedService { const scale = col.typeParams.scale === undefined ? 0 : col.typeParams.scale; const maxAbsoluteValue = Math.pow(10, precision - scale) - Math.pow(10, -scale); - const generator = new Generators.GenerateNumber({ + const generator = new generatorsMap.GenerateNumber[0]!({ minValue: -maxAbsoluteValue, maxValue: maxAbsoluteValue, precision: Math.pow(10, scale), @@ -817,7 +834,7 @@ export class SeedService { return generator; } - const generator = new Generators.GenerateNumber({}); + const generator = new generatorsMap.GenerateNumber[0]!({}); return generator; } @@ -831,7 +848,7 @@ export class SeedService { || col.columnType.startsWith('varbinary')) && table.primaryKeys.includes(col.name) ) { - const generator = new Generators.GenerateUniqueString({}); + const generator = new generatorsMap.GenerateUniqueString[0]!({}); return generator; } @@ -844,7 +861,7 @@ export class SeedService { || col.columnType.startsWith('varbinary')) && col.name.toLowerCase().includes('name') ) { - const generator = new Generators.GenerateFirstName({}); + const generator = new generatorsMap.GenerateFirstName[0]!({}); return generator; } @@ -857,7 +874,7 @@ export class SeedService { || col.columnType.startsWith('varbinary')) && col.name.toLowerCase().includes('email') ) { - const generator = new Generators.GenerateEmail({}); + const generator = new generatorsMap.GenerateEmail[0]!({}); return generator; } @@ -869,58 +886,58 @@ export class SeedService { || col.columnType.startsWith('binary') || col.columnType.startsWith('varbinary') ) { - const generator = new Generators.GenerateString({}); + const generator = new generatorsMap.GenerateString[0]!({}); return generator; } // BOOLEAN if (col.columnType === 'boolean') { - const generator = new Generators.GenerateBoolean({}); + const generator = new generatorsMap.GenerateBoolean[0]!({}); return generator; } // DATE, TIME, TIMESTAMP, DATETIME, YEAR if (col.columnType.includes('datetime')) { - const generator = new Generators.GenerateDatetime({}); + const generator = new generatorsMap.GenerateDatetime[0]!({}); return generator; } if (col.columnType.includes('date')) { - const generator = new Generators.GenerateDate({}); + const generator = new generatorsMap.GenerateDate[0]!({}); return generator; } if (col.columnType === 'time') { - const generator = new Generators.GenerateTime({}); + const generator = new generatorsMap.GenerateTime[0]!({}); return generator; } if (col.columnType.includes('timestamp')) { - const generator = new Generators.GenerateTimestamp({}); + const generator = new generatorsMap.GenerateTimestamp[0]!({}); return generator; } if (col.columnType === 'year') { - const generator = new Generators.GenerateYear({}); + const generator = new generatorsMap.GenerateYear[0]!({}); return generator; } // JSON if (col.columnType === 'json') { - const generator = new Generators.GenerateJson({}); + const generator = new generatorsMap.GenerateJson[0]!({}); return generator; } // ENUM if (col.enumValues !== undefined) { - const generator = new Generators.GenerateEnum({ + const generator = new generatorsMap.GenerateEnum[0]!({ enumValues: col.enumValues, }); return generator; } if (col.hasDefault && col.default !== undefined) { - const generator = new Generators.GenerateDefault({ + const generator = new generatorsMap.GenerateDefault[0]!({ defaultValue: col.default, }); return generator; @@ -929,15 +946,7 @@ export class SeedService { return; }; - let generator = pickGenerator(table, col) as AbstractGenerator || undefined; - if (generator !== undefined) { - const generatorConstructor = this.selectGeneratorOfVersion( - version, - generator.getEntityKind(), - ); - - generator = new generatorConstructor(generator.params); - } + const generator = pickGenerator(table, col); return generator; }; @@ -945,7 +954,6 @@ export class SeedService { selectGeneratorForSqlite = ( table: Table, col: Column, - version: number, ) => { const pickGenerator = (table: Table, col: Column) => { // int section --------------------------------------------------------------------------------------- @@ -953,17 +961,17 @@ export class SeedService { (col.columnType === 'integer' || col.columnType === 'numeric') && table.primaryKeys.includes(col.name) ) { - const generator = new Generators.GenerateIntPrimaryKey({}); + const generator = new generatorsMap.GenerateIntPrimaryKey[0]!({}); return generator; } if (col.columnType === 'integer' && col.dataType === 'boolean') { - const generator = new Generators.GenerateBoolean({}); + const generator = new generatorsMap.GenerateBoolean[0]!({}); return generator; } if ((col.columnType === 'integer' && col.dataType === 'date')) { - const generator = new Generators.GenerateTimestamp({}); + const generator = new generatorsMap.GenerateTimestamp[0]!({}); return generator; } @@ -971,7 +979,7 @@ export class SeedService { col.columnType === 'integer' || (col.dataType === 'bigint' && col.columnType === 'blob') ) { - const generator = new Generators.GenerateInt({}); + const generator = new generatorsMap.GenerateInt[0]!({}); return generator; } @@ -982,7 +990,7 @@ export class SeedService { const scale = col.typeParams.scale === undefined ? 0 : col.typeParams.scale; const maxAbsoluteValue = Math.pow(10, precision - scale) - Math.pow(10, -scale); - const generator = new Generators.GenerateNumber({ + const generator = new generatorsMap.GenerateNumber[0]!({ minValue: -maxAbsoluteValue, maxValue: maxAbsoluteValue, precision: Math.pow(10, scale), @@ -990,7 +998,7 @@ export class SeedService { return generator; } - const generator = new Generators.GenerateNumber({}); + const generator = new generatorsMap.GenerateNumber[0]!({}); return generator; } @@ -1001,7 +1009,7 @@ export class SeedService { || col.columnType.startsWith('blob')) && table.primaryKeys.includes(col.name) ) { - const generator = new Generators.GenerateUniqueString({}); + const generator = new generatorsMap.GenerateUniqueString[0]!({}); return generator; } @@ -1011,7 +1019,7 @@ export class SeedService { || col.columnType.startsWith('blob')) && col.name.toLowerCase().includes('name') ) { - const generator = new Generators.GenerateFirstName({}); + const generator = new generatorsMap.GenerateFirstName[0]!({}); return generator; } @@ -1021,7 +1029,7 @@ export class SeedService { || col.columnType.startsWith('blob')) && col.name.toLowerCase().includes('email') ) { - const generator = new Generators.GenerateEmail({}); + const generator = new generatorsMap.GenerateEmail[0]!({}); return generator; } @@ -1031,7 +1039,7 @@ export class SeedService { || col.columnType.startsWith('blob') || col.columnType.startsWith('blobbuffer') ) { - const generator = new Generators.GenerateString({}); + const generator = new generatorsMap.GenerateString[0]!({}); return generator; } @@ -1039,12 +1047,12 @@ export class SeedService { (col.columnType.startsWith('text') && col.dataType === 'json') || (col.columnType.startsWith('blob') && col.dataType === 'json') ) { - const generator = new Generators.GenerateJson({}); + const generator = new generatorsMap.GenerateJson[0]!({}); return generator; } if (col.hasDefault && col.default !== undefined) { - const generator = new Generators.GenerateDefault({ + const generator = new generatorsMap.GenerateDefault[0]!({ defaultValue: col.default, }); return generator; @@ -1053,15 +1061,7 @@ export class SeedService { return; }; - let generator = pickGenerator(table, col) as AbstractGenerator || undefined; - if (generator !== undefined) { - const generatorConstructor = this.selectGeneratorOfVersion( - version, - generator.getEntityKind(), - ); - - generator = new generatorConstructor(generator.params); - } + const generator = pickGenerator(table, col); return generator; }; @@ -1220,9 +1220,13 @@ export class SeedService { }))!.map((rows) => rows[refColName]) as (string | number | boolean)[]; hasSelfRelation = true; - genObj = new Generators.GenerateSelfRelationsValuesFromArray({ + genObj = new generatorsMap.GenerateSelfRelationsValuesFromArray[0]!({ values: refColumnValues, }); + genObj = this.selectVersionOfGenerator(genObj); + // genObj = new GenerateSelfRelationsValuesFromArray({ + // values: refColumnValues, + // }); } else if ( tableGenerators[rel.columns[colIdx]!]?.wasDefinedBefore === false && tableGenerators[rel.columns[colIdx]!]?.wasRefined === false @@ -1240,10 +1244,11 @@ export class SeedService { weightedCountSeed = table.withFromTable[rel.refTable]!.weightedCountSeed; } - genObj = new Generators.GenerateValuesFromArray({ values: refColumnValues }); - (genObj as Generators.GenerateValuesFromArray).notNull = tableGenerators[rel.columns[colIdx]!]!.notNull; - (genObj as Generators.GenerateValuesFromArray).weightedCountSeed = weightedCountSeed; - (genObj as Generators.GenerateValuesFromArray).maxRepeatedValuesCount = repeatedValuesCount; + // TODO: revise maybe need to select version of generator here too + genObj = new generatorsMap.GenerateValuesFromArray[0]!({ values: refColumnValues }); + genObj.notNull = tableGenerators[rel.columns[colIdx]!]!.notNull; + genObj.weightedCountSeed = weightedCountSeed; + genObj.maxRepeatedValuesCount = repeatedValuesCount; } if (genObj !== undefined) { @@ -1360,15 +1365,15 @@ export class SeedService { seed: columnGenerator.pRNGSeed, }); - const arrayGen = columnsGenerators[columnName]!.replaceIfArray({ count, seed: columnGenerator.pRNGSeed }); - if (arrayGen !== undefined) { - columnsGenerators[columnName] = arrayGen; - } + // const arrayGen = columnsGenerators[columnName]!.replaceIfArray({ count, seed: columnGenerator.pRNGSeed }); + // if (arrayGen !== undefined) { + // columnsGenerators[columnName] = arrayGen; + // } - const uniqueGen = columnsGenerators[columnName]!.replaceIfUnique({ count, seed: columnGenerator.pRNGSeed }); - if (uniqueGen !== undefined) { - columnsGenerators[columnName] = uniqueGen; - } + // const uniqueGen = columnsGenerators[columnName]!.replaceIfUnique({ count, seed: columnGenerator.pRNGSeed }); + // if (uniqueGen !== undefined) { + // columnsGenerators[columnName] = uniqueGen; + // } } let maxParametersNumber: number; if (is(db, PgDatabase)) { diff --git a/drizzle-seed/src/services/versioning/v2.ts b/drizzle-seed/src/services/versioning/v2.ts index 96451b2f8..e09cc45f6 100644 --- a/drizzle-seed/src/services/versioning/v2.ts +++ b/drizzle-seed/src/services/versioning/v2.ts @@ -1,12 +1,6 @@ -import { entityKind } from 'drizzle-orm'; +/* eslint-disable drizzle-internal/require-entity-kind */ import prand from 'pure-rand'; -import { AbstractGenerator, GenerateInterval } from '../Generators.ts'; - -export class GenerateIntervalV2 extends GenerateInterval { - static override readonly [entityKind]: string = 'GenerateInterval'; - override readonly version: number = 2; - override uniqueVersionOfGen = GenerateUniqueIntervalV2; -} +import { AbstractGenerator } from '../Generators.ts'; export class GenerateUniqueIntervalV2 extends AbstractGenerator<{ fields?: @@ -25,8 +19,8 @@ export class GenerateUniqueIntervalV2 extends AbstractGenerator<{ | 'minute to second'; isUnique?: boolean; }> { - static override readonly [entityKind]: string = 'GenerateUniqueInterval'; - override readonly version: number = 2; + static override readonly 'entityKind': string = 'GenerateUniqueInterval'; + static override readonly version: number = 2; private state: { rng: prand.RandomGenerator; @@ -119,8 +113,8 @@ export class GenerateStringV2 extends AbstractGenerator<{ isUnique?: boolean; arraySize?: number; }> { - static override readonly [entityKind]: string = 'GenerateString'; - override readonly version: number = 2; + static override readonly 'entityKind': string = 'GenerateString'; + static override readonly version: number = 2; private state: { rng: prand.RandomGenerator; @@ -175,8 +169,8 @@ export class GenerateStringV2 extends AbstractGenerator<{ } export class GenerateUniqueStringV2 extends AbstractGenerator<{ isUnique?: boolean }> { - static override readonly [entityKind]: string = 'GenerateUniqueString'; - override readonly version: number = 2; + static override readonly 'entityKind': string = 'GenerateUniqueString'; + static override readonly version: number = 2; private state: { rng: prand.RandomGenerator; diff --git a/drizzle-seed/tests/benchmarks/generatorsBenchmark.ts b/drizzle-seed/tests/benchmarks/generatorsBenchmark.ts index 3473e555a..23fca0c6c 100644 --- a/drizzle-seed/tests/benchmarks/generatorsBenchmark.ts +++ b/drizzle-seed/tests/benchmarks/generatorsBenchmark.ts @@ -23,7 +23,7 @@ import { GeneratePoint, GeneratePostcode, GenerateState, - GenerateStreetAdddress, + GenerateStreetAddress, GenerateString, GenerateTime, GenerateTimestamp, @@ -35,7 +35,7 @@ import { GenerateUniqueNumber, GenerateUniquePoint, GenerateUniquePostcode, - GenerateUniqueStreetAdddress, + GenerateUniqueStreetAddress, GenerateUniqueString, GenerateValuesFromArray, GenerateYear, @@ -107,8 +107,8 @@ const generatorsFuncs = { // uniqueCountry: new GenerateUniqueCountry({}), city: new GenerateCity({}), // uniqueCity: new GenerateUniqueCity({}), - streetAddress: new GenerateStreetAdddress({}), - uniqueStreetAddress: new GenerateUniqueStreetAdddress({}), + streetAddress: new GenerateStreetAddress({}), + uniqueStreetAddress: new GenerateUniqueStreetAddress({}), jobTitle: new GenerateJobTitle({}), postcode: new GeneratePostcode({}), uniquePostcode: new GenerateUniquePostcode({}), diff --git a/drizzle-seed/tests/pg/allDataTypesTest/pgSchema.ts b/drizzle-seed/tests/pg/allDataTypesTest/pgSchema.ts index 16a55baf4..d4f45de22 100644 --- a/drizzle-seed/tests/pg/allDataTypesTest/pgSchema.ts +++ b/drizzle-seed/tests/pg/allDataTypesTest/pgSchema.ts @@ -64,31 +64,31 @@ export const allDataTypes = schema.table('all_data_types', { }); export const allArrayDataTypes = schema.table('all_array_data_types', { - integerArray: integer('integer_array').array(), - smallintArray: smallint('smallint_array').array(), - bigintegerArray: bigint('bigint_array', { mode: 'bigint' }).array(), - bigintNumberArray: bigint('bigint_number_array', { mode: 'number' }).array(), - booleanArray: boolean('boolean_array').array(), - textArray: text('text_array').array(), - varcharArray: varchar('varchar_array', { length: 256 }).array(), - charArray: char('char_array', { length: 256 }).array(), - numericArray: numeric('numeric_array').array(), - decimalArray: decimal('decimal_array').array(), - realArray: real('real_array').array(), - doublePrecisionArray: doublePrecision('double_precision_array').array(), - jsonArray: json('json_array').array(), - jsonbArray: jsonb('jsonb_array').array(), - timeArray: time('time_array').array(), - timestampDateArray: timestamp('timestamp_date_array', { mode: 'date' }).array(), - timestampStringArray: timestamp('timestamp_string_array', { mode: 'string' }).array(), + // integerArray: integer('integer_array').array(), + // smallintArray: smallint('smallint_array').array(), + // bigintegerArray: bigint('bigint_array', { mode: 'bigint' }).array(), + // bigintNumberArray: bigint('bigint_number_array', { mode: 'number' }).array(), + // booleanArray: boolean('boolean_array').array(), + // textArray: text('text_array').array(), + // varcharArray: varchar('varchar_array', { length: 256 }).array(), + // charArray: char('char_array', { length: 256 }).array(), + // numericArray: numeric('numeric_array').array(), + // decimalArray: decimal('decimal_array').array(), + // realArray: real('real_array').array(), + // doublePrecisionArray: doublePrecision('double_precision_array').array(), + // jsonArray: json('json_array').array(), + // jsonbArray: jsonb('jsonb_array').array(), + // timeArray: time('time_array').array(), + // timestampDateArray: timestamp('timestamp_date_array', { mode: 'date' }).array(), + // timestampStringArray: timestamp('timestamp_string_array', { mode: 'string' }).array(), dateStringArray: date('date_string_array', { mode: 'string' }).array(), dateArray: date('date_array', { mode: 'date' }).array(), - intervalArray: interval('interval_array').array(), - pointArray: point('point_array', { mode: 'xy' }).array(), - pointTupleArray: point('point_tuple_array', { mode: 'tuple' }).array(), - lineArray: line('line_array', { mode: 'abc' }).array(), - lineTupleArray: line('line_tuple_array', { mode: 'tuple' }).array(), - moodEnumArray: moodEnum('mood_enum_array').array(), + // intervalArray: interval('interval_array').array(), + // pointArray: point('point_array', { mode: 'xy' }).array(), + // pointTupleArray: point('point_tuple_array', { mode: 'tuple' }).array(), + // lineArray: line('line_array', { mode: 'abc' }).array(), + // lineTupleArray: line('line_tuple_array', { mode: 'tuple' }).array(), + // moodEnumArray: moodEnum('mood_enum_array').array(), }); export const ndArrays = schema.table('nd_arrays', { From eb72873330b7822ff95cdfb7af4e9b24fb28b7fb Mon Sep 17 00:00:00 2001 From: OleksiiKH0240 Date: Mon, 23 Dec 2024 15:51:18 +0200 Subject: [PATCH 22/32] bug fix, refine functions versioning --- changelogs/drizzle-seed/0.2.0.md | 5 ++- drizzle-seed/src/index.ts | 36 +++++++++++++++------ drizzle-seed/src/services/GeneratorFuncs.ts | 5 +-- drizzle-seed/src/services/SeedService.ts | 31 ++++++++++++++---- 4 files changed, 58 insertions(+), 19 deletions(-) diff --git a/changelogs/drizzle-seed/0.2.0.md b/changelogs/drizzle-seed/0.2.0.md index be19a8949..5bb9d6007 100644 --- a/changelogs/drizzle-seed/0.2.0.md +++ b/changelogs/drizzle-seed/0.2.0.md @@ -3,7 +3,7 @@ We are introducing a new parameter, `version`, to the `seed` function options. This parameter, which controls generator versioning, has been added to make it easier to update deterministic generators in the future. Since values should remain consistent after each regeneration, it is crucial to provide a well-designed API for gradual updates ```ts -await seed(db, schema, { version: 2 }); +await seed(db, schema, { version: '2' }); ``` #### Example: @@ -121,3 +121,6 @@ However, after inserting the `1 minute 60 second` interval, PostgreSQL database **Generating strings based on text-like column length:** Now (in version 2), the maximum length of a string depends on the length of the text column (e.g., `varchar(20)`). + +## Bug fixes +- seeding table with foreign key referencing another table, without including the second table in schema, will cause seed process to get stuck. \ No newline at end of file diff --git a/drizzle-seed/src/index.ts b/drizzle-seed/src/index.ts index 48aad2973..de99eb746 100644 --- a/drizzle-seed/src/index.ts +++ b/drizzle-seed/src/index.ts @@ -10,7 +10,7 @@ import { getTableConfig as getPgTableConfig, PgDatabase, PgTable } from 'drizzle import type { SQLiteColumn } from 'drizzle-orm/sqlite-core'; import { BaseSQLiteDatabase, getTableConfig as getSqliteTableConfig, SQLiteTable } from 'drizzle-orm/sqlite-core'; -import { generatorsFuncs } from './services/GeneratorFuncs.ts'; +import { generatorsFuncs, generatorsFuncsV2 } from './services/GeneratorFuncs.ts'; import type { AbstractGenerator } from './services/Generators.ts'; import { SeedService } from './services/SeedService.ts'; import type { DrizzleStudioObjectType, DrizzleStudioRelationType } from './types/drizzleStudio.ts'; @@ -131,6 +131,7 @@ class SeedPromise< SCHEMA extends { [key: string]: PgTable | PgSchema | MySqlTable | MySqlSchema | SQLiteTable; }, + VERSION extends string | undefined, > implements Promise { static readonly entityKind: string = 'SeedPromise'; @@ -139,7 +140,7 @@ class SeedPromise< constructor( private db: DB, private schema: SCHEMA, - private options?: { count?: number; seed?: number; version?: number }, + private options?: { count?: number; seed?: number; version?: VERSION }, ) {} then( @@ -181,14 +182,23 @@ class SeedPromise< } async refine( - callback: (funcs: typeof generatorsFuncs) => InferCallbackType, + callback: ( + funcs: FunctionsVersioning, + ) => InferCallbackType, ): Promise { - const refinements = callback(generatorsFuncs) as RefinementsType; + const refinements = this.options?.version === undefined || this.options.version === '2' + ? callback(generatorsFuncsV2 as FunctionsVersioning) as RefinementsType + : callback(generatorsFuncs as FunctionsVersioning) as RefinementsType; + // const refinements = callback(generatorsFuncs) as RefinementsType; await seedFunc(this.db, this.schema, this.options, refinements); } } +type FunctionsVersioning = VERSION extends `1` ? typeof generatorsFuncs + : VERSION extends `2` ? typeof generatorsFuncsV2 + : typeof generatorsFuncsV2; + export function getGeneratorsFunctions() { return generatorsFuncs; } @@ -337,8 +347,9 @@ export function seed< | SQLiteTable | any; }, ->(db: DB, schema: SCHEMA, options?: { count?: number; seed?: number; version?: number }) { - return new SeedPromise(db, schema, options); + VERSION extends '2' | '1' | undefined, +>(db: DB, schema: SCHEMA, options?: { count?: number; seed?: number; version?: VERSION }) { + return new SeedPromise(db, schema, options); } const seedFunc = async ( @@ -352,21 +363,26 @@ const seedFunc = async ( | SQLiteTable | any; }, - options: { count?: number; seed?: number; version?: number } = {}, + options: { count?: number; seed?: number; version?: string } = {}, refinements?: RefinementsType, ) => { + let version: number | undefined; + if (options?.version !== undefined) { + version = Number(options?.version); + } + if (is(db, PgDatabase)) { const { pgSchema } = filterPgTables(schema); - await seedPostgres(db, pgSchema, options, refinements); + await seedPostgres(db, pgSchema, { ...options, version }, refinements); } else if (is(db, MySqlDatabase)) { const { mySqlSchema } = filterMySqlTables(schema); - await seedMySql(db, mySqlSchema, options, refinements); + await seedMySql(db, mySqlSchema, { ...options, version }, refinements); } else if (is(db, BaseSQLiteDatabase)) { const { sqliteSchema } = filterSqliteTables(schema); - await seedSqlite(db, sqliteSchema, options, refinements); + await seedSqlite(db, sqliteSchema, { ...options, version }, refinements); } else { throw new Error( 'The drizzle-seed package currently supports only PostgreSQL, MySQL, and SQLite databases. Please ensure your database is one of these supported types', diff --git a/drizzle-seed/src/services/GeneratorFuncs.ts b/drizzle-seed/src/services/GeneratorFuncs.ts index be715a33b..f0687b5ed 100644 --- a/drizzle-seed/src/services/GeneratorFuncs.ts +++ b/drizzle-seed/src/services/GeneratorFuncs.ts @@ -756,9 +756,10 @@ export const generatorsFuncs = { weightedRandom: createGenerator(WeightedRandomGenerator), }; -// TODO: revise // so far, version changes don’t affect generator parameters. -export const generatorsFuncsV2 = { ...generatorsFuncs }; +export const generatorsFuncsV2 = { + ...generatorsFuncs, +}; export const generatorsMap = { HollowGenerator: [ diff --git a/drizzle-seed/src/services/SeedService.ts b/drizzle-seed/src/services/SeedService.ts index 35cba442f..6c045f368 100644 --- a/drizzle-seed/src/services/SeedService.ts +++ b/drizzle-seed/src/services/SeedService.ts @@ -42,7 +42,7 @@ export class SeedService { let tablePossibleGenerators: Prettify; const customSeed = options?.seed === undefined ? 0 : options.seed; this.version = options?.version === undefined ? latestVersion : options.version; - if (this.version < 1 || this.version > latestVersion) { + if (Number.isNaN(this.version) || this.version < 1 || this.version > latestVersion) { throw new Error(`Version should be in range [1, ${latestVersion}].`); } @@ -209,9 +209,23 @@ export class SeedService { if (cyclicRelation !== undefined) { columnPossibleGenerator.isCyclic = true; } - const predicate = cyclicRelation !== undefined && col.notNull === false; + + if (foreignKeyColumns[col.name]?.table === undefined && col.notNull === true) { + throw new Error( + `Column '${col.name}' cannot be nullable, and you didn't specify a table for foreign key on column '${col.name}' in '${table.name}' table.`, + ); + } + + const predicate = (cyclicRelation !== undefined || foreignKeyColumns[col.name]?.table === undefined) + && col.notNull === false; if (predicate === true) { + if (foreignKeyColumns[col.name]?.table === undefined && col.notNull === false) { + console.warn( + `Column '${col.name}' in '${table.name}' table will be filled with null values` + + `\nbecause you specified neither a table for foreign key on column '${col.name}' nor a function for '${col.name}' column in refinements.`, + ); + } columnPossibleGenerator.generator = new generatorsMap.GenerateDefault[0]!({ defaultValue: null }); columnPossibleGenerator.wasDefinedBefore = true; } else { @@ -417,7 +431,10 @@ export class SeedService { }; } - if (tablesInOutRelations[rel.refTable] === undefined) { + if ( + rel.refTable !== undefined + && tablesInOutRelations[rel.refTable] === undefined + ) { tablesInOutRelations[rel.refTable] = { out: 0, in: 0, @@ -428,13 +445,15 @@ export class SeedService { }; } - tablesInOutRelations[rel.table]!.out += 1; - tablesInOutRelations[rel.refTable]!.in += 1; + if (rel.refTable !== undefined) { + tablesInOutRelations[rel.table]!.out += 1; + tablesInOutRelations[rel.refTable]!.in += 1; + } if (rel.refTable === rel.table) { tablesInOutRelations[rel.table]!.selfRelation = true; tablesInOutRelations[rel.table]!.selfRelCount = rel.columns.length; - } else { + } else if (rel.refTable !== undefined) { tablesInOutRelations[rel.table]!.requiredTableNames.add(rel.refTable); tablesInOutRelations[rel.refTable]!.dependantTableNames.add(rel.table); } From 280720064f258b8cd83b5592deece91e5e2bc7e7 Mon Sep 17 00:00:00 2001 From: OleksiiKH0240 Date: Mon, 23 Dec 2024 16:04:08 +0200 Subject: [PATCH 23/32] fixes --- drizzle-seed/src/index.ts | 5 +---- 1 file changed, 1 insertion(+), 4 deletions(-) diff --git a/drizzle-seed/src/index.ts b/drizzle-seed/src/index.ts index de99eb746..e3fc5494a 100644 --- a/drizzle-seed/src/index.ts +++ b/drizzle-seed/src/index.ts @@ -587,10 +587,7 @@ const getPostgresInfo = (schema: { [key: string]: PgTable }) => { // might be empty list const newRelations = tableConfig.foreignKeys.map((fk) => { const table = dbToTsTableNamesMap[tableConfig.name] as string; - const refTableName0 = fk.reference(); - const refTableName1 = refTableName0.foreignTable; - const refTableName2 = getTableName(refTableName1); - const refTable = dbToTsTableNamesMap[refTableName2] as string; + const refTable = dbToTsTableNamesMap[getTableName(fk.reference().foreignTable)] as string; const dbToTsColumnNamesMapForRefTable = getDbToTsColumnNamesMap( fk.reference().foreignTable, From 0db3d1334263f6badc4ed9d9fd9a127ba0f90fad Mon Sep 17 00:00:00 2001 From: OleksiiKH0240 Date: Mon, 23 Dec 2024 17:32:10 +0200 Subject: [PATCH 24/32] fixes --- drizzle-seed/src/index.ts | 1 - drizzle-seed/src/services/GeneratorFuncs.ts | 4 +- drizzle-seed/src/services/Generators.ts | 6 +- drizzle-seed/src/services/SeedService.ts | 129 ++++++++++---------- 4 files changed, 70 insertions(+), 70 deletions(-) diff --git a/drizzle-seed/src/index.ts b/drizzle-seed/src/index.ts index e3fc5494a..98c449095 100644 --- a/drizzle-seed/src/index.ts +++ b/drizzle-seed/src/index.ts @@ -189,7 +189,6 @@ class SeedPromise< const refinements = this.options?.version === undefined || this.options.version === '2' ? callback(generatorsFuncsV2 as FunctionsVersioning) as RefinementsType : callback(generatorsFuncs as FunctionsVersioning) as RefinementsType; - // const refinements = callback(generatorsFuncs) as RefinementsType; await seedFunc(this.db, this.schema, this.options, refinements); } diff --git a/drizzle-seed/src/services/GeneratorFuncs.ts b/drizzle-seed/src/services/GeneratorFuncs.ts index f0687b5ed..10d0d10f7 100644 --- a/drizzle-seed/src/services/GeneratorFuncs.ts +++ b/drizzle-seed/src/services/GeneratorFuncs.ts @@ -54,7 +54,7 @@ import { import { GenerateStringV2, GenerateUniqueIntervalV2, GenerateUniqueStringV2 } from './versioning/v2.ts'; function createGenerator, T>( - generatorConstructor: new(params: T) => GeneratorType, + generatorConstructor: new(params?: T) => GeneratorType, ) { return ( ...args: GeneratorType extends GenerateValuesFromArray | GenerateDefault | WeightedRandomGenerator ? [T] @@ -915,4 +915,4 @@ export const generatorsMap = { GenerateWeightedCount: [ GenerateWeightedCount, ], -}; +} as const; diff --git a/drizzle-seed/src/services/Generators.ts b/drizzle-seed/src/services/Generators.ts index fff543739..0d285540e 100644 --- a/drizzle-seed/src/services/Generators.ts +++ b/drizzle-seed/src/services/Generators.ts @@ -38,9 +38,11 @@ export abstract class AbstractGenerator { public weightedCountSeed?: number | undefined; public maxRepeatedValuesCount?: number | { weight: number; count: number | number[] }[] | undefined; - // param for GenerateIntP + public params: T; - constructor(public params: T) {} + constructor(params?: T) { + this.params = params === undefined ? {} as T : params as T; + } init(params: { count: number | { weight: number; count: number | number[] }[]; seed: number }): void; init() { diff --git a/drizzle-seed/src/services/SeedService.ts b/drizzle-seed/src/services/SeedService.ts index 6c045f368..84165169e 100644 --- a/drizzle-seed/src/services/SeedService.ts +++ b/drizzle-seed/src/services/SeedService.ts @@ -212,7 +212,7 @@ export class SeedService { if (foreignKeyColumns[col.name]?.table === undefined && col.notNull === true) { throw new Error( - `Column '${col.name}' cannot be nullable, and you didn't specify a table for foreign key on column '${col.name}' in '${table.name}' table.`, + `Column '${col.name}' has no null contraint, and you didn't specify a table for foreign key on column '${col.name}' in '${table.name}' table. You should pass `, ); } @@ -222,14 +222,14 @@ export class SeedService { if (predicate === true) { if (foreignKeyColumns[col.name]?.table === undefined && col.notNull === false) { console.warn( - `Column '${col.name}' in '${table.name}' table will be filled with null values` + `Column '${col.name}' in '${table.name}' table will be filled with Null values` + `\nbecause you specified neither a table for foreign key on column '${col.name}' nor a function for '${col.name}' column in refinements.`, ); } - columnPossibleGenerator.generator = new generatorsMap.GenerateDefault[0]!({ defaultValue: null }); + columnPossibleGenerator.generator = new generatorsMap.GenerateDefault[0]({ defaultValue: null }); columnPossibleGenerator.wasDefinedBefore = true; } else { - columnPossibleGenerator.generator = new generatorsMap.HollowGenerator[0]!({}); + columnPossibleGenerator.generator = new generatorsMap.HollowGenerator[0](); } } // TODO: rewrite pickGeneratorFor... using new col properties: isUnique and notNull else if (connectionType === 'postgresql') { @@ -294,13 +294,12 @@ export class SeedService { (generator as GenerateArray).params.baseColumnGen = newBaseColumnGen; } - let possibleGeneratorConstructors = - (generatorsMap as { [entityKind: string]: (typeof AbstractGenerator)[] })[entityKind]; + const possibleGeneratorConstructors = generatorsMap[entityKind as keyof typeof generatorsMap]; - possibleGeneratorConstructors = possibleGeneratorConstructors?.filter((possGenCon) => + const possibleGeneratorConstructorsFiltered = possibleGeneratorConstructors?.filter((possGenCon) => possGenCon.version <= this.version! // sorting in ascending order by version ).sort((a, b) => a.version - b.version); - const generatorConstructor = possibleGeneratorConstructors?.at(-1) as + const generatorConstructor = possibleGeneratorConstructorsFiltered?.at(-1) as | (new(params: any) => AbstractGenerator) | undefined; if (generatorConstructor === undefined) { @@ -467,7 +466,7 @@ export class SeedService { count: number, seed: number, ) => { - let gen = new generatorsMap.GenerateWeightedCount[0]!({}); + let gen = new generatorsMap.GenerateWeightedCount[0](); gen = this.selectVersionOfGenerator(gen) as GenerateWeightedCount; // const gen = new GenerateWeightedCount({}); gen.init({ count: weightedCount, seed }); @@ -504,7 +503,7 @@ export class SeedService { // }; // const baseColumnDataType = getBaseColumnDataType(col.baseColumn); - const generator = new generatorsMap.GenerateArray[0]!({ baseColumnGen, size: col.size }); + const generator = new generatorsMap.GenerateArray[0]({ baseColumnGen, size: col.size }); // generator.baseColumnDataType = baseColumnDataType; return generator; @@ -524,10 +523,10 @@ export class SeedService { throw new Error(`column with type ${col.baseColumn!.columnType} is not supported for now.`); } - let generator = new generatorsMap.GenerateArray[0]!({ baseColumnGen }); + let generator = new generatorsMap.GenerateArray[0]({ baseColumnGen }); for (let i = 0; i < col.typeParams.dimensions! - 1; i++) { - generator = new generatorsMap.GenerateArray[0]!({ baseColumnGen: generator }); + generator = new generatorsMap.GenerateArray[0]({ baseColumnGen: generator }); } return generator; @@ -541,7 +540,7 @@ export class SeedService { || col.columnType.includes('bigint')) && table.primaryKeys.includes(col.name) ) { - const generator = new generatorsMap.GenerateIntPrimaryKey[0]!({}); + const generator = new generatorsMap.GenerateIntPrimaryKey[0](); return generator; } @@ -589,7 +588,7 @@ export class SeedService { && !col.columnType.includes('interval') && !col.columnType.includes('point') ) { - const generator = new generatorsMap.GenerateInt[0]!({ + const generator = new generatorsMap.GenerateInt[0]({ minValue, maxValue, }); @@ -598,7 +597,7 @@ export class SeedService { } if (col.columnType.includes('serial')) { - const generator = new generatorsMap.GenerateIntPrimaryKey[0]!({}); + const generator = new generatorsMap.GenerateIntPrimaryKey[0](); generator.maxValue = maxValue; @@ -617,14 +616,14 @@ export class SeedService { const scale = col.typeParams.scale === undefined ? 0 : col.typeParams.scale; const maxAbsoluteValue = Math.pow(10, precision - scale) - Math.pow(10, -scale); - const generator = new generatorsMap.GenerateNumber[0]!({ + const generator = new generatorsMap.GenerateNumber[0]({ minValue: -maxAbsoluteValue, maxValue: maxAbsoluteValue, precision: Math.pow(10, scale), }); return generator; } - const generator = new generatorsMap.GenerateNumber[0]!({}); + const generator = new generatorsMap.GenerateNumber[0](); return generator; } @@ -636,7 +635,7 @@ export class SeedService { || col.columnType.startsWith('char')) && table.primaryKeys.includes(col.name) ) { - const generator = new generatorsMap.GenerateUniqueString[0]!({}); + const generator = new generatorsMap.GenerateUniqueString[0](); return generator; } @@ -647,7 +646,7 @@ export class SeedService { || col.columnType.startsWith('char')) && col.name.toLowerCase().includes('name') ) { - const generator = new generatorsMap.GenerateFirstName[0]!({}); + const generator = new generatorsMap.GenerateFirstName[0](); return generator; } @@ -658,7 +657,7 @@ export class SeedService { || col.columnType.startsWith('char')) && col.name.toLowerCase().includes('email') ) { - const generator = new generatorsMap.GenerateEmail[0]!({}); + const generator = new generatorsMap.GenerateEmail[0](); return generator; } @@ -668,47 +667,47 @@ export class SeedService { || col.columnType.startsWith('varchar') || col.columnType.startsWith('char') ) { - const generator = new generatorsMap.GenerateString[0]!({}); + const generator = new generatorsMap.GenerateString[0](); return generator; } // UUID if (col.columnType === 'uuid') { - const generator = new generatorsMap.GenerateUUID[0]!({}); + const generator = new generatorsMap.GenerateUUID[0](); return generator; } // BOOLEAN if (col.columnType === 'boolean') { - const generator = new generatorsMap.GenerateBoolean[0]!({}); + const generator = new generatorsMap.GenerateBoolean[0](); return generator; } // DATE, TIME, TIMESTAMP if (col.columnType.includes('date')) { - const generator = new generatorsMap.GenerateDate[0]!({}); + const generator = new generatorsMap.GenerateDate[0](); return generator; } if (col.columnType === 'time') { - const generator = new generatorsMap.GenerateTime[0]!({}); + const generator = new generatorsMap.GenerateTime[0](); return generator; } if (col.columnType.includes('timestamp')) { - const generator = new generatorsMap.GenerateTimestamp[0]!({}); + const generator = new generatorsMap.GenerateTimestamp[0](); return generator; } // JSON, JSONB if (col.columnType === 'json' || col.columnType === 'jsonb') { - const generator = new generatorsMap.GenerateJson[0]!({}); + const generator = new generatorsMap.GenerateJson[0](); return generator; } @@ -720,7 +719,7 @@ export class SeedService { // ENUM if (col.enumValues !== undefined) { - const generator = new generatorsMap.GenerateEnum[0]!({ + const generator = new generatorsMap.GenerateEnum[0]({ enumValues: col.enumValues, }); @@ -730,32 +729,32 @@ export class SeedService { // INTERVAL if (col.columnType.startsWith('interval')) { if (col.columnType === 'interval') { - const generator = new generatorsMap.GenerateInterval[0]!({}); + const generator = new generatorsMap.GenerateInterval[0](); return generator; } const fields = col.columnType.replace('interval ', '') as GenerateInterval['params']['fields']; - const generator = new generatorsMap.GenerateInterval[0]!({ fields }); + const generator = new generatorsMap.GenerateInterval[0]({ fields }); return generator; } // POINT, LINE if (col.columnType.includes('point')) { - const generator = new generatorsMap.GeneratePoint[0]!({}); + const generator = new generatorsMap.GeneratePoint[0](); return generator; } if (col.columnType.includes('line')) { - const generator = new generatorsMap.GenerateLine[0]!({}); + const generator = new generatorsMap.GenerateLine[0](); return generator; } if (col.hasDefault && col.default !== undefined) { - const generator = new generatorsMap.GenerateDefault[0]!({ + const generator = new generatorsMap.GenerateDefault[0]({ defaultValue: col.default, }); return generator; @@ -784,7 +783,7 @@ export class SeedService { (col.columnType.includes('serial') || col.columnType.includes('int')) && table.primaryKeys.includes(col.name) ) { - const generator = new generatorsMap.GenerateIntPrimaryKey[0]!({}); + const generator = new generatorsMap.GenerateIntPrimaryKey[0](); return generator; } @@ -819,7 +818,7 @@ export class SeedService { } if (col.columnType.includes('int')) { - const generator = new generatorsMap.GenerateInt[0]!({ + const generator = new generatorsMap.GenerateInt[0]({ minValue, maxValue, }); @@ -827,7 +826,7 @@ export class SeedService { } if (col.columnType.includes('serial')) { - const generator = new generatorsMap.GenerateIntPrimaryKey[0]!({}); + const generator = new generatorsMap.GenerateIntPrimaryKey[0](); generator.maxValue = maxValue; return generator; } @@ -845,7 +844,7 @@ export class SeedService { const scale = col.typeParams.scale === undefined ? 0 : col.typeParams.scale; const maxAbsoluteValue = Math.pow(10, precision - scale) - Math.pow(10, -scale); - const generator = new generatorsMap.GenerateNumber[0]!({ + const generator = new generatorsMap.GenerateNumber[0]({ minValue: -maxAbsoluteValue, maxValue: maxAbsoluteValue, precision: Math.pow(10, scale), @@ -853,7 +852,7 @@ export class SeedService { return generator; } - const generator = new generatorsMap.GenerateNumber[0]!({}); + const generator = new generatorsMap.GenerateNumber[0](); return generator; } @@ -867,7 +866,7 @@ export class SeedService { || col.columnType.startsWith('varbinary')) && table.primaryKeys.includes(col.name) ) { - const generator = new generatorsMap.GenerateUniqueString[0]!({}); + const generator = new generatorsMap.GenerateUniqueString[0](); return generator; } @@ -880,7 +879,7 @@ export class SeedService { || col.columnType.startsWith('varbinary')) && col.name.toLowerCase().includes('name') ) { - const generator = new generatorsMap.GenerateFirstName[0]!({}); + const generator = new generatorsMap.GenerateFirstName[0](); return generator; } @@ -893,7 +892,7 @@ export class SeedService { || col.columnType.startsWith('varbinary')) && col.name.toLowerCase().includes('email') ) { - const generator = new generatorsMap.GenerateEmail[0]!({}); + const generator = new generatorsMap.GenerateEmail[0](); return generator; } @@ -905,58 +904,58 @@ export class SeedService { || col.columnType.startsWith('binary') || col.columnType.startsWith('varbinary') ) { - const generator = new generatorsMap.GenerateString[0]!({}); + const generator = new generatorsMap.GenerateString[0](); return generator; } // BOOLEAN if (col.columnType === 'boolean') { - const generator = new generatorsMap.GenerateBoolean[0]!({}); + const generator = new generatorsMap.GenerateBoolean[0](); return generator; } // DATE, TIME, TIMESTAMP, DATETIME, YEAR if (col.columnType.includes('datetime')) { - const generator = new generatorsMap.GenerateDatetime[0]!({}); + const generator = new generatorsMap.GenerateDatetime[0](); return generator; } if (col.columnType.includes('date')) { - const generator = new generatorsMap.GenerateDate[0]!({}); + const generator = new generatorsMap.GenerateDate[0](); return generator; } if (col.columnType === 'time') { - const generator = new generatorsMap.GenerateTime[0]!({}); + const generator = new generatorsMap.GenerateTime[0](); return generator; } if (col.columnType.includes('timestamp')) { - const generator = new generatorsMap.GenerateTimestamp[0]!({}); + const generator = new generatorsMap.GenerateTimestamp[0](); return generator; } if (col.columnType === 'year') { - const generator = new generatorsMap.GenerateYear[0]!({}); + const generator = new generatorsMap.GenerateYear[0](); return generator; } // JSON if (col.columnType === 'json') { - const generator = new generatorsMap.GenerateJson[0]!({}); + const generator = new generatorsMap.GenerateJson[0](); return generator; } // ENUM if (col.enumValues !== undefined) { - const generator = new generatorsMap.GenerateEnum[0]!({ + const generator = new generatorsMap.GenerateEnum[0]({ enumValues: col.enumValues, }); return generator; } if (col.hasDefault && col.default !== undefined) { - const generator = new generatorsMap.GenerateDefault[0]!({ + const generator = new generatorsMap.GenerateDefault[0]({ defaultValue: col.default, }); return generator; @@ -980,17 +979,17 @@ export class SeedService { (col.columnType === 'integer' || col.columnType === 'numeric') && table.primaryKeys.includes(col.name) ) { - const generator = new generatorsMap.GenerateIntPrimaryKey[0]!({}); + const generator = new generatorsMap.GenerateIntPrimaryKey[0](); return generator; } if (col.columnType === 'integer' && col.dataType === 'boolean') { - const generator = new generatorsMap.GenerateBoolean[0]!({}); + const generator = new generatorsMap.GenerateBoolean[0](); return generator; } if ((col.columnType === 'integer' && col.dataType === 'date')) { - const generator = new generatorsMap.GenerateTimestamp[0]!({}); + const generator = new generatorsMap.GenerateTimestamp[0](); return generator; } @@ -998,7 +997,7 @@ export class SeedService { col.columnType === 'integer' || (col.dataType === 'bigint' && col.columnType === 'blob') ) { - const generator = new generatorsMap.GenerateInt[0]!({}); + const generator = new generatorsMap.GenerateInt[0](); return generator; } @@ -1009,7 +1008,7 @@ export class SeedService { const scale = col.typeParams.scale === undefined ? 0 : col.typeParams.scale; const maxAbsoluteValue = Math.pow(10, precision - scale) - Math.pow(10, -scale); - const generator = new generatorsMap.GenerateNumber[0]!({ + const generator = new generatorsMap.GenerateNumber[0]({ minValue: -maxAbsoluteValue, maxValue: maxAbsoluteValue, precision: Math.pow(10, scale), @@ -1017,7 +1016,7 @@ export class SeedService { return generator; } - const generator = new generatorsMap.GenerateNumber[0]!({}); + const generator = new generatorsMap.GenerateNumber[0](); return generator; } @@ -1028,7 +1027,7 @@ export class SeedService { || col.columnType.startsWith('blob')) && table.primaryKeys.includes(col.name) ) { - const generator = new generatorsMap.GenerateUniqueString[0]!({}); + const generator = new generatorsMap.GenerateUniqueString[0](); return generator; } @@ -1038,7 +1037,7 @@ export class SeedService { || col.columnType.startsWith('blob')) && col.name.toLowerCase().includes('name') ) { - const generator = new generatorsMap.GenerateFirstName[0]!({}); + const generator = new generatorsMap.GenerateFirstName[0](); return generator; } @@ -1048,7 +1047,7 @@ export class SeedService { || col.columnType.startsWith('blob')) && col.name.toLowerCase().includes('email') ) { - const generator = new generatorsMap.GenerateEmail[0]!({}); + const generator = new generatorsMap.GenerateEmail[0](); return generator; } @@ -1058,7 +1057,7 @@ export class SeedService { || col.columnType.startsWith('blob') || col.columnType.startsWith('blobbuffer') ) { - const generator = new generatorsMap.GenerateString[0]!({}); + const generator = new generatorsMap.GenerateString[0](); return generator; } @@ -1066,12 +1065,12 @@ export class SeedService { (col.columnType.startsWith('text') && col.dataType === 'json') || (col.columnType.startsWith('blob') && col.dataType === 'json') ) { - const generator = new generatorsMap.GenerateJson[0]!({}); + const generator = new generatorsMap.GenerateJson[0](); return generator; } if (col.hasDefault && col.default !== undefined) { - const generator = new generatorsMap.GenerateDefault[0]!({ + const generator = new generatorsMap.GenerateDefault[0]({ defaultValue: col.default, }); return generator; @@ -1239,7 +1238,7 @@ export class SeedService { }))!.map((rows) => rows[refColName]) as (string | number | boolean)[]; hasSelfRelation = true; - genObj = new generatorsMap.GenerateSelfRelationsValuesFromArray[0]!({ + genObj = new generatorsMap.GenerateSelfRelationsValuesFromArray[0]({ values: refColumnValues, }); genObj = this.selectVersionOfGenerator(genObj); @@ -1264,7 +1263,7 @@ export class SeedService { } // TODO: revise maybe need to select version of generator here too - genObj = new generatorsMap.GenerateValuesFromArray[0]!({ values: refColumnValues }); + genObj = new generatorsMap.GenerateValuesFromArray[0]({ values: refColumnValues }); genObj.notNull = tableGenerators[rel.columns[colIdx]!]!.notNull; genObj.weightedCountSeed = weightedCountSeed; genObj.maxRepeatedValuesCount = repeatedValuesCount; From 414ffd95154df20ce017384dffc5c493b8b13d75 Mon Sep 17 00:00:00 2001 From: Oleksii Khomenko <47694554+OleksiiKH0240@users.noreply.github.com> Date: Wed, 25 Dec 2024 09:30:17 +0200 Subject: [PATCH 25/32] Drizzle seed/studio (#3808) * support varchar for postgres using sql column type * added checks to string-like generators, updated sqlite and mysql seeding using sql column type * added array support for studio * updated postgres interval generators to work with fields, added tests * fixes, partial implementation of generator versioning * added generator versioning with version v1 for string and interval generators * updated changelogs * updated generator versioning * updated changelogs * Review * changes in generator selection * updated generator versioning * bug fix, refine functions versioning * fixes * fixes * Change release notes text * Bump version --------- Co-authored-by: Andrii Sherman --- changelogs/drizzle-seed/0.2.0.md | 166 +++ drizzle-seed/package.json | 4 +- drizzle-seed/src/datasets/adjectives.ts | 2 + drizzle-seed/src/datasets/cityNames.ts | 2 + .../src/datasets/companyNameSuffixes.ts | 2 + drizzle-seed/src/datasets/countries.ts | 2 + drizzle-seed/src/datasets/emailDomains.ts | 2 + drizzle-seed/src/datasets/firstNames.ts | 2 + drizzle-seed/src/datasets/jobsTitles.ts | 2 + drizzle-seed/src/datasets/lastNames.ts | 2 + .../src/datasets/loremIpsumSentences.ts | 2 + drizzle-seed/src/datasets/states.ts | 2 + drizzle-seed/src/datasets/streetSuffix.ts | 2 + drizzle-seed/src/index.ts | 146 +- drizzle-seed/src/services/GeneratorFuncs.ts | 918 ++++++++++++ .../{GeneratorsWrappers.ts => Generators.ts} | 1232 ++++++----------- drizzle-seed/src/services/SeedService.ts | 1142 ++++++++------- drizzle-seed/src/services/apiVersion.ts | 1 + drizzle-seed/src/services/versioning/v2.ts | 232 ++++ drizzle-seed/src/types/seedService.ts | 2 +- drizzle-seed/src/types/tables.ts | 6 + .../tests/benchmarks/generatorsBenchmark.ts | 10 +- .../mysql_all_data_types.test.ts | 2 +- .../tests/pg/allDataTypesTest/pgSchema.ts | 62 +- .../pg_all_data_types.test.ts | 30 + 25 files changed, 2567 insertions(+), 1408 deletions(-) create mode 100644 changelogs/drizzle-seed/0.2.0.md create mode 100644 drizzle-seed/src/services/GeneratorFuncs.ts rename drizzle-seed/src/services/{GeneratorsWrappers.ts => Generators.ts} (73%) create mode 100644 drizzle-seed/src/services/apiVersion.ts create mode 100644 drizzle-seed/src/services/versioning/v2.ts diff --git a/changelogs/drizzle-seed/0.2.0.md b/changelogs/drizzle-seed/0.2.0.md new file mode 100644 index 000000000..a34509510 --- /dev/null +++ b/changelogs/drizzle-seed/0.2.0.md @@ -0,0 +1,166 @@ +## API updates + +We are introducing a new parameter, `version`, to the `seed` function options. This parameter, which controls generator versioning, has been added to make it easier to update deterministic generators in the future. Since values should remain consistent after each regeneration, it is crucial to provide a well-designed API for gradual updates + +```ts +await seed(db, schema, { version: '2' }); +``` + +#### Example: + +> This is not an actual API change; it is just an example of how we will proceed with `drizzle-seed` versioning + +For example, `lastName` generator was changed, and new version, `V2`, of this generator became available. + +Later, `firstName` generator was changed, making `V3` version of this generator available. + +| | `V1` | `V2` | `V3(latest)` | +| :--------------: | :--------------: | :-------------: | :--------------: | +| **LastNameGen** | `LastNameGenV1` | `LastNameGenV2` | | +| **FirstNameGen** | `FirstNameGenV1` | | `FirstNameGenV3` | + + +##### Use the `firstName` generator of version 3 and the `lastName` generator of version 2 +```ts +await seed(db, schema); +``` + +If you are not ready to use latest generator version right away, you can specify max version to use + +##### Use the `firstName` generator of version 1 and the `lastName` generator of version 2 +```ts +await seed(db, schema, { version: '2' }); +``` + +##### Use the `firstName` generator of version 1 and the `lastName` generator of version 1. +```ts +await seed(db, schema, { version: '1' }); +``` + +Each update with breaking changes for generators will be documented on our docs and in release notes, explaining which version you should use, if you are not ready to upgrade the way generators works + +## Breaking changes + +### `interval` unique generator was changed and upgraded to v2 + +```ts +await seed(db, { table }).refine((f) => ({ + table: { + columns: { + // this function usage will output different values with the same `seed` number from previous version + column1: f.interval({ isUnique: true }), + } + } +})) +``` + +**Reason for upgrade** +An older version of the generator could produce intervals like `1 minute 60 seconds` and `2 minutes 0 seconds`, treating them as distinct intervals. +However, when the `1 minute 60 seconds` interval is inserted into a PostgreSQL database, it is automatically converted to `2 minutes 0 seconds`. As a result, attempting to insert the `2 minutes 0 seconds` interval into a unique column afterwards will cause an error + +**Usage** +```ts +await seed(db, schema); +// or explicit +await seed(db, schema, { version: '2' }); +``` + +**Switch to the old version** +```ts +await seed(db, schema, { version: '1' }); +``` + +### `string` generators were changed and upgraded to v2 + +```ts +await seed(db, { table }).refine((f) => ({ + table: { + columns: { + // this function will output different values with the same `seed` number from previous version + column1: f.string(), + } + } +})) +``` + +**Reason to upgrade** + +Ability to generate a unique string based on the length of the text column (e.g., `varchar(20)`) + +#### PostgreSQL changes + +Default generators for `text`, `varchar`, `char` will output different values with the same `seed` number from previous version. + +```ts +// schema.ts +import * as p from 'drizzle-orm/pg-core' + +export const table = p.pgTable('table', { + column1: p.text(), + column2: p.varchar(), + column3: p.char() +}); + +// index.ts +... +// this will be affected with new changes +await seed(db, { table }); +``` + +**Switch to the old version** +```ts +await seed(db, schema, { version: '' }); +``` + +#### MySQL changes + +Default generators for `text`, `char`, `varchar`, `binary`, `varbinary` will output different values with the same `seed` number. + +```ts +// schema.ts +import * as p from 'drizzle-orm/mysql-core' + +export const table = p.mysqlTable('table', { + column1: p.text(), + column2: p.char(), + column3: p.varchar({ length: 256 }), + column4: p.binary(), + column5: p.varbinary({ length: 256 }), +}); + +// index.ts +... +// this will be affected with new changes +await seed(db, {table}) +``` + +**Switch to the old version** +```ts +await seed(db, schema, { version: '1' }); +``` + +#### SQLite changes + +Default generators for `text`, `numeric`, `blob`, `blobbuffer` will output different values with the same `seed` number. +```ts +// schema.ts +import * as p from 'drizzle-orm/sqlite-core' + +export const table = p.sqliteTable('table', { + column1: p.text(), + column2: p.numeric(), + column3: p.blob({ mode:'buffer' }), + column4: p.blob(), +}); + +// index.ts +... +// this will be affected with new changes +await seed(db, { table }) +``` + +Functions from refine that will be affected by this change: `` + +## Bug fixes +- Seeding a table with a foreign key referencing another table, without including the second table in the schema, will cause the seeding process to get stuck +- [[BUG]: seeding postgresql char column doesn't respect length option](https://github.com/drizzle-team/drizzle-orm/issues/3774) \ No newline at end of file diff --git a/drizzle-seed/package.json b/drizzle-seed/package.json index bfa0e39ad..ae5648cdb 100644 --- a/drizzle-seed/package.json +++ b/drizzle-seed/package.json @@ -1,6 +1,6 @@ { "name": "drizzle-seed", - "version": "0.1.3", + "version": "0.2.0", "main": "index.js", "type": "module", "scripts": { @@ -102,4 +102,4 @@ "dependencies": { "pure-rand": "^6.1.0" } -} +} \ No newline at end of file diff --git a/drizzle-seed/src/datasets/adjectives.ts b/drizzle-seed/src/datasets/adjectives.ts index c2b152af0..880e52636 100644 --- a/drizzle-seed/src/datasets/adjectives.ts +++ b/drizzle-seed/src/datasets/adjectives.ts @@ -4844,3 +4844,5 @@ export default [ 'zonked', 'zoological', ]; + +export const maxStringLength = 22; diff --git a/drizzle-seed/src/datasets/cityNames.ts b/drizzle-seed/src/datasets/cityNames.ts index 780b55213..3ea80747e 100644 --- a/drizzle-seed/src/datasets/cityNames.ts +++ b/drizzle-seed/src/datasets/cityNames.ts @@ -42857,3 +42857,5 @@ export default [ 'Garches', 'Chemini', ]; + +export const maxStringLength = 49; diff --git a/drizzle-seed/src/datasets/companyNameSuffixes.ts b/drizzle-seed/src/datasets/companyNameSuffixes.ts index ae8ce6163..1ce31a9c3 100644 --- a/drizzle-seed/src/datasets/companyNameSuffixes.ts +++ b/drizzle-seed/src/datasets/companyNameSuffixes.ts @@ -25,3 +25,5 @@ export default [ 'Co.', 'SCC', ]; + +export const maxStringLength = 7; diff --git a/drizzle-seed/src/datasets/countries.ts b/drizzle-seed/src/datasets/countries.ts index 4808fc5e5..eb1c001d0 100644 --- a/drizzle-seed/src/datasets/countries.ts +++ b/drizzle-seed/src/datasets/countries.ts @@ -169,3 +169,5 @@ export default [ 'Yemen', 'Zambia', ]; + +export const maxStringLength = 30; diff --git a/drizzle-seed/src/datasets/emailDomains.ts b/drizzle-seed/src/datasets/emailDomains.ts index 9904aad3e..ea323ed41 100644 --- a/drizzle-seed/src/datasets/emailDomains.ts +++ b/drizzle-seed/src/datasets/emailDomains.ts @@ -22,3 +22,5 @@ export default [ 'ymail.com', 'libero.it', ]; + +export const maxStringLength = 14; diff --git a/drizzle-seed/src/datasets/firstNames.ts b/drizzle-seed/src/datasets/firstNames.ts index 7ca0ff928..d719aa4e2 100644 --- a/drizzle-seed/src/datasets/firstNames.ts +++ b/drizzle-seed/src/datasets/firstNames.ts @@ -30277,3 +30277,5 @@ export default [ 'Lavasia', 'Laniqua', ]; + +export const maxStringLength = 15; diff --git a/drizzle-seed/src/datasets/jobsTitles.ts b/drizzle-seed/src/datasets/jobsTitles.ts index 3a38e3244..e7993da2a 100644 --- a/drizzle-seed/src/datasets/jobsTitles.ts +++ b/drizzle-seed/src/datasets/jobsTitles.ts @@ -150,3 +150,5 @@ export default [ 'Legal secretary', 'Market analyst', ]; + +export const maxStringLength = 35; diff --git a/drizzle-seed/src/datasets/lastNames.ts b/drizzle-seed/src/datasets/lastNames.ts index 117c5fe28..9d35f7cf7 100644 --- a/drizzle-seed/src/datasets/lastNames.ts +++ b/drizzle-seed/src/datasets/lastNames.ts @@ -50001,3 +50001,5 @@ export default [ 'Thagard', 'Leavelle', ]; + +export const maxStringLength = 15; diff --git a/drizzle-seed/src/datasets/loremIpsumSentences.ts b/drizzle-seed/src/datasets/loremIpsumSentences.ts index f03277d86..64fe59f71 100644 --- a/drizzle-seed/src/datasets/loremIpsumSentences.ts +++ b/drizzle-seed/src/datasets/loremIpsumSentences.ts @@ -1637,3 +1637,5 @@ export default [ 'Sed gravida enim quis nunc interdum imperdiet.', 'Proin cursus odio ac dolor blandit, quis sollicitudin ante rutrum.', ]; + +export const maxStringLength = 190; diff --git a/drizzle-seed/src/datasets/states.ts b/drizzle-seed/src/datasets/states.ts index 1de77160d..cd66cf330 100644 --- a/drizzle-seed/src/datasets/states.ts +++ b/drizzle-seed/src/datasets/states.ts @@ -50,3 +50,5 @@ export default [ 'Wisconsin', 'Wyoming', ]; + +export const maxStringLength = 14; diff --git a/drizzle-seed/src/datasets/streetSuffix.ts b/drizzle-seed/src/datasets/streetSuffix.ts index e9b20c392..90a70c2c6 100644 --- a/drizzle-seed/src/datasets/streetSuffix.ts +++ b/drizzle-seed/src/datasets/streetSuffix.ts @@ -198,3 +198,5 @@ export default [ 'Well', 'Wells', ]; + +export const maxStringLength = 10; diff --git a/drizzle-seed/src/index.ts b/drizzle-seed/src/index.ts index 603512fd3..98c449095 100644 --- a/drizzle-seed/src/index.ts +++ b/drizzle-seed/src/index.ts @@ -1,4 +1,5 @@ -import { entityKind, getTableName, is, sql } from 'drizzle-orm'; +/* eslint-disable drizzle-internal/require-entity-kind */ +import { getTableName, is, sql } from 'drizzle-orm'; import type { MySqlColumn, MySqlSchema } from 'drizzle-orm/mysql-core'; import { getTableConfig as getMysqlTableConfig, MySqlDatabase, MySqlTable } from 'drizzle-orm/mysql-core'; @@ -9,8 +10,8 @@ import { getTableConfig as getPgTableConfig, PgDatabase, PgTable } from 'drizzle import type { SQLiteColumn } from 'drizzle-orm/sqlite-core'; import { BaseSQLiteDatabase, getTableConfig as getSqliteTableConfig, SQLiteTable } from 'drizzle-orm/sqlite-core'; -import type { AbstractGenerator } from './services/GeneratorsWrappers.ts'; -import { generatorsFuncs } from './services/GeneratorsWrappers.ts'; +import { generatorsFuncs, generatorsFuncsV2 } from './services/GeneratorFuncs.ts'; +import type { AbstractGenerator } from './services/Generators.ts'; import { SeedService } from './services/SeedService.ts'; import type { DrizzleStudioObjectType, DrizzleStudioRelationType } from './types/drizzleStudio.ts'; import type { RefinementsType } from './types/seedService.ts'; @@ -130,15 +131,16 @@ class SeedPromise< SCHEMA extends { [key: string]: PgTable | PgSchema | MySqlTable | MySqlSchema | SQLiteTable; }, + VERSION extends string | undefined, > implements Promise { - static readonly [entityKind]: string = 'SeedPromise'; + static readonly entityKind: string = 'SeedPromise'; [Symbol.toStringTag] = 'SeedPromise'; constructor( private db: DB, private schema: SCHEMA, - private options?: { count?: number; seed?: number }, + private options?: { count?: number; seed?: number; version?: VERSION }, ) {} then( @@ -180,14 +182,22 @@ class SeedPromise< } async refine( - callback: (funcs: typeof generatorsFuncs) => InferCallbackType, + callback: ( + funcs: FunctionsVersioning, + ) => InferCallbackType, ): Promise { - const refinements = callback(generatorsFuncs) as RefinementsType; + const refinements = this.options?.version === undefined || this.options.version === '2' + ? callback(generatorsFuncsV2 as FunctionsVersioning) as RefinementsType + : callback(generatorsFuncs as FunctionsVersioning) as RefinementsType; await seedFunc(this.db, this.schema, this.options, refinements); } } +type FunctionsVersioning = VERSION extends `1` ? typeof generatorsFuncs + : VERSION extends `2` ? typeof generatorsFuncsV2 + : typeof generatorsFuncsV2; + export function getGeneratorsFunctions() { return generatorsFuncs; } @@ -222,6 +232,8 @@ export async function seedForDrizzleStudio( name: col.name, dataType: 'string', columnType: col.type, + // TODO: revise later + typeParams: {}, default: col.default, hasDefault: col.default === undefined ? false : true, isUnique: col.isUnique === undefined ? false : col.isUnique, @@ -257,7 +269,6 @@ export async function seedForDrizzleStudio( sqlDialect, tables, isCyclicRelations, - {}, // TODO: fix later refinements, options, ); @@ -324,7 +335,7 @@ export async function seedForDrizzleStudio( export function seed< DB extends | PgDatabase - | MySqlDatabase + | MySqlDatabase | BaseSQLiteDatabase, SCHEMA extends { [key: string]: @@ -335,8 +346,9 @@ export function seed< | SQLiteTable | any; }, ->(db: DB, schema: SCHEMA, options?: { count?: number; seed?: number }) { - return new SeedPromise(db, schema, options); + VERSION extends '2' | '1' | undefined, +>(db: DB, schema: SCHEMA, options?: { count?: number; seed?: number; version?: VERSION }) { + return new SeedPromise(db, schema, options); } const seedFunc = async ( @@ -350,21 +362,26 @@ const seedFunc = async ( | SQLiteTable | any; }, - options: { count?: number; seed?: number } = {}, + options: { count?: number; seed?: number; version?: string } = {}, refinements?: RefinementsType, ) => { + let version: number | undefined; + if (options?.version !== undefined) { + version = Number(options?.version); + } + if (is(db, PgDatabase)) { const { pgSchema } = filterPgTables(schema); - await seedPostgres(db, pgSchema, options, refinements); + await seedPostgres(db, pgSchema, { ...options, version }, refinements); } else if (is(db, MySqlDatabase)) { const { mySqlSchema } = filterMySqlTables(schema); - await seedMySql(db, mySqlSchema, options, refinements); + await seedMySql(db, mySqlSchema, { ...options, version }, refinements); } else if (is(db, BaseSQLiteDatabase)) { const { sqliteSchema } = filterSqliteTables(schema); - await seedSqlite(db, sqliteSchema, options, refinements); + await seedSqlite(db, sqliteSchema, { ...options, version }, refinements); } else { throw new Error( 'The drizzle-seed package currently supports only PostgreSQL, MySQL, and SQLite databases. Please ensure your database is one of these supported types', @@ -417,7 +434,7 @@ const seedFunc = async ( export async function reset< DB extends | PgDatabase - | MySqlDatabase + | MySqlDatabase | BaseSQLiteDatabase, SCHEMA extends { [key: string]: @@ -488,17 +505,16 @@ const filterPgTables = (schema: { const seedPostgres = async ( db: PgDatabase, schema: { [key: string]: PgTable }, - options: { count?: number; seed?: number } = {}, + options: { count?: number; seed?: number; version?: number } = {}, refinements?: RefinementsType, ) => { const seedService = new SeedService(); - const { tables, relations, tableRelations } = getPostgresInfo(schema); + const { tables, relations } = getPostgresInfo(schema); const generatedTablesGenerators = seedService.generatePossibleGenerators( 'postgresql', tables, relations, - tableRelations, refinements, options, ); @@ -608,7 +624,8 @@ const getPostgresInfo = (schema: { [key: string]: PgTable }) => { ): Column['baseColumn'] => { const baseColumnResult: Column['baseColumn'] = { name: baseColumn.name, - columnType: baseColumn.columnType.replace('Pg', '').toLowerCase(), + columnType: baseColumn.getSQLType(), + typeParams: getTypeParams(baseColumn.getSQLType()), dataType: baseColumn.dataType, size: (baseColumn as PgArray).size, hasDefault: baseColumn.hasDefault, @@ -623,12 +640,54 @@ const getPostgresInfo = (schema: { [key: string]: PgTable }) => { return baseColumnResult; }; + const getTypeParams = (sqlType: string) => { + // get type params + const typeParams: Column['typeParams'] = {}; + + // handle dimensions + if (sqlType.includes('[')) { + const match = sqlType.match(/\[\w*]/g); + if (match) { + typeParams['dimensions'] = match.length; + } + } + + if ( + sqlType.startsWith('numeric') + || sqlType.startsWith('decimal') + || sqlType.startsWith('double precision') + || sqlType.startsWith('real') + ) { + const match = sqlType.match(/\((\d+), *(\d+)\)/); + if (match) { + typeParams['precision'] = Number(match[1]); + typeParams['scale'] = Number(match[2]); + } + } else if ( + sqlType.startsWith('varchar') + || sqlType.startsWith('bpchar') + || sqlType.startsWith('char') + || sqlType.startsWith('bit') + || sqlType.startsWith('time') + || sqlType.startsWith('timestamp') + || sqlType.startsWith('interval') + ) { + const match = sqlType.match(/\((\d+)\)/); + if (match) { + typeParams['length'] = Number(match[1]); + } + } + + return typeParams; + }; + // console.log(tableConfig.columns); tables.push({ name: dbToTsTableNamesMap[tableConfig.name] as string, columns: tableConfig.columns.map((column) => ({ name: dbToTsColumnNamesMap[column.name] as string, - columnType: column.columnType.replace('Pg', '').toLowerCase(), + columnType: column.getSQLType(), + typeParams: getTypeParams(column.getSQLType()), dataType: column.dataType, size: (column as PgArray).size, hasDefault: column.hasDefault, @@ -739,10 +798,10 @@ const filterMySqlTables = (schema: { const seedMySql = async ( db: MySqlDatabase, schema: { [key: string]: MySqlTable }, - options: { count?: number; seed?: number } = {}, + options: { count?: number; seed?: number; version?: number } = {}, refinements?: RefinementsType, ) => { - const { tables, relations, tableRelations } = getMySqlInfo(schema); + const { tables, relations } = getMySqlInfo(schema); const seedService = new SeedService(); @@ -750,7 +809,6 @@ const seedMySql = async ( 'mysql', tables, relations, - tableRelations, refinements, options, ); @@ -853,11 +911,41 @@ const getMySqlInfo = (schema: { [key: string]: MySqlTable }) => { } tableRelations[dbToTsTableNamesMap[tableConfig.name] as string]!.push(...newRelations); + const getTypeParams = (sqlType: string) => { + // get type params and set only type + const typeParams: Column['typeParams'] = {}; + + if ( + sqlType.startsWith('decimal') + || sqlType.startsWith('real') + || sqlType.startsWith('double') + || sqlType.startsWith('float') + ) { + const match = sqlType.match(/\((\d+), *(\d+)\)/); + if (match) { + typeParams['precision'] = Number(match[1]); + typeParams['scale'] = Number(match[2]); + } + } else if ( + sqlType.startsWith('varchar') + || sqlType.startsWith('binary') + || sqlType.startsWith('varbinary') + ) { + const match = sqlType.match(/\((\d+)\)/); + if (match) { + typeParams['length'] = Number(match[1]); + } + } + + return typeParams; + }; + tables.push({ name: dbToTsTableNamesMap[tableConfig.name] as string, columns: tableConfig.columns.map((column) => ({ name: dbToTsColumnNamesMap[column.name] as string, - columnType: column.columnType.replace('MySql', '').toLowerCase(), + columnType: column.getSQLType(), + typeParams: getTypeParams(column.getSQLType()), dataType: column.dataType, hasDefault: column.hasDefault, default: column.default, @@ -928,10 +1016,10 @@ const filterSqliteTables = (schema: { const seedSqlite = async ( db: BaseSQLiteDatabase, schema: { [key: string]: SQLiteTable }, - options: { count?: number; seed?: number } = {}, + options: { count?: number; seed?: number; version?: number } = {}, refinements?: RefinementsType, ) => { - const { tables, relations, tableRelations } = getSqliteInfo(schema); + const { tables, relations } = getSqliteInfo(schema); const seedService = new SeedService(); @@ -939,7 +1027,6 @@ const seedSqlite = async ( 'sqlite', tables, relations, - tableRelations, refinements, options, ); @@ -1046,7 +1133,8 @@ const getSqliteInfo = (schema: { [key: string]: SQLiteTable }) => { name: dbToTsTableNamesMap[tableConfig.name] as string, columns: tableConfig.columns.map((column) => ({ name: dbToTsColumnNamesMap[column.name] as string, - columnType: column.columnType.replace('SQLite', '').toLowerCase(), + columnType: column.getSQLType(), + typeParams: {}, dataType: column.dataType, hasDefault: column.hasDefault, default: column.default, diff --git a/drizzle-seed/src/services/GeneratorFuncs.ts b/drizzle-seed/src/services/GeneratorFuncs.ts new file mode 100644 index 000000000..10d0d10f7 --- /dev/null +++ b/drizzle-seed/src/services/GeneratorFuncs.ts @@ -0,0 +1,918 @@ +import type { AbstractGenerator } from './Generators.ts'; +import { + GenerateArray, + GenerateBoolean, + GenerateCity, + GenerateCompanyName, + GenerateCountry, + GenerateDate, + GenerateDatetime, + GenerateDefault, + GenerateEmail, + GenerateEnum, + GenerateFirstName, + GenerateFullName, + GenerateInt, + GenerateInterval, + GenerateIntPrimaryKey, + GenerateJobTitle, + GenerateJson, + GenerateLastName, + GenerateLine, + GenerateLoremIpsum, + GenerateNumber, + GeneratePhoneNumber, + GeneratePoint, + GeneratePostcode, + GenerateSelfRelationsValuesFromArray, + GenerateState, + GenerateStreetAddress, + GenerateString, + GenerateTime, + GenerateTimestamp, + GenerateUniqueCity, + GenerateUniqueCompanyName, + GenerateUniqueCountry, + GenerateUniqueFirstName, + GenerateUniqueFullName, + GenerateUniqueInt, + GenerateUniqueInterval, + GenerateUniqueLastName, + GenerateUniqueLine, + GenerateUniqueNumber, + GenerateUniquePoint, + GenerateUniquePostcode, + GenerateUniqueStreetAddress, + GenerateUniqueString, + GenerateUUID, + GenerateValuesFromArray, + GenerateWeightedCount, + GenerateYear, + HollowGenerator, + WeightedRandomGenerator, +} from './Generators.ts'; +import { GenerateStringV2, GenerateUniqueIntervalV2, GenerateUniqueStringV2 } from './versioning/v2.ts'; + +function createGenerator, T>( + generatorConstructor: new(params?: T) => GeneratorType, +) { + return ( + ...args: GeneratorType extends GenerateValuesFromArray | GenerateDefault | WeightedRandomGenerator ? [T] + : ([] | [T]) + ): GeneratorType => { + let params = args[0]; + if (params === undefined) params = {} as T; + return new generatorConstructor(params); + }; +} + +export const generatorsFuncs = { + /** + * generates same given value each time the generator is called. + * @param defaultValue - value you want to generate + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * posts: { + * columns: { + * content: funcs.default({ defaultValue: "post content" }), + * }, + * }, + * })); + * ``` + */ + default: createGenerator(GenerateDefault), + + /** + * generates values from given array + * @param values - array of values you want to generate. can be array of weighted values. + * @param isUnique - property that controls if generated values gonna be unique or not. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * posts: { + * columns: { + * title: funcs.valuesFromArray({ + * values: ["Title1", "Title2", "Title3", "Title4", "Title5"], + * isUnique: true + * }), + * }, + * }, + * })); + * + * ``` + * weighted values example + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * posts: { + * columns: { + * title: funcs.valuesFromArray({ + * values: [ + * { weight: 0.35, values: ["Title1", "Title2"] }, + * { weight: 0.5, values: ["Title3", "Title4"] }, + * { weight: 0.15, values: ["Title5"] }, + * ], + * isUnique: false + * }), + * }, + * }, + * })); + * + * ``` + */ + valuesFromArray: createGenerator(GenerateValuesFromArray), + + /** + * generates sequential integers starting with 1. + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * posts: { + * columns: { + * id: funcs.intPrimaryKey(), + * }, + * }, + * })); + * + * ``` + */ + intPrimaryKey: createGenerator(GenerateIntPrimaryKey), + + /** + * generates numbers with floating point in given range. + * @param minValue - lower border of range. + * @param maxValue - upper border of range. + * @param precision - precision of generated number: + * precision equals 10 means that values will be accurate to one tenth (1.2, 34.6); + * precision equals 100 means that values will be accurate to one hundredth (1.23, 34.67). + * @param isUnique - property that controls if generated values gonna be unique or not. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * products: { + * columns: { + * unitPrice: funcs.number({ minValue: 10, maxValue: 120, precision: 100, isUnique: false }), + * }, + * }, + * })); + * + * ``` + */ + number: createGenerator(GenerateNumber), + // uniqueNumber: createGenerator(GenerateUniqueNumber), + + /** + * generates integers within given range. + * @param minValue - lower border of range. + * @param maxValue - upper border of range. + * @param isUnique - property that controls if generated values gonna be unique or not. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * products: { + * columns: { + * unitsInStock: funcs.number({ minValue: 0, maxValue: 100, isUnique: false }), + * }, + * }, + * })); + * + * ``` + */ + int: createGenerator(GenerateInt), + // uniqueInt: createGenerator(GenerateUniqueInt), + + /** + * generates boolean values(true or false) + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * users: { + * columns: { + * isAvailable: funcs.boolean() + * }, + * }, + * })); + * + * ``` + */ + boolean: createGenerator(GenerateBoolean), + + /** + * generates date within given range. + * @param minDate - lower border of range. + * @param maxDate - upper border of range. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * users: { + * columns: { + * birthDate: funcs.date({ minDate: "1990-01-01", maxDate: "2010-12-31" }) + * }, + * }, + * })); + * + * ``` + */ + date: createGenerator(GenerateDate), + + /** + * generates time in 24 hours style. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * users: { + * columns: { + * birthTime: funcs.time() + * }, + * }, + * })); + * + * ``` + */ + time: createGenerator(GenerateTime), + + /** + * generates timestamps. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * orders: { + * columns: { + * shippedDate: funcs.timestamp() + * }, + * }, + * })); + * + * ``` + */ + timestamp: createGenerator(GenerateTimestamp), + + /** + * generates datetime objects. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * orders: { + * columns: { + * shippedDate: funcs.datetime() + * }, + * }, + * })); + * + * ``` + */ + datetime: createGenerator(GenerateDatetime), + + /** + * generates years. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * users: { + * columns: { + * birthYear: funcs.year() + * }, + * }, + * })); + * + * ``` + */ + year: createGenerator(GenerateYear), + + /** + * generates json objects with fixed structure. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * json structure can equal this: + * ``` + * { + * email, + * name, + * isGraduated, + * hasJob, + * salary, + * startedWorking, + * visitedCountries, + * } + * ``` + * or this + * ``` + * { + * email, + * name, + * isGraduated, + * hasJob, + * visitedCountries, + * } + * ``` + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * users: { + * columns: { + * metadata: funcs.json() + * }, + * }, + * })); + * ``` + */ + json: createGenerator(GenerateJson), + // jsonb: createGenerator(GenerateJsonb), + + /** + * generates time intervals. + * + * interval example: "1 years 12 days 5 minutes" + * + * @param isUnique - property that controls if generated values gonna be unique or not. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * @param fields - range of values you want to see in your intervals. + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * users: { + * columns: { + * timeSpentOnWebsite: funcs.interval() + * }, + * }, + * })); + * ``` + */ + interval: createGenerator(GenerateInterval), + // uniqueInterval: createGenerator(GenerateUniqueInterval), + + /** + * generates random strings. + * @param isUnique - property that controls if generated values gonna be unique or not. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * users: { + * columns: { + * hashedPassword: funcs.string({isUnique: false}) + * }, + * }, + * })); + * ``` + */ + string: createGenerator(GenerateString), + // uniqueString: createGenerator(GenerateUniqueString), + + /** + * generates v4 UUID strings if arraySize is not specified, or v4 UUID 1D arrays if it is. + * + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * users: { + * columns: { + * uuid: funcs.uuid({ + * arraySize: 4 + * }) + * }, + * }, + * })); + * ``` + */ + uuid: createGenerator(GenerateUUID), + + /** + * generates person's first names. + * @param isUnique - property that controls if generated values gonna be unique or not. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * users: { + * columns: { + * firstName: funcs.firstName({isUnique: true}) + * }, + * }, + * })); + * ``` + */ + firstName: createGenerator(GenerateFirstName), + // uniqueFirstName: createGenerator(GenerateUniqueName), + + /** + * generates person's last names. + * @param isUnique - property that controls if generated values gonna be unique or not. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * users: { + * columns: { + * lastName: funcs.lastName({isUnique: false}) + * }, + * }, + * })); + * ``` + */ + lastName: createGenerator(GenerateLastName), + // uniqueLastName: createGenerator(GenerateUniqueSurname), + + /** + * generates person's full names. + * @param isUnique - property that controls if generated values gonna be unique or not. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * users: { + * columns: { + * fullName: funcs.fullName({isUnique: true}) + * }, + * }, + * })); + * ``` + */ + fullName: createGenerator(GenerateFullName), + // uniqueFullName: createGenerator(GenerateUniqueFullName), + + /** + * generates unique emails. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * users: { + * columns: { + * email: funcs.email() + * }, + * }, + * })); + * ``` + */ + email: createGenerator(GenerateEmail), + + /** + * generates unique phone numbers. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @param template - phone number template, where all '#' symbols will be substituted with generated digits. + * @param prefixes - array of any string you want to be your phone number prefixes.(not compatible with template property) + * @param generatedDigitsNumbers - number of digits that will be added at the end of prefixes.(not compatible with template property) + * @example + * ```ts + * //generate phone number using template property + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * users: { + * columns: { + * phoneNumber: funcs.phoneNumber({template: "+(380) ###-####"}) + * }, + * }, + * })); + * + * //generate phone number using prefixes and generatedDigitsNumbers properties + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * users: { + * columns: { + * phoneNumber: funcs.phoneNumber({prefixes: [ "+380 99", "+380 67" ], generatedDigitsNumbers: 7}) + * }, + * }, + * })); + * + * //generate phone number using prefixes and generatedDigitsNumbers properties but with different generatedDigitsNumbers for prefixes + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * users: { + * columns: { + * phoneNumber: funcs.phoneNumber({prefixes: [ "+380 99", "+380 67", "+1" ], generatedDigitsNumbers: [7, 7, 10]}) + * }, + * }, + * })); + * + * ``` + */ + phoneNumber: createGenerator(GeneratePhoneNumber), + + /** + * generates country's names. + * @param isUnique - property that controls if generated values gonna be unique or not. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * users: { + * columns: { + * country: funcs.country({isUnique: false}) + * }, + * }, + * })); + * ``` + */ + country: createGenerator(GenerateCountry), + // uniqueCountry: createGenerator(GenerateUniqueCountry), + + /** + * generates city's names. + * @param isUnique - property that controls if generated values gonna be unique or not. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * users: { + * columns: { + * city: funcs.city({isUnique: false}) + * }, + * }, + * })); + * ``` + */ + city: createGenerator(GenerateCity), + // uniqueCity: createGenerator(GenerateUniqueCityName), + + /** + * generates street address. + * @param isUnique - property that controls if generated values gonna be unique or not. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * users: { + * columns: { + * streetAddress: funcs.streetAddress({isUnique: true}) + * }, + * }, + * })); + * ``` + */ + streetAddress: createGenerator(GenerateStreetAddress), + // uniqueStreetAddress: createGenerator(GenerateUniqueStreetAddress), + + /** + * generates job titles. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * users: { + * columns: { + * jobTitle: funcs.jobTitle() + * }, + * }, + * })); + * ``` + */ + jobTitle: createGenerator(GenerateJobTitle), + + /** + * generates postal codes. + * + * @param isUnique - property that controls if generated values gonna be unique or not. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * users: { + * columns: { + * postcode: funcs.postcode({isUnique: true}) + * }, + * }, + * })); + * ``` + */ + postcode: createGenerator(GeneratePostcode), + // uniquePostcoe: createGenerator(GenerateUniquePostcode), + + /** + * generates states of America. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * users: { + * columns: { + * state: funcs.state() + * }, + * }, + * })); + * ``` + */ + state: createGenerator(GenerateState), + + /** + * generates company's names. + * + * @param isUnique - property that controls if generated values gonna be unique or not. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * users: { + * columns: { + * company: funcs.companyName({isUnique: true}) + * }, + * }, + * })); + * ``` + */ + companyName: createGenerator(GenerateCompanyName), + // uniqueCompanyName: createGenerator(GenerateUniqueCompanyName), + + /** + * generates 'lorem ipsum' text sentences. + * + * @param sentencesCount - number of sentences you want to generate as one generated value(string). + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * posts: { + * columns: { + * content: funcs.loremIpsum({sentencesCount: 2}) + * }, + * }, + * })); + * ``` + */ + loremIpsum: createGenerator(GenerateLoremIpsum), + + /** + * generates 2D points within specified ranges for x and y coordinates. + * + * @param isUnique - property that controls if generated values gonna be unique or not. + * @param minXValue - lower bound of range for x coordinate. + * @param maxXValue - upper bound of range for x coordinate. + * @param minYValue - lower bound of range for y coordinate. + * @param maxYValue - upper bound of range for y coordinate. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * triangles: { + * columns: { + * pointCoords: funcs.point({ + * isUnique: true, + * minXValue: -5, maxXValue:20, + * minYValue: 0, maxYValue: 30 + * }) + * }, + * }, + * })); + * ``` + */ + point: createGenerator(GeneratePoint), + // uniquePoint: createGenerator(GenerateUniquePoint), + + /** + * generates 2D lines within specified ranges for a, b and c parameters of line. + * + * ``` + * line equation: a*x + b*y + c = 0 + * ``` + * + * @param isUnique - property that controls if generated values gonna be unique or not. + * @param minAValue - lower bound of range for a parameter. + * @param maxAValue - upper bound of range for x parameter. + * @param minBValue - lower bound of range for y parameter. + * @param maxBValue - upper bound of range for y parameter. + * @param minCValue - lower bound of range for y parameter. + * @param maxCValue - upper bound of range for y parameter. + * @param arraySize - number of elements in each one-dimensional array. (If specified, arrays will be generated.) + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * lines: { + * columns: { + * lineParams: funcs.point({ + * isUnique: true, + * minAValue: -5, maxAValue:20, + * minBValue: 0, maxBValue: 30, + * minCValue: 0, maxCValue: 10 + * }) + * }, + * }, + * })); + * ``` + */ + line: createGenerator(GenerateLine), + // uniqueLine: createGenerator(GenerateUniqueLine), + + /** + * gives you the opportunity to call different generators with different probabilities to generate values for one column. + * @param params - array of generators with probabilities you would like to call them to generate values. + * + * @example + * ```ts + * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ + * posts: { + * columns: { + * content: funcs.weightedRandom([ + * { + * weight: 0.6, + * value: funcs.loremIpsum({ sentencesCount: 3 }), + * }, + * { + * weight: 0.4, + * value: funcs.default({ defaultValue: "TODO" }), + * }, + * ]), + * }, + * }, + * })); + * ``` + */ + weightedRandom: createGenerator(WeightedRandomGenerator), +}; + +// so far, version changes don’t affect generator parameters. +export const generatorsFuncsV2 = { + ...generatorsFuncs, +}; + +export const generatorsMap = { + HollowGenerator: [ + HollowGenerator, + ], + GenerateDefault: [ + GenerateDefault, + ], + GenerateValuesFromArray: [ + GenerateValuesFromArray, + ], + GenerateSelfRelationsValuesFromArray: [ + GenerateSelfRelationsValuesFromArray, + ], + GenerateIntPrimaryKey: [ + GenerateIntPrimaryKey, + ], + GenerateNumber: [ + GenerateNumber, + ], + GenerateUniqueNumber: [ + GenerateUniqueNumber, + ], + GenerateInt: [ + GenerateInt, + ], + GenerateUniqueInt: [ + GenerateUniqueInt, + ], + GenerateBoolean: [ + GenerateBoolean, + ], + GenerateDate: [ + GenerateDate, + ], + GenerateTime: [ + GenerateTime, + ], + GenerateTimestamp: [ + GenerateTimestamp, + ], + GenerateDatetime: [ + GenerateDatetime, + ], + GenerateYear: [ + GenerateYear, + ], + GenerateJson: [ + GenerateJson, + ], + GenerateEnum: [ + GenerateEnum, + ], + GenerateInterval: [ + GenerateInterval, + ], + GenerateUniqueInterval: [ + GenerateUniqueInterval, + GenerateUniqueIntervalV2, + ], + GenerateString: [ + GenerateString, + GenerateStringV2, + ], + GenerateUniqueString: [ + GenerateUniqueString, + GenerateUniqueStringV2, + ], + GenerateUUID: [ + GenerateUUID, + ], + GenerateFirstName: [ + GenerateFirstName, + ], + GenerateUniqueFirstName: [ + GenerateUniqueFirstName, + ], + GenerateLastName: [ + GenerateLastName, + ], + GenerateUniqueLastName: [ + GenerateUniqueLastName, + ], + GenerateFullName: [ + GenerateFullName, + ], + GenerateUniqueFullName: [ + GenerateUniqueFullName, + ], + GenerateEmail: [ + GenerateEmail, + ], + GeneratePhoneNumber: [ + GeneratePhoneNumber, + ], + GenerateCountry: [ + GenerateCountry, + ], + GenerateUniqueCountry: [ + GenerateUniqueCountry, + ], + GenerateCity: [ + GenerateCity, + ], + GenerateUniqueCity: [ + GenerateUniqueCity, + ], + GenerateStreetAddress: [ + GenerateStreetAddress, + ], + GenerateUniqueStreetAddress: [ + GenerateUniqueStreetAddress, + ], + GenerateJobTitle: [ + GenerateJobTitle, + ], + GeneratePostcode: [ + GeneratePostcode, + ], + GenerateUniquePostcode: [ + GenerateUniquePostcode, + ], + GenerateState: [ + GenerateState, + ], + GenerateCompanyName: [ + GenerateCompanyName, + ], + GenerateUniqueCompanyName: [ + GenerateUniqueCompanyName, + ], + GenerateLoremIpsum: [ + GenerateLoremIpsum, + ], + GeneratePoint: [ + GeneratePoint, + ], + GenerateUniquePoint: [ + GenerateUniquePoint, + ], + GenerateLine: [ + GenerateLine, + ], + GenerateUniqueLine: [ + GenerateUniqueLine, + ], + WeightedRandomGenerator: [ + WeightedRandomGenerator, + ], + GenerateArray: [ + GenerateArray, + ], + GenerateWeightedCount: [ + GenerateWeightedCount, + ], +} as const; diff --git a/drizzle-seed/src/services/GeneratorsWrappers.ts b/drizzle-seed/src/services/Generators.ts similarity index 73% rename from drizzle-seed/src/services/GeneratorsWrappers.ts rename to drizzle-seed/src/services/Generators.ts index 06d6adeb5..0d285540e 100644 --- a/drizzle-seed/src/services/GeneratorsWrappers.ts +++ b/drizzle-seed/src/services/Generators.ts @@ -1,34 +1,55 @@ -import { entityKind } from 'drizzle-orm'; +/* eslint-disable drizzle-internal/require-entity-kind */ import prand from 'pure-rand'; -import adjectives from '../datasets/adjectives.ts'; -import cityNames from '../datasets/cityNames.ts'; -import companyNameSuffixes from '../datasets/companyNameSuffixes.ts'; -import countries from '../datasets/countries.ts'; -import emailDomains from '../datasets/emailDomains.ts'; -import firstNames from '../datasets/firstNames.ts'; -import jobsTitles from '../datasets/jobsTitles.ts'; -import lastNames from '../datasets/lastNames.ts'; -import loremIpsumSentences from '../datasets/loremIpsumSentences.ts'; +import adjectives, { maxStringLength as maxAdjectiveLength } from '../datasets/adjectives.ts'; +import cityNames, { maxStringLength as maxCityNameLength } from '../datasets/cityNames.ts'; +import companyNameSuffixes, { maxStringLength as maxCompanyNameSuffixLength } from '../datasets/companyNameSuffixes.ts'; +import countries, { maxStringLength as maxCountryLength } from '../datasets/countries.ts'; +import emailDomains, { maxStringLength as maxEmailDomainLength } from '../datasets/emailDomains.ts'; +import firstNames, { maxStringLength as maxFirstNameLength } from '../datasets/firstNames.ts'; +import jobsTitles, { maxStringLength as maxJobTitleLength } from '../datasets/jobsTitles.ts'; +import lastNames, { maxStringLength as maxLastNameLength } from '../datasets/lastNames.ts'; +import loremIpsumSentences, { maxStringLength as maxLoremIpsumLength } from '../datasets/loremIpsumSentences.ts'; import phonesInfo from '../datasets/phonesInfo.ts'; -import states from '../datasets/states.ts'; -import streetSuffix from '../datasets/streetSuffix.ts'; +import states, { maxStringLength as maxStateLength } from '../datasets/states.ts'; +import streetSuffix, { maxStringLength as maxStreetSuffixLength } from '../datasets/streetSuffix.ts'; import { fastCartesianProduct, fillTemplate, getWeightedIndices, isObject } from './utils.ts'; export abstract class AbstractGenerator { - static readonly [entityKind]: string = 'AbstractGenerator'; + static readonly entityKind: string = 'AbstractGenerator'; + static readonly version: number = 1; public isUnique = false; public notNull = false; + + // param for generators which have a unique version of themselves public uniqueVersionOfGen?: new(params: T) => AbstractGenerator; + public dataType?: string; public timeSpent?: number; + + // public arraySize?: number; public baseColumnDataType?: string; - constructor(public params: T) {} + // param for text-like generators + public stringLength?: number; + + // params for GenerateValuesFromArray + public weightedCountSeed?: number | undefined; + public maxRepeatedValuesCount?: number | { weight: number; count: number | number[] }[] | undefined; + + public params: T; + + constructor(params?: T) { + this.params = params === undefined ? {} as T : params as T; + } init(params: { count: number | { weight: number; count: number | number[] }[]; seed: number }): void; init() { + this.updateParams(); + } + + updateParams() { if ((this.params as any).arraySize !== undefined) { this.arraySize = (this.params as any).arraySize; } @@ -46,10 +67,11 @@ export abstract class AbstractGenerator { getEntityKind(): string { const constructor = this.constructor as typeof AbstractGenerator; - return constructor[entityKind]; + return constructor.entityKind; } - replaceIfUnique({ count, seed }: { count: number; seed: number }) { + replaceIfUnique() { + this.updateParams(); if ( this.uniqueVersionOfGen !== undefined && this.isUnique === true @@ -57,10 +79,7 @@ export abstract class AbstractGenerator { const uniqueGen = new this.uniqueVersionOfGen({ ...this.params, }); - uniqueGen.init({ - count, - seed, - }); + uniqueGen.isUnique = this.isUnique; uniqueGen.dataType = this.dataType; @@ -69,9 +88,10 @@ export abstract class AbstractGenerator { return; } - replaceIfArray({ count, seed }: { count: number; seed: number }) { + replaceIfArray() { + this.updateParams(); if (!(this.getEntityKind() === 'GenerateArray') && this.arraySize !== undefined) { - const uniqueGen = this.replaceIfUnique({ count, seed }); + const uniqueGen = this.replaceIfUnique(); const baseColumnGen = uniqueGen === undefined ? this : uniqueGen; baseColumnGen.dataType = this.baseColumnDataType; const arrayGen = new GenerateArray( @@ -80,7 +100,6 @@ export abstract class AbstractGenerator { size: this.arraySize, }, ); - arrayGen.init({ count, seed }); return arrayGen; } @@ -89,22 +108,9 @@ export abstract class AbstractGenerator { } } -function createGenerator, T>( - generatorConstructor: new(params: T) => GeneratorType, -) { - return ( - ...args: GeneratorType extends GenerateValuesFromArray | GenerateDefault | WeightedRandomGenerator ? [T] - : ([] | [T]) - ): GeneratorType => { - let params = args[0]; - if (params === undefined) params = {} as T; - return new generatorConstructor(params); - }; -} - // Generators Classes ----------------------------------------------------------------------------------------------------------------------- export class GenerateArray extends AbstractGenerator<{ baseColumnGen: AbstractGenerator; size?: number }> { - static override readonly [entityKind]: string = 'GenerateArray'; + static override readonly entityKind: string = 'GenerateArray'; public override arraySize = 10; override init({ count, seed }: { count: number; seed: number }) { @@ -124,7 +130,7 @@ export class GenerateArray extends AbstractGenerator<{ baseColumnGen: AbstractGe } export class GenerateWeightedCount extends AbstractGenerator<{}> { - static override readonly [entityKind]: string = 'GenerateWeightedCount'; + static override readonly entityKind: string = 'GenerateWeightedCount'; private state: { rng: prand.RandomGenerator; @@ -162,7 +168,7 @@ export class GenerateWeightedCount extends AbstractGenerator<{}> { } export class HollowGenerator extends AbstractGenerator<{}> { - static override readonly [entityKind]: string = 'HollowGenerator'; + static override readonly entityKind: string = 'HollowGenerator'; override init() {} @@ -173,7 +179,7 @@ export class GenerateDefault extends AbstractGenerator<{ defaultValue: unknown; arraySize?: number; }> { - static override readonly [entityKind]: string = 'GenerateDefault'; + static override readonly entityKind: string = 'GenerateDefault'; generate() { return this.params.defaultValue; @@ -189,10 +195,8 @@ export class GenerateValuesFromArray extends AbstractGenerator< arraySize?: number; } > { - static override readonly [entityKind]: string = 'GenerateValuesFromArray'; + static override readonly entityKind: string = 'GenerateValuesFromArray'; - public weightedCountSeed: number | undefined = undefined; - public maxRepeatedValuesCount?: number | { weight: number; count: number | number[] }[] = undefined; private state: { rng: prand.RandomGenerator; values: @@ -400,7 +404,7 @@ export class GenerateValuesFromArray extends AbstractGenerator< } export class GenerateSelfRelationsValuesFromArray extends AbstractGenerator<{ values: (number | string | boolean)[] }> { - static override readonly [entityKind]: string = 'GenerateSelfRelationsValuesFromArray'; + static override readonly entityKind: string = 'GenerateSelfRelationsValuesFromArray'; private state: { rng: prand.RandomGenerator; @@ -438,7 +442,7 @@ export class GenerateSelfRelationsValuesFromArray extends AbstractGenerator<{ va } export class GenerateIntPrimaryKey extends AbstractGenerator<{}> { - static override readonly [entityKind]: string = 'GenerateIntPrimaryKey'; + static override readonly entityKind: string = 'GenerateIntPrimaryKey'; public maxValue?: number | bigint; @@ -466,7 +470,7 @@ export class GenerateNumber extends AbstractGenerator< arraySize?: number; } > { - static override readonly [entityKind]: string = 'GenerateNumber'; + static override readonly entityKind: string = 'GenerateNumber'; private state: { rng: prand.RandomGenerator; @@ -521,7 +525,7 @@ export class GenerateUniqueNumber extends AbstractGenerator< isUnique?: boolean; } > { - static override readonly [entityKind]: string = 'GenerateUniqueNumber'; + static override readonly entityKind: string = 'GenerateUniqueNumber'; private state: { genUniqueIntObj: GenerateUniqueInt; @@ -573,7 +577,7 @@ export class GenerateInt extends AbstractGenerator<{ isUnique?: boolean; arraySize?: number; }> { - static override readonly [entityKind]: string = 'GenerateInt'; + static override readonly entityKind: string = 'GenerateInt'; private state: { rng: prand.RandomGenerator; @@ -641,7 +645,7 @@ export class GenerateUniqueInt extends AbstractGenerator<{ maxValue?: number | bigint; isUnique?: boolean; }> { - static override readonly [entityKind]: string = 'GenerateUniqueInt'; + static override readonly entityKind: string = 'GenerateUniqueInt'; public genMaxRepeatedValuesCount: GenerateDefault | GenerateWeightedCount | undefined; public skipCheck?: boolean = false; @@ -809,7 +813,7 @@ export class GenerateUniqueInt extends AbstractGenerator<{ } export class GenerateBoolean extends AbstractGenerator<{ arraySize?: number }> { - static override readonly [entityKind]: string = 'GenerateBoolean'; + static override readonly entityKind: string = 'GenerateBoolean'; private state: { rng: prand.RandomGenerator; @@ -840,7 +844,7 @@ export class GenerateDate extends AbstractGenerator<{ maxDate?: string | Date; arraySize?: number; }> { - static override readonly [entityKind]: string = 'GenerateDate'; + static override readonly entityKind: string = 'GenerateDate'; private state: { rng: prand.RandomGenerator; @@ -902,7 +906,7 @@ export class GenerateDate extends AbstractGenerator<{ } } export class GenerateTime extends AbstractGenerator<{ arraySize?: number }> { - static override readonly [entityKind]: string = 'GenerateTime'; + static override readonly entityKind: string = 'GenerateTime'; private state: { rng: prand.RandomGenerator; @@ -938,7 +942,7 @@ export class GenerateTime extends AbstractGenerator<{ arraySize?: number }> { } } export class GenerateTimestampInt extends AbstractGenerator<{ unitOfTime?: 'seconds' | 'milliseconds' }> { - static override readonly [entityKind]: string = 'GenerateTimestampInt'; + static override readonly entityKind: string = 'GenerateTimestampInt'; private state: { generateTimestampObj: GenerateTimestamp; @@ -971,7 +975,7 @@ export class GenerateTimestampInt extends AbstractGenerator<{ unitOfTime?: 'seco } export class GenerateTimestamp extends AbstractGenerator<{ arraySize?: number }> { - static override readonly [entityKind]: string = 'GenerateTimestamp'; + static override readonly entityKind: string = 'GenerateTimestamp'; private state: { rng: prand.RandomGenerator; @@ -1015,7 +1019,7 @@ export class GenerateTimestamp extends AbstractGenerator<{ arraySize?: number }> } export class GenerateDatetime extends AbstractGenerator<{ arraySize?: number }> { - static override readonly [entityKind]: string = 'GenerateDatetime'; + static override readonly entityKind: string = 'GenerateDatetime'; private state: { rng: prand.RandomGenerator; @@ -1059,7 +1063,7 @@ export class GenerateDatetime extends AbstractGenerator<{ arraySize?: number }> } export class GenerateYear extends AbstractGenerator<{ arraySize?: number }> { - static override readonly [entityKind]: string = 'GenerateYear'; + static override readonly entityKind: string = 'GenerateYear'; private state: { rng: prand.RandomGenerator; @@ -1094,7 +1098,7 @@ export class GenerateYear extends AbstractGenerator<{ arraySize?: number }> { } export class GenerateJson extends AbstractGenerator<{ arraySize?: number }> { - static override readonly [entityKind]: string = 'GenerateJson'; + static override readonly entityKind: string = 'GenerateJson'; private state: { emailGeneratorObj: GenerateEmail; @@ -1198,7 +1202,7 @@ export class GenerateJson extends AbstractGenerator<{ arraySize?: number }> { } export class GenerateEnum extends AbstractGenerator<{ enumValues: (string | number | boolean)[] }> { - static override readonly [entityKind]: string = 'GenerateEnum'; + static override readonly entityKind: string = 'GenerateEnum'; private state: { enumValuesGenerator: GenerateValuesFromArray; @@ -1221,19 +1225,74 @@ export class GenerateEnum extends AbstractGenerator<{ enumValues: (string | numb } export class GenerateInterval extends AbstractGenerator<{ + fields?: + | 'year' + | 'month' + | 'day' + | 'hour' + | 'minute' + | 'second' + | 'year to month' + | 'day to hour' + | 'day to minute' + | 'day to second' + | 'hour to minute' + | 'hour to second' + | 'minute to second'; isUnique?: boolean; arraySize?: number; }> { - static override readonly [entityKind]: string = 'GenerateInterval'; + static override readonly entityKind: string = 'GenerateInterval'; - private state: { rng: prand.RandomGenerator } | undefined; - override uniqueVersionOfGen = GenerateUniqueInterval; + private state: { + rng: prand.RandomGenerator; + fieldsToGenerate: string[]; + } | undefined; + override uniqueVersionOfGen: new(params: any) => AbstractGenerator = GenerateUniqueInterval; + private config: { [key: string]: { from: number; to: number } } = { + year: { + from: 0, + to: 5, + }, + month: { + from: 0, + to: 12, + }, + day: { + from: 1, + to: 29, + }, + hour: { + from: 0, + to: 24, + }, + minute: { + from: 0, + to: 60, + }, + second: { + from: 0, + to: 60, + }, + }; override init({ count, seed }: { count: number; seed: number }) { super.init({ count, seed }); + const allFields = ['year', 'month', 'day', 'hour', 'minute', 'second']; + let fieldsToGenerate: string[] = allFields; + + if (this.params.fields !== undefined && this.params.fields?.includes(' to ')) { + const tokens = this.params.fields.split(' to '); + const endIdx = allFields.indexOf(tokens[1]!); + fieldsToGenerate = allFields.slice(0, endIdx + 1); + } else if (this.params.fields !== undefined) { + const endIdx = allFields.indexOf(this.params.fields); + fieldsToGenerate = allFields.slice(0, endIdx + 1); + } + const rng = prand.xoroshiro128plus(seed); - this.state = { rng }; + this.state = { rng, fieldsToGenerate }; } generate() { @@ -1241,55 +1300,97 @@ export class GenerateInterval extends AbstractGenerator<{ throw new Error('state is not defined.'); } - let yearsNumb: number, - monthsNumb: number, - daysNumb: number, - hoursNumb: number, - minutesNumb: number, - secondsNumb: number; - - let interval = ''; - - [yearsNumb, this.state.rng] = prand.uniformIntDistribution(0, 5, this.state.rng); - [monthsNumb, this.state.rng] = prand.uniformIntDistribution(0, 12, this.state.rng); + let interval = '', numb: number; - [daysNumb, this.state.rng] = prand.uniformIntDistribution(1, 29, this.state.rng); - - [hoursNumb, this.state.rng] = prand.uniformIntDistribution(0, 24, this.state.rng); - - [minutesNumb, this.state.rng] = prand.uniformIntDistribution(0, 60, this.state.rng); - - [secondsNumb, this.state.rng] = prand.uniformIntDistribution(0, 60, this.state.rng); - - interval = `${yearsNumb === 0 ? '' : `${yearsNumb} years `}` - + `${monthsNumb === 0 ? '' : `${monthsNumb} months `}` - + `${daysNumb === 0 ? '' : `${daysNumb} days `}` - + `${hoursNumb === 0 ? '' : `${hoursNumb} hours `}` - + `${minutesNumb === 0 ? '' : `${minutesNumb} minutes `}` - + `${secondsNumb === 0 ? '' : `${secondsNumb} seconds`}`; + for (const field of this.state.fieldsToGenerate) { + const from = this.config[field]!.from, to = this.config[field]!.to; + [numb, this.state.rng] = prand.uniformIntDistribution(from, to, this.state.rng); + interval += `${numb} ${field} `; + } return interval; } } -export class GenerateUniqueInterval extends AbstractGenerator<{ isUnique?: boolean }> { - static override readonly [entityKind]: string = 'GenerateUniqueInterval'; +// has a newer version +export class GenerateUniqueInterval extends AbstractGenerator<{ + fields?: + | 'year' + | 'month' + | 'day' + | 'hour' + | 'minute' + | 'second' + | 'year to month' + | 'day to hour' + | 'day to minute' + | 'day to second' + | 'hour to minute' + | 'hour to second' + | 'minute to second'; + isUnique?: boolean; +}> { + static override readonly 'entityKind': string = 'GenerateUniqueInterval'; private state: { rng: prand.RandomGenerator; + fieldsToGenerate: string[]; intervalSet: Set; } | undefined; public override isUnique = true; + private config: { [key: string]: { from: number; to: number } } = { + year: { + from: 0, + to: 5, + }, + month: { + from: 0, + to: 12, + }, + day: { + from: 1, + to: 29, + }, + hour: { + from: 0, + to: 24, + }, + minute: { + from: 0, + to: 60, + }, + second: { + from: 0, + to: 60, + }, + }; override init({ count, seed }: { count: number; seed: number }) { - const maxUniqueIntervalsNumber = 6 * 13 * 29 * 25 * 61 * 61; + const allFields = ['year', 'month', 'day', 'hour', 'minute', 'second']; + let fieldsToGenerate: string[] = allFields; + + if (this.params.fields !== undefined && this.params.fields?.includes(' to ')) { + const tokens = this.params.fields.split(' to '); + const endIdx = allFields.indexOf(tokens[1]!); + fieldsToGenerate = allFields.slice(0, endIdx + 1); + } else if (this.params.fields !== undefined) { + const endIdx = allFields.indexOf(this.params.fields); + fieldsToGenerate = allFields.slice(0, endIdx + 1); + } + + let maxUniqueIntervalsNumber = 1; + for (const field of fieldsToGenerate) { + const from = this.config[field]!.from, to = this.config[field]!.to; + maxUniqueIntervalsNumber *= from - to + 1; + } + if (count > maxUniqueIntervalsNumber) { throw new RangeError(`count exceeds max number of unique intervals(${maxUniqueIntervalsNumber})`); } const rng = prand.xoroshiro128plus(seed); const intervalSet = new Set(); - this.state = { rng, intervalSet }; + this.state = { rng, fieldsToGenerate, intervalSet }; } generate() { @@ -1297,29 +1398,16 @@ export class GenerateUniqueInterval extends AbstractGenerator<{ isUnique?: boole throw new Error('state is not defined.'); } - let yearsNumb: number, - monthsNumb: number, - daysNumb: number, - hoursNumb: number, - minutesNumb: number, - secondsNumb: number; - - let interval = ''; + let interval, numb: number; for (;;) { - [yearsNumb, this.state.rng] = prand.uniformIntDistribution(0, 5, this.state.rng); - [monthsNumb, this.state.rng] = prand.uniformIntDistribution(0, 12, this.state.rng); - [daysNumb, this.state.rng] = prand.uniformIntDistribution(1, 29, this.state.rng); - [hoursNumb, this.state.rng] = prand.uniformIntDistribution(0, 24, this.state.rng); - [minutesNumb, this.state.rng] = prand.uniformIntDistribution(0, 60, this.state.rng); - [secondsNumb, this.state.rng] = prand.uniformIntDistribution(0, 60, this.state.rng); - - interval = `${yearsNumb === 0 ? '' : `${yearsNumb} years `}` - + `${monthsNumb === 0 ? '' : `${monthsNumb} months `}` - + `${daysNumb === 0 ? '' : `${daysNumb} days `}` - + `${hoursNumb === 0 ? '' : `${hoursNumb} hours `}` - + `${minutesNumb === 0 ? '' : `${minutesNumb} minutes `}` - + `${secondsNumb === 0 ? '' : `${secondsNumb} seconds`}`; + interval = ''; + + for (const field of this.state.fieldsToGenerate) { + const from = this.config[field]!.from, to = this.config[field]!.to; + [numb, this.state.rng] = prand.uniformIntDistribution(from, to, this.state.rng); + interval += `${numb} ${field} `; + } if (!this.state.intervalSet.has(interval)) { this.state.intervalSet.add(interval); @@ -1331,11 +1419,12 @@ export class GenerateUniqueInterval extends AbstractGenerator<{ isUnique?: boole } } +// has a newer version export class GenerateString extends AbstractGenerator<{ isUnique?: boolean; arraySize?: number; }> { - static override readonly [entityKind]: string = 'GenerateString'; + static override readonly entityKind: string = 'GenerateString'; private state: { rng: prand.RandomGenerator } | undefined; override uniqueVersionOfGen = GenerateUniqueString; @@ -1377,8 +1466,9 @@ export class GenerateString extends AbstractGenerator<{ } } +// has a newer version export class GenerateUniqueString extends AbstractGenerator<{ isUnique?: boolean }> { - static override readonly [entityKind]: string = 'GenerateUniqueString'; + static override readonly entityKind: string = 'GenerateUniqueString'; private state: { rng: prand.RandomGenerator } | undefined; public override isUnique = true; @@ -1423,7 +1513,7 @@ export class GenerateUniqueString extends AbstractGenerator<{ isUnique?: boolean export class GenerateUUID extends AbstractGenerator<{ arraySize?: number; }> { - static override readonly [entityKind]: string = 'GenerateUUID'; + static override readonly entityKind: string = 'GenerateUUID'; public override isUnique = true; @@ -1470,7 +1560,7 @@ export class GenerateFirstName extends AbstractGenerator<{ isUnique?: boolean; arraySize?: number; }> { - static override readonly [entityKind]: string = 'GenerateFirstName'; + static override readonly entityKind: string = 'GenerateFirstName'; override timeSpent: number = 0; private state: { @@ -1483,6 +1573,12 @@ export class GenerateFirstName extends AbstractGenerator<{ const rng = prand.xoroshiro128plus(seed); + if (this.stringLength !== undefined && this.stringLength < maxFirstNameLength) { + throw new Error( + `You can't use first name generator with a db column length restriction of ${this.stringLength}. Set the maximum string length to at least ${maxFirstNameLength}.`, + ); + } + this.state = { rng }; } @@ -1493,7 +1589,6 @@ export class GenerateFirstName extends AbstractGenerator<{ // logic for this generator // names dataset contains about 30000 unique names. - // TODO: generate names accordingly to max column length let idx: number; [idx, this.state.rng] = prand.uniformIntDistribution(0, firstNames.length - 1, this.state.rng); @@ -1504,7 +1599,7 @@ export class GenerateFirstName extends AbstractGenerator<{ export class GenerateUniqueFirstName extends AbstractGenerator<{ isUnique?: boolean; }> { - static override readonly [entityKind]: string = 'GenerateUniqueFirstName'; + static override readonly entityKind: string = 'GenerateUniqueFirstName'; private state: { genIndicesObj: GenerateUniqueInt; @@ -1515,6 +1610,13 @@ export class GenerateUniqueFirstName extends AbstractGenerator<{ if (count > firstNames.length) { throw new Error('count exceeds max number of unique first names.'); } + + if (this.stringLength !== undefined && this.stringLength < maxFirstNameLength) { + throw new Error( + `You can't use first name generator with a db column length restriction of ${this.stringLength}. Set the maximum string length to at least ${maxFirstNameLength}.`, + ); + } + const genIndicesObj = new GenerateUniqueInt({ minValue: 0, maxValue: firstNames.length - 1 }); genIndicesObj.init({ count, seed }); @@ -1538,7 +1640,7 @@ export class GenerateLastName extends AbstractGenerator<{ isUnique?: boolean; arraySize?: number; }> { - static override readonly [entityKind]: string = 'GenerateLastName'; + static override readonly entityKind: string = 'GenerateLastName'; private state: { rng: prand.RandomGenerator; @@ -1550,6 +1652,12 @@ export class GenerateLastName extends AbstractGenerator<{ const rng = prand.xoroshiro128plus(seed); + if (this.stringLength !== undefined && this.stringLength < maxLastNameLength) { + throw new Error( + `You can't use last name generator with a db column length restriction of ${this.stringLength}. Set the maximum string length to at least ${maxLastNameLength}.`, + ); + } + this.state = { rng }; } generate() { @@ -1565,7 +1673,7 @@ export class GenerateLastName extends AbstractGenerator<{ } export class GenerateUniqueLastName extends AbstractGenerator<{ isUnique?: boolean }> { - static override readonly [entityKind]: string = 'GenerateUniqueLastName'; + static override readonly entityKind: string = 'GenerateUniqueLastName'; private state: { genIndicesObj: GenerateUniqueInt; @@ -1577,6 +1685,12 @@ export class GenerateUniqueLastName extends AbstractGenerator<{ isUnique?: boole throw new Error('count exceeds max number of unique last names.'); } + if (this.stringLength !== undefined && this.stringLength < maxLastNameLength) { + throw new Error( + `You can't use last name generator with a db column length restriction of ${this.stringLength}. Set the maximum string length to at least ${maxLastNameLength}.`, + ); + } + const genIndicesObj = new GenerateUniqueInt({ minValue: 0, maxValue: lastNames.length - 1 }); genIndicesObj.init({ count, seed }); @@ -1599,7 +1713,7 @@ export class GenerateFullName extends AbstractGenerator<{ isUnique?: boolean; arraySize?: number; }> { - static override readonly [entityKind]: string = 'GenerateFullName'; + static override readonly entityKind: string = 'GenerateFullName'; private state: { rng: prand.RandomGenerator; @@ -1611,6 +1725,14 @@ export class GenerateFullName extends AbstractGenerator<{ const rng = prand.xoroshiro128plus(seed); + if (this.stringLength !== undefined && this.stringLength < (maxFirstNameLength + maxLastNameLength + 1)) { + throw new Error( + `You can't use full name generator with a db column length restriction of ${this.stringLength}. Set the maximum string length to at least ${ + maxFirstNameLength + maxLastNameLength + 1 + }.`, + ); + } + this.state = { rng }; } @@ -1636,7 +1758,7 @@ export class GenerateFullName extends AbstractGenerator<{ export class GenerateUniqueFullName extends AbstractGenerator<{ isUnique?: boolean; }> { - static override readonly [entityKind]: string = 'GenerateUniqueFullName'; + static override readonly entityKind: string = 'GenerateUniqueFullName'; private state: { fullnameSet: Set; @@ -1654,6 +1776,15 @@ export class GenerateUniqueFullName extends AbstractGenerator<{ `count exceeds max number of unique full names(${maxUniqueFullNamesNumber}).`, ); } + + if (this.stringLength !== undefined && this.stringLength < (maxFirstNameLength + maxLastNameLength + 1)) { + throw new Error( + `You can't use full name generator with a db column length restriction of ${this.stringLength}. Set the maximum string length to at least ${ + maxFirstNameLength + maxLastNameLength + 1 + }.`, + ); + } + const rng = prand.xoroshiro128plus(seed); const fullnameSet = new Set(); @@ -1692,7 +1823,7 @@ export class GenerateUniqueFullName extends AbstractGenerator<{ export class GenerateEmail extends AbstractGenerator<{ arraySize?: number; }> { - static override readonly [entityKind]: string = 'GenerateEmail'; + static override readonly entityKind: string = 'GenerateEmail'; private state: { genIndicesObj: GenerateUniqueInt; @@ -1715,6 +1846,13 @@ export class GenerateEmail extends AbstractGenerator<{ ); } + const maxEmailLength = maxAdjectiveLength + maxFirstNameLength + maxEmailDomainLength + 2; + if (this.stringLength !== undefined && this.stringLength < maxEmailLength) { + throw new Error( + `You can't use email generator with a db column length restriction of ${this.stringLength}. Set the maximum string length to at least ${maxEmailLength}.`, + ); + } + const arraysToGenerateFrom = [adjectivesArray, namesArray, domainsArray]; const genIndicesObj = new GenerateUniqueInt({ minValue: 0, @@ -1752,7 +1890,7 @@ export class GeneratePhoneNumber extends AbstractGenerator<{ generatedDigitsNumbers?: number | number[]; arraySize?: number; }> { - static override readonly [entityKind]: string = 'GeneratePhoneNumber'; + static override readonly entityKind: string = 'GeneratePhoneNumber'; private state: { rng: prand.RandomGenerator; @@ -1773,6 +1911,13 @@ export class GeneratePhoneNumber extends AbstractGenerator<{ const rng = prand.xoroshiro128plus(seed); if (template !== undefined) { + if (this.stringLength !== undefined && this.stringLength < template.length) { + throw new Error( + `Length of phone number template is shorter than db column length restriction: ${this.stringLength}. + Set the maximum string length to at least ${template.length}.`, + ); + } + const iterArray = [...template.matchAll(/#/g)]; const placeholdersCount = iterArray.length; @@ -1828,6 +1973,17 @@ export class GeneratePhoneNumber extends AbstractGenerator<{ } } + const maxPrefixLength = Math.max(...prefixesArray.map((prefix) => prefix.length)); + const maxGeneratedDigits = Math.max(...generatedDigitsNumbers); + + if (this.stringLength !== undefined && this.stringLength < (maxPrefixLength + maxGeneratedDigits)) { + throw new Error( + `You can't use phone number generator with a db column length restriction of ${this.stringLength}. Set the maximum string length to at least ${ + maxPrefixLength + maxGeneratedDigits + }.`, + ); + } + if (new Set(prefixesArray).size !== prefixesArray.length) { throw new Error('prefixes are not unique.'); } @@ -1907,7 +2063,7 @@ export class GeneratePhoneNumber extends AbstractGenerator<{ numberBody = '0'.repeat(digitsNumberDiff) + numberBody; } - phoneNumber = (prefix.includes('+') ? '' : '+') + prefix + '' + numberBody; + phoneNumber = prefix + '' + numberBody; return phoneNumber; } else { @@ -1928,7 +2084,7 @@ export class GenerateCountry extends AbstractGenerator<{ isUnique?: boolean; arraySize?: number; }> { - static override readonly [entityKind]: string = 'GenerateCountry'; + static override readonly entityKind: string = 'GenerateCountry'; private state: { rng: prand.RandomGenerator; @@ -1940,6 +2096,12 @@ export class GenerateCountry extends AbstractGenerator<{ const rng = prand.xoroshiro128plus(seed); + if (this.stringLength !== undefined && this.stringLength < maxCountryLength) { + throw new Error( + `You can't use country generator with a db column length restriction of ${this.stringLength}. Set the maximum string length to at least ${maxCountryLength}.`, + ); + } + this.state = { rng }; } @@ -1958,7 +2120,7 @@ export class GenerateCountry extends AbstractGenerator<{ } export class GenerateUniqueCountry extends AbstractGenerator<{ isUnique?: boolean }> { - static override readonly [entityKind]: string = 'GenerateUniqueCountry'; + static override readonly entityKind: string = 'GenerateUniqueCountry'; private state: { genIndicesObj: GenerateUniqueInt; @@ -1970,6 +2132,12 @@ export class GenerateUniqueCountry extends AbstractGenerator<{ isUnique?: boolea throw new Error('count exceeds max number of unique countries.'); } + if (this.stringLength !== undefined && this.stringLength < maxCountryLength) { + throw new Error( + `You can't use country generator with a db column length restriction of ${this.stringLength}. Set the maximum string length to at least ${maxCountryLength}.`, + ); + } + const genIndicesObj = new GenerateUniqueInt({ minValue: 0, maxValue: countries.length - 1 }); genIndicesObj.init({ count, seed }); @@ -1991,7 +2159,7 @@ export class GenerateUniqueCountry extends AbstractGenerator<{ isUnique?: boolea export class GenerateJobTitle extends AbstractGenerator<{ arraySize?: number; }> { - static override readonly [entityKind]: string = 'GenerateJobTitle'; + static override readonly entityKind: string = 'GenerateJobTitle'; private state: { rng: prand.RandomGenerator; @@ -2002,6 +2170,12 @@ export class GenerateJobTitle extends AbstractGenerator<{ const rng = prand.xoroshiro128plus(seed); + if (this.stringLength !== undefined && this.stringLength < maxJobTitleLength) { + throw new Error( + `You can't use job title generator with a db column length restriction of ${this.stringLength}. Set the maximum string length to at least ${maxJobTitleLength}.`, + ); + } + this.state = { rng }; } @@ -2017,23 +2191,31 @@ export class GenerateJobTitle extends AbstractGenerator<{ } } -export class GenerateStreetAdddress extends AbstractGenerator<{ +export class GenerateStreetAddress extends AbstractGenerator<{ isUnique?: boolean; arraySize?: number; }> { - static override readonly [entityKind]: string = 'GenerateStreetAdddress'; + static override readonly entityKind: string = 'GenerateStreetAddress'; private state: { rng: prand.RandomGenerator; possStreetNames: string[][]; } | undefined; - override uniqueVersionOfGen = GenerateUniqueStreetAdddress; + override uniqueVersionOfGen = GenerateUniqueStreetAddress; override init({ count, seed }: { count: number; seed: number }) { super.init({ count, seed }); const rng = prand.xoroshiro128plus(seed); const possStreetNames = [firstNames, lastNames]; + + const maxStreetAddressLength = 4 + Math.max(maxFirstNameLength, maxLastNameLength) + 1 + maxStreetSuffixLength; + if (this.stringLength !== undefined && this.stringLength < maxStreetAddressLength) { + throw new Error( + `You can't use street address generator with a db column length restriction of ${this.stringLength}. Set the maximum string length to at least ${maxStreetAddressLength}.`, + ); + } + this.state = { rng, possStreetNames }; } @@ -2059,8 +2241,8 @@ export class GenerateStreetAdddress extends AbstractGenerator<{ } } -export class GenerateUniqueStreetAdddress extends AbstractGenerator<{ isUnique?: boolean }> { - static override readonly [entityKind]: string = 'GenerateUniqueStreetAdddress'; +export class GenerateUniqueStreetAddress extends AbstractGenerator<{ isUnique?: boolean }> { + static override readonly entityKind: string = 'GenerateUniqueStreetAddress'; private state: { rng: prand.RandomGenerator; @@ -2084,6 +2266,13 @@ export class GenerateUniqueStreetAdddress extends AbstractGenerator<{ isUnique?: ); } + const maxStreetAddressLength = 4 + Math.max(maxFirstNameLength, maxLastNameLength) + 1 + maxStreetSuffixLength; + if (this.stringLength !== undefined && this.stringLength < maxStreetAddressLength) { + throw new Error( + `You can't use street address generator with a db column length restriction of ${this.stringLength}. Set the maximum string length to at least ${maxStreetAddressLength}.`, + ); + } + const rng = prand.xoroshiro128plus(seed); // ["1", "2", ..., "999"] @@ -2149,7 +2338,7 @@ export class GenerateCity extends AbstractGenerator<{ isUnique?: boolean; arraySize?: number; }> { - static override readonly [entityKind]: string = 'GenerateCity'; + static override readonly entityKind: string = 'GenerateCity'; private state: { rng: prand.RandomGenerator; @@ -2161,6 +2350,12 @@ export class GenerateCity extends AbstractGenerator<{ const rng = prand.xoroshiro128plus(seed); + if (this.stringLength !== undefined && this.stringLength < maxCityNameLength) { + throw new Error( + `You can't use city generator with a db column length restriction of ${this.stringLength}. Set the maximum string length to at least ${maxCityNameLength}.`, + ); + } + this.state = { rng }; } @@ -2177,7 +2372,7 @@ export class GenerateCity extends AbstractGenerator<{ } export class GenerateUniqueCity extends AbstractGenerator<{ isUnique?: boolean }> { - static override readonly [entityKind]: string = 'GenerateUniqueCity'; + static override readonly entityKind: string = 'GenerateUniqueCity'; private state: { genIndicesObj: GenerateUniqueInt; @@ -2189,6 +2384,12 @@ export class GenerateUniqueCity extends AbstractGenerator<{ isUnique?: boolean } throw new Error('count exceeds max number of unique cities.'); } + if (this.stringLength !== undefined && this.stringLength < maxCityNameLength) { + throw new Error( + `You can't use city generator with a db column length restriction of ${this.stringLength}. Set the maximum string length to at least ${maxCityNameLength}.`, + ); + } + const genIndicesObj = new GenerateUniqueInt({ minValue: 0, maxValue: cityNames.length - 1 }); genIndicesObj.init({ count, seed }); @@ -2211,7 +2412,7 @@ export class GeneratePostcode extends AbstractGenerator<{ isUnique?: boolean; arraySize?: number; }> { - static override readonly [entityKind]: string = 'GeneratePostcode'; + static override readonly entityKind: string = 'GeneratePostcode'; private state: { rng: prand.RandomGenerator; @@ -2225,6 +2426,13 @@ export class GeneratePostcode extends AbstractGenerator<{ const rng = prand.xoroshiro128plus(seed); const templates = ['#####', '#####-####']; + const maxPostcodeLength = Math.max(...templates.map((template) => template.length)); + if (this.stringLength !== undefined && this.stringLength < maxPostcodeLength) { + throw new Error( + `You can't use postcode generator with a db column length restriction of ${this.stringLength}. Set the maximum string length to at least ${maxPostcodeLength}.`, + ); + } + this.state = { rng, templates }; } @@ -2258,7 +2466,7 @@ export class GeneratePostcode extends AbstractGenerator<{ } export class GenerateUniquePostcode extends AbstractGenerator<{ isUnique?: boolean }> { - static override readonly [entityKind]: string = 'GenerateUniquePostcode'; + static override readonly entityKind: string = 'GenerateUniquePostcode'; private state: { rng: prand.RandomGenerator; @@ -2298,6 +2506,13 @@ export class GenerateUniquePostcode extends AbstractGenerator<{ isUnique?: boole }, ]; + const maxPostcodeLength = Math.max(...templates.map((template) => template.template.length)); + if (this.stringLength !== undefined && this.stringLength < maxPostcodeLength) { + throw new Error( + `You can't use postcode generator with a db column length restriction of ${this.stringLength}. Set the maximum string length to at least ${maxPostcodeLength}.`, + ); + } + for (const templateObj of templates) { templateObj.indicesGen.skipCheck = true; templateObj.indicesGen.init({ count, seed }); @@ -2338,7 +2553,7 @@ export class GenerateUniquePostcode extends AbstractGenerator<{ isUnique?: boole export class GenerateState extends AbstractGenerator<{ arraySize?: number; }> { - static override readonly [entityKind]: string = 'GenerateState'; + static override readonly entityKind: string = 'GenerateState'; private state: { rng: prand.RandomGenerator; @@ -2349,6 +2564,12 @@ export class GenerateState extends AbstractGenerator<{ const rng = prand.xoroshiro128plus(seed); + if (this.stringLength !== undefined && this.stringLength < maxStateLength) { + throw new Error( + `You can't use state generator with a db column length restriction of ${this.stringLength}. Set the maximum string length to at least ${maxStateLength}.`, + ); + } + this.state = { rng }; } @@ -2368,7 +2589,7 @@ export class GenerateCompanyName extends AbstractGenerator<{ isUnique?: boolean; arraySize?: number; }> { - static override readonly [entityKind]: string = 'GenerateCompanyName'; + static override readonly entityKind: string = 'GenerateCompanyName'; private state: { rng: prand.RandomGenerator; @@ -2387,6 +2608,17 @@ export class GenerateCompanyName extends AbstractGenerator<{ { template: '#, # and #', placeholdersCount: 3 }, ]; + // max( { template: '#', placeholdersCount: 1 }, { template: '#, # and #', placeholdersCount: 3 } ) + const maxCompanyNameLength = Math.max( + maxLastNameLength + maxCompanyNameSuffixLength + 1, + 3 * maxLastNameLength + 7, + ); + if (this.stringLength !== undefined && this.stringLength < maxCompanyNameLength) { + throw new Error( + `You can't use company name generator with a db column length restriction of ${this.stringLength}. Set the maximum string length to at least ${maxCompanyNameLength}.`, + ); + } + this.state = { rng, templates }; } @@ -2426,7 +2658,7 @@ export class GenerateCompanyName extends AbstractGenerator<{ } export class GenerateUniqueCompanyName extends AbstractGenerator<{ isUnique?: boolean }> { - static override readonly [entityKind]: string = 'GenerateUniqueCompanyName'; + static override readonly entityKind: string = 'GenerateUniqueCompanyName'; private state: { rng: prand.RandomGenerator; @@ -2450,6 +2682,17 @@ export class GenerateUniqueCompanyName extends AbstractGenerator<{ isUnique?: bo ); } + // max( { template: '#', placeholdersCount: 1 }, { template: '#, # and #', placeholdersCount: 3 } ) + const maxCompanyNameLength = Math.max( + maxLastNameLength + maxCompanyNameSuffixLength + 1, + 3 * maxLastNameLength + 7, + ); + if (this.stringLength !== undefined && this.stringLength < maxCompanyNameLength) { + throw new Error( + `You can't use company name generator with a db column length restriction of ${this.stringLength}. Set the maximum string length to at least ${maxCompanyNameLength}.`, + ); + } + const rng = prand.xoroshiro128plus(seed); // when count reach maxUniqueCompanyNameNumber template will be deleted from array const templates = [ @@ -2526,7 +2769,7 @@ export class GenerateLoremIpsum extends AbstractGenerator<{ sentencesCount?: number; arraySize?: number; }> { - static override readonly [entityKind]: string = 'GenerateLoremIpsum'; + static override readonly entityKind: string = 'GenerateLoremIpsum'; private state: { rng: prand.RandomGenerator; @@ -2538,6 +2781,14 @@ export class GenerateLoremIpsum extends AbstractGenerator<{ const rng = prand.xoroshiro128plus(seed); if (this.params.sentencesCount === undefined) this.params.sentencesCount = 1; + const maxLoremIpsumSentencesLength = maxLoremIpsumLength * this.params.sentencesCount + this.params.sentencesCount + - 1; + if (this.stringLength !== undefined && this.stringLength < maxLoremIpsumSentencesLength) { + throw new Error( + `You can't use lorem ipsum generator with a db column length restriction of ${this.stringLength}. Set the maximum string length to at least ${maxLoremIpsumSentencesLength}.`, + ); + } + this.state = { rng }; } @@ -2557,7 +2808,7 @@ export class GenerateLoremIpsum extends AbstractGenerator<{ } export class WeightedRandomGenerator extends AbstractGenerator<{ weight: number; value: AbstractGenerator }[]> { - static override readonly [entityKind]: string = 'WeightedRandomGenerator'; + static override readonly entityKind: string = 'WeightedRandomGenerator'; private state: { rng: prand.RandomGenerator; @@ -2627,7 +2878,7 @@ export class GeneratePoint extends AbstractGenerator<{ maxYValue?: number; arraySize?: number; }> { - static override readonly [entityKind]: string = 'GeneratePoint'; + static override readonly entityKind: string = 'GeneratePoint'; private state: { xCoordinateGen: GenerateNumber; @@ -2681,7 +2932,7 @@ export class GenerateUniquePoint extends AbstractGenerator<{ maxYValue?: number; isUnique?: boolean; }> { - static override readonly [entityKind]: string = 'GenerateUniquePoint'; + static override readonly entityKind: string = 'GenerateUniquePoint'; private state: { xCoordinateGen: GenerateUniqueNumber; @@ -2736,7 +2987,7 @@ export class GenerateLine extends AbstractGenerator<{ maxCValue?: number; arraySize?: number; }> { - static override readonly [entityKind]: string = 'GenerateLine'; + static override readonly entityKind: string = 'GenerateLine'; private state: { aCoefficientGen: GenerateNumber; @@ -2807,7 +3058,7 @@ export class GenerateUniqueLine extends AbstractGenerator<{ maxCValue?: number; isUnique?: boolean; }> { - static override readonly [entityKind]: string = 'GenerateUniqueLine'; + static override readonly entityKind: string = 'GenerateUniqueLine'; private state: { aCoefficientGen: GenerateUniqueNumber; @@ -2866,692 +3117,3 @@ export class GenerateUniqueLine extends AbstractGenerator<{ } } } - -export const generatorsFuncs = { - /** - * generates same given value each time the generator is called. - * @param defaultValue - value you want to generate - * @param arraySize - number of elements in each one-dimensional array. - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * posts: { - * columns: { - * content: funcs.default({ defaultValue: "post content" }), - * }, - * }, - * })); - * ``` - */ - default: createGenerator(GenerateDefault), - - /** - * generates values from given array - * @param values - array of values you want to generate. can be array of weighted values. - * @param isUnique - property that controls if generated values gonna be unique or not. - * @param arraySize - number of elements in each one-dimensional array. - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * posts: { - * columns: { - * title: funcs.valuesFromArray({ - * values: ["Title1", "Title2", "Title3", "Title4", "Title5"], - * isUnique: true - * }), - * }, - * }, - * })); - * - * ``` - * weighted values example - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * posts: { - * columns: { - * title: funcs.valuesFromArray({ - * values: [ - * { weight: 0.35, values: ["Title1", "Title2"] }, - * { weight: 0.5, values: ["Title3", "Title4"] }, - * { weight: 0.15, values: ["Title5"] }, - * ], - * isUnique: false - * }), - * }, - * }, - * })); - * - * ``` - */ - valuesFromArray: createGenerator(GenerateValuesFromArray), - - /** - * generates sequential integers starting with 1. - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * posts: { - * columns: { - * id: funcs.intPrimaryKey(), - * }, - * }, - * })); - * - * ``` - */ - intPrimaryKey: createGenerator(GenerateIntPrimaryKey), - - /** - * generates numbers with floating point in given range. - * @param minValue - lower border of range. - * @param maxValue - upper border of range. - * @param precision - precision of generated number: - * precision equals 10 means that values will be accurate to one tenth (1.2, 34.6); - * precision equals 100 means that values will be accurate to one hundredth (1.23, 34.67). - * @param isUnique - property that controls if generated values gonna be unique or not. - * @param arraySize - number of elements in each one-dimensional array. - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * products: { - * columns: { - * unitPrice: funcs.number({ minValue: 10, maxValue: 120, precision: 100, isUnique: false }), - * }, - * }, - * })); - * - * ``` - */ - number: createGenerator(GenerateNumber), - // uniqueNumber: createGenerator(GenerateUniqueNumber), - - /** - * generates integers within given range. - * @param minValue - lower border of range. - * @param maxValue - upper border of range. - * @param isUnique - property that controls if generated values gonna be unique or not. - * @param arraySize - number of elements in each one-dimensional array. - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * products: { - * columns: { - * unitsInStock: funcs.number({ minValue: 0, maxValue: 100, isUnique: false }), - * }, - * }, - * })); - * - * ``` - */ - int: createGenerator(GenerateInt), - // uniqueInt: createGenerator(GenerateUniqueInt), - - /** - * generates boolean values(true or false) - * @param arraySize - number of elements in each one-dimensional array. - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * users: { - * columns: { - * isAvailable: funcs.boolean() - * }, - * }, - * })); - * - * ``` - */ - boolean: createGenerator(GenerateBoolean), - - /** - * generates date within given range. - * @param minDate - lower border of range. - * @param maxDate - upper border of range. - * @param arraySize - number of elements in each one-dimensional array. - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * users: { - * columns: { - * birthDate: funcs.date({ minDate: "1990-01-01", maxDate: "2010-12-31" }) - * }, - * }, - * })); - * - * ``` - */ - date: createGenerator(GenerateDate), - - /** - * generates time in 24 hours style. - * @param arraySize - number of elements in each one-dimensional array. - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * users: { - * columns: { - * birthTime: funcs.time() - * }, - * }, - * })); - * - * ``` - */ - time: createGenerator(GenerateTime), - - /** - * generates timestamps. - * @param arraySize - number of elements in each one-dimensional array. - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * orders: { - * columns: { - * shippedDate: funcs.timestamp() - * }, - * }, - * })); - * - * ``` - */ - timestamp: createGenerator(GenerateTimestamp), - - /** - * generates datetime objects. - * @param arraySize - number of elements in each one-dimensional array. - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * orders: { - * columns: { - * shippedDate: funcs.datetime() - * }, - * }, - * })); - * - * ``` - */ - datetime: createGenerator(GenerateDatetime), - - /** - * generates years. - * @param arraySize - number of elements in each one-dimensional array. - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * users: { - * columns: { - * birthYear: funcs.year() - * }, - * }, - * })); - * - * ``` - */ - year: createGenerator(GenerateYear), - - /** - * generates json objects with fixed structure. - * @param arraySize - number of elements in each one-dimensional array. - * - * json structure can equal this: - * ``` - * { - * email, - * name, - * isGraduated, - * hasJob, - * salary, - * startedWorking, - * visitedCountries, - * } - * ``` - * or this - * ``` - * { - * email, - * name, - * isGraduated, - * hasJob, - * visitedCountries, - * } - * ``` - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * users: { - * columns: { - * metadata: funcs.json() - * }, - * }, - * })); - * ``` - */ - json: createGenerator(GenerateJson), - // jsonb: createGenerator(GenerateJsonb), - - /** - * generates time intervals. - * - * interval example: "1 years 12 days 5 minutes" - * - * @param isUnique - property that controls if generated values gonna be unique or not. - * @param arraySize - number of elements in each one-dimensional array. - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * users: { - * columns: { - * timeSpentOnWebsite: funcs.interval() - * }, - * }, - * })); - * ``` - */ - interval: createGenerator(GenerateInterval), - // uniqueInterval: createGenerator(GenerateUniqueInterval), - - /** - * generates random strings. - * @param isUnique - property that controls if generated values gonna be unique or not. - * @param arraySize - number of elements in each one-dimensional array. - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * users: { - * columns: { - * hashedPassword: funcs.string({isUnique: false}) - * }, - * }, - * })); - * ``` - */ - string: createGenerator(GenerateString), - // uniqueString: createGenerator(GenerateUniqueString), - - /** - * generates v4 UUID strings if arraySize is not specified, or v4 UUID 1D arrays if it is. - * - * @param arraySize - number of elements in each one-dimensional array. - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * users: { - * columns: { - * uuid: funcs.uuid({ - * arraySize: 4 - * }) - * }, - * }, - * })); - * ``` - */ - uuid: createGenerator(GenerateUUID), - - /** - * generates person's first names. - * @param isUnique - property that controls if generated values gonna be unique or not. - * @param arraySize - number of elements in each one-dimensional array. - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * users: { - * columns: { - * firstName: funcs.firstName({isUnique: true}) - * }, - * }, - * })); - * ``` - */ - firstName: createGenerator(GenerateFirstName), - // uniqueFirstName: createGenerator(GenerateUniqueName), - - /** - * generates person's last names. - * @param isUnique - property that controls if generated values gonna be unique or not. - * @param arraySize - number of elements in each one-dimensional array. - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * users: { - * columns: { - * lastName: funcs.lastName({isUnique: false}) - * }, - * }, - * })); - * ``` - */ - lastName: createGenerator(GenerateLastName), - // uniqueLastName: createGenerator(GenerateUniqueSurname), - - /** - * generates person's full names. - * @param isUnique - property that controls if generated values gonna be unique or not. - * @param arraySize - number of elements in each one-dimensional array. - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * users: { - * columns: { - * fullName: funcs.fullName({isUnique: true}) - * }, - * }, - * })); - * ``` - */ - fullName: createGenerator(GenerateFullName), - // uniqueFullName: createGenerator(GenerateUniqueFullName), - - /** - * generates unique emails. - * @param arraySize - number of elements in each one-dimensional array. - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * users: { - * columns: { - * email: funcs.email() - * }, - * }, - * })); - * ``` - */ - email: createGenerator(GenerateEmail), - - /** - * generates unique phone numbers. - * @param arraySize - number of elements in each one-dimensional array. - * - * @param template - phone number template, where all '#' symbols will be substituted with generated digits. - * @param prefixes - array of any string you want to be your phone number prefixes.(not compatible with template property) - * @param generatedDigitsNumbers - number of digits that will be added at the end of prefixes.(not compatible with template property) - * @example - * ```ts - * //generate phone number using template property - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * users: { - * columns: { - * phoneNumber: funcs.phoneNumber({template: "+(380) ###-####"}) - * }, - * }, - * })); - * - * //generate phone number using prefixes and generatedDigitsNumbers properties - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * users: { - * columns: { - * phoneNumber: funcs.phoneNumber({prefixes: [ "+380 99", "+380 67" ], generatedDigitsNumbers: 7}) - * }, - * }, - * })); - * - * //generate phone number using prefixes and generatedDigitsNumbers properties but with different generatedDigitsNumbers for prefixes - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * users: { - * columns: { - * phoneNumber: funcs.phoneNumber({prefixes: [ "+380 99", "+380 67", "+1" ], generatedDigitsNumbers: [7, 7, 10]}) - * }, - * }, - * })); - * - * ``` - */ - phoneNumber: createGenerator(GeneratePhoneNumber), - - /** - * generates country's names. - * @param isUnique - property that controls if generated values gonna be unique or not. - * @param arraySize - number of elements in each one-dimensional array. - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * users: { - * columns: { - * country: funcs.country({isUnique: false}) - * }, - * }, - * })); - * ``` - */ - country: createGenerator(GenerateCountry), - // uniqueCountry: createGenerator(GenerateUniqueCountry), - - /** - * generates city's names. - * @param isUnique - property that controls if generated values gonna be unique or not. - * @param arraySize - number of elements in each one-dimensional array. - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * users: { - * columns: { - * city: funcs.city({isUnique: false}) - * }, - * }, - * })); - * ``` - */ - city: createGenerator(GenerateCity), - // uniqueCity: createGenerator(GenerateUniqueCityName), - - /** - * generates street address. - * @param isUnique - property that controls if generated values gonna be unique or not. - * @param arraySize - number of elements in each one-dimensional array. - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * users: { - * columns: { - * streetAddress: funcs.streetAddress({isUnique: true}) - * }, - * }, - * })); - * ``` - */ - streetAddress: createGenerator(GenerateStreetAdddress), - // uniqueStreetAddress: createGenerator(GenerateUniqueStreetAdddress), - - /** - * generates job titles. - * @param arraySize - number of elements in each one-dimensional array. - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * users: { - * columns: { - * jobTitle: funcs.jobTitle() - * }, - * }, - * })); - * ``` - */ - jobTitle: createGenerator(GenerateJobTitle), - - /** - * generates postal codes. - * - * @param isUnique - property that controls if generated values gonna be unique or not. - * @param arraySize - number of elements in each one-dimensional array. - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * users: { - * columns: { - * postcode: funcs.postcode({isUnique: true}) - * }, - * }, - * })); - * ``` - */ - postcode: createGenerator(GeneratePostcode), - // uniquePostcoe: createGenerator(GenerateUniquePostcode), - - /** - * generates states of America. - * @param arraySize - number of elements in each one-dimensional array. - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * users: { - * columns: { - * state: funcs.state() - * }, - * }, - * })); - * ``` - */ - state: createGenerator(GenerateState), - - /** - * generates company's names. - * - * @param isUnique - property that controls if generated values gonna be unique or not. - * @param arraySize - number of elements in each one-dimensional array. - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * users: { - * columns: { - * company: funcs.companyName({isUnique: true}) - * }, - * }, - * })); - * ``` - */ - companyName: createGenerator(GenerateCompanyName), - // uniqueCompanyName: createGenerator(GenerateUniqueCompanyName), - - /** - * generates 'lorem ipsum' text sentences. - * - * @param sentencesCount - number of sentences you want to generate as one generated value(string). - * @param arraySize - number of elements in each one-dimensional array. - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * posts: { - * columns: { - * content: funcs.loremIpsum({sentencesCount: 2}) - * }, - * }, - * })); - * ``` - */ - loremIpsum: createGenerator(GenerateLoremIpsum), - - /** - * generates 2D points within specified ranges for x and y coordinates. - * - * @param isUnique - property that controls if generated values gonna be unique or not. - * @param minXValue - lower bound of range for x coordinate. - * @param maxXValue - upper bound of range for x coordinate. - * @param minYValue - lower bound of range for y coordinate. - * @param maxYValue - upper bound of range for y coordinate. - * @param arraySize - number of elements in each one-dimensional array. - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * triangles: { - * columns: { - * pointCoords: funcs.point({ - * isUnique: true, - * minXValue: -5, maxXValue:20, - * minYValue: 0, maxYValue: 30 - * }) - * }, - * }, - * })); - * ``` - */ - point: createGenerator(GeneratePoint), - // uniquePoint: createGenerator(GenerateUniquePoint), - - /** - * generates 2D lines within specified ranges for a, b and c parameters of line. - * - * ``` - * line equation: a*x + b*y + c = 0 - * ``` - * - * @param isUnique - property that controls if generated values gonna be unique or not. - * @param minAValue - lower bound of range for a parameter. - * @param maxAValue - upper bound of range for x parameter. - * @param minBValue - lower bound of range for y parameter. - * @param maxBValue - upper bound of range for y parameter. - * @param minCValue - lower bound of range for y parameter. - * @param maxCValue - upper bound of range for y parameter. - * @param arraySize - number of elements in each one-dimensional array. - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * lines: { - * columns: { - * lineParams: funcs.point({ - * isUnique: true, - * minAValue: -5, maxAValue:20, - * minBValue: 0, maxBValue: 30, - * minCValue: 0, maxCValue: 10 - * }) - * }, - * }, - * })); - * ``` - */ - line: createGenerator(GenerateLine), - // uniqueLine: createGenerator(GenerateUniqueLine), - - /** - * gives you the opportunity to call different generators with different probabilities to generate values for one column. - * @param params - array of generators with probabilities you would like to call them to generate values. - * - * @example - * ```ts - * await seed(db, schema, { count: 1000 }).refine((funcs) => ({ - * posts: { - * columns: { - * content: funcs.weightedRandom([ - * { - * weight: 0.6, - * value: funcs.loremIpsum({ sentencesCount: 3 }), - * }, - * { - * weight: 0.4, - * value: funcs.default({ defaultValue: "TODO" }), - * }, - * ]), - * }, - * }, - * })); - * ``` - */ - weightedRandom: createGenerator(WeightedRandomGenerator), -}; diff --git a/drizzle-seed/src/services/SeedService.ts b/drizzle-seed/src/services/SeedService.ts index 87569babb..84165169e 100644 --- a/drizzle-seed/src/services/SeedService.ts +++ b/drizzle-seed/src/services/SeedService.ts @@ -1,3 +1,4 @@ +/* eslint-disable drizzle-internal/require-entity-kind */ import { entityKind, eq, is } from 'drizzle-orm'; import type { MySqlTable, MySqlTableWithColumns } from 'drizzle-orm/mysql-core'; import { MySqlDatabase } from 'drizzle-orm/mysql-core'; @@ -11,39 +12,15 @@ import type { RefinementsType, TableGeneratorsType, } from '../types/seedService.ts'; -import type { Column, Prettify, Relation, RelationWithReferences, Table } from '../types/tables.ts'; -import type { AbstractGenerator } from './GeneratorsWrappers.ts'; -import { - GenerateArray, - GenerateBoolean, - GenerateDate, - GenerateDatetime, - GenerateDefault, - GenerateEmail, - GenerateEnum, - GenerateFirstName, - GenerateInt, - GenerateInterval, - GenerateIntPrimaryKey, - GenerateJson, - GenerateLine, - GenerateNumber, - GeneratePoint, - GenerateSelfRelationsValuesFromArray, - GenerateString, - GenerateTime, - GenerateTimestamp, - GenerateUniqueString, - GenerateUUID, - GenerateValuesFromArray, - GenerateWeightedCount, - GenerateYear, - HollowGenerator, -} from './GeneratorsWrappers.ts'; +import type { Column, Prettify, Relation, Table } from '../types/tables.ts'; +import { generatorsMap } from './GeneratorFuncs.ts'; +import type { AbstractGenerator, GenerateArray, GenerateInterval, GenerateWeightedCount } from './Generators.ts'; + +import { latestVersion } from './apiVersion.ts'; import { equalSets, generateHashFromString } from './utils.ts'; export class SeedService { - static readonly [entityKind]: string = 'SeedService'; + static readonly entityKind: string = 'SeedService'; private defaultCountForTable = 10; private postgresPgLiteMaxParametersNumber = 32740; @@ -52,21 +29,24 @@ export class SeedService { private mysqlMaxParametersNumber = 100000; // SQLITE_MAX_VARIABLE_NUMBER, which by default equals to 999 for SQLite versions prior to 3.32.0 (2020-05-22) or 32766 for SQLite versions after 3.32.0. private sqliteMaxParametersNumber = 32766; + private version?: number; generatePossibleGenerators = ( connectionType: 'postgresql' | 'mysql' | 'sqlite', tables: Table[], relations: (Relation & { isCyclic: boolean })[], - tableRelations: { [tableName: string]: RelationWithReferences[] }, refinements?: RefinementsType, - options?: { count?: number; seed?: number }, + options?: { count?: number; seed?: number; version?: number }, ) => { let columnPossibleGenerator: Prettify; let tablePossibleGenerators: Prettify; const customSeed = options?.seed === undefined ? 0 : options.seed; + this.version = options?.version === undefined ? latestVersion : options.version; + if (Number.isNaN(this.version) || this.version < 1 || this.version > latestVersion) { + throw new Error(`Version should be in range [1, ${latestVersion}].`); + } // sorting table in order which they will be filled up (tables with foreign keys case) - // relations = relations.filter(rel => rel.type === "one"); const { tablesInOutRelations } = this.getInfoFromRelations(relations); const orderedTablesNames = this.getOrderedTablesList(tablesInOutRelations); tables = tables.sort((table1, table2) => { @@ -203,11 +183,14 @@ export class SeedService { && refinements[table.name]!.columns[col.name] !== undefined ) { const genObj = refinements[table.name]!.columns[col.name]!; - // TODO: for now only GenerateValuesFromArray support notNull property - genObj.notNull = col.notNull; - if (col.dataType === 'array') { - if (col.baseColumn?.dataType === 'array' && col.baseColumn?.columnType === 'array') { - throw new Error("for now you can't specify generators for columns of dimensition greater than 1."); + + if (col.columnType.match(/\[\w*]/g) !== null) { + if ( + (col.baseColumn?.dataType === 'array' && col.baseColumn.columnType.match(/\[\w*]/g) !== null) + // studio case + || (col.typeParams.dimensions !== undefined && col.typeParams.dimensions > 1) + ) { + throw new Error("for now you can't specify generators for columns of dimension greater than 1."); } genObj.baseColumnDataType = col.baseColumn?.dataType; @@ -217,49 +200,79 @@ export class SeedService { columnPossibleGenerator.wasRefined = true; } else if (Object.hasOwn(foreignKeyColumns, col.name)) { // TODO: I might need to assign repeatedValuesCount to column there instead of doing so in generateTablesValues - const cyclicRelation = tableRelations[table.name]!.find((rel) => - rel.isCyclic === true + const cyclicRelation = relations.find((rel) => + rel.table === table.name + && rel.isCyclic === true && rel.columns.includes(col.name) ); if (cyclicRelation !== undefined) { columnPossibleGenerator.isCyclic = true; } - const predicate = cyclicRelation !== undefined && col.notNull === false; + + if (foreignKeyColumns[col.name]?.table === undefined && col.notNull === true) { + throw new Error( + `Column '${col.name}' has no null contraint, and you didn't specify a table for foreign key on column '${col.name}' in '${table.name}' table. You should pass `, + ); + } + + const predicate = (cyclicRelation !== undefined || foreignKeyColumns[col.name]?.table === undefined) + && col.notNull === false; if (predicate === true) { - columnPossibleGenerator.generator = new GenerateDefault({ defaultValue: null }); + if (foreignKeyColumns[col.name]?.table === undefined && col.notNull === false) { + console.warn( + `Column '${col.name}' in '${table.name}' table will be filled with Null values` + + `\nbecause you specified neither a table for foreign key on column '${col.name}' nor a function for '${col.name}' column in refinements.`, + ); + } + columnPossibleGenerator.generator = new generatorsMap.GenerateDefault[0]({ defaultValue: null }); columnPossibleGenerator.wasDefinedBefore = true; + } else { + columnPossibleGenerator.generator = new generatorsMap.HollowGenerator[0](); } - - columnPossibleGenerator.generator = new HollowGenerator({}); } // TODO: rewrite pickGeneratorFor... using new col properties: isUnique and notNull else if (connectionType === 'postgresql') { - columnPossibleGenerator.generator = this.pickGeneratorForPostgresColumn( + columnPossibleGenerator.generator = this.selectGeneratorForPostgresColumn( table, col, ); } else if (connectionType === 'mysql') { - columnPossibleGenerator.generator = this.pickGeneratorForMysqlColumn( + columnPossibleGenerator.generator = this.selectGeneratorForMysqlColumn( table, col, ); } else if (connectionType === 'sqlite') { - columnPossibleGenerator.generator = this.pickGeneratorForSqlite( + columnPossibleGenerator.generator = this.selectGeneratorForSqlite( table, col, ); } if (columnPossibleGenerator.generator === undefined) { - console.log(col); throw new Error( `column with type ${col.columnType} is not supported for now.`, ); } + const arrayGen = columnPossibleGenerator.generator.replaceIfArray(); + if (arrayGen !== undefined) { + columnPossibleGenerator.generator = arrayGen; + } + + const uniqueGen = columnPossibleGenerator.generator.replaceIfUnique(); + if (uniqueGen !== undefined) { + columnPossibleGenerator.generator = uniqueGen; + } + + // selecting version of generator + columnPossibleGenerator.generator = this.selectVersionOfGenerator(columnPossibleGenerator.generator); + columnPossibleGenerator.generator.isUnique = col.isUnique; + // TODO: for now only GenerateValuesFromArray support notNull property + columnPossibleGenerator.generator.notNull = col.notNull; columnPossibleGenerator.generator.dataType = col.dataType; + columnPossibleGenerator.generator.stringLength = col.typeParams.length; tablePossibleGenerators.columnsPossibleGenerators.push( columnPossibleGenerator, @@ -270,6 +283,38 @@ export class SeedService { return tablesPossibleGenerators; }; + selectVersionOfGenerator = (generator: AbstractGenerator) => { + const entityKind = generator.getEntityKind(); + if (entityKind === 'GenerateArray') { + const oldBaseColumnGen = (generator as GenerateArray).params.baseColumnGen; + + const newBaseColumnGen = this.selectVersionOfGenerator(oldBaseColumnGen); + // newGenerator.baseColumnDataType = oldGenerator.baseColumnDataType; + + (generator as GenerateArray).params.baseColumnGen = newBaseColumnGen; + } + + const possibleGeneratorConstructors = generatorsMap[entityKind as keyof typeof generatorsMap]; + + const possibleGeneratorConstructorsFiltered = possibleGeneratorConstructors?.filter((possGenCon) => + possGenCon.version <= this.version! // sorting in ascending order by version + ).sort((a, b) => a.version - b.version); + const generatorConstructor = possibleGeneratorConstructorsFiltered?.at(-1) as + | (new(params: any) => AbstractGenerator) + | undefined; + if (generatorConstructor === undefined) { + throw new Error(`Can't select ${entityKind} generator for ${this.version} version.`); + } + + const newGenerator = new generatorConstructor(generator.params); + newGenerator.baseColumnDataType = generator.baseColumnDataType; + newGenerator.isUnique = generator.isUnique; + newGenerator.dataType = generator.dataType; + newGenerator.stringLength = generator.stringLength; + + return newGenerator; + }; + cyclicTablesCompare = ( table1: Table, table2: Table, @@ -385,7 +430,10 @@ export class SeedService { }; } - if (tablesInOutRelations[rel.refTable] === undefined) { + if ( + rel.refTable !== undefined + && tablesInOutRelations[rel.refTable] === undefined + ) { tablesInOutRelations[rel.refTable] = { out: 0, in: 0, @@ -396,13 +444,15 @@ export class SeedService { }; } - tablesInOutRelations[rel.table]!.out += 1; - tablesInOutRelations[rel.refTable]!.in += 1; + if (rel.refTable !== undefined) { + tablesInOutRelations[rel.table]!.out += 1; + tablesInOutRelations[rel.refTable]!.in += 1; + } if (rel.refTable === rel.table) { tablesInOutRelations[rel.table]!.selfRelation = true; tablesInOutRelations[rel.table]!.selfRelCount = rel.columns.length; - } else { + } else if (rel.refTable !== undefined) { tablesInOutRelations[rel.table]!.requiredTableNames.add(rel.refTable); tablesInOutRelations[rel.refTable]!.dependantTableNames.add(rel.table); } @@ -416,7 +466,9 @@ export class SeedService { count: number, seed: number, ) => { - const gen = new GenerateWeightedCount({}); + let gen = new generatorsMap.GenerateWeightedCount[0](); + gen = this.selectVersionOfGenerator(gen) as GenerateWeightedCount; + // const gen = new GenerateWeightedCount({}); gen.init({ count: weightedCount, seed }); let weightedWithCount = 0; for (let i = 0; i < count; i++) { @@ -427,543 +479,609 @@ export class SeedService { }; // TODO: revise serial part generators - - pickGeneratorForPostgresColumn = ( + selectGeneratorForPostgresColumn = ( table: Table, col: Column, ) => { - let generator: AbstractGenerator | undefined; + const pickGenerator = (table: Table, col: Column) => { + // ARRAY + if (col.columnType.match(/\[\w*]/g) !== null && col.baseColumn !== undefined) { + const baseColumnGen = this.selectGeneratorForPostgresColumn( + table, + col.baseColumn!, + ) as AbstractGenerator; + if (baseColumnGen === undefined) { + throw new Error(`column with type ${col.baseColumn!.columnType} is not supported for now.`); + } - // INT ------------------------------------------------------------------------------------------------------------ - if ( - (col.columnType.includes('serial') - || col.columnType === 'integer' - || col.columnType === 'smallint' - || col.columnType.includes('bigint')) - && table.primaryKeys.includes(col.name) - ) { - generator = new GenerateIntPrimaryKey({}); + // const getBaseColumnDataType = (baseColumn: Column) => { + // if (baseColumn.baseColumn !== undefined) { + // return getBaseColumnDataType(baseColumn.baseColumn); + // } - generator.isUnique = col.isUnique; - generator.dataType = col.dataType; - return generator; - } + // return baseColumn.dataType; + // }; + // const baseColumnDataType = getBaseColumnDataType(col.baseColumn); - let minValue: number | bigint | undefined; - let maxValue: number | bigint | undefined; - if (col.columnType.includes('serial')) { - minValue = 1; - if (col.columnType === 'smallserial') { - // 2^16 / 2 - 1, 2 bytes - maxValue = 32767; - } else if (col.columnType === 'serial') { - // 2^32 / 2 - 1, 4 bytes - maxValue = 2147483647; - } else if (col.columnType === 'bigserial') { - // 2^64 / 2 - 1, 8 bytes - minValue = BigInt(1); - maxValue = BigInt('9223372036854775807'); + const generator = new generatorsMap.GenerateArray[0]({ baseColumnGen, size: col.size }); + // generator.baseColumnDataType = baseColumnDataType; + + return generator; } - } else if (col.columnType.includes('int')) { - if (col.columnType === 'smallint') { - // 2^16 / 2 - 1, 2 bytes - minValue = -32768; - maxValue = 32767; - } else if (col.columnType === 'integer') { - // 2^32 / 2 - 1, 4 bytes - minValue = -2147483648; - maxValue = 2147483647; - } else if (col.columnType.includes('bigint')) { - if (col.dataType === 'bigint') { + + // ARRAY for studio + if (col.columnType.match(/\[\w*]/g) !== null) { + // remove dimensions from type + const baseColumnType = col.columnType.replace(/\[\w*]/g, ''); + const baseColumn: Column = { + ...col, + }; + baseColumn.columnType = baseColumnType; + + const baseColumnGen = this.selectGeneratorForPostgresColumn(table, baseColumn) as AbstractGenerator; + if (baseColumnGen === undefined) { + throw new Error(`column with type ${col.baseColumn!.columnType} is not supported for now.`); + } + + let generator = new generatorsMap.GenerateArray[0]({ baseColumnGen }); + + for (let i = 0; i < col.typeParams.dimensions! - 1; i++) { + generator = new generatorsMap.GenerateArray[0]({ baseColumnGen: generator }); + } + + return generator; + } + + // INT ------------------------------------------------------------------------------------------------------------ + if ( + (col.columnType.includes('serial') + || col.columnType === 'integer' + || col.columnType === 'smallint' + || col.columnType.includes('bigint')) + && table.primaryKeys.includes(col.name) + ) { + const generator = new generatorsMap.GenerateIntPrimaryKey[0](); + + return generator; + } + + let minValue: number | bigint | undefined; + let maxValue: number | bigint | undefined; + if (col.columnType.includes('serial')) { + minValue = 1; + if (col.columnType === 'smallserial') { + // 2^16 / 2 - 1, 2 bytes + maxValue = 32767; + } else if (col.columnType === 'serial') { + // 2^32 / 2 - 1, 4 bytes + maxValue = 2147483647; + } else if (col.columnType === 'bigserial') { // 2^64 / 2 - 1, 8 bytes - minValue = BigInt('-9223372036854775808'); + minValue = BigInt(1); maxValue = BigInt('9223372036854775807'); - } else if (col.dataType === 'number') { - // if you’re expecting values above 2^31 but below 2^53 - minValue = -9007199254740991; - maxValue = 9007199254740991; + } + } else if (col.columnType.includes('int')) { + if (col.columnType === 'smallint') { + // 2^16 / 2 - 1, 2 bytes + minValue = -32768; + maxValue = 32767; + } else if (col.columnType === 'integer') { + // 2^32 / 2 - 1, 4 bytes + minValue = -2147483648; + maxValue = 2147483647; + } else if (col.columnType.includes('bigint')) { + if (col.dataType === 'bigint') { + // 2^64 / 2 - 1, 8 bytes + minValue = BigInt('-9223372036854775808'); + maxValue = BigInt('9223372036854775807'); + } else { + // if (col.dataType === 'number') + // if you’re expecting values above 2^31 but below 2^53 + minValue = -9007199254740991; + maxValue = 9007199254740991; + } } } - } - if ( - col.columnType.includes('int') - && !col.columnType.includes('interval') - && !col.columnType.includes('point') - ) { - generator = new GenerateInt({ - minValue, - maxValue, - }); + if ( + col.columnType.includes('int') + && !col.columnType.includes('interval') + && !col.columnType.includes('point') + ) { + const generator = new generatorsMap.GenerateInt[0]({ + minValue, + maxValue, + }); - generator.isUnique = col.isUnique; - generator.dataType = col.dataType; - return generator; - } + return generator; + } - if (col.columnType.includes('serial')) { - generator = new GenerateIntPrimaryKey({}); + if (col.columnType.includes('serial')) { + const generator = new generatorsMap.GenerateIntPrimaryKey[0](); - (generator as GenerateIntPrimaryKey).maxValue = maxValue; - generator.isUnique = col.isUnique; - generator.dataType = col.dataType; - return generator; - } + generator.maxValue = maxValue; - // NUMBER(real, double, decimal, numeric) - if ( - col.columnType === 'real' - || col.columnType === 'doubleprecision' - || col.columnType === 'decimal' - || col.columnType === 'numeric' - ) { - generator = new GenerateNumber({}); + return generator; + } - generator.isUnique = col.isUnique; - generator.dataType = col.dataType; - return generator; - } + // NUMBER(real, double, decimal, numeric) + if ( + col.columnType.startsWith('real') + || col.columnType.startsWith('double precision') + || col.columnType.startsWith('decimal') + || col.columnType.startsWith('numeric') + ) { + if (col.typeParams.precision !== undefined) { + const precision = col.typeParams.precision; + const scale = col.typeParams.scale === undefined ? 0 : col.typeParams.scale; + + const maxAbsoluteValue = Math.pow(10, precision - scale) - Math.pow(10, -scale); + const generator = new generatorsMap.GenerateNumber[0]({ + minValue: -maxAbsoluteValue, + maxValue: maxAbsoluteValue, + precision: Math.pow(10, scale), + }); + return generator; + } + const generator = new generatorsMap.GenerateNumber[0](); - // STRING - if ( - (col.columnType === 'text' - || col.columnType === 'varchar' - || col.columnType === 'char') - && table.primaryKeys.includes(col.name) - ) { - generator = new GenerateUniqueString({}); + return generator; + } - generator.isUnique = col.isUnique; - generator.dataType = col.dataType; - return generator; - } + // STRING + if ( + (col.columnType === 'text' + || col.columnType.startsWith('varchar') + || col.columnType.startsWith('char')) + && table.primaryKeys.includes(col.name) + ) { + const generator = new generatorsMap.GenerateUniqueString[0](); - if ( - (col.columnType === 'text' - || col.columnType === 'varchar' - || col.columnType === 'char') - && col.name.toLowerCase().includes('name') - ) { - generator = new GenerateFirstName({}); + return generator; + } - generator.isUnique = col.isUnique; - generator.dataType = col.dataType; - return generator; - } + if ( + (col.columnType === 'text' + || col.columnType.startsWith('varchar') + || col.columnType.startsWith('char')) + && col.name.toLowerCase().includes('name') + ) { + const generator = new generatorsMap.GenerateFirstName[0](); - if ( - (col.columnType === 'text' - || col.columnType === 'varchar' - || col.columnType === 'char') - && col.name.toLowerCase().includes('email') - ) { - generator = new GenerateEmail({}); + return generator; + } - generator.isUnique = col.isUnique; - generator.dataType = col.dataType; - return generator; - } + if ( + (col.columnType === 'text' + || col.columnType.startsWith('varchar') + || col.columnType.startsWith('char')) + && col.name.toLowerCase().includes('email') + ) { + const generator = new generatorsMap.GenerateEmail[0](); - if ( - col.columnType === 'text' - || col.columnType === 'varchar' - || col.columnType === 'char' - ) { - // console.log(col, table) - generator = new GenerateString({}); + return generator; + } - generator.isUnique = col.isUnique; - generator.dataType = col.dataType; - return generator; - } + if ( + col.columnType === 'text' + || col.columnType.startsWith('varchar') + || col.columnType.startsWith('char') + ) { + const generator = new generatorsMap.GenerateString[0](); - // UUID - if (col.columnType === 'uuid') { - generator = new GenerateUUID({}); + return generator; + } - generator.isUnique = col.isUnique; - generator.dataType = col.dataType; - return generator; - } + // UUID + if (col.columnType === 'uuid') { + const generator = new generatorsMap.GenerateUUID[0](); - // BOOLEAN - if (col.columnType === 'boolean') { - generator = new GenerateBoolean({}); + return generator; + } - generator.isUnique = col.isUnique; - generator.dataType = col.dataType; - return generator; - } + // BOOLEAN + if (col.columnType === 'boolean') { + const generator = new generatorsMap.GenerateBoolean[0](); - // DATE, TIME, TIMESTAMP - if (col.columnType.includes('date')) { - generator = new GenerateDate({}); + return generator; + } - generator.isUnique = col.isUnique; - generator.dataType = col.dataType; - return generator; - } + // DATE, TIME, TIMESTAMP + if (col.columnType.includes('date')) { + const generator = new generatorsMap.GenerateDate[0](); - if (col.columnType === 'time') { - generator = new GenerateTime({}); + return generator; + } - generator.isUnique = col.isUnique; - generator.dataType = col.dataType; - return generator; - } + if (col.columnType === 'time') { + const generator = new generatorsMap.GenerateTime[0](); - if (col.columnType.includes('timestamp')) { - generator = new GenerateTimestamp({}); + return generator; + } - generator.isUnique = col.isUnique; - generator.dataType = col.dataType; - return generator; - } + if (col.columnType.includes('timestamp')) { + const generator = new generatorsMap.GenerateTimestamp[0](); - // JSON, JSONB - if (col.columnType === 'json' || col.columnType === 'jsonb') { - generator = new GenerateJson({}); + return generator; + } - generator.isUnique = col.isUnique; - generator.dataType = col.dataType; - return generator; - } + // JSON, JSONB + if (col.columnType === 'json' || col.columnType === 'jsonb') { + const generator = new generatorsMap.GenerateJson[0](); - // if (col.columnType === "jsonb") { - // const generator = new GenerateJsonb({}); - // return generator; - // } + return generator; + } - // ENUM - if (col.enumValues !== undefined) { - generator = new GenerateEnum({ - enumValues: col.enumValues, - }); + // if (col.columnType === "jsonb") { + // const generator = new GenerateJsonb({}); + // return generator; + // } - generator.isUnique = col.isUnique; - generator.dataType = col.dataType; - return generator; - } + // ENUM + if (col.enumValues !== undefined) { + const generator = new generatorsMap.GenerateEnum[0]({ + enumValues: col.enumValues, + }); - // INTERVAL - if (col.columnType === 'interval') { - generator = new GenerateInterval({}); + return generator; + } - generator.isUnique = col.isUnique; - generator.dataType = col.dataType; - return generator; - } + // INTERVAL + if (col.columnType.startsWith('interval')) { + if (col.columnType === 'interval') { + const generator = new generatorsMap.GenerateInterval[0](); - // POINT, LINE - if (col.columnType.includes('point')) { - generator = new GeneratePoint({}); + return generator; + } - generator.isUnique = col.isUnique; - generator.dataType = col.dataType; - return generator; - } + const fields = col.columnType.replace('interval ', '') as GenerateInterval['params']['fields']; + const generator = new generatorsMap.GenerateInterval[0]({ fields }); - if (col.columnType.includes('line')) { - generator = new GenerateLine({}); + return generator; + } - generator.isUnique = col.isUnique; - generator.dataType = col.dataType; - return generator; - } + // POINT, LINE + if (col.columnType.includes('point')) { + const generator = new generatorsMap.GeneratePoint[0](); + + return generator; + } + + if (col.columnType.includes('line')) { + const generator = new generatorsMap.GenerateLine[0](); - // ARRAY - if (col.columnType.includes('array') && col.baseColumn !== undefined) { - const baseColumnGen = this.pickGeneratorForPostgresColumn(table, col.baseColumn!) as AbstractGenerator; - if (baseColumnGen === undefined) { - throw new Error(`column with type ${col.baseColumn!.columnType} is not supported for now.`); + return generator; + } + + if (col.hasDefault && col.default !== undefined) { + const generator = new generatorsMap.GenerateDefault[0]({ + defaultValue: col.default, + }); + return generator; } - generator = new GenerateArray({ baseColumnGen, size: col.size }); + return; + }; + + const generator = pickGenerator(table, col); + if (generator !== undefined) { generator.isUnique = col.isUnique; generator.dataType = col.dataType; - return generator; - } - - if (col.hasDefault && col.default !== undefined) { - generator = new GenerateDefault({ - defaultValue: col.default, - }); - return generator; + generator.stringLength = col.typeParams.length; } return generator; }; - pickGeneratorForMysqlColumn = ( + selectGeneratorForMysqlColumn = ( table: Table, col: Column, ) => { - // console.log(col); - // INT ------------------------------------------------------------------------------------------------------------ - if ( - (col.columnType.includes('serial') || col.columnType.includes('int')) - && table.primaryKeys.includes(col.name) - ) { - const generator = new GenerateIntPrimaryKey({}); - return generator; - } + const pickGenerator = (table: Table, col: Column) => { + // INT ------------------------------------------------------------------------------------------------------------ + if ( + (col.columnType.includes('serial') || col.columnType.includes('int')) + && table.primaryKeys.includes(col.name) + ) { + const generator = new generatorsMap.GenerateIntPrimaryKey[0](); + return generator; + } - let minValue: number | bigint | undefined; - let maxValue: number | bigint | undefined; - if (col.columnType === 'serial') { - // 2^64 % 2 - 1, 8 bytes - minValue = BigInt(0); - maxValue = BigInt('9223372036854775807'); - } else if (col.columnType.includes('int')) { - if (col.columnType === 'tinyint') { - // 2^8 / 2 - 1, 1 bytes - minValue = -128; - maxValue = 127; - } else if (col.columnType === 'smallint') { - // 2^16 / 2 - 1, 2 bytes - minValue = -32768; - maxValue = 32767; - } else if (col.columnType === 'mediumint') { - // 2^16 / 2 - 1, 2 bytes - minValue = -8388608; - maxValue = 8388607; - } else if (col.columnType === 'int') { - // 2^32 / 2 - 1, 4 bytes - minValue = -2147483648; - maxValue = 2147483647; - } else if (col.columnType === 'bigint') { - // 2^64 / 2 - 1, 8 bytes - minValue = BigInt('-9223372036854775808'); + let minValue: number | bigint | undefined; + let maxValue: number | bigint | undefined; + if (col.columnType === 'serial') { + // 2^64 % 2 - 1, 8 bytes + minValue = BigInt(0); maxValue = BigInt('9223372036854775807'); + } else if (col.columnType.includes('int')) { + if (col.columnType === 'tinyint') { + // 2^8 / 2 - 1, 1 bytes + minValue = -128; + maxValue = 127; + } else if (col.columnType === 'smallint') { + // 2^16 / 2 - 1, 2 bytes + minValue = -32768; + maxValue = 32767; + } else if (col.columnType === 'mediumint') { + // 2^16 / 2 - 1, 2 bytes + minValue = -8388608; + maxValue = 8388607; + } else if (col.columnType === 'int') { + // 2^32 / 2 - 1, 4 bytes + minValue = -2147483648; + maxValue = 2147483647; + } else if (col.columnType === 'bigint') { + // 2^64 / 2 - 1, 8 bytes + minValue = BigInt('-9223372036854775808'); + maxValue = BigInt('9223372036854775807'); + } } - } - if (col.columnType.includes('int')) { - const generator = new GenerateInt({ - minValue, - maxValue, - }); - return generator; - } + if (col.columnType.includes('int')) { + const generator = new generatorsMap.GenerateInt[0]({ + minValue, + maxValue, + }); + return generator; + } - if (col.columnType.includes('serial')) { - const generator = new GenerateIntPrimaryKey({}); - generator.maxValue = maxValue; - return generator; - } + if (col.columnType.includes('serial')) { + const generator = new generatorsMap.GenerateIntPrimaryKey[0](); + generator.maxValue = maxValue; + return generator; + } - // NUMBER(real, double, decimal, float) - if ( - col.columnType === 'real' - || col.columnType === 'double' - || col.columnType === 'decimal' - || col.columnType === 'float' - ) { - const generator = new GenerateNumber({}); - return generator; - } + // NUMBER(real, double, decimal, float) + if ( + col.columnType.startsWith('real') + || col.columnType.startsWith('double') + || col.columnType.startsWith('decimal') + || col.columnType.startsWith('float') + || col.columnType.startsWith('numeric') + ) { + if (col.typeParams.precision !== undefined) { + const precision = col.typeParams.precision; + const scale = col.typeParams.scale === undefined ? 0 : col.typeParams.scale; + + const maxAbsoluteValue = Math.pow(10, precision - scale) - Math.pow(10, -scale); + const generator = new generatorsMap.GenerateNumber[0]({ + minValue: -maxAbsoluteValue, + maxValue: maxAbsoluteValue, + precision: Math.pow(10, scale), + }); + return generator; + } - // STRING - if ( - (col.columnType === 'text' - || col.columnType === 'blob' - || col.columnType.includes('char') - || col.columnType.includes('varchar') - || col.columnType.includes('binary') - || col.columnType.includes('varbinary')) - && table.primaryKeys.includes(col.name) - ) { - const generator = new GenerateUniqueString({}); - return generator; - } + const generator = new generatorsMap.GenerateNumber[0](); + return generator; + } - if ( - (col.columnType === 'text' - || col.columnType === 'blob' - || col.columnType.includes('char') - || col.columnType.includes('varchar') - || col.columnType.includes('binary') - || col.columnType.includes('varbinary')) - && col.name.toLowerCase().includes('name') - ) { - const generator = new GenerateFirstName({}); - return generator; - } + // STRING + if ( + (col.columnType === 'text' + || col.columnType === 'blob' + || col.columnType.startsWith('char') + || col.columnType.startsWith('varchar') + || col.columnType.startsWith('binary') + || col.columnType.startsWith('varbinary')) + && table.primaryKeys.includes(col.name) + ) { + const generator = new generatorsMap.GenerateUniqueString[0](); + return generator; + } - if ( - (col.columnType === 'text' + if ( + (col.columnType === 'text' + || col.columnType === 'blob' + || col.columnType.startsWith('char') + || col.columnType.startsWith('varchar') + || col.columnType.startsWith('binary') + || col.columnType.startsWith('varbinary')) + && col.name.toLowerCase().includes('name') + ) { + const generator = new generatorsMap.GenerateFirstName[0](); + return generator; + } + + if ( + (col.columnType === 'text' + || col.columnType === 'blob' + || col.columnType.startsWith('char') + || col.columnType.startsWith('varchar') + || col.columnType.startsWith('binary') + || col.columnType.startsWith('varbinary')) + && col.name.toLowerCase().includes('email') + ) { + const generator = new generatorsMap.GenerateEmail[0](); + return generator; + } + + if ( + col.columnType === 'text' || col.columnType === 'blob' - || col.columnType.includes('char') - || col.columnType.includes('varchar') - || col.columnType.includes('binary') - || col.columnType.includes('varbinary')) - && col.name.toLowerCase().includes('email') - ) { - const generator = new GenerateEmail({}); - return generator; - } + || col.columnType.startsWith('char') + || col.columnType.startsWith('varchar') + || col.columnType.startsWith('binary') + || col.columnType.startsWith('varbinary') + ) { + const generator = new generatorsMap.GenerateString[0](); + return generator; + } - if ( - col.columnType === 'text' - || col.columnType === 'blob' - || col.columnType.includes('char') - || col.columnType.includes('varchar') - || col.columnType.includes('binary') - || col.columnType.includes('varbinary') - ) { - // console.log(col, table); - const generator = new GenerateString({}); - return generator; - } + // BOOLEAN + if (col.columnType === 'boolean') { + const generator = new generatorsMap.GenerateBoolean[0](); + return generator; + } - // BOOLEAN - if (col.columnType === 'boolean') { - const generator = new GenerateBoolean({}); - return generator; - } + // DATE, TIME, TIMESTAMP, DATETIME, YEAR + if (col.columnType.includes('datetime')) { + const generator = new generatorsMap.GenerateDatetime[0](); + return generator; + } - // DATE, TIME, TIMESTAMP, DATETIME, YEAR - if (col.columnType.includes('datetime')) { - const generator = new GenerateDatetime({}); - return generator; - } + if (col.columnType.includes('date')) { + const generator = new generatorsMap.GenerateDate[0](); + return generator; + } - if (col.columnType.includes('date')) { - const generator = new GenerateDate({}); - return generator; - } + if (col.columnType === 'time') { + const generator = new generatorsMap.GenerateTime[0](); + return generator; + } - if (col.columnType === 'time') { - const generator = new GenerateTime({}); - return generator; - } + if (col.columnType.includes('timestamp')) { + const generator = new generatorsMap.GenerateTimestamp[0](); + return generator; + } - if (col.columnType.includes('timestamp')) { - const generator = new GenerateTimestamp({}); - return generator; - } + if (col.columnType === 'year') { + const generator = new generatorsMap.GenerateYear[0](); + return generator; + } - if (col.columnType === 'year') { - const generator = new GenerateYear({}); - return generator; - } + // JSON + if (col.columnType === 'json') { + const generator = new generatorsMap.GenerateJson[0](); + return generator; + } - // JSON - if (col.columnType === 'json') { - const generator = new GenerateJson({}); - return generator; - } + // ENUM + if (col.enumValues !== undefined) { + const generator = new generatorsMap.GenerateEnum[0]({ + enumValues: col.enumValues, + }); + return generator; + } - // ENUM - if (col.enumValues !== undefined) { - const generator = new GenerateEnum({ - enumValues: col.enumValues, - }); - return generator; - } + if (col.hasDefault && col.default !== undefined) { + const generator = new generatorsMap.GenerateDefault[0]({ + defaultValue: col.default, + }); + return generator; + } - if (col.hasDefault && col.default !== undefined) { - const generator = new GenerateDefault({ - defaultValue: col.default, - }); - return generator; - } + return; + }; + + const generator = pickGenerator(table, col); - return; + return generator; }; - pickGeneratorForSqlite = ( + selectGeneratorForSqlite = ( table: Table, col: Column, ) => { - // int section --------------------------------------------------------------------------------------- - if ( - (col.columnType === 'integer' || col.columnType === 'numeric') - && table.primaryKeys.includes(col.name) - ) { - const generator = new GenerateIntPrimaryKey({}); - return generator; - } + const pickGenerator = (table: Table, col: Column) => { + // int section --------------------------------------------------------------------------------------- + if ( + (col.columnType === 'integer' || col.columnType === 'numeric') + && table.primaryKeys.includes(col.name) + ) { + const generator = new generatorsMap.GenerateIntPrimaryKey[0](); + return generator; + } - if ( - col.columnType === 'integer' - || col.columnType === 'numeric' - || col.columnType === 'bigint' - ) { - const generator = new GenerateInt({}); - return generator; - } + if (col.columnType === 'integer' && col.dataType === 'boolean') { + const generator = new generatorsMap.GenerateBoolean[0](); + return generator; + } - if (col.columnType === 'boolean') { - const generator = new GenerateBoolean({}); - return generator; - } + if ((col.columnType === 'integer' && col.dataType === 'date')) { + const generator = new generatorsMap.GenerateTimestamp[0](); + return generator; + } - // number section ------------------------------------------------------------------------------------ - if (col.columnType === 'real' || col.columnType === 'numeric') { - const generator = new GenerateNumber({}); - return generator; - } + if ( + col.columnType === 'integer' + || (col.dataType === 'bigint' && col.columnType === 'blob') + ) { + const generator = new generatorsMap.GenerateInt[0](); + return generator; + } - // string section ------------------------------------------------------------------------------------ - if ( - (col.columnType === 'text' - || col.columnType === 'numeric' - || col.columnType === 'blob') - && table.primaryKeys.includes(col.name) - ) { - const generator = new GenerateUniqueString({}); - return generator; - } + // number section ------------------------------------------------------------------------------------ + if (col.columnType.startsWith('real') || col.columnType.startsWith('numeric')) { + if (col.typeParams.precision !== undefined) { + const precision = col.typeParams.precision; + const scale = col.typeParams.scale === undefined ? 0 : col.typeParams.scale; + + const maxAbsoluteValue = Math.pow(10, precision - scale) - Math.pow(10, -scale); + const generator = new generatorsMap.GenerateNumber[0]({ + minValue: -maxAbsoluteValue, + maxValue: maxAbsoluteValue, + precision: Math.pow(10, scale), + }); + return generator; + } - if ( - (col.columnType === 'text' - || col.columnType === 'numeric' - || col.columnType === 'blob') - && col.name.toLowerCase().includes('name') - ) { - const generator = new GenerateFirstName({}); - return generator; - } + const generator = new generatorsMap.GenerateNumber[0](); + return generator; + } - if ( - (col.columnType === 'text' - || col.columnType === 'numeric' - || col.columnType === 'blob') - && col.name.toLowerCase().includes('email') - ) { - const generator = new GenerateEmail({}); - return generator; - } + // string section ------------------------------------------------------------------------------------ + if ( + (col.columnType.startsWith('text') + || col.columnType.startsWith('numeric') + || col.columnType.startsWith('blob')) + && table.primaryKeys.includes(col.name) + ) { + const generator = new generatorsMap.GenerateUniqueString[0](); + return generator; + } - if ( - col.columnType === 'text' - || col.columnType === 'numeric' - || col.columnType === 'blob' - || col.columnType === 'blobbuffer' - ) { - const generator = new GenerateString({}); - return generator; - } + if ( + (col.columnType.startsWith('text') + || col.columnType.startsWith('numeric') + || col.columnType.startsWith('blob')) + && col.name.toLowerCase().includes('name') + ) { + const generator = new generatorsMap.GenerateFirstName[0](); + return generator; + } - if (col.columnType === 'textjson' || col.columnType === 'blobjson') { - const generator = new GenerateJson({}); - return generator; - } + if ( + (col.columnType.startsWith('text') + || col.columnType.startsWith('numeric') + || col.columnType.startsWith('blob')) + && col.name.toLowerCase().includes('email') + ) { + const generator = new generatorsMap.GenerateEmail[0](); + return generator; + } - if (col.columnType === 'timestamp' || col.columnType === 'timestamp_ms') { - const generator = new GenerateTimestamp({}); - return generator; - } + if ( + col.columnType.startsWith('text') + || col.columnType.startsWith('numeric') + || col.columnType.startsWith('blob') + || col.columnType.startsWith('blobbuffer') + ) { + const generator = new generatorsMap.GenerateString[0](); + return generator; + } - if (col.hasDefault && col.default !== undefined) { - const generator = new GenerateDefault({ - defaultValue: col.default, - }); - return generator; - } + if ( + (col.columnType.startsWith('text') && col.dataType === 'json') + || (col.columnType.startsWith('blob') && col.dataType === 'json') + ) { + const generator = new generatorsMap.GenerateJson[0](); + return generator; + } - return; + if (col.hasDefault && col.default !== undefined) { + const generator = new generatorsMap.GenerateDefault[0]({ + defaultValue: col.default, + }); + return generator; + } + + return; + }; + + const generator = pickGenerator(table, col); + + return generator; }; filterCyclicTables = (tablesGenerators: ReturnType) => { @@ -1024,9 +1142,6 @@ export class SeedService { tablesUniqueNotNullColumn?: { [tableName: string]: { uniqueNotNullColName: string } }; }, ) => { - // console.time( - // "generateTablesValues-----------------------------------------------------" - // ); const customSeed = options?.seed === undefined ? 0 : options.seed; let tableCount: number | undefined; let columnsGenerators: Prettify[]; @@ -1042,7 +1157,6 @@ export class SeedService { }[] = options?.tablesValues === undefined ? [] : options.tablesValues; let pRNGSeed: number; - // relations = relations.filter(rel => rel.type === "one"); let filteredRelations: typeof relations; let preserveData: boolean, insertDataInDb: boolean = true, updateDataInDb: boolean = false; @@ -1124,9 +1238,13 @@ export class SeedService { }))!.map((rows) => rows[refColName]) as (string | number | boolean)[]; hasSelfRelation = true; - genObj = new GenerateSelfRelationsValuesFromArray({ + genObj = new generatorsMap.GenerateSelfRelationsValuesFromArray[0]({ values: refColumnValues, }); + genObj = this.selectVersionOfGenerator(genObj); + // genObj = new GenerateSelfRelationsValuesFromArray({ + // values: refColumnValues, + // }); } else if ( tableGenerators[rel.columns[colIdx]!]?.wasDefinedBefore === false && tableGenerators[rel.columns[colIdx]!]?.wasRefined === false @@ -1144,13 +1262,13 @@ export class SeedService { weightedCountSeed = table.withFromTable[rel.refTable]!.weightedCountSeed; } - genObj = new GenerateValuesFromArray({ values: refColumnValues }); - (genObj as GenerateValuesFromArray).notNull = tableGenerators[rel.columns[colIdx]!]!.notNull; - (genObj as GenerateValuesFromArray).weightedCountSeed = weightedCountSeed; - (genObj as GenerateValuesFromArray).maxRepeatedValuesCount = repeatedValuesCount; + // TODO: revise maybe need to select version of generator here too + genObj = new generatorsMap.GenerateValuesFromArray[0]({ values: refColumnValues }); + genObj.notNull = tableGenerators[rel.columns[colIdx]!]!.notNull; + genObj.weightedCountSeed = weightedCountSeed; + genObj.maxRepeatedValuesCount = repeatedValuesCount; } - // console.log(rel.columns[colIdx], tableGenerators) if (genObj !== undefined) { tableGenerators[rel.columns[colIdx]!]!.generator = genObj; } @@ -1265,15 +1383,15 @@ export class SeedService { seed: columnGenerator.pRNGSeed, }); - const arrayGen = columnsGenerators[columnName]!.replaceIfArray({ count, seed: columnGenerator.pRNGSeed }); - if (arrayGen !== undefined) { - columnsGenerators[columnName] = arrayGen; - } + // const arrayGen = columnsGenerators[columnName]!.replaceIfArray({ count, seed: columnGenerator.pRNGSeed }); + // if (arrayGen !== undefined) { + // columnsGenerators[columnName] = arrayGen; + // } - const uniqueGen = columnsGenerators[columnName]!.replaceIfUnique({ count, seed: columnGenerator.pRNGSeed }); - if (uniqueGen !== undefined) { - columnsGenerators[columnName] = uniqueGen; - } + // const uniqueGen = columnsGenerators[columnName]!.replaceIfUnique({ count, seed: columnGenerator.pRNGSeed }); + // if (uniqueGen !== undefined) { + // columnsGenerators[columnName] = uniqueGen; + // } } let maxParametersNumber: number; if (is(db, PgDatabase)) { diff --git a/drizzle-seed/src/services/apiVersion.ts b/drizzle-seed/src/services/apiVersion.ts new file mode 100644 index 000000000..6cda0267e --- /dev/null +++ b/drizzle-seed/src/services/apiVersion.ts @@ -0,0 +1 @@ +export const latestVersion = 2; diff --git a/drizzle-seed/src/services/versioning/v2.ts b/drizzle-seed/src/services/versioning/v2.ts new file mode 100644 index 000000000..e09cc45f6 --- /dev/null +++ b/drizzle-seed/src/services/versioning/v2.ts @@ -0,0 +1,232 @@ +/* eslint-disable drizzle-internal/require-entity-kind */ +import prand from 'pure-rand'; +import { AbstractGenerator } from '../Generators.ts'; + +export class GenerateUniqueIntervalV2 extends AbstractGenerator<{ + fields?: + | 'year' + | 'month' + | 'day' + | 'hour' + | 'minute' + | 'second' + | 'year to month' + | 'day to hour' + | 'day to minute' + | 'day to second' + | 'hour to minute' + | 'hour to second' + | 'minute to second'; + isUnique?: boolean; +}> { + static override readonly 'entityKind': string = 'GenerateUniqueInterval'; + static override readonly version: number = 2; + + private state: { + rng: prand.RandomGenerator; + fieldsToGenerate: string[]; + intervalSet: Set; + } | undefined; + public override isUnique = true; + private config: { [key: string]: { from: number; to: number } } = { + year: { + from: 0, + to: 5, + }, + month: { + from: 0, + to: 11, + }, + day: { + from: 0, + to: 29, + }, + hour: { + from: 0, + to: 23, + }, + minute: { + from: 0, + to: 59, + }, + second: { + from: 0, + to: 59, + }, + }; + + override init({ count, seed }: { count: number; seed: number }) { + const allFields = ['year', 'month', 'day', 'hour', 'minute', 'second']; + let fieldsToGenerate: string[] = allFields; + + if (this.params.fields !== undefined && this.params.fields?.includes(' to ')) { + const tokens = this.params.fields.split(' to '); + const endIdx = allFields.indexOf(tokens[1]!); + fieldsToGenerate = allFields.slice(0, endIdx + 1); + } else if (this.params.fields !== undefined) { + const endIdx = allFields.indexOf(this.params.fields); + fieldsToGenerate = allFields.slice(0, endIdx + 1); + } + + let maxUniqueIntervalsNumber = 1; + for (const field of fieldsToGenerate) { + const from = this.config[field]!.from, to = this.config[field]!.to; + maxUniqueIntervalsNumber *= from - to + 1; + } + + if (count > maxUniqueIntervalsNumber) { + throw new RangeError(`count exceeds max number of unique intervals(${maxUniqueIntervalsNumber})`); + } + + const rng = prand.xoroshiro128plus(seed); + const intervalSet = new Set(); + this.state = { rng, fieldsToGenerate, intervalSet }; + } + + generate() { + if (this.state === undefined) { + throw new Error('state is not defined.'); + } + + let interval, numb: number; + + for (;;) { + interval = ''; + + for (const field of this.state.fieldsToGenerate) { + const from = this.config[field]!.from, to = this.config[field]!.to; + [numb, this.state.rng] = prand.uniformIntDistribution(from, to, this.state.rng); + interval += `${numb} ${field} `; + } + + if (!this.state.intervalSet.has(interval)) { + this.state.intervalSet.add(interval); + break; + } + } + + return interval; + } +} + +export class GenerateStringV2 extends AbstractGenerator<{ + isUnique?: boolean; + arraySize?: number; +}> { + static override readonly 'entityKind': string = 'GenerateString'; + static override readonly version: number = 2; + + private state: { + rng: prand.RandomGenerator; + minStringLength: number; + maxStringLength: number; + } | undefined; + override uniqueVersionOfGen = GenerateUniqueStringV2; + + override init({ count, seed }: { count: number; seed: number }) { + super.init({ count, seed }); + + let minStringLength = 8; + let maxStringLength = 20; + if (this.stringLength !== undefined) { + maxStringLength = this.stringLength; + if (maxStringLength === 1) minStringLength = maxStringLength; + if (maxStringLength < minStringLength) minStringLength = 1; + } + + const rng = prand.xoroshiro128plus(seed); + this.state = { rng, minStringLength, maxStringLength }; + } + + generate() { + if (this.state === undefined) { + throw new Error('state is not defined.'); + } + + const minStringLength = this.state.minStringLength, + maxStringLength = this.state.maxStringLength; + const stringChars = '1234567890abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ'; + let idx: number, + strLength: number, + currStr: string; + + currStr = ''; + [strLength, this.state.rng] = prand.uniformIntDistribution( + minStringLength, + maxStringLength, + this.state.rng, + ); + for (let j = 0; j < strLength; j++) { + [idx, this.state.rng] = prand.uniformIntDistribution( + 0, + stringChars.length - 1, + this.state.rng, + ); + currStr += stringChars[idx]; + } + return currStr; + } +} + +export class GenerateUniqueStringV2 extends AbstractGenerator<{ isUnique?: boolean }> { + static override readonly 'entityKind': string = 'GenerateUniqueString'; + static override readonly version: number = 2; + + private state: { + rng: prand.RandomGenerator; + minStringLength: number; + maxStringLength: number; + } | undefined; + public override isUnique = true; + + override init({ seed, count }: { seed: number; count: number }) { + const rng = prand.xoroshiro128plus(seed); + + let minStringLength = 8; + let maxStringLength = 20; + // TODO: revise later + if (this.stringLength !== undefined) { + maxStringLength = this.stringLength; + if (maxStringLength === 1 || maxStringLength < minStringLength) minStringLength = maxStringLength; + } + + if (maxStringLength < count.toString(16).length) { + throw new Error( + `You can't generate ${count} unique strings, with a maximum string length of ${maxStringLength}.`, + ); + } + + this.state = { rng, minStringLength, maxStringLength }; + } + + generate({ i }: { i: number }) { + if (this.state === undefined) { + throw new Error('state is not defined.'); + } + + const minStringLength = this.state.minStringLength, + maxStringLength = this.state.maxStringLength; + const stringChars = '1234567890abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ'; + let idx: number, + strLength: number; + let currStr: string; + + currStr = ''; + const uniqueStr = i.toString(16); + [strLength, this.state.rng] = prand.uniformIntDistribution( + minStringLength, + maxStringLength - uniqueStr.length, + this.state.rng, + ); + for (let j = 0; j < strLength - uniqueStr.length; j++) { + [idx, this.state.rng] = prand.uniformIntDistribution( + 0, + stringChars.length - 1, + this.state.rng, + ); + currStr += stringChars[idx]; + } + + return uniqueStr + currStr; + } +} diff --git a/drizzle-seed/src/types/seedService.ts b/drizzle-seed/src/types/seedService.ts index 0b5237468..1ae06f44c 100644 --- a/drizzle-seed/src/types/seedService.ts +++ b/drizzle-seed/src/types/seedService.ts @@ -1,4 +1,4 @@ -import type { AbstractGenerator } from '../services/GeneratorsWrappers.ts'; +import type { AbstractGenerator } from '../services/Generators.ts'; import type { Prettify } from './tables.ts'; export type TableGeneratorsType = { diff --git a/drizzle-seed/src/types/tables.ts b/drizzle-seed/src/types/tables.ts index 8473179ed..dc28c748d 100644 --- a/drizzle-seed/src/types/tables.ts +++ b/drizzle-seed/src/types/tables.ts @@ -4,6 +4,12 @@ export type Column = { name: string; dataType: string; columnType: string; + typeParams: { + precision?: number; + scale?: number; + length?: number; + dimensions?: number; + }; size?: number; default?: any; hasDefault: boolean; diff --git a/drizzle-seed/tests/benchmarks/generatorsBenchmark.ts b/drizzle-seed/tests/benchmarks/generatorsBenchmark.ts index 9d79ebaa8..23fca0c6c 100644 --- a/drizzle-seed/tests/benchmarks/generatorsBenchmark.ts +++ b/drizzle-seed/tests/benchmarks/generatorsBenchmark.ts @@ -23,7 +23,7 @@ import { GeneratePoint, GeneratePostcode, GenerateState, - GenerateStreetAdddress, + GenerateStreetAddress, GenerateString, GenerateTime, GenerateTimestamp, @@ -35,12 +35,12 @@ import { GenerateUniqueNumber, GenerateUniquePoint, GenerateUniquePostcode, - GenerateUniqueStreetAdddress, + GenerateUniqueStreetAddress, GenerateUniqueString, GenerateValuesFromArray, GenerateYear, WeightedRandomGenerator, -} from '../../src/services/GeneratorsWrappers.ts'; +} from '../../src/services/Generators.ts'; const benchmark = ({ generatorName, generator, count = 100000, seed = 1 }: { generatorName: string; @@ -107,8 +107,8 @@ const generatorsFuncs = { // uniqueCountry: new GenerateUniqueCountry({}), city: new GenerateCity({}), // uniqueCity: new GenerateUniqueCity({}), - streetAddress: new GenerateStreetAdddress({}), - uniqueStreetAddress: new GenerateUniqueStreetAdddress({}), + streetAddress: new GenerateStreetAddress({}), + uniqueStreetAddress: new GenerateUniqueStreetAddress({}), jobTitle: new GenerateJobTitle({}), postcode: new GeneratePostcode({}), uniquePostcode: new GenerateUniquePostcode({}), diff --git a/drizzle-seed/tests/mysql/allDataTypesTest/mysql_all_data_types.test.ts b/drizzle-seed/tests/mysql/allDataTypesTest/mysql_all_data_types.test.ts index 5a9ba9908..f39a55fef 100644 --- a/drizzle-seed/tests/mysql/allDataTypesTest/mysql_all_data_types.test.ts +++ b/drizzle-seed/tests/mysql/allDataTypesTest/mysql_all_data_types.test.ts @@ -11,7 +11,7 @@ import { seed } from '../../../src/index.ts'; import * as schema from './mysqlSchema.ts'; let mysqlContainer: Docker.Container; -let client: Connection; +let client: Connection | undefined; let db: MySql2Database; async function createDockerDB(): Promise { diff --git a/drizzle-seed/tests/pg/allDataTypesTest/pgSchema.ts b/drizzle-seed/tests/pg/allDataTypesTest/pgSchema.ts index 75ba20a43..d4f45de22 100644 --- a/drizzle-seed/tests/pg/allDataTypesTest/pgSchema.ts +++ b/drizzle-seed/tests/pg/allDataTypesTest/pgSchema.ts @@ -64,31 +64,31 @@ export const allDataTypes = schema.table('all_data_types', { }); export const allArrayDataTypes = schema.table('all_array_data_types', { - integerArray: integer('integer_array').array(), - smallintArray: smallint('smallint_array').array(), - bigintegerArray: bigint('bigint_array', { mode: 'bigint' }).array(), - bigintNumberArray: bigint('bigint_number_array', { mode: 'number' }).array(), - booleanArray: boolean('boolean_array').array(), - textArray: text('text_array').array(), - varcharArray: varchar('varchar_array', { length: 256 }).array(), - charArray: char('char_array', { length: 256 }).array(), - numericArray: numeric('numeric_array').array(), - decimalArray: decimal('decimal_array').array(), - realArray: real('real_array').array(), - doublePrecisionArray: doublePrecision('double_precision_array').array(), - jsonArray: json('json_array').array(), - jsonbArray: jsonb('jsonb_array').array(), - timeArray: time('time_array').array(), - timestampDateArray: timestamp('timestamp_date_array', { mode: 'date' }).array(), - timestampStringArray: timestamp('timestamp_string_array', { mode: 'string' }).array(), + // integerArray: integer('integer_array').array(), + // smallintArray: smallint('smallint_array').array(), + // bigintegerArray: bigint('bigint_array', { mode: 'bigint' }).array(), + // bigintNumberArray: bigint('bigint_number_array', { mode: 'number' }).array(), + // booleanArray: boolean('boolean_array').array(), + // textArray: text('text_array').array(), + // varcharArray: varchar('varchar_array', { length: 256 }).array(), + // charArray: char('char_array', { length: 256 }).array(), + // numericArray: numeric('numeric_array').array(), + // decimalArray: decimal('decimal_array').array(), + // realArray: real('real_array').array(), + // doublePrecisionArray: doublePrecision('double_precision_array').array(), + // jsonArray: json('json_array').array(), + // jsonbArray: jsonb('jsonb_array').array(), + // timeArray: time('time_array').array(), + // timestampDateArray: timestamp('timestamp_date_array', { mode: 'date' }).array(), + // timestampStringArray: timestamp('timestamp_string_array', { mode: 'string' }).array(), dateStringArray: date('date_string_array', { mode: 'string' }).array(), dateArray: date('date_array', { mode: 'date' }).array(), - intervalArray: interval('interval_array').array(), - pointArray: point('point_array', { mode: 'xy' }).array(), - pointTupleArray: point('point_tuple_array', { mode: 'tuple' }).array(), - lineArray: line('line_array', { mode: 'abc' }).array(), - lineTupleArray: line('line_tuple_array', { mode: 'tuple' }).array(), - moodEnumArray: moodEnum('mood_enum_array').array(), + // intervalArray: interval('interval_array').array(), + // pointArray: point('point_array', { mode: 'xy' }).array(), + // pointTupleArray: point('point_tuple_array', { mode: 'tuple' }).array(), + // lineArray: line('line_array', { mode: 'abc' }).array(), + // lineTupleArray: line('line_tuple_array', { mode: 'tuple' }).array(), + // moodEnumArray: moodEnum('mood_enum_array').array(), }); export const ndArrays = schema.table('nd_arrays', { @@ -97,3 +97,19 @@ export const ndArrays = schema.table('nd_arrays', { integer3DArray: integer('integer_3d_array').array(3).array(4).array(5), integer4DArray: integer('integer_4d_array').array(3).array(4).array(5).array(6), }); + +export const intervals = schema.table('intervals', { + intervalYear: interval({ fields: 'year' }), + intervalYearToMonth: interval({ fields: 'year to month' }), + intervalMonth: interval({ fields: 'month' }), + intervalDay: interval({ fields: 'day' }), + intervalDayToHour: interval({ fields: 'day to hour' }), + intervalDayToMinute: interval({ fields: 'day to minute' }), + intervalDayToSecond: interval({ fields: 'day to second' }), + intervalHour: interval({ fields: 'hour' }), + intervalHourToMinute: interval({ fields: 'hour to minute' }), + intervalHourToSecond: interval({ fields: 'hour to second' }), + intervalMinute: interval({ fields: 'minute' }), + intervalMinuteToSecond: interval({ fields: 'minute to second' }), + intervalSecond: interval({ fields: 'second' }), +}); diff --git a/drizzle-seed/tests/pg/allDataTypesTest/pg_all_data_types.test.ts b/drizzle-seed/tests/pg/allDataTypesTest/pg_all_data_types.test.ts index 7dfbc089b..62d0895c0 100644 --- a/drizzle-seed/tests/pg/allDataTypesTest/pg_all_data_types.test.ts +++ b/drizzle-seed/tests/pg/allDataTypesTest/pg_all_data_types.test.ts @@ -105,6 +105,26 @@ beforeAll(async () => { ); `, ); + + await db.execute( + sql` + CREATE TABLE IF NOT EXISTS "seeder_lib_pg"."intervals" ( + "intervalYear" interval year, + "intervalYearToMonth" interval year to month, + "intervalMonth" interval month, + "intervalDay" interval day, + "intervalDayToHour" interval day to hour, + "intervalDayToMinute" interval day to minute, + "intervalDayToSecond" interval day to second, + "intervalHour" interval hour, + "intervalHourToMinute" interval hour to minute, + "intervalHourToSecond" interval hour to second, + "intervalMinute" interval minute, + "intervalMinuteToSecond" interval minute to second, + "intervalSecond" interval second + ); + `, + ); }); afterAll(async () => { @@ -157,3 +177,13 @@ test('nd arrays', async () => { expect(predicate0 && predicate1 && predicate2 && predicate3 && predicate4).toBe(true); }); + +test('intervals test', async () => { + await seed(db, { intervals: schema.intervals }, { count: 1000 }); + + const intervals = await db.select().from(schema.intervals); + // every value in each rows does not equal undefined. + const predicate = intervals.every((row) => Object.values(row).every((val) => val !== undefined && val !== null)); + + expect(predicate).toBe(true); +}); From 31c6a0ffb850d4e5ea9d990377476ed9deb67043 Mon Sep 17 00:00:00 2001 From: AndriiSherman Date: Wed, 25 Dec 2024 09:49:50 +0200 Subject: [PATCH 26/32] Add release notes and bump versions --- changelogs/drizzle-orm/0.38.3.md | 1 + changelogs/drizzle-typebox/0.2.1.md | 25 ++++++++++++ changelogs/drizzle-valibot/0.3.1.md | 23 +++++++++++ changelogs/drizzle-zod/0.6.1.md | 26 ++++++++++++ drizzle-orm/package.json | 4 +- drizzle-orm/src/mysql-core/table.ts | 59 ++++++++++++++-------------- drizzle-orm/src/pg-core/table.ts | 59 ++++++++++++++-------------- drizzle-orm/src/sqlite-core/table.ts | 59 ++++++++++++++-------------- drizzle-typebox/package.json | 4 +- drizzle-valibot/package.json | 4 +- drizzle-zod/package.json | 4 +- 11 files changed, 170 insertions(+), 98 deletions(-) create mode 100644 changelogs/drizzle-orm/0.38.3.md create mode 100644 changelogs/drizzle-typebox/0.2.1.md create mode 100644 changelogs/drizzle-valibot/0.3.1.md create mode 100644 changelogs/drizzle-zod/0.6.1.md diff --git a/changelogs/drizzle-orm/0.38.3.md b/changelogs/drizzle-orm/0.38.3.md new file mode 100644 index 000000000..3f94386b6 --- /dev/null +++ b/changelogs/drizzle-orm/0.38.3.md @@ -0,0 +1 @@ +- Fix incorrect deprecation detection for table declarations \ No newline at end of file diff --git a/changelogs/drizzle-typebox/0.2.1.md b/changelogs/drizzle-typebox/0.2.1.md new file mode 100644 index 000000000..2d9e762c3 --- /dev/null +++ b/changelogs/drizzle-typebox/0.2.1.md @@ -0,0 +1,25 @@ +# Added support for SingleStore dialect + +```ts +import { singlestoreTable, text, int } from 'drizzle-orm/singlestore-core'; +import { createSelectSchema } from 'drizzle-typebox'; +import { Value } from '@sinclair/typebox/value'; + +const users = singlestoreTable('users', { + id: int().primaryKey(), + name: text().notNull(), + age: int().notNull() +}); + +const userSelectSchema = createSelectSchema(users); + +const rows = await db.select({ id: users.id, name: users.name }).from(users).limit(1); +const parsed: { id: number; name: string; age: number } = Value.Parse(userSelectSchema, rows[0]); // Error: `age` is not returned in the above query + +const rows = await db.select().from(users).limit(1); +const parsed: { id: number; name: string; age: number } = Value.Parse(userSelectSchema, rows[0]); // Will parse successfully +``` + +# Bug fixes + +- [[BUG]: drizzle-typebox infers integer() as TString](https://github.com/drizzle-team/drizzle-orm/issues/3756) \ No newline at end of file diff --git a/changelogs/drizzle-valibot/0.3.1.md b/changelogs/drizzle-valibot/0.3.1.md new file mode 100644 index 000000000..21f05270b --- /dev/null +++ b/changelogs/drizzle-valibot/0.3.1.md @@ -0,0 +1,23 @@ +# Added support for SingleStore dialect + +```ts +import { singlestoreTable, text, int } from 'drizzle-orm/singlestore-core'; +import { createSelectSchema } from 'drizzle-valibot'; +import { parse } from 'valibot'; + +const users = singlestoreTable('users', { + id: int().primaryKey(), + name: text().notNull(), + age: int().notNull() +}); + +const userSelectSchema = createSelectSchema(users); +const rows = await db.select({ id: users.id, name: users.name }).from(users).limit(1); +const parsed: { id: number; name: string; age: number } = parse(userSelectSchema, rows[0]); // Error: `age` is not returned in the above query +const rows = await db.select().from(users).limit(1); +const parsed: { id: number; name: string; age: number } = parse(userSelectSchema, rows[0]); // Will parse successfully +``` + +# Bug fixes + +- [[BUG]: drizzle-valibot throws Type instantiation is excessively deep and possibly infinite. for refinements](https://github.com/drizzle-team/drizzle-orm/issues/3751) diff --git a/changelogs/drizzle-zod/0.6.1.md b/changelogs/drizzle-zod/0.6.1.md new file mode 100644 index 000000000..dd6740660 --- /dev/null +++ b/changelogs/drizzle-zod/0.6.1.md @@ -0,0 +1,26 @@ +# New Features + +## Added support for SingleStore dialect + +```ts +import { singlestoreTable, text, int } from 'drizzle-orm/singlestore-core'; +import { createSelectSchema } from 'drizzle-zod'; + +const users = singlestoreTable('users', { + id: int().primaryKey(), + name: text().notNull(), + age: int().notNull() +}); + +const userSelectSchema = createSelectSchema(users); +const rows = await db.select({ id: users.id, name: users.name }).from(users).limit(1); +const parsed: { id: number; name: string; age: number } = userSelectSchema.parse(rows[0]); // Error: `age` is not returned in the above query + +const rows = await db.select().from(users).limit(1); +const parsed: { id: number; name: string; age: number } = userSelectSchema.parse(rows[0]); // Will parse successfully +``` + +# Bug fixes + +- [[BUG]: refining schema using createSelectSchema is not working with drizzle-kit 0.6.0](https://github.com/drizzle-team/drizzle-orm/issues/3735) +- [[BUG]: drizzle-zod inferring types incorrectly](https://github.com/drizzle-team/drizzle-orm/issues/3734) \ No newline at end of file diff --git a/drizzle-orm/package.json b/drizzle-orm/package.json index 72e641467..4385e34fd 100644 --- a/drizzle-orm/package.json +++ b/drizzle-orm/package.json @@ -1,6 +1,6 @@ { "name": "drizzle-orm", - "version": "0.38.2", + "version": "0.38.3", "description": "Drizzle ORM package for SQL databases", "type": "module", "scripts": { @@ -204,4 +204,4 @@ "zod": "^3.20.2", "zx": "^7.2.2" } -} +} \ No newline at end of file diff --git a/drizzle-orm/src/mysql-core/table.ts b/drizzle-orm/src/mysql-core/table.ts index c3d3e581a..2616e7159 100644 --- a/drizzle-orm/src/mysql-core/table.ts +++ b/drizzle-orm/src/mysql-core/table.ts @@ -116,6 +116,35 @@ export function mysqlTableWithSchema< } export interface MySqlTableFn { + < + TTableName extends string, + TColumnsMap extends Record, + >( + name: TTableName, + columns: TColumnsMap, + extraConfig?: ( + self: BuildColumns, + ) => MySqlTableExtraConfigValue[], + ): MySqlTableWithColumns<{ + name: TTableName; + schema: TSchemaName; + columns: BuildColumns; + dialect: 'mysql'; + }>; + + < + TTableName extends string, + TColumnsMap extends Record, + >( + name: TTableName, + columns: (columnTypes: MySqlColumnBuilders) => TColumnsMap, + extraConfig?: (self: BuildColumns) => MySqlTableExtraConfigValue[], + ): MySqlTableWithColumns<{ + name: TTableName; + schema: TSchemaName; + columns: BuildColumns; + dialect: 'mysql'; + }>; /** * @deprecated The third parameter of mysqlTable is changing and will only accept an array instead of an object * @@ -187,36 +216,6 @@ export interface MySqlTableFn; dialect: 'mysql'; }>; - - < - TTableName extends string, - TColumnsMap extends Record, - >( - name: TTableName, - columns: TColumnsMap, - extraConfig?: ( - self: BuildColumns, - ) => MySqlTableExtraConfigValue[], - ): MySqlTableWithColumns<{ - name: TTableName; - schema: TSchemaName; - columns: BuildColumns; - dialect: 'mysql'; - }>; - - < - TTableName extends string, - TColumnsMap extends Record, - >( - name: TTableName, - columns: (columnTypes: MySqlColumnBuilders) => TColumnsMap, - extraConfig?: (self: BuildColumns) => MySqlTableExtraConfigValue[], - ): MySqlTableWithColumns<{ - name: TTableName; - schema: TSchemaName; - columns: BuildColumns; - dialect: 'mysql'; - }>; } export const mysqlTable: MySqlTableFn = (name, columns, extraConfig) => { diff --git a/drizzle-orm/src/pg-core/table.ts b/drizzle-orm/src/pg-core/table.ts index 7f12e634d..b5a60e91a 100644 --- a/drizzle-orm/src/pg-core/table.ts +++ b/drizzle-orm/src/pg-core/table.ts @@ -134,6 +134,35 @@ export function pgTableWithSchema< } export interface PgTableFn { + < + TTableName extends string, + TColumnsMap extends Record, + >( + name: TTableName, + columns: TColumnsMap, + extraConfig?: ( + self: BuildExtraConfigColumns, + ) => PgTableExtraConfigValue[], + ): PgTableWithColumns<{ + name: TTableName; + schema: TSchema; + columns: BuildColumns; + dialect: 'pg'; + }>; + + < + TTableName extends string, + TColumnsMap extends Record, + >( + name: TTableName, + columns: (columnTypes: PgColumnsBuilders) => TColumnsMap, + extraConfig?: (self: BuildExtraConfigColumns) => PgTableExtraConfigValue[], + ): PgTableWithColumns<{ + name: TTableName; + schema: TSchema; + columns: BuildColumns; + dialect: 'pg'; + }>; /** * @deprecated The third parameter of pgTable is changing and will only accept an array instead of an object * @@ -207,36 +236,6 @@ export interface PgTableFn { columns: BuildColumns; dialect: 'pg'; }>; - - < - TTableName extends string, - TColumnsMap extends Record, - >( - name: TTableName, - columns: TColumnsMap, - extraConfig?: ( - self: BuildExtraConfigColumns, - ) => PgTableExtraConfigValue[], - ): PgTableWithColumns<{ - name: TTableName; - schema: TSchema; - columns: BuildColumns; - dialect: 'pg'; - }>; - - < - TTableName extends string, - TColumnsMap extends Record, - >( - name: TTableName, - columns: (columnTypes: PgColumnsBuilders) => TColumnsMap, - extraConfig?: (self: BuildExtraConfigColumns) => PgTableExtraConfigValue[], - ): PgTableWithColumns<{ - name: TTableName; - schema: TSchema; - columns: BuildColumns; - dialect: 'pg'; - }>; } export const pgTable: PgTableFn = (name, columns, extraConfig) => { diff --git a/drizzle-orm/src/sqlite-core/table.ts b/drizzle-orm/src/sqlite-core/table.ts index 5a68f9cb1..290605b66 100644 --- a/drizzle-orm/src/sqlite-core/table.ts +++ b/drizzle-orm/src/sqlite-core/table.ts @@ -57,6 +57,35 @@ export type SQLiteTableWithColumns = }; export interface SQLiteTableFn { + < + TTableName extends string, + TColumnsMap extends Record, + >( + name: TTableName, + columns: TColumnsMap, + extraConfig?: ( + self: BuildColumns, + ) => SQLiteTableExtraConfigValue[], + ): SQLiteTableWithColumns<{ + name: TTableName; + schema: TSchema; + columns: BuildColumns; + dialect: 'sqlite'; + }>; + + < + TTableName extends string, + TColumnsMap extends Record, + >( + name: TTableName, + columns: (columnTypes: SQLiteColumnBuilders) => TColumnsMap, + extraConfig?: (self: BuildColumns) => SQLiteTableExtraConfigValue[], + ): SQLiteTableWithColumns<{ + name: TTableName; + schema: TSchema; + columns: BuildColumns; + dialect: 'sqlite'; + }>; /** * @deprecated The third parameter of sqliteTable is changing and will only accept an array instead of an object * @@ -128,36 +157,6 @@ export interface SQLiteTableFn { columns: BuildColumns; dialect: 'sqlite'; }>; - - < - TTableName extends string, - TColumnsMap extends Record, - >( - name: TTableName, - columns: TColumnsMap, - extraConfig?: ( - self: BuildColumns, - ) => SQLiteTableExtraConfigValue[], - ): SQLiteTableWithColumns<{ - name: TTableName; - schema: TSchema; - columns: BuildColumns; - dialect: 'sqlite'; - }>; - - < - TTableName extends string, - TColumnsMap extends Record, - >( - name: TTableName, - columns: (columnTypes: SQLiteColumnBuilders) => TColumnsMap, - extraConfig?: (self: BuildColumns) => SQLiteTableExtraConfigValue[], - ): SQLiteTableWithColumns<{ - name: TTableName; - schema: TSchema; - columns: BuildColumns; - dialect: 'sqlite'; - }>; } function sqliteTableBase< diff --git a/drizzle-typebox/package.json b/drizzle-typebox/package.json index a6c34fc69..6ba6f2235 100644 --- a/drizzle-typebox/package.json +++ b/drizzle-typebox/package.json @@ -1,6 +1,6 @@ { "name": "drizzle-typebox", - "version": "0.2.0", + "version": "0.2.1", "description": "Generate Typebox schemas from Drizzle ORM schemas", "type": "module", "scripts": { @@ -70,4 +70,4 @@ "vitest": "^1.6.0", "zx": "^7.2.2" } -} +} \ No newline at end of file diff --git a/drizzle-valibot/package.json b/drizzle-valibot/package.json index c9e6a02e9..d68e55531 100644 --- a/drizzle-valibot/package.json +++ b/drizzle-valibot/package.json @@ -1,6 +1,6 @@ { "name": "drizzle-valibot", - "version": "0.3.0", + "version": "0.3.1", "description": "Generate valibot schemas from Drizzle ORM schemas", "type": "module", "scripts": { @@ -70,4 +70,4 @@ "vitest": "^1.6.0", "zx": "^7.2.2" } -} +} \ No newline at end of file diff --git a/drizzle-zod/package.json b/drizzle-zod/package.json index ebbec398e..3d1df1919 100644 --- a/drizzle-zod/package.json +++ b/drizzle-zod/package.json @@ -1,6 +1,6 @@ { "name": "drizzle-zod", - "version": "0.6.0", + "version": "0.6.1", "description": "Generate Zod schemas from Drizzle ORM schemas", "type": "module", "scripts": { @@ -79,4 +79,4 @@ "zod": "^3.20.2", "zx": "^7.2.2" } -} +} \ No newline at end of file From be0f8337a831b3d0d17177e6b770734a4d7c2b62 Mon Sep 17 00:00:00 2001 From: AndriiSherman Date: Wed, 25 Dec 2024 09:56:37 +0200 Subject: [PATCH 27/32] dprint --- drizzle-orm/package.json | 2 +- drizzle-seed/package.json | 2 +- drizzle-typebox/package.json | 2 +- drizzle-valibot/package.json | 2 +- drizzle-zod/package.json | 2 +- 5 files changed, 5 insertions(+), 5 deletions(-) diff --git a/drizzle-orm/package.json b/drizzle-orm/package.json index 4385e34fd..5ab4de9b2 100644 --- a/drizzle-orm/package.json +++ b/drizzle-orm/package.json @@ -204,4 +204,4 @@ "zod": "^3.20.2", "zx": "^7.2.2" } -} \ No newline at end of file +} diff --git a/drizzle-seed/package.json b/drizzle-seed/package.json index ae5648cdb..cbc3b570c 100644 --- a/drizzle-seed/package.json +++ b/drizzle-seed/package.json @@ -102,4 +102,4 @@ "dependencies": { "pure-rand": "^6.1.0" } -} \ No newline at end of file +} diff --git a/drizzle-typebox/package.json b/drizzle-typebox/package.json index 6ba6f2235..c03d64105 100644 --- a/drizzle-typebox/package.json +++ b/drizzle-typebox/package.json @@ -70,4 +70,4 @@ "vitest": "^1.6.0", "zx": "^7.2.2" } -} \ No newline at end of file +} diff --git a/drizzle-valibot/package.json b/drizzle-valibot/package.json index d68e55531..621d36782 100644 --- a/drizzle-valibot/package.json +++ b/drizzle-valibot/package.json @@ -70,4 +70,4 @@ "vitest": "^1.6.0", "zx": "^7.2.2" } -} \ No newline at end of file +} diff --git a/drizzle-zod/package.json b/drizzle-zod/package.json index 3d1df1919..cb1e472fa 100644 --- a/drizzle-zod/package.json +++ b/drizzle-zod/package.json @@ -79,4 +79,4 @@ "zod": "^3.20.2", "zx": "^7.2.2" } -} \ No newline at end of file +} From fedc800b2c66504562c2bd8247185929a8434636 Mon Sep 17 00:00:00 2001 From: OleksiiKH0240 Date: Wed, 25 Dec 2024 12:44:54 +0200 Subject: [PATCH 28/32] fixes --- .gitignore | 2 ++ drizzle-seed/src/index.ts | 31 ++++++++++++++++++++-- drizzle-seed/src/services/SeedService.ts | 4 ++- drizzle-seed/src/services/versioning/v2.ts | 4 +-- 4 files changed, 36 insertions(+), 5 deletions(-) diff --git a/.gitignore b/.gitignore index 570a706f8..c266f115f 100644 --- a/.gitignore +++ b/.gitignore @@ -13,4 +13,6 @@ rollup.config-*.mjs *.log .DS_Store drizzle-seed/src/test.ts +drizzle-seed/src/testMysql.ts +drizzle-seed/src/testSqlite.ts drizzle-seed/src/schemaTest.ts \ No newline at end of file diff --git a/drizzle-seed/src/index.ts b/drizzle-seed/src/index.ts index 98c449095..c73e497cb 100644 --- a/drizzle-seed/src/index.ts +++ b/drizzle-seed/src/index.ts @@ -927,7 +927,8 @@ const getMySqlInfo = (schema: { [key: string]: MySqlTable }) => { typeParams['scale'] = Number(match[2]); } } else if ( - sqlType.startsWith('varchar') + sqlType.startsWith('char') + || sqlType.startsWith('varchar') || sqlType.startsWith('binary') || sqlType.startsWith('varbinary') ) { @@ -1129,12 +1130,38 @@ const getSqliteInfo = (schema: { [key: string]: SQLiteTable }) => { } tableRelations[dbToTsTableNamesMap[tableConfig.name] as string]!.push(...newRelations); + const getTypeParams = (sqlType: string) => { + // get type params and set only type + const typeParams: Column['typeParams'] = {}; + + if ( + sqlType.startsWith('decimal') + ) { + const match = sqlType.match(/\((\d+), *(\d+)\)/); + if (match) { + typeParams['precision'] = Number(match[1]); + typeParams['scale'] = Number(match[2]); + } + } else if ( + sqlType.startsWith('char') + || sqlType.startsWith('varchar') + || sqlType.startsWith('text') + ) { + const match = sqlType.match(/\((\d+)\)/); + if (match) { + typeParams['length'] = Number(match[1]); + } + } + + return typeParams; + }; + tables.push({ name: dbToTsTableNamesMap[tableConfig.name] as string, columns: tableConfig.columns.map((column) => ({ name: dbToTsColumnNamesMap[column.name] as string, columnType: column.getSQLType(), - typeParams: {}, + typeParams: getTypeParams(column.getSQLType()), dataType: column.dataType, hasDefault: column.hasDefault, default: column.default, diff --git a/drizzle-seed/src/services/SeedService.ts b/drizzle-seed/src/services/SeedService.ts index 84165169e..1370138b4 100644 --- a/drizzle-seed/src/services/SeedService.ts +++ b/drizzle-seed/src/services/SeedService.ts @@ -260,6 +260,7 @@ export class SeedService { columnPossibleGenerator.generator = arrayGen; } + columnPossibleGenerator.generator.isUnique = col.isUnique; const uniqueGen = columnPossibleGenerator.generator.replaceIfUnique(); if (uniqueGen !== undefined) { columnPossibleGenerator.generator = uniqueGen; @@ -268,7 +269,6 @@ export class SeedService { // selecting version of generator columnPossibleGenerator.generator = this.selectVersionOfGenerator(columnPossibleGenerator.generator); - columnPossibleGenerator.generator.isUnique = col.isUnique; // TODO: for now only GenerateValuesFromArray support notNull property columnPossibleGenerator.generator.notNull = col.notNull; columnPossibleGenerator.generator.dataType = col.dataType; @@ -309,6 +309,8 @@ export class SeedService { const newGenerator = new generatorConstructor(generator.params); newGenerator.baseColumnDataType = generator.baseColumnDataType; newGenerator.isUnique = generator.isUnique; + // TODO: for now only GenerateValuesFromArray support notNull property + newGenerator.notNull = generator.notNull; newGenerator.dataType = generator.dataType; newGenerator.stringLength = generator.stringLength; diff --git a/drizzle-seed/src/services/versioning/v2.ts b/drizzle-seed/src/services/versioning/v2.ts index e09cc45f6..f4dbf32f4 100644 --- a/drizzle-seed/src/services/versioning/v2.ts +++ b/drizzle-seed/src/services/versioning/v2.ts @@ -126,7 +126,7 @@ export class GenerateStringV2 extends AbstractGenerator<{ override init({ count, seed }: { count: number; seed: number }) { super.init({ count, seed }); - let minStringLength = 8; + let minStringLength = 7; let maxStringLength = 20; if (this.stringLength !== undefined) { maxStringLength = this.stringLength; @@ -182,7 +182,7 @@ export class GenerateUniqueStringV2 extends AbstractGenerator<{ isUnique?: boole override init({ seed, count }: { seed: number; count: number }) { const rng = prand.xoroshiro128plus(seed); - let minStringLength = 8; + let minStringLength = 7; let maxStringLength = 20; // TODO: revise later if (this.stringLength !== undefined) { From fa9f4b7acaeacc6f2eaa9fb19a5bd1c007bfe283 Mon Sep 17 00:00:00 2001 From: OleksiiKH0240 Date: Wed, 25 Dec 2024 12:53:05 +0200 Subject: [PATCH 29/32] updated changelogs --- changelogs/drizzle-seed/0.2.0.md | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/changelogs/drizzle-seed/0.2.0.md b/changelogs/drizzle-seed/0.2.0.md index 5bb9d6007..bd2a5260e 100644 --- a/changelogs/drizzle-seed/0.2.0.md +++ b/changelogs/drizzle-seed/0.2.0.md @@ -123,4 +123,5 @@ However, after inserting the `1 minute 60 second` interval, PostgreSQL database Now (in version 2), the maximum length of a string depends on the length of the text column (e.g., `varchar(20)`). ## Bug fixes -- seeding table with foreign key referencing another table, without including the second table in schema, will cause seed process to get stuck. \ No newline at end of file +- seeding table with foreign key referencing another table, without including the second table in schema, will cause seed process to get stuck. +- not setting unique generators for unique database columns. \ No newline at end of file From 6854ac6c5fda8d34f2ea9980fed24ceb6a3b04d4 Mon Sep 17 00:00:00 2001 From: OleksiiKH0240 Date: Wed, 25 Dec 2024 13:17:33 +0200 Subject: [PATCH 30/32] fix --- changelogs/drizzle-seed/0.2.0.md | 1 - 1 file changed, 1 deletion(-) diff --git a/changelogs/drizzle-seed/0.2.0.md b/changelogs/drizzle-seed/0.2.0.md index 6c8573ee4..0978c7a98 100644 --- a/changelogs/drizzle-seed/0.2.0.md +++ b/changelogs/drizzle-seed/0.2.0.md @@ -159,7 +159,6 @@ export const table = p.sqliteTable('table', { await seed(db, { table }) ``` -Functions from refine that will be affected by this change: `` ## Bug fixes - Seeding a table with a foreign key referencing another table, without including the second table in the schema, will cause the seeding process to get stuck From ca569cb0a86b6059067f321ec51f0f9132a153b2 Mon Sep 17 00:00:00 2001 From: AndriiSherman Date: Wed, 25 Dec 2024 13:35:11 +0200 Subject: [PATCH 31/32] Bump version --- changelogs/drizzle-seed/{0.2.0.md => 0.2.1.md} | 0 drizzle-seed/package.json | 4 ++-- 2 files changed, 2 insertions(+), 2 deletions(-) rename changelogs/drizzle-seed/{0.2.0.md => 0.2.1.md} (100%) diff --git a/changelogs/drizzle-seed/0.2.0.md b/changelogs/drizzle-seed/0.2.1.md similarity index 100% rename from changelogs/drizzle-seed/0.2.0.md rename to changelogs/drizzle-seed/0.2.1.md diff --git a/drizzle-seed/package.json b/drizzle-seed/package.json index cbc3b570c..b0606f04e 100644 --- a/drizzle-seed/package.json +++ b/drizzle-seed/package.json @@ -1,6 +1,6 @@ { "name": "drizzle-seed", - "version": "0.2.0", + "version": "0.2.1", "main": "index.js", "type": "module", "scripts": { @@ -102,4 +102,4 @@ "dependencies": { "pure-rand": "^6.1.0" } -} +} \ No newline at end of file From 06be106f3d111339466eb596d000d7580c89b01f Mon Sep 17 00:00:00 2001 From: AndriiSherman Date: Wed, 25 Dec 2024 13:38:37 +0200 Subject: [PATCH 32/32] dprint --- drizzle-seed/package.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/drizzle-seed/package.json b/drizzle-seed/package.json index b0606f04e..29ceae587 100644 --- a/drizzle-seed/package.json +++ b/drizzle-seed/package.json @@ -102,4 +102,4 @@ "dependencies": { "pure-rand": "^6.1.0" } -} \ No newline at end of file +}