2022-12-19 03:04:34 +00:00
// Copyright © 2022 Roberto Hidalgo <chinampa@un.rob.mx>
2022-12-31 05:53:24 +00:00
// SPDX-License-Identifier: Apache-2.0
2022-12-19 03:04:34 +00:00
package command
import (
"fmt"
"strconv"
"strings"
2022-12-31 05:53:24 +00:00
"git.rob.mx/nidito/chinampa/pkg/errors"
2022-12-19 03:04:34 +00:00
"github.com/spf13/cobra"
"github.com/spf13/pflag"
)
// Options is a map of name to Option.
type Options map [ string ] * Option
func ( opts * Options ) AllKnown ( ) map [ string ] any {
col := map [ string ] any { }
for name , opt := range * opts {
col [ name ] = opt . ToValue ( )
}
return col
}
func ( opts * Options ) AllKnownStr ( ) map [ string ] string {
col := map [ string ] string { }
for name , opt := range * opts {
col [ name ] = opt . ToString ( )
}
return col
}
func ( opts * Options ) Parse ( supplied * pflag . FlagSet ) {
// logrus.Debugf("Parsing supplied flags, %v", supplied)
for name , opt := range * opts {
switch opt . Type {
case ValueTypeBoolean :
if val , err := supplied . GetBool ( name ) ; err == nil {
opt . provided = val
continue
}
2022-12-31 05:53:24 +00:00
case ValueTypeInt :
if val , err := supplied . GetInt ( name ) ; err == nil {
opt . provided = val
continue
}
2022-12-19 03:04:34 +00:00
default :
opt . Type = ValueTypeString
if val , err := supplied . GetString ( name ) ; err == nil {
opt . provided = val
continue
}
}
}
}
func ( opts * Options ) AreValid ( ) error {
for name , opt := range * opts {
if err := opt . Validate ( name ) ; err != nil {
return err
}
}
return nil
}
// Option represents a command line flag.
type Option struct {
ShortName string ` json:"short-name,omitempty" yaml:"short-name,omitempty" ` // nolint:tagliatelle
2022-12-31 05:53:24 +00:00
Type ValueType ` json:"type" yaml:"type" validate:"omitempty,oneof=string bool int" `
2022-12-19 03:04:34 +00:00
Description string ` json:"description" yaml:"description" validate:"required" `
Default any ` json:"default,omitempty" yaml:"default,omitempty" `
Values * ValueSource ` json:"values,omitempty" yaml:"values,omitempty" validate:"omitempty" `
Repeated bool ` json:"repeated" yaml:"repeated" validate:"omitempty" `
Command * Command ` json:"-" yaml:"-" validate:"-" `
provided any
}
func ( opt * Option ) IsKnown ( ) bool {
return opt . provided != nil
}
func ( opt * Option ) ToValue ( ) any {
if opt . IsKnown ( ) {
return opt . provided
}
return opt . Default
}
func ( opt * Option ) ToString ( ) string {
value := opt . ToValue ( )
stringValue := ""
2022-12-31 05:53:24 +00:00
switch opt . Type {
case ValueTypeBoolean :
2022-12-19 03:04:34 +00:00
if value == nil {
stringValue = ""
} else {
stringValue = strconv . FormatBool ( value . ( bool ) )
}
2022-12-31 05:53:24 +00:00
case ValueTypeInt :
if value == nil {
stringValue = ""
} else {
stringValue = fmt . Sprintf ( "%i" , value )
}
default :
2022-12-19 03:04:34 +00:00
if value != nil {
stringValue = value . ( string )
}
}
return stringValue
}
func ( opt * Option ) Validate ( name string ) error {
if ! opt . Validates ( ) {
return nil
}
current := opt . ToString ( ) // nolint:ifshort
if current == "" {
return nil
}
validValues , _ , err := opt . Resolve ( current )
if err != nil {
return err
}
if ! contains ( validValues , current ) {
return errors . BadArguments { Msg : fmt . Sprintf ( "%s is not a valid value for option <%s>. Valid options are: %s" , current , name , strings . Join ( validValues , ", " ) ) }
}
return nil
}
// Validates tells if the user-supplied value needs validation.
func ( opt * Option ) Validates ( ) bool {
return opt . Values != nil && opt . Values . Validates ( )
}
// providesAutocomplete tells if this option provides autocomplete values.
func ( opt * Option ) providesAutocomplete ( ) bool {
return opt . Values != nil
}
// Resolve returns autocomplete values for an option.
func ( opt * Option ) Resolve ( currentValue string ) ( values [ ] string , flag cobra . ShellCompDirective , err error ) {
if opt . Values != nil {
if opt . Values . command == nil {
opt . Values . command = opt . Command
}
return opt . Values . Resolve ( currentValue )
}
return
}
// CompletionFunction is called by cobra when asked to complete an option.
func ( opt * Option ) CompletionFunction ( cmd * cobra . Command , args [ ] string , toComplete string ) ( values [ ] string , flag cobra . ShellCompDirective ) {
if ! opt . providesAutocomplete ( ) {
flag = cobra . ShellCompDirectiveNoFileComp
return
}
opt . Command . Arguments . Parse ( args )
opt . Command . Options . Parse ( cmd . Flags ( ) )
var err error
values , flag , err = opt . Resolve ( toComplete )
if err != nil {
return values , cobra . ShellCompDirectiveError
}
2023-01-14 23:10:59 +00:00
if toComplete != "" && flag != cobra . ShellCompDirectiveFilterFileExt && flag != cobra . ShellCompDirectiveFilterDirs {
2022-12-19 03:04:34 +00:00
filtered := [ ] string { }
for _ , value := range values {
if strings . HasPrefix ( value , toComplete ) {
filtered = append ( filtered , value )
}
}
values = filtered
}
return cobra . AppendActiveHelp ( values , opt . Description ) , flag
}