databricks-cli/libs/flags/log_file_flag.go

111 lines
1.9 KiB
Go

package flags
import (
"io"
"os"
"strings"
"github.com/spf13/cobra"
)
// Abstract over files that are already open (e.g. stderr) and
// files that need to be opened before use.
type logFile interface {
Writer() io.Writer
Open() error
Close() error
}
// nopLogFile implements [logFile] for [os.Stderr] and [os.Stdout].
// The [logFile.Open] and [logFile.Close] functions do nothing.
type nopLogFile struct {
f *os.File
}
func (f *nopLogFile) Writer() io.Writer {
return f.f
}
func (f *nopLogFile) Open() error {
return nil
}
func (f *nopLogFile) Close() error {
return nil
}
// nopLogFile implements [logFile] for actual files.
type realLogFile struct {
s string
f *os.File
}
func (f *realLogFile) Writer() io.Writer {
if f.f == nil {
panic("file hasn't been opened")
}
return f.f
}
func (f *realLogFile) Open() error {
file, err := os.OpenFile(f.s, os.O_CREATE|os.O_WRONLY|os.O_APPEND, 0o600)
if err != nil {
return err
}
f.f = file
return nil
}
func (f *realLogFile) Close() error {
if f.f == nil {
return nil
}
return f.f.Close()
}
type LogFileFlag struct {
name string
logFile
}
func NewLogFileFlag() LogFileFlag {
return LogFileFlag{
name: "stderr",
logFile: &nopLogFile{os.Stderr},
}
}
func (f *LogFileFlag) String() string {
return f.name
}
func (f *LogFileFlag) Set(s string) error {
lower := strings.ToLower(s)
switch lower {
case "stderr":
f.name = lower
f.logFile = &nopLogFile{os.Stderr}
case "stdout":
f.name = lower
f.logFile = &nopLogFile{os.Stdout}
default:
f.name = s
f.logFile = &realLogFile{s: s}
}
return nil
}
func (f *LogFileFlag) Type() string {
return "file"
}
// Complete is the Cobra compatible completion function for this flag.
func (f *LogFileFlag) Complete(cmd *cobra.Command, args []string, toComplete string) ([]string, cobra.ShellCompDirective) {
return []string{
"stdout",
"stderr",
}, cobra.ShellCompDirectiveDefault
}