From cc1e03efccc280e90e80587d84a1c799e2f36773 Mon Sep 17 00:00:00 2001 From: hoshinotsuyoshi Date: Wed, 4 Dec 2024 19:57:43 +0900 Subject: [PATCH 1/3] Parse and handle of table comments in `postgres` --- .../parser/__snapshots__/index.test.ts.snap | 2 +- .../sql/input/postgresql_schema1.in.sql | 1 + .../src/parser/sql/postgresql/converter.ts | 39 +++++++++++++++++++ .../src/parser/sql/postgresql/index.test.ts | 17 ++++++++ 4 files changed, 58 insertions(+), 1 deletion(-) diff --git a/frontend/packages/db-structure/src/parser/__snapshots__/index.test.ts.snap b/frontend/packages/db-structure/src/parser/__snapshots__/index.test.ts.snap index 225a1c719..5be04a643 100644 --- a/frontend/packages/db-structure/src/parser/__snapshots__/index.test.ts.snap +++ b/frontend/packages/db-structure/src/parser/__snapshots__/index.test.ts.snap @@ -91,7 +91,7 @@ exports[`parse > should parse postgresql to JSON correctly 1`] = ` "unique": true, }, }, - "comment": null, + "comment": "store our users.", "indices": { "index_users_on_id_and_email": { "columns": [ diff --git a/frontend/packages/db-structure/src/parser/sql/input/postgresql_schema1.in.sql b/frontend/packages/db-structure/src/parser/sql/input/postgresql_schema1.in.sql index d1d06ad53..706f7d638 100644 --- a/frontend/packages/db-structure/src/parser/sql/input/postgresql_schema1.in.sql +++ b/frontend/packages/db-structure/src/parser/sql/input/postgresql_schema1.in.sql @@ -11,3 +11,4 @@ CREATE TABLE posts ( ); CREATE UNIQUE INDEX index_users_on_id_and_email ON public.users USING btree (id, email); +COMMENT ON TABLE users IS 'store our users.'; diff --git a/frontend/packages/db-structure/src/parser/sql/postgresql/converter.ts b/frontend/packages/db-structure/src/parser/sql/postgresql/converter.ts index e539fedd0..c363bd86d 100644 --- a/frontend/packages/db-structure/src/parser/sql/postgresql/converter.ts +++ b/frontend/packages/db-structure/src/parser/sql/postgresql/converter.ts @@ -1,7 +1,9 @@ import type { + CommentStmt, Constraint, CreateStmt, IndexStmt, + List, Node, String as PgString, } from '@pgsql/types' @@ -40,6 +42,10 @@ export const convertToDBStructure = (ast: RawStmtWrapper[]): DBStructure => { return 'IndexStmt' in stmt } + function isCommentStmt(stmt: Node): stmt is { CommentStmt: CommentStmt } { + return 'CommentStmt' in stmt + } + function handleCreateStmt(createStmt: CreateStmt) { if (!createStmt || !createStmt.relation || !createStmt.tableElts) return @@ -212,6 +218,37 @@ export const convertToDBStructure = (ast: RawStmtWrapper[]): DBStructure => { } } + function handleCommentStmt(commentStmt: CommentStmt) { + if (commentStmt.objtype !== 'OBJECT_TABLE') return + const objectNode = commentStmt.object + if (!objectNode) return + + const isList = (stmt: Node): stmt is { List: List } => 'List' in stmt + if (!isList(objectNode)) return + + // Handles statements like `COMMENT ON TABLE IS '';`. + // NOTE: As far as I know, PostgreSQL only allows adding a comment to one table per statement, + // so we can reasonably assume the number of `` elements is 1. + const item = objectNode.List?.items?.[0] + if (!item) return + const tableName = + 'String' in item && + typeof item.String === 'object' && + item.String !== null && + 'sval' in item.String && + item.String.sval + + if (!tableName) return + if (!tables[tableName]) return + const comment = commentStmt.comment + if (!comment) return + + tables[tableName] = { + ...tables[tableName], + comment, + } + } + if (!ast) { return { tables: {}, @@ -229,6 +266,8 @@ export const convertToDBStructure = (ast: RawStmtWrapper[]): DBStructure => { handleCreateStmt(stmt.CreateStmt) } else if (isIndexStmt(stmt)) { handleIndexStmt(stmt.IndexStmt) + } else if (isCommentStmt(stmt)) { + handleCommentStmt(stmt.CommentStmt) } } diff --git a/frontend/packages/db-structure/src/parser/sql/postgresql/index.test.ts b/frontend/packages/db-structure/src/parser/sql/postgresql/index.test.ts index ae2aa7e0e..fc242b5d5 100644 --- a/frontend/packages/db-structure/src/parser/sql/postgresql/index.test.ts +++ b/frontend/packages/db-structure/src/parser/sql/postgresql/index.test.ts @@ -28,10 +28,27 @@ describe(processor, () => { indices: { ...override?.indices, }, + comment: override?.comment ?? null, }), }, }) + it('comment', async () => { + const result = await processor(/* PostgreSQL */ ` + CREATE TABLE users ( + id SERIAL PRIMARY KEY, + name VARCHAR(255) + ); + COMMENT ON TABLE users IS 'store our users.'; + `) + + const expected = userTable({ + comment: 'store our users.', + }) + + expect(result).toEqual(expected) + }) + it('not null', async () => { const result = await processor(/* sql */ ` CREATE TABLE users ( From f1eb2742cdf1de1f9c08fab67b2527d935790800 Mon Sep 17 00:00:00 2001 From: hoshinotsuyoshi Date: Wed, 4 Dec 2024 21:34:12 +0900 Subject: [PATCH 2/3] Fix typos --- .../packages/db-structure/src/parser/schemarb/index.test.ts | 2 +- .../db-structure/src/parser/sql/postgresql/index.test.ts | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/frontend/packages/db-structure/src/parser/schemarb/index.test.ts b/frontend/packages/db-structure/src/parser/schemarb/index.test.ts index 978efc87d..d7c3385f4 100644 --- a/frontend/packages/db-structure/src/parser/schemarb/index.test.ts +++ b/frontend/packages/db-structure/src/parser/schemarb/index.test.ts @@ -4,7 +4,7 @@ import { aColumn, aDBStructure, aTable, anIndex } from '../../schema/index.js' import { processor } from './index.js' describe(processor, () => { - describe('should parse create_table correctry', () => { + describe('should parse create_table correctly', () => { const userTable = (override?: Partial) => aDBStructure({ tables: { diff --git a/frontend/packages/db-structure/src/parser/sql/postgresql/index.test.ts b/frontend/packages/db-structure/src/parser/sql/postgresql/index.test.ts index fc242b5d5..1b16ac8d1 100644 --- a/frontend/packages/db-structure/src/parser/sql/postgresql/index.test.ts +++ b/frontend/packages/db-structure/src/parser/sql/postgresql/index.test.ts @@ -4,7 +4,7 @@ import { aColumn, aDBStructure, aTable } from '../../../schema/index.js' import { processor } from './index.js' describe(processor, () => { - describe('should parse create_table correctry', () => { + describe('should parse create_table correctly', () => { const userTable = (override?: Partial
) => aDBStructure({ tables: { From e5a04a49a795e9936637f89648d9b14e04d54f2c Mon Sep 17 00:00:00 2001 From: hoshino tsuyoshi Date: Thu, 5 Dec 2024 12:09:26 +0900 Subject: [PATCH 3/3] Update frontend/packages/db-structure/src/parser/sql/postgresql/converter.ts Co-authored-by: Ryota Sasazawa <65695538+sasamuku@users.noreply.github.com> --- .../db-structure/src/parser/sql/postgresql/converter.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/frontend/packages/db-structure/src/parser/sql/postgresql/converter.ts b/frontend/packages/db-structure/src/parser/sql/postgresql/converter.ts index c363bd86d..30721f020 100644 --- a/frontend/packages/db-structure/src/parser/sql/postgresql/converter.ts +++ b/frontend/packages/db-structure/src/parser/sql/postgresql/converter.ts @@ -227,7 +227,7 @@ export const convertToDBStructure = (ast: RawStmtWrapper[]): DBStructure => { if (!isList(objectNode)) return // Handles statements like `COMMENT ON TABLE IS '';`. - // NOTE: As far as I know, PostgreSQL only allows adding a comment to one table per statement, + // NOTE: PostgreSQL allows only one comment to be added to one table per statement, // so we can reasonably assume the number of `` elements is 1. const item = objectNode.List?.items?.[0] if (!item) return