@conorroberts/utils 0.0.98 → 0.0.101
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
|
@@ -36,9 +36,13 @@ const getLatestSnapshot = (metaDir) => {
|
|
|
36
36
|
return JSON.parse(readFileSync(snapshotPath, "utf-8"));
|
|
37
37
|
};
|
|
38
38
|
const validateDatabaseSchema = (options) => {
|
|
39
|
-
const { metaDir, verbose = false } = options;
|
|
39
|
+
const { metaDir, verbose = false, exitOnComplete = false } = options;
|
|
40
40
|
const issues = [];
|
|
41
41
|
const allConstraints = [];
|
|
42
|
+
const finalize = (exitCode) => {
|
|
43
|
+
if (exitOnComplete) process.exit(exitCode);
|
|
44
|
+
return exitCode;
|
|
45
|
+
};
|
|
42
46
|
const snapshot = getLatestSnapshot(metaDir);
|
|
43
47
|
const tables = Object.values(snapshot.tables);
|
|
44
48
|
tables.forEach((table) => {
|
|
@@ -157,7 +161,7 @@ const validateDatabaseSchema = (options) => {
|
|
|
157
161
|
consola.log("");
|
|
158
162
|
if (issues.length === 0) {
|
|
159
163
|
consola.success("No name length issues found!");
|
|
160
|
-
return 0;
|
|
164
|
+
return finalize(0);
|
|
161
165
|
}
|
|
162
166
|
consola.warn(`Found ${issues.length} name length issue(s):`);
|
|
163
167
|
consola.log("");
|
|
@@ -242,7 +246,7 @@ const validateDatabaseSchema = (options) => {
|
|
|
242
246
|
consola.log("4. Run 'pnpm generate' again after making changes");
|
|
243
247
|
consola.log("5. Run this check with --verbose to see all constraint names");
|
|
244
248
|
consola.log("");
|
|
245
|
-
return 1;
|
|
249
|
+
return finalize(1);
|
|
246
250
|
};
|
|
247
251
|
|
|
248
252
|
//#endregion
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"validate-schema.mjs","names":["entries: JournalEntry[]","issues: NameLengthIssue[]","allConstraints: Array<{ table: string; type: string; name: string; length: number }>","parts: string[]"],"sources":["../../src/db/validateSchema.ts"],"sourcesContent":["/**\r\n * Database Schema Validator\r\n *\r\n * This script validates the database schema by analyzing all MySQL table definitions\r\n * from Drizzle migration metadata and reports any table, column, or constraint names\r\n * that exceed MySQL's maximum length limits.\r\n *\r\n * MySQL Limits:\r\n * - Table/Column names: 64 characters\r\n * - Constraint names: 64 characters\r\n *\r\n * Usage:\r\n * - Basic: pnpm -F scripts validate:db-schema\r\n * - Verbose: pnpm -F scripts validate:db-schema -- --verbose\r\n *\r\n * The verbose mode shows all constraint names being checked, sorted by length.\r\n *\r\n * This validation runs automatically:\r\n * - After `pnpm generate` (blocks if violations found)\r\n * - In CI/CD as part of PR checks\r\n */\r\n\r\nimport { consola } from \"consola\";\r\nimport { readFileSync } from \"node:fs\";\r\nimport { join, resolve } from \"node:path\";\r\n\r\ninterface NameLengthIssue {\r\n type: \"table\" | \"column\" | \"constraint\";\r\n tableName: string;\r\n name: string;\r\n actualLength: number;\r\n maxLength: number;\r\n constraintType?: \"foreign_key\" | \"primary_key\" | \"unique\" | \"index\";\r\n}\r\n\r\ninterface DrizzleSnapshot {\r\n tables: Record<string, DrizzleTable>;\r\n}\r\n\r\ninterface DrizzleTable {\r\n name: string;\r\n columns: Record<string, DrizzleColumn>;\r\n indexes: Record<string, DrizzleIndex>;\r\n foreignKeys: Record<string, DrizzleForeignKey>;\r\n compositePrimaryKeys: Record<string, DrizzlePrimaryKey>;\r\n uniqueConstraints: Record<string, DrizzleUniqueConstraint>;\r\n}\r\n\r\ninterface DrizzleColumn {\r\n name: string;\r\n type: string;\r\n primaryKey: boolean;\r\n notNull: boolean;\r\n autoincrement: boolean;\r\n}\r\n\r\ninterface DrizzleIndex {\r\n name: string;\r\n columns: string[];\r\n}\r\n\r\ninterface DrizzleForeignKey {\r\n name: string;\r\n tableFrom: string;\r\n tableTo: string;\r\n columnsFrom: string[];\r\n columnsTo: string[];\r\n}\r\n\r\ninterface DrizzlePrimaryKey {\r\n name: string;\r\n columns: string[];\r\n}\r\n\r\ninterface DrizzleUniqueConstraint {\r\n name: string;\r\n columns: string[];\r\n}\r\n\r\nconst MAX_IDENTIFIER_LENGTH = 64;\r\nconst MAX_CONSTRAINT_NAME_LENGTH = 64;\r\n\r\nconst getLatestSnapshot = (metaDir: string): DrizzleSnapshot => {\r\n const resolvedMetaDir = resolve(metaDir);\r\n\r\n // Read the journal to find the latest migration\r\n const journalPath = join(resolvedMetaDir, \"_journal.json\");\r\n const parsedJournal = JSON.parse(readFileSync(journalPath, \"utf-8\"));\r\n\r\n interface JournalEntry {\r\n tag: string;\r\n idx: number;\r\n }\r\n\r\n const entries: JournalEntry[] = parsedJournal.entries;\r\n const latestEntry = entries[entries.length - 1];\r\n\r\n if (!latestEntry) {\r\n throw new Error(\"No migrations found in journal\");\r\n }\r\n\r\n // Read the latest snapshot\r\n const snapshotPath = join(resolvedMetaDir, `${latestEntry.idx.toString().padStart(4, \"0\")}_snapshot.json`);\r\n const parsedSnapshot = JSON.parse(readFileSync(snapshotPath, \"utf-8\"));\r\n const snapshot: DrizzleSnapshot = parsedSnapshot;\r\n\r\n return snapshot;\r\n};\r\n\r\ninterface ValidateDatabaseSchemaOptions {\r\n metaDir: string;\r\n verbose?: boolean;\r\n}\r\n\r\nconst validateDatabaseSchema = (options: ValidateDatabaseSchemaOptions): number => {\r\n const { metaDir, verbose = false } = options;\r\n const issues: NameLengthIssue[] = [];\r\n const allConstraints: Array<{ table: string; type: string; name: string; length: number }> = [];\r\n\r\n const snapshot = getLatestSnapshot(metaDir);\r\n const tables = Object.values(snapshot.tables);\r\n\r\n tables.forEach((table) => {\r\n // Check table name length\r\n if (table.name.length > MAX_IDENTIFIER_LENGTH) {\r\n issues.push({\r\n type: \"table\",\r\n tableName: table.name,\r\n name: table.name,\r\n actualLength: table.name.length,\r\n maxLength: MAX_IDENTIFIER_LENGTH,\r\n });\r\n }\r\n\r\n // Check column name lengths\r\n Object.values(table.columns).forEach((column) => {\r\n if (column.name.length > MAX_IDENTIFIER_LENGTH) {\r\n issues.push({\r\n type: \"column\",\r\n tableName: table.name,\r\n name: column.name,\r\n actualLength: column.name.length,\r\n maxLength: MAX_IDENTIFIER_LENGTH,\r\n });\r\n }\r\n });\r\n\r\n // Check foreign key constraint names\r\n Object.values(table.foreignKeys).forEach((fk) => {\r\n allConstraints.push({ table: table.name, type: \"FK\", name: fk.name, length: fk.name.length });\r\n if (fk.name.length > MAX_CONSTRAINT_NAME_LENGTH) {\r\n issues.push({\r\n type: \"constraint\",\r\n tableName: table.name,\r\n name: fk.name,\r\n actualLength: fk.name.length,\r\n maxLength: MAX_CONSTRAINT_NAME_LENGTH,\r\n constraintType: \"foreign_key\",\r\n });\r\n }\r\n });\r\n\r\n // Check primary key constraint names\r\n Object.values(table.compositePrimaryKeys).forEach((pk) => {\r\n const pkType = pk.columns.length > 1 ? \"PK (composite)\" : \"PK\";\r\n allConstraints.push({ table: table.name, type: pkType, name: pk.name, length: pk.name.length });\r\n if (pk.name.length > MAX_CONSTRAINT_NAME_LENGTH) {\r\n issues.push({\r\n type: \"constraint\",\r\n tableName: table.name,\r\n name: pk.name,\r\n actualLength: pk.name.length,\r\n maxLength: MAX_CONSTRAINT_NAME_LENGTH,\r\n constraintType: \"primary_key\",\r\n });\r\n }\r\n });\r\n\r\n // Check unique constraint names\r\n Object.values(table.uniqueConstraints).forEach((unique) => {\r\n allConstraints.push({ table: table.name, type: \"UNIQUE\", name: unique.name, length: unique.name.length });\r\n if (unique.name.length > MAX_CONSTRAINT_NAME_LENGTH) {\r\n issues.push({\r\n type: \"constraint\",\r\n tableName: table.name,\r\n name: unique.name,\r\n actualLength: unique.name.length,\r\n maxLength: MAX_CONSTRAINT_NAME_LENGTH,\r\n constraintType: \"unique\",\r\n });\r\n }\r\n });\r\n\r\n // Check index names\r\n Object.values(table.indexes).forEach((index) => {\r\n allConstraints.push({ table: table.name, type: \"INDEX\", name: index.name, length: index.name.length });\r\n if (index.name.length > MAX_CONSTRAINT_NAME_LENGTH) {\r\n issues.push({\r\n type: \"constraint\",\r\n tableName: table.name,\r\n name: index.name,\r\n actualLength: index.name.length,\r\n maxLength: MAX_CONSTRAINT_NAME_LENGTH,\r\n constraintType: \"index\",\r\n });\r\n }\r\n });\r\n });\r\n\r\n // Verbose mode: show all constraints checked\r\n if (verbose) {\r\n consola.box(\"All Constraints Checked\");\r\n const sortedConstraints = allConstraints.sort((a, b) => b.length - a.length);\r\n sortedConstraints.forEach((constraint) => {\r\n const exceeds = constraint.length > MAX_CONSTRAINT_NAME_LENGTH;\r\n const lengthStr = `[${constraint.length.toString().padStart(2)} chars]`;\r\n const typeStr = constraint.type.padEnd(15);\r\n const message = `${lengthStr} ${typeStr} ${constraint.name} (${constraint.table})`;\r\n if (exceeds) {\r\n consola.error(message);\r\n } else {\r\n consola.success(message);\r\n }\r\n });\r\n consola.log(\"\");\r\n }\r\n\r\n // Print report\r\n consola.box(\"MySQL Name Length Analysis\");\r\n consola.info(`Maximum identifier length: ${MAX_IDENTIFIER_LENGTH} characters`);\r\n consola.info(`Maximum constraint name length: ${MAX_CONSTRAINT_NAME_LENGTH} characters`);\r\n consola.info(`Analyzed ${tables.length} tables`);\r\n consola.log(\"\");\r\n\r\n // Show table summary\r\n consola.start(\"Tables analyzed:\");\r\n tables.forEach((table) => {\r\n const columnCount = Object.keys(table.columns).length;\r\n const fkCount = Object.keys(table.foreignKeys).length;\r\n const pkCount = Object.keys(table.compositePrimaryKeys).length;\r\n const uniqueCount = Object.keys(table.uniqueConstraints).length;\r\n const indexCount = Object.keys(table.indexes).length;\r\n\r\n const parts: string[] = [];\r\n parts.push(`${columnCount} columns`);\r\n if (fkCount > 0) {\r\n parts.push(`${fkCount} FKs`);\r\n }\r\n if (pkCount > 0) {\r\n parts.push(`${pkCount} PKs`);\r\n }\r\n if (uniqueCount > 0) {\r\n parts.push(`${uniqueCount} unique`);\r\n }\r\n if (indexCount > 0) {\r\n parts.push(`${indexCount} indexes`);\r\n }\r\n\r\n consola.log(` - ${table.name} (${parts.join(\", \")})`);\r\n });\r\n consola.log(\"\");\r\n\r\n if (issues.length === 0) {\r\n consola.success(\"No name length issues found!\");\r\n return 0;\r\n }\r\n\r\n consola.warn(`Found ${issues.length} name length issue(s):`);\r\n consola.log(\"\");\r\n\r\n const tableIssues = issues.filter((i) => i.type === \"table\");\r\n const columnIssues = issues.filter((i) => i.type === \"column\");\r\n const constraintIssues = issues.filter((i) => i.type === \"constraint\");\r\n\r\n if (tableIssues.length > 0) {\r\n consola.error(\"Table Name Issues:\");\r\n tableIssues.forEach((issue) => {\r\n consola.log(\r\n ` - Table: ${issue.name} (${issue.actualLength} chars, exceeds max by ${issue.actualLength - issue.maxLength})`,\r\n );\r\n });\r\n consola.log(\"\");\r\n }\r\n\r\n if (columnIssues.length > 0) {\r\n consola.error(\"Column Name Issues:\");\r\n columnIssues.forEach((issue) => {\r\n consola.log(\r\n ` - ${issue.tableName}.${issue.name} (${issue.actualLength} chars, exceeds max by ${issue.actualLength - issue.maxLength})`,\r\n );\r\n });\r\n consola.log(\"\");\r\n }\r\n\r\n if (constraintIssues.length > 0) {\r\n consola.error(\"Constraint Name Issues:\");\r\n const fkIssues = constraintIssues.filter((i) => i.constraintType === \"foreign_key\");\r\n const pkIssues = constraintIssues.filter((i) => i.constraintType === \"primary_key\");\r\n const uniqueIssues = constraintIssues.filter((i) => i.constraintType === \"unique\");\r\n const indexIssues = constraintIssues.filter((i) => i.constraintType === \"index\");\r\n\r\n if (fkIssues.length > 0) {\r\n consola.log(\" Foreign Keys:\");\r\n fkIssues.forEach((issue) => {\r\n consola.log(\r\n ` - ${issue.name} (${issue.actualLength} chars, exceeds max by ${issue.actualLength - issue.maxLength})`,\r\n );\r\n consola.log(` Table: ${issue.tableName}`);\r\n });\r\n }\r\n\r\n if (pkIssues.length > 0) {\r\n consola.log(\" Primary Keys:\");\r\n pkIssues.forEach((issue) => {\r\n consola.log(\r\n ` - ${issue.name} (${issue.actualLength} chars, exceeds max by ${issue.actualLength - issue.maxLength})`,\r\n );\r\n consola.log(` Table: ${issue.tableName}`);\r\n });\r\n }\r\n\r\n if (uniqueIssues.length > 0) {\r\n consola.log(\" Unique Constraints:\");\r\n uniqueIssues.forEach((issue) => {\r\n consola.log(\r\n ` - ${issue.name} (${issue.actualLength} chars, exceeds max by ${issue.actualLength - issue.maxLength})`,\r\n );\r\n consola.log(` Table: ${issue.tableName}`);\r\n });\r\n }\r\n\r\n if (indexIssues.length > 0) {\r\n consola.log(\" Indexes:\");\r\n indexIssues.forEach((issue) => {\r\n consola.log(\r\n ` - ${issue.name} (${issue.actualLength} chars, exceeds max by ${issue.actualLength - issue.maxLength})`,\r\n );\r\n consola.log(` Table: ${issue.tableName}`);\r\n });\r\n }\r\n consola.log(\"\");\r\n }\r\n\r\n consola.box(\"Summary\");\r\n consola.info(`Total issues: ${issues.length}`);\r\n consola.info(` - Table names: ${tableIssues.length}`);\r\n consola.info(` - Column names: ${columnIssues.length}`);\r\n consola.info(` - Constraint names: ${constraintIssues.length}`);\r\n if (constraintIssues.length > 0) {\r\n const fkCount = constraintIssues.filter((i) => i.constraintType === \"foreign_key\").length;\r\n const pkCount = constraintIssues.filter((i) => i.constraintType === \"primary_key\").length;\r\n const uniqueCount = constraintIssues.filter((i) => i.constraintType === \"unique\").length;\r\n const indexCount = constraintIssues.filter((i) => i.constraintType === \"index\").length;\r\n consola.info(` - Foreign keys: ${fkCount}`);\r\n consola.info(` - Primary keys: ${pkCount}`);\r\n consola.info(` - Unique constraints: ${uniqueCount}`);\r\n consola.info(` - Indexes: ${indexCount}`);\r\n }\r\n consola.log(\"\");\r\n\r\n // Print actionable feedback\r\n consola.fail(\"MIGRATION BLOCKED: Name length violations detected!\");\r\n consola.log(\"\");\r\n consola.start(\"Action required:\");\r\n consola.log(\"1. Review the issues listed above\");\r\n consola.log(\"2. Shorten table/column names in your schema definitions\");\r\n consola.log(\"3. For foreign keys, consider:\");\r\n consola.log(\" - Shortening table names\");\r\n consola.log(\" - Shortening column names\");\r\n consola.log(\" - Using shorter referenced table names\");\r\n consola.log(\"4. Run 'pnpm generate' again after making changes\");\r\n consola.log(\"5. Run this check with --verbose to see all constraint names\");\r\n consola.log(\"\");\r\n\r\n return 1;\r\n};\r\n\r\nexport { validateDatabaseSchema };\r\n"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"validate-schema.mjs","names":["entries: JournalEntry[]","issues: NameLengthIssue[]","allConstraints: Array<{ table: string; type: string; name: string; length: number }>","parts: string[]"],"sources":["../../src/db/validateSchema.ts"],"sourcesContent":["/**\r\n * Database Schema Validator\r\n *\r\n * This script validates the database schema by analyzing all MySQL table definitions\r\n * from Drizzle migration metadata and reports any table, column, or constraint names\r\n * that exceed MySQL's maximum length limits.\r\n *\r\n * MySQL Limits:\r\n * - Table/Column names: 64 characters\r\n * - Constraint names: 64 characters\r\n *\r\n * Usage:\r\n * - Basic: pnpm -F scripts validate:db-schema\r\n * - Verbose: pnpm -F scripts validate:db-schema -- --verbose\r\n *\r\n * The verbose mode shows all constraint names being checked, sorted by length.\r\n *\r\n * This validation runs automatically:\r\n * - After `pnpm generate` (blocks if violations found)\r\n * - In CI/CD as part of PR checks\r\n */\r\n\r\nimport { consola } from \"consola\";\r\nimport { readFileSync } from \"node:fs\";\r\nimport { join, resolve } from \"node:path\";\r\n\r\ninterface NameLengthIssue {\r\n type: \"table\" | \"column\" | \"constraint\";\r\n tableName: string;\r\n name: string;\r\n actualLength: number;\r\n maxLength: number;\r\n constraintType?: \"foreign_key\" | \"primary_key\" | \"unique\" | \"index\";\r\n}\r\n\r\ninterface DrizzleSnapshot {\r\n tables: Record<string, DrizzleTable>;\r\n}\r\n\r\ninterface DrizzleTable {\r\n name: string;\r\n columns: Record<string, DrizzleColumn>;\r\n indexes: Record<string, DrizzleIndex>;\r\n foreignKeys: Record<string, DrizzleForeignKey>;\r\n compositePrimaryKeys: Record<string, DrizzlePrimaryKey>;\r\n uniqueConstraints: Record<string, DrizzleUniqueConstraint>;\r\n}\r\n\r\ninterface DrizzleColumn {\r\n name: string;\r\n type: string;\r\n primaryKey: boolean;\r\n notNull: boolean;\r\n autoincrement: boolean;\r\n}\r\n\r\ninterface DrizzleIndex {\r\n name: string;\r\n columns: string[];\r\n}\r\n\r\ninterface DrizzleForeignKey {\r\n name: string;\r\n tableFrom: string;\r\n tableTo: string;\r\n columnsFrom: string[];\r\n columnsTo: string[];\r\n}\r\n\r\ninterface DrizzlePrimaryKey {\r\n name: string;\r\n columns: string[];\r\n}\r\n\r\ninterface DrizzleUniqueConstraint {\r\n name: string;\r\n columns: string[];\r\n}\r\n\r\nconst MAX_IDENTIFIER_LENGTH = 64;\r\nconst MAX_CONSTRAINT_NAME_LENGTH = 64;\r\n\r\nconst getLatestSnapshot = (metaDir: string): DrizzleSnapshot => {\r\n const resolvedMetaDir = resolve(metaDir);\r\n\r\n // Read the journal to find the latest migration\r\n const journalPath = join(resolvedMetaDir, \"_journal.json\");\r\n const parsedJournal = JSON.parse(readFileSync(journalPath, \"utf-8\"));\r\n\r\n interface JournalEntry {\r\n tag: string;\r\n idx: number;\r\n }\r\n\r\n const entries: JournalEntry[] = parsedJournal.entries;\r\n const latestEntry = entries[entries.length - 1];\r\n\r\n if (!latestEntry) {\r\n throw new Error(\"No migrations found in journal\");\r\n }\r\n\r\n // Read the latest snapshot\r\n const snapshotPath = join(resolvedMetaDir, `${latestEntry.idx.toString().padStart(4, \"0\")}_snapshot.json`);\r\n const parsedSnapshot = JSON.parse(readFileSync(snapshotPath, \"utf-8\"));\r\n const snapshot: DrizzleSnapshot = parsedSnapshot;\r\n\r\n return snapshot;\r\n};\r\n\r\ninterface ValidateDatabaseSchemaOptions {\n metaDir: string;\n verbose?: boolean;\n exitOnComplete?: boolean;\n}\n\nconst validateDatabaseSchema = (options: ValidateDatabaseSchemaOptions): number => {\n const { metaDir, verbose = false, exitOnComplete = false } = options;\n const issues: NameLengthIssue[] = [];\n const allConstraints: Array<{ table: string; type: string; name: string; length: number }> = [];\n const finalize = (exitCode: number): number => {\n if (exitOnComplete) {\n process.exit(exitCode);\n }\n\n return exitCode;\n };\n\r\n const snapshot = getLatestSnapshot(metaDir);\r\n const tables = Object.values(snapshot.tables);\r\n\r\n tables.forEach((table) => {\r\n // Check table name length\r\n if (table.name.length > MAX_IDENTIFIER_LENGTH) {\r\n issues.push({\r\n type: \"table\",\r\n tableName: table.name,\r\n name: table.name,\r\n actualLength: table.name.length,\r\n maxLength: MAX_IDENTIFIER_LENGTH,\r\n });\r\n }\r\n\r\n // Check column name lengths\r\n Object.values(table.columns).forEach((column) => {\r\n if (column.name.length > MAX_IDENTIFIER_LENGTH) {\r\n issues.push({\r\n type: \"column\",\r\n tableName: table.name,\r\n name: column.name,\r\n actualLength: column.name.length,\r\n maxLength: MAX_IDENTIFIER_LENGTH,\r\n });\r\n }\r\n });\r\n\r\n // Check foreign key constraint names\r\n Object.values(table.foreignKeys).forEach((fk) => {\r\n allConstraints.push({ table: table.name, type: \"FK\", name: fk.name, length: fk.name.length });\r\n if (fk.name.length > MAX_CONSTRAINT_NAME_LENGTH) {\r\n issues.push({\r\n type: \"constraint\",\r\n tableName: table.name,\r\n name: fk.name,\r\n actualLength: fk.name.length,\r\n maxLength: MAX_CONSTRAINT_NAME_LENGTH,\r\n constraintType: \"foreign_key\",\r\n });\r\n }\r\n });\r\n\r\n // Check primary key constraint names\r\n Object.values(table.compositePrimaryKeys).forEach((pk) => {\r\n const pkType = pk.columns.length > 1 ? \"PK (composite)\" : \"PK\";\r\n allConstraints.push({ table: table.name, type: pkType, name: pk.name, length: pk.name.length });\r\n if (pk.name.length > MAX_CONSTRAINT_NAME_LENGTH) {\r\n issues.push({\r\n type: \"constraint\",\r\n tableName: table.name,\r\n name: pk.name,\r\n actualLength: pk.name.length,\r\n maxLength: MAX_CONSTRAINT_NAME_LENGTH,\r\n constraintType: \"primary_key\",\r\n });\r\n }\r\n });\r\n\r\n // Check unique constraint names\r\n Object.values(table.uniqueConstraints).forEach((unique) => {\r\n allConstraints.push({ table: table.name, type: \"UNIQUE\", name: unique.name, length: unique.name.length });\r\n if (unique.name.length > MAX_CONSTRAINT_NAME_LENGTH) {\r\n issues.push({\r\n type: \"constraint\",\r\n tableName: table.name,\r\n name: unique.name,\r\n actualLength: unique.name.length,\r\n maxLength: MAX_CONSTRAINT_NAME_LENGTH,\r\n constraintType: \"unique\",\r\n });\r\n }\r\n });\r\n\r\n // Check index names\r\n Object.values(table.indexes).forEach((index) => {\r\n allConstraints.push({ table: table.name, type: \"INDEX\", name: index.name, length: index.name.length });\r\n if (index.name.length > MAX_CONSTRAINT_NAME_LENGTH) {\r\n issues.push({\r\n type: \"constraint\",\r\n tableName: table.name,\r\n name: index.name,\r\n actualLength: index.name.length,\r\n maxLength: MAX_CONSTRAINT_NAME_LENGTH,\r\n constraintType: \"index\",\r\n });\r\n }\r\n });\r\n });\r\n\r\n // Verbose mode: show all constraints checked\r\n if (verbose) {\r\n consola.box(\"All Constraints Checked\");\r\n const sortedConstraints = allConstraints.sort((a, b) => b.length - a.length);\r\n sortedConstraints.forEach((constraint) => {\r\n const exceeds = constraint.length > MAX_CONSTRAINT_NAME_LENGTH;\r\n const lengthStr = `[${constraint.length.toString().padStart(2)} chars]`;\r\n const typeStr = constraint.type.padEnd(15);\r\n const message = `${lengthStr} ${typeStr} ${constraint.name} (${constraint.table})`;\r\n if (exceeds) {\r\n consola.error(message);\r\n } else {\r\n consola.success(message);\r\n }\r\n });\r\n consola.log(\"\");\r\n }\r\n\r\n // Print report\r\n consola.box(\"MySQL Name Length Analysis\");\r\n consola.info(`Maximum identifier length: ${MAX_IDENTIFIER_LENGTH} characters`);\r\n consola.info(`Maximum constraint name length: ${MAX_CONSTRAINT_NAME_LENGTH} characters`);\r\n consola.info(`Analyzed ${tables.length} tables`);\r\n consola.log(\"\");\r\n\r\n // Show table summary\r\n consola.start(\"Tables analyzed:\");\r\n tables.forEach((table) => {\r\n const columnCount = Object.keys(table.columns).length;\r\n const fkCount = Object.keys(table.foreignKeys).length;\r\n const pkCount = Object.keys(table.compositePrimaryKeys).length;\r\n const uniqueCount = Object.keys(table.uniqueConstraints).length;\r\n const indexCount = Object.keys(table.indexes).length;\r\n\r\n const parts: string[] = [];\r\n parts.push(`${columnCount} columns`);\r\n if (fkCount > 0) {\r\n parts.push(`${fkCount} FKs`);\r\n }\r\n if (pkCount > 0) {\r\n parts.push(`${pkCount} PKs`);\r\n }\r\n if (uniqueCount > 0) {\r\n parts.push(`${uniqueCount} unique`);\r\n }\r\n if (indexCount > 0) {\r\n parts.push(`${indexCount} indexes`);\r\n }\r\n\r\n consola.log(` - ${table.name} (${parts.join(\", \")})`);\r\n });\r\n consola.log(\"\");\r\n\r\n if (issues.length === 0) {\n consola.success(\"No name length issues found!\");\n return finalize(0);\n }\n\r\n consola.warn(`Found ${issues.length} name length issue(s):`);\r\n consola.log(\"\");\r\n\r\n const tableIssues = issues.filter((i) => i.type === \"table\");\r\n const columnIssues = issues.filter((i) => i.type === \"column\");\r\n const constraintIssues = issues.filter((i) => i.type === \"constraint\");\r\n\r\n if (tableIssues.length > 0) {\r\n consola.error(\"Table Name Issues:\");\r\n tableIssues.forEach((issue) => {\r\n consola.log(\r\n ` - Table: ${issue.name} (${issue.actualLength} chars, exceeds max by ${issue.actualLength - issue.maxLength})`,\r\n );\r\n });\r\n consola.log(\"\");\r\n }\r\n\r\n if (columnIssues.length > 0) {\r\n consola.error(\"Column Name Issues:\");\r\n columnIssues.forEach((issue) => {\r\n consola.log(\r\n ` - ${issue.tableName}.${issue.name} (${issue.actualLength} chars, exceeds max by ${issue.actualLength - issue.maxLength})`,\r\n );\r\n });\r\n consola.log(\"\");\r\n }\r\n\r\n if (constraintIssues.length > 0) {\r\n consola.error(\"Constraint Name Issues:\");\r\n const fkIssues = constraintIssues.filter((i) => i.constraintType === \"foreign_key\");\r\n const pkIssues = constraintIssues.filter((i) => i.constraintType === \"primary_key\");\r\n const uniqueIssues = constraintIssues.filter((i) => i.constraintType === \"unique\");\r\n const indexIssues = constraintIssues.filter((i) => i.constraintType === \"index\");\r\n\r\n if (fkIssues.length > 0) {\r\n consola.log(\" Foreign Keys:\");\r\n fkIssues.forEach((issue) => {\r\n consola.log(\r\n ` - ${issue.name} (${issue.actualLength} chars, exceeds max by ${issue.actualLength - issue.maxLength})`,\r\n );\r\n consola.log(` Table: ${issue.tableName}`);\r\n });\r\n }\r\n\r\n if (pkIssues.length > 0) {\r\n consola.log(\" Primary Keys:\");\r\n pkIssues.forEach((issue) => {\r\n consola.log(\r\n ` - ${issue.name} (${issue.actualLength} chars, exceeds max by ${issue.actualLength - issue.maxLength})`,\r\n );\r\n consola.log(` Table: ${issue.tableName}`);\r\n });\r\n }\r\n\r\n if (uniqueIssues.length > 0) {\r\n consola.log(\" Unique Constraints:\");\r\n uniqueIssues.forEach((issue) => {\r\n consola.log(\r\n ` - ${issue.name} (${issue.actualLength} chars, exceeds max by ${issue.actualLength - issue.maxLength})`,\r\n );\r\n consola.log(` Table: ${issue.tableName}`);\r\n });\r\n }\r\n\r\n if (indexIssues.length > 0) {\r\n consola.log(\" Indexes:\");\r\n indexIssues.forEach((issue) => {\r\n consola.log(\r\n ` - ${issue.name} (${issue.actualLength} chars, exceeds max by ${issue.actualLength - issue.maxLength})`,\r\n );\r\n consola.log(` Table: ${issue.tableName}`);\r\n });\r\n }\r\n consola.log(\"\");\r\n }\r\n\r\n consola.box(\"Summary\");\r\n consola.info(`Total issues: ${issues.length}`);\r\n consola.info(` - Table names: ${tableIssues.length}`);\r\n consola.info(` - Column names: ${columnIssues.length}`);\r\n consola.info(` - Constraint names: ${constraintIssues.length}`);\r\n if (constraintIssues.length > 0) {\r\n const fkCount = constraintIssues.filter((i) => i.constraintType === \"foreign_key\").length;\r\n const pkCount = constraintIssues.filter((i) => i.constraintType === \"primary_key\").length;\r\n const uniqueCount = constraintIssues.filter((i) => i.constraintType === \"unique\").length;\r\n const indexCount = constraintIssues.filter((i) => i.constraintType === \"index\").length;\r\n consola.info(` - Foreign keys: ${fkCount}`);\r\n consola.info(` - Primary keys: ${pkCount}`);\r\n consola.info(` - Unique constraints: ${uniqueCount}`);\r\n consola.info(` - Indexes: ${indexCount}`);\r\n }\r\n consola.log(\"\");\r\n\r\n // Print actionable feedback\r\n consola.fail(\"MIGRATION BLOCKED: Name length violations detected!\");\r\n consola.log(\"\");\r\n consola.start(\"Action required:\");\r\n consola.log(\"1. Review the issues listed above\");\r\n consola.log(\"2. Shorten table/column names in your schema definitions\");\r\n consola.log(\"3. For foreign keys, consider:\");\r\n consola.log(\" - Shortening table names\");\r\n consola.log(\" - Shortening column names\");\r\n consola.log(\" - Using shorter referenced table names\");\r\n consola.log(\"4. Run 'pnpm generate' again after making changes\");\r\n consola.log(\"5. Run this check with --verbose to see all constraint names\");\r\n consola.log(\"\");\r\n\r\n return finalize(1);\n};\n\r\nexport { validateDatabaseSchema };\r\n"],"mappings":"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"}
|