mirror of
https://github.com/rclone/rclone.git
synced 2025-08-17 17:11:37 +02:00
fs: create fs.Enum for easy creation of parameters from a list of choices
This commit is contained in:
107
fs/enum.go
Normal file
107
fs/enum.go
Normal file
@ -0,0 +1,107 @@
|
||||
package fs
|
||||
|
||||
import (
|
||||
"encoding/json"
|
||||
"fmt"
|
||||
"strings"
|
||||
)
|
||||
|
||||
// Enum is an option which can only be one of the Choices.
|
||||
//
|
||||
// Suggested implementation is something like this:
|
||||
//
|
||||
// type choice = Enum[choices]
|
||||
//
|
||||
// const (
|
||||
// choiceA choice = iota
|
||||
// choiceB
|
||||
// choiceC
|
||||
// )
|
||||
//
|
||||
// type choices struct{}
|
||||
//
|
||||
// func (choices) Choices() []string {
|
||||
// return []string{
|
||||
// choiceA: "A",
|
||||
// choiceB: "B",
|
||||
// choiceC: "C",
|
||||
// }
|
||||
// }
|
||||
type Enum[C Choices] byte
|
||||
|
||||
// Choices returns the valid choices for this type.
|
||||
//
|
||||
// It must work on the zero value.
|
||||
//
|
||||
// Note that when using this in an Option the ExampleChoices will be
|
||||
// filled in automatically.
|
||||
type Choices interface {
|
||||
// Choices returns the valid choices for this type
|
||||
Choices() []string
|
||||
}
|
||||
|
||||
// String renders the Enum as a string
|
||||
func (e Enum[C]) String() string {
|
||||
choices := e.Choices()
|
||||
if int(e) >= len(choices) {
|
||||
return fmt.Sprintf("Unknown(%d)", e)
|
||||
}
|
||||
return choices[e]
|
||||
}
|
||||
|
||||
// Choices returns the possible values of the Enum.
|
||||
func (e Enum[C]) Choices() []string {
|
||||
var c C
|
||||
return c.Choices()
|
||||
}
|
||||
|
||||
// Help returns a comma separated list of all possible states.
|
||||
func (e Enum[C]) Help() string {
|
||||
return strings.Join(e.Choices(), ", ")
|
||||
}
|
||||
|
||||
// Set the Enum entries
|
||||
func (e *Enum[C]) Set(s string) error {
|
||||
for i, choice := range e.Choices() {
|
||||
if strings.EqualFold(s, choice) {
|
||||
*e = Enum[C](i)
|
||||
return nil
|
||||
}
|
||||
}
|
||||
return fmt.Errorf("invalid choice %q from: %s", s, e.Help())
|
||||
}
|
||||
|
||||
// Type of the value.
|
||||
//
|
||||
// If C has a Type() string method then it will be used instead.
|
||||
func (e Enum[C]) Type() string {
|
||||
var c C
|
||||
if do, ok := any(c).(typer); ok {
|
||||
return do.Type()
|
||||
}
|
||||
return strings.Join(e.Choices(), "|")
|
||||
}
|
||||
|
||||
// Scan implements the fmt.Scanner interface
|
||||
func (e *Enum[C]) Scan(s fmt.ScanState, ch rune) error {
|
||||
token, err := s.Token(true, nil)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
return e.Set(string(token))
|
||||
}
|
||||
|
||||
// UnmarshalJSON parses it as a string
|
||||
func (e *Enum[C]) UnmarshalJSON(in []byte) error {
|
||||
var choice string
|
||||
err := json.Unmarshal(in, &choice)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
return e.Set(choice)
|
||||
}
|
||||
|
||||
// MarshalJSON encodes it as string
|
||||
func (e *Enum[C]) MarshalJSON() ([]byte, error) {
|
||||
return json.Marshal(e.String())
|
||||
}
|
Reference in New Issue
Block a user