mirror of
https://github.com/neilotoole/sq.git
synced 2024-12-19 22:21:56 +03:00
98b47a2666
* refactor: moved cli flags to pkg cli/flag * testh: add OptLongDB for long-running tests * implement 'sq config dir' * legacy dir migration: probably a bad idea * cleanup * Refactored SQ_CONFIG and --config * added yaml writer * Dialing in tests * YAML output for 'sq driver ls' * Significant refactoring of config * Minor test for ioz * Rename source.Set to source.Collection * Cleaning up references to source.Set
147 lines
4.0 KiB
Go
147 lines
4.0 KiB
Go
// Package config holds CLI configuration.
|
|
package config
|
|
|
|
import (
|
|
"time"
|
|
|
|
"github.com/neilotoole/sq/cli/buildinfo"
|
|
|
|
"github.com/neilotoole/sq/drivers/userdriver"
|
|
"github.com/neilotoole/sq/libsq/core/errz"
|
|
"github.com/neilotoole/sq/libsq/core/stringz"
|
|
"github.com/neilotoole/sq/libsq/source"
|
|
)
|
|
|
|
const (
|
|
EnvarLogPath = "SQ_LOGFILE"
|
|
EnvarLogTruncate = "SQ_LOGFILE_TRUNCATE"
|
|
|
|
// EnvarConfigDir is the legacy envar for config location.
|
|
// Instead use EnvarConfig.
|
|
EnvarConfigDir = "SQ_CONFIGDIR"
|
|
|
|
// EnvarConfig is the envar for config location.
|
|
EnvarConfig = "SQ_CONFIG"
|
|
)
|
|
|
|
// Config holds application config/session data.
|
|
type Config struct {
|
|
// Version is the config version. This will allow sq to
|
|
// upgrade config files if needed. It must be a valid semver.
|
|
Version string `yaml:"config_version" json:"config_version"`
|
|
|
|
// Options contains default settings, such as output format.
|
|
Options Options `yaml:"options" json:"options"`
|
|
|
|
// Collection is the set of data sources.
|
|
Collection *source.Collection `yaml:"collection" json:"collection"`
|
|
|
|
// Ext holds sq config extensions, such as user driver config.
|
|
Ext Ext `yaml:"-" json:"-"`
|
|
}
|
|
|
|
// String returns a log/debug-friendly representation.
|
|
func (c *Config) String() string {
|
|
return stringz.SprintJSON(c)
|
|
}
|
|
|
|
// Ext holds additional config (extensions) loaded from other
|
|
// config files, e.g. ~/.config/sq/ext/*.sq.yml.
|
|
type Ext struct {
|
|
UserDrivers []*userdriver.DriverDef `yaml:"user_drivers" json:"user_drivers"`
|
|
}
|
|
|
|
// Options contains default config values.
|
|
type Options struct {
|
|
// Format is the default output format: json, table, etc.
|
|
Format Format `yaml:"output_format" json:"output_format"`
|
|
|
|
// Header determines if a header should be printed (if relevant
|
|
// for the output format).
|
|
Header bool `yaml:"output_header" json:"output_header"`
|
|
|
|
// PingTimeout is the allowed time for a ping.
|
|
PingTimeout time.Duration `yaml:"ping_timeout" json:"ping_timeout"`
|
|
|
|
// ShellCompletionTimeout is the time allowed for the shell
|
|
// completion callback to execute.
|
|
ShellCompletionTimeout time.Duration `yaml:"shell_completion_timeout" json:"shell_completion_timeout"`
|
|
}
|
|
|
|
// New returns a config instance with default options set.
|
|
func New() *Config {
|
|
cfg := &Config{}
|
|
|
|
// By default, we want header to be true; this is
|
|
// ugly wrt initCfg, as the zero value of a bool
|
|
// is false, but we actually want it to be true for Header.
|
|
cfg.Options.Header = true
|
|
|
|
initCfg(cfg)
|
|
return cfg
|
|
}
|
|
|
|
// initCfg checks if required values are present, and if not, sets them.
|
|
func initCfg(cfg *Config) {
|
|
if cfg.Collection == nil {
|
|
cfg.Collection = &source.Collection{}
|
|
}
|
|
|
|
if cfg.Version == "" {
|
|
cfg.Version = buildinfo.Version
|
|
}
|
|
|
|
if cfg.Options.Format == "" {
|
|
cfg.Options.Format = FormatTable
|
|
}
|
|
|
|
if cfg.Options.PingTimeout == 0 {
|
|
// Probably should be setting this in the New function,
|
|
// but we haven't yet defined cli's behavior wrt
|
|
// a zero timeout. Does it mean no timeout?
|
|
cfg.Options.PingTimeout = 10 * time.Second
|
|
}
|
|
|
|
if cfg.Options.ShellCompletionTimeout == 0 {
|
|
cfg.Options.ShellCompletionTimeout = time.Millisecond * 500
|
|
}
|
|
}
|
|
|
|
// Format is an output format such as json or xml.
|
|
type Format string
|
|
|
|
// UnmarshalText implements encoding.TextUnmarshaler.
|
|
func (f *Format) UnmarshalText(text []byte) error {
|
|
switch Format(text) {
|
|
default:
|
|
return errz.Errorf("unknown output format {%s}", string(text))
|
|
case FormatJSON, FormatJSONA, FormatJSONL, FormatTable, FormatRaw,
|
|
FormatHTML, FormatMarkdown, FormatXLSX, FormatXML,
|
|
FormatCSV, FormatTSV, FormatYAML:
|
|
}
|
|
|
|
*f = Format(text)
|
|
return nil
|
|
}
|
|
|
|
// String returns the format value.
|
|
func (f Format) String() string {
|
|
return string(f)
|
|
}
|
|
|
|
// Output format values.
|
|
const (
|
|
FormatJSON Format = "json"
|
|
FormatJSONL Format = "jsonl"
|
|
FormatJSONA Format = "jsona"
|
|
FormatTable Format = "table"
|
|
FormatRaw Format = "raw"
|
|
FormatHTML Format = "html"
|
|
FormatMarkdown Format = "markdown"
|
|
FormatXLSX Format = "xlsx"
|
|
FormatXML Format = "xml"
|
|
FormatCSV Format = "csv"
|
|
FormatTSV Format = "tsv"
|
|
FormatYAML Format = "yaml"
|
|
)
|