mirror of https://github.com/databricks/cli.git
111 lines
1.9 KiB
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
|
||
|
}
|