-
-
Notifications
You must be signed in to change notification settings - Fork 102
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Feature idea: configuration file alternative to CLI | env arg #225
Comments
The approach I'm hoping to take to enable this is the one from #197. Need to get that PR up to date with all recent changes to this repo, and finish a few things before publishing it as "github.com/alexflint/go-arg/v2" |
Heya Alex! I've been doing a little exploration of the command-line parsing libraries in golang... I'm currently using urfave/cli for https://github.com/purpleidea/mgmt/ ... I've noticed their new "v3" API is generics ridden and really clunky. I thought there must be a better way... I found this project... ...And I think the "struct populating" approach is excellent and very close to what I was looking for! My two main concerns were:
I'd actually love to be more involved, but I'm fairly busy working on https://github.com/purpleidea/mgmt/ so, I don't expect too many hours. In any case, I just figured I'd check in, say that I like what you've been doing, and in case you're still hacking at your monastery, please keep it up! Cheers @SamuelMarks -- if you're hacking on adding your config work into core or helping with Alex's v2 approach and need a hand, please LMK. |
@purpleidea Actually after my #RewriteInRust I decided to #RewriteInC |
Thanks for the note @purpleidea. Care to jump on a call some time to say hi / chat? Any chance you're free at 3pm US Eastern time one day this week? @SamuelMarks would be great to meet you face-to-face at some point, too, though I know you're busy! |
Good idea! Send me an invite: my name at gmail.com |
Sure! I'm free either today or tomorrow at that time, otherwise TBD. Send me your preferred meeting thing, I've got signal, jitsi, gmeet or anything that works in a web browser. my username at gmail. Cheers! |
I've been thinking about how to pull values from config file into go-arg. Here's what I've come up with.
When the library is looking for a value to use, it calls that function with the key from the struct. If it gets a value back, it passes it through the usual parsing that we use for the That way everyone can have their own parser. And of course if we really want to, we can add one or two common ones as a util package. HTH |
In the next little while, I'd really like to be able to do this for https://github.com/purpleidea/mgmt/ ... I've landed all my new CLI patches, and they look great! Mostly here: purpleidea/mgmt@589a5f9 But also to show how other code then simplifies, look at: purpleidea/mgmt@d537c3d It's really an improvement! Thanks Alex for this great library! So instead of bolting config file reading onto mgmt, I'd rather bolt it onto this go-arg lib where it belongs. I would send a patch, but if that would conflict with your v2 patch, then maybe that generates more work for you which I don't want to do if you're not up for it. This patch also wouldn't need a v2 API change, so I think it's safe to add in before it. I also don't know how receptive to this idea you are and how many cycles you have to review and merge this. So please let me know how you'd like me to proceed. Thanks again! |
I needed a quick way to support config through a file and just used an ".env" file approach and env variables. To make this an easy add I just hacked the ".env" variables into my program using some simple load. Here is the initial code I generated using claude.ai, because I was feeling very lazy. package main
import (
"bufio"
"os"
"strings"
)
func init() {
loadEnv()
}
func loadEnv() {
file, err := os.Open(".env")
if err != nil {
// Silently ignore if the file doesn't exist
return
}
defer file.Close()
scanner := bufio.NewScanner(file)
for scanner.Scan() {
line := scanner.Text()
if equal := strings.Index(line, "="); equal >= 0 {
if key := strings.TrimSpace(line[:equal]); len(key) > 0 {
value := ""
if len(line) > equal {
value = strings.TrimSpace(line[equal+1:])
}
os.Setenv(key, value)
}
}
}
}
func main() {
var args struct {
Server string `arg:"env:GITEA_SERVER" default:"localhost:3000" help:"gitea server address"`
UserName string `arg:"required,env:GITEA_USERNAME" help:"gitea user name"`
Password string `arg:"required,env:GITEA_PASSWORD" help:"gitea password"`
}
arg.MustParse(&args)
// Your program logic here
} |
Another workaround would involve using a configuration management library like Viper, which can handle config file unmarshalling and integrate with Below is one such example: type App struct {
Writer io.Writer `mapstructure:"-" arg:"-"` // ignore in flags
Message string `arg:"-m,--message" default:"World" help:"recipient"`
Times int `arg:"-t,--times" default:"1" help:"number of times to print the message"`
}
// initApp initializes the app using Viper (for config files)
// and go-arg (for flags and environment variables).
func initApp(app *App) error {
// Init Viper
v := viper.New()
v.SetConfigName("config") // specify the config file name (without extension)
v.AddConfigPath(".") // look for config in the current directory
// Read config file, ignore missing file error
if err := v.ReadInConfig(); err != nil {
if _, ok := err.(viper.ConfigFileNotFoundError); !ok {
return fmt.Errorf("error reading config file: %w", err)
}
}
// Unmarshal config values into the app struct
if err := v.Unmarshal(app); err != nil {
return fmt.Errorf("error unmarshaling config: %w", err)
}
// Parse command-line flags and environment variables
p, err := arg.NewParser(arg.Config{EnvPrefix: "MYAPP"}, app)
if err != nil {
return fmt.Errorf("error creating parser: %w", err)
}
// Parse flags and env variables
if err := p.Parse(os.Args[1:]); err != nil {
if err == arg.ErrHelp {
p.WriteHelp(os.Stdout)
os.Exit(0)
}
return fmt.Errorf("error parsing arguments: %w", err)
}
return nil
} |
I added config file support to my go-arg using project but due to #94 I could not determine when to read the config.
My project hacks support with the builtin json package. Specifically here: https://github.com/offscale/postgres-version-manager-go/blob/master/pvm/config_utils.go (with go-arg annotated
struct
here: config.go)Any chance we can get support for a configuration file in your project directly?
Thanks
EDIT: Got it working in my latest commit
48cba8
but it was a major hack; here's a taste [from that commit]:Issues with this design include:
The text was updated successfully, but these errors were encountered: