sq/cli/output/jsonw/metadatawriter.go
Neil O'Toole 26f0c9a381
Refactor source.Files (#363)
* Moved `source.Files` to its own package, thus the type is now `files.Files`.
* Moved much of the location functionality from pkg `source` to its own package `location`.
2024-01-24 23:29:55 -07:00

107 lines
2.7 KiB
Go

package jsonw
import (
"io"
"github.com/samber/lo"
"github.com/neilotoole/sq/cli/output"
"github.com/neilotoole/sq/libsq/driver"
"github.com/neilotoole/sq/libsq/source/location"
"github.com/neilotoole/sq/libsq/source/metadata"
)
// mdWriter implements output.MetadataWriter for JSON.
type mdWriter struct {
out io.Writer
pr *output.Printing
}
// NewMetadataWriter returns a new output.MetadataWriter instance
// that outputs metadata in JSON.
func NewMetadataWriter(out io.Writer, pr *output.Printing) output.MetadataWriter {
return &mdWriter{out: out, pr: pr}
}
// DriverMetadata implements output.MetadataWriter.
func (w *mdWriter) DriverMetadata(md []driver.Metadata) error {
return writeJSON(w.out, w.pr, md)
}
// TableMetadata implements output.MetadataWriter.
func (w *mdWriter) TableMetadata(md *metadata.Table) error {
return writeJSON(w.out, w.pr, md)
}
// SourceMetadata implements output.MetadataWriter.
func (w *mdWriter) SourceMetadata(md *metadata.Source, showSchema bool) error {
md2 := *md // Shallow copy is fine
md2.Location = location.Redact(md2.Location)
if showSchema {
return writeJSON(w.out, w.pr, &md2)
}
// Don't render "tables", "table_count", and "view_count"
type mdNoSchema struct {
metadata.Source
Tables *[]*metadata.Table `json:"tables,omitempty"`
TableCount *int64 `json:"table_count,omitempty"`
ViewCount *int64 `json:"view_count,omitempty"`
}
return writeJSON(w.out, w.pr, &mdNoSchema{Source: md2})
}
// DBProperties implements output.MetadataWriter.
func (w *mdWriter) DBProperties(props map[string]any) error {
if len(props) == 0 {
return nil
}
return writeJSON(w.out, w.pr, props)
}
// Catalogs implements output.MetadataWriter.
func (w *mdWriter) Catalogs(currentCatalog string, catalogs []string) error {
if len(catalogs) == 0 {
return nil
}
type cat struct {
Name string `json:"catalog"`
Active *bool `json:"active,omitempty"`
}
cats := make([]cat, len(catalogs))
for i, c := range catalogs {
cats[i] = cat{Name: c}
if c == currentCatalog {
cats[i].Active = lo.ToPtr(true)
}
}
return writeJSON(w.out, w.pr, cats)
}
// Schemata implements output.MetadataWriter.
func (w *mdWriter) Schemata(currentSchema string, schemas []*metadata.Schema) error {
if len(schemas) == 0 {
return nil
}
// We wrap each schema in a struct that has an "active" field,
// because we need to show the current schema in the output.
type wrapper struct {
metadata.Schema `json:",omitempty,inline"`
Active *bool `json:"active,omitempty"`
}
a := make([]*wrapper, len(schemas))
for i, s := range schemas {
a[i] = &wrapper{Schema: *s}
if s.Name == currentSchema {
a[i].Active = lo.ToPtr(true)
}
}
return writeJSON(w.out, w.pr, a)
}