UNPKG

kysely

Version:
94 lines (93 loc) 3.66 kB
/// <reference types="./postgres-introspector.d.ts" /> import { DEFAULT_MIGRATION_LOCK_TABLE, DEFAULT_MIGRATION_TABLE, } from '../../migration/migrator.js'; import { freeze } from '../../util/object-utils.js'; import { sql } from '../../raw-builder/sql.js'; export class PostgresIntrospector { #db; constructor(db) { this.#db = db; } async getSchemas() { let rawSchemas = await this.#db .selectFrom('pg_catalog.pg_namespace') .select('nspname') .$castTo() .execute(); return rawSchemas.map((it) => ({ name: it.nspname })); } async getTables(options = { withInternalKyselyTables: false }) { let query = this.#db // column .selectFrom('pg_catalog.pg_attribute as a') // table .innerJoin('pg_catalog.pg_class as c', 'a.attrelid', 'c.oid') // table schema .innerJoin('pg_catalog.pg_namespace as ns', 'c.relnamespace', 'ns.oid') // column data type .innerJoin('pg_catalog.pg_type as typ', 'a.atttypid', 'typ.oid') // column data type schema .innerJoin('pg_catalog.pg_namespace as dtns', 'typ.typnamespace', 'dtns.oid') .select([ 'a.attname as column', 'a.attnotnull as not_null', 'a.atthasdef as has_default', 'c.relname as table', 'c.relkind as table_type', 'ns.nspname as schema', 'typ.typname as type', 'dtns.nspname as type_schema', sql `col_description(a.attrelid, a.attnum)`.as('column_description'), sql `pg_get_serial_sequence(quote_ident(ns.nspname) || '.' || quote_ident(c.relname), a.attname)`.as('auto_incrementing'), ]) .where('c.relkind', 'in', [ 'r' /*regular table*/, 'v' /*view*/, 'p' /*partitioned table*/, ]) .where('ns.nspname', '!~', '^pg_') .where('ns.nspname', '!=', 'information_schema') // No system columns .where('a.attnum', '>=', 0) .where('a.attisdropped', '!=', true) .orderBy('ns.nspname') .orderBy('c.relname') .orderBy('a.attnum') .$castTo(); if (!options.withInternalKyselyTables) { query = query .where('c.relname', '!=', DEFAULT_MIGRATION_TABLE) .where('c.relname', '!=', DEFAULT_MIGRATION_LOCK_TABLE); } const rawColumns = await query.execute(); return this.#parseTableMetadata(rawColumns); } async getMetadata(options) { return { tables: await this.getTables(options), }; } #parseTableMetadata(columns) { return columns.reduce((tables, it) => { let table = tables.find((tbl) => tbl.name === it.table && tbl.schema === it.schema); if (!table) { table = freeze({ name: it.table, isView: it.table_type === 'v', schema: it.schema, columns: [], }); tables.push(table); } table.columns.push(freeze({ name: it.column, dataType: it.type, dataTypeSchema: it.type_schema, isNullable: !it.not_null, isAutoIncrementing: it.auto_incrementing !== null, hasDefaultValue: it.has_default, comment: it.column_description ?? undefined, })); return tables; }, []); } }