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 { //nolint:govet // field alignment 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 { //nolint:govet // field alignment 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) }