From af18d5e70d86b91cdb163cb691a9db81fb95de8b Mon Sep 17 00:00:00 2001 From: Jennifer Beckett Date: Tue, 25 Jun 2024 15:36:40 +0100 Subject: [PATCH] Add support to output Postgres enum definitions in the db doc README --- config/templates.go | 1 + drivers/postgres/postgres.go | 32 +++++++++++++++++ output/md/md.go | 34 +++++++++++++++++++ output/md/templates/index.md.tmpl | 9 +++++ schema/json.go | 13 +++++++ schema/schema.go | 6 ++++ testdata/json_output_schema.golden | 10 ++++++ testdata/md_test_README.md.adjust.golden | 6 ++++ testdata/md_test_README.md.first_para.golden | 6 ++++ testdata/md_test_README.md.golden | 6 ++++ testdata/md_test_README.md.mermaid.golden | 6 ++++ testdata/md_test_README.md.number.golden | 6 ++++ ..._test_README.md.space_in_table_name.golden | 6 ++++ testdata/yaml_output_schema.golden | 6 ++++ testutil/schema.go | 9 +++++ 15 files changed, 156 insertions(+) diff --git a/config/templates.go b/config/templates.go index 0e38ef48b..5051b4e38 100644 --- a/config/templates.go +++ b/config/templates.go @@ -15,6 +15,7 @@ type MD struct { Index string `yaml:"index,omitempty"` Table string `yaml:"table,omitempty"` Viewpoint string `yaml:"viewpoint,omitempty"` + Enum string `yaml:"enum,omitempty"` } // Dot holds the paths to the dot template files. diff --git a/drivers/postgres/postgres.go b/drivers/postgres/postgres.go index b6ab4f358..efc57e22d 100644 --- a/drivers/postgres/postgres.go +++ b/drivers/postgres/postgres.go @@ -329,6 +329,13 @@ ORDER BY tgrelid } s.Functions = functions + // Enums + enums, err := p.getEnums() + if err != nil { + return err + } + s.Enums = enums + s.Tables = tables // Relations @@ -490,6 +497,31 @@ func (p *Postgres) getFunctionsByQuery(query string) ([]*schema.Function, error) return functions, nil } +func (p *Postgres) getEnums() ([]*schema.Enum, error) { + enums := []*schema.Enum{} + + enumsResult, err := p.db.Query(`SELECT t.typname AS enum_name, ARRAY_AGG(e.enumlabel) AS enum_values + FROM pg_type t, pg_enum e + WHERE t.typcategory = 'E' + AND t.oid = e.enumtypid + GROUP BY t.typname `) + + if err != nil { + return nil, errors.WithStack(err) + } + defer enumsResult.Close() + + for enumsResult.Next() { + var enum schema.Enum + err := enumsResult.Scan(&enum.Name, pq.Array(&enum.Values)) + if err != nil { + return enums, errors.WithStack(err) + } + enums = append(enums, &enum) + } + return enums, nil +} + func fullTableName(owner string, tableName string) string { return fmt.Sprintf("%s.%s", owner, tableName) } diff --git a/output/md/md.go b/output/md/md.go index fc3ebf550..f115ca8eb 100644 --- a/output/md/md.go +++ b/output/md/md.go @@ -7,6 +7,7 @@ import ( "io" "os" "path/filepath" + "sort" "strconv" "strings" "text/template" @@ -521,11 +522,15 @@ func (m *Md) makeSchemaTemplateData(s *schema.Schema) map[string]interface{} { // Viewpoints viewpointsData := m.viewpointsData(s.Viewpoints, number, adjust, showOnlyFirstParagraph) + // Enums + enumData := m.enumData(s.Enums) + return map[string]interface{}{ "Schema": s, "Tables": tablesData, "Functions": functionsData, "Viewpoints": viewpointsData, + "Enums": enumData, } } @@ -873,6 +878,35 @@ func (m *Md) functionsData(functions []*schema.Function, number, adjust, showOnl return data } +func (m *Md) enumData(enums []*schema.Enum) [][]string { + data := [][]string{} + + if len(enums) == 0 { + return data + } + + header := []string{ + m.config.MergedDict.Lookup("Name"), + m.config.MergedDict.Lookup("Values"), + } + headerLine := []string{"----", "-------"} + data = append(data, + header, + headerLine, + ) + + for _, e := range enums { + sort.Strings(e.Values) + d := []string{ + e.Name, + strings.Join(e.Values, ", "), + } + data = append(data, d) + } + + return data +} + func (m *Md) viewpointsData(viewpoints []*schema.Viewpoint, number, adjust, showOnlyFirstParagraph bool) [][]string { data := [][]string{} header := []string{ diff --git a/output/md/templates/index.md.tmpl b/output/md/templates/index.md.tmpl index 19f8ab12c..461262507 100644 --- a/output/md/templates/index.md.tmpl +++ b/output/md/templates/index.md.tmpl @@ -31,6 +31,15 @@ |{{ range $d := $t }} {{ $d | nl2br }} |{{ end }} {{- end -}} {{- end -}} + +{{ if ne (len .Enums) 0 }} + +## {{ "Enums" | lookup }} +{{ range $t := .Enums }} +|{{ range $d := $t }} {{ $d | nl2br }} |{{ end }} +{{- end -}} +{{- end -}} + {{- if .er }} ## {{ "Relations" | lookup }} diff --git a/schema/json.go b/schema/json.go index cd635ac45..e62f88543 100644 --- a/schema/json.go +++ b/schema/json.go @@ -19,6 +19,7 @@ func (s Schema) MarshalJSON() ([]byte, error) { Tables []*Table `json:"tables"` Relations []*Relation `json:"relations"` Functions []*Function `json:"functions"` + Enums []*Enum `json:"enums,omitempty"` Driver *Driver `json:"driver"` Labels Labels `json:"labels,omitempty"` Viewpoints []*Viewpoint `json:"viewpoints,omitempty"` @@ -29,6 +30,7 @@ func (s Schema) MarshalJSON() ([]byte, error) { Relations: s.Relations, Driver: s.Driver, Functions: s.Functions, + Enums: s.Enums, Labels: s.Labels, Viewpoints: s.Viewpoints, }) @@ -49,6 +51,17 @@ func (d Function) MarshalJSON() ([]byte, error) { }) } +// MarshalJSON return custom JSON byte +func (e Enum) MarshalJSON() ([]byte, error) { + return json.Marshal(&struct { + Name string `json:"name"` + Values []string `json:"values"` + }{ + Name: e.Name, + Values: e.Values, + }) +} + // MarshalJSON return custom JSON byte func (d Driver) MarshalJSON() ([]byte, error) { if d.Meta == nil { diff --git a/schema/schema.go b/schema/schema.go index 38b7b0d19..721254170 100644 --- a/schema/schema.go +++ b/schema/schema.go @@ -180,6 +180,11 @@ type Function struct { Type string `json:"type"` } +type Enum struct { + Name string `json:"name"` + Values []string `json:"values"` +} + // Driver is the struct for tbls driver information type Driver struct { Name string `json:"name"` @@ -194,6 +199,7 @@ type Schema struct { Tables []*Table `json:"tables"` Relations []*Relation `json:"relations"` Functions []*Function `json:"functions"` + Enums []*Enum `json:"enums,omitempty"` Driver *Driver `json:"driver"` Labels Labels `json:"labels,omitempty"` Viewpoints Viewpoints `json:"viewpoints,omitempty"` diff --git a/testdata/json_output_schema.golden b/testdata/json_output_schema.golden index 30755d3f2..34ea9768e 100644 --- a/testdata/json_output_schema.golden +++ b/testdata/json_output_schema.golden @@ -139,6 +139,16 @@ } ], "functions": null, + "enums": [ + { + "name": "enum", + "values": [ + "one", + "two", + "three" + ] + } + ], "driver": { "name": "testdriver", "database_version": "1.0.0", diff --git a/testdata/md_test_README.md.adjust.golden b/testdata/md_test_README.md.adjust.golden index 15987e201..6438e9a7e 100644 --- a/testdata/md_test_README.md.adjust.golden +++ b/testdata/md_test_README.md.adjust.golden @@ -17,6 +17,12 @@ | [b](b.md) | 2 | table b | | `red` `green` | | [view](view.md) | 1 | view | VIEW | | +## Enums + +| Name | Values | +| ---- | ------- | +| enum | one, three, two | + --- > Generated by [tbls](https://github.com/k1LoW/tbls) diff --git a/testdata/md_test_README.md.first_para.golden b/testdata/md_test_README.md.first_para.golden index fa9983221..46bd9f467 100644 --- a/testdata/md_test_README.md.first_para.golden +++ b/testdata/md_test_README.md.first_para.golden @@ -17,6 +17,12 @@ | [b](b.md) | 2 | table b | | `red` `green` | | [view](view.md) | 1 | view | VIEW | | +## Enums + +| Name | Values | +| ---- | ------- | +| enum | one, three, two | + ## Relations ![er](schema.png) diff --git a/testdata/md_test_README.md.golden b/testdata/md_test_README.md.golden index f4b0c3469..03c914fb5 100644 --- a/testdata/md_test_README.md.golden +++ b/testdata/md_test_README.md.golden @@ -17,6 +17,12 @@ | [b](b.md) | 2 | table b | | `red` `green` | | [view](view.md) | 1 | view | VIEW | | +## Enums + +| Name | Values | +| ---- | ------- | +| enum | one, three, two | + ## Relations ![er](schema.png) diff --git a/testdata/md_test_README.md.mermaid.golden b/testdata/md_test_README.md.mermaid.golden index 565168938..80248c4fc 100644 --- a/testdata/md_test_README.md.mermaid.golden +++ b/testdata/md_test_README.md.mermaid.golden @@ -17,6 +17,12 @@ | [b](b.md) | 2 | table b | | `red` `green` | | [view](view.md) | 1 | view | VIEW | | +## Enums + +| Name | Values | +| ---- | ------- | +| enum | one, three, two | + ## Relations ```mermaid diff --git a/testdata/md_test_README.md.number.golden b/testdata/md_test_README.md.number.golden index 48c7890a9..c392c62df 100644 --- a/testdata/md_test_README.md.number.golden +++ b/testdata/md_test_README.md.number.golden @@ -17,6 +17,12 @@ | 2 | [b](b.md) | 2 | table b | | `red` `green` | | 3 | [view](view.md) | 1 | view | VIEW | | +## Enums + +| Name | Values | +| ---- | ------- | +| enum | one, three, two | + --- > Generated by [tbls](https://github.com/k1LoW/tbls) diff --git a/testdata/md_test_README.md.space_in_table_name.golden b/testdata/md_test_README.md.space_in_table_name.golden index 21d19b13c..009ff3d98 100644 --- a/testdata/md_test_README.md.space_in_table_name.golden +++ b/testdata/md_test_README.md.space_in_table_name.golden @@ -17,6 +17,12 @@ | [a b](a%20b.md) | 2 | table b | | `red` `green` | | [view](view.md) | 1 | view | VIEW | | +## Enums + +| Name | Values | +| ---- | ------- | +| enum | one, three, two | + --- > Generated by [tbls](https://github.com/k1LoW/tbls) diff --git a/testdata/yaml_output_schema.golden b/testdata/yaml_output_schema.golden index 6cd14c80b..e976bb8e6 100644 --- a/testdata/yaml_output_schema.golden +++ b/testdata/yaml_output_schema.golden @@ -92,6 +92,12 @@ relations: def: FOREIGN KEY (b) REFERENCES a(a) virtual: false functions: [] +enums: +- name: enum + values: + - one + - two + - three driver: name: testdriver databaseVersion: 1.0.0 diff --git a/testutil/schema.go b/testutil/schema.go index fdbb6c5df..ecb9000e7 100644 --- a/testutil/schema.go +++ b/testutil/schema.go @@ -14,6 +14,7 @@ func NewSchema(t *testing.T) *schema.Schema { labelBlueName = "blue" labelRedName = "red" labelGreenName = "green" + enumName = "enum" ) labelBlue := &schema.Label{ @@ -108,6 +109,11 @@ func NewSchema(t *testing.T) *schema.Schema { }, } + enum := &schema.Enum{ + Name: enumName, + Values: []string{"one", "two", "three"}, + } + r := &schema.Relation{ Table: tb, Columns: []*schema.Column{cb}, @@ -128,6 +134,9 @@ func NewSchema(t *testing.T) *schema.Schema { tb, tView, }, + Enums: []*schema.Enum{ + enum, + }, Relations: []*schema.Relation{ r, },