forgepb/config.go

173 lines
4.8 KiB
Go
Raw Normal View History

2024-11-20 09:31:24 -06:00
package forgepb
// functions to import and export the protobuf
// data to and from config files
import (
"errors"
2025-01-05 01:20:12 -06:00
"fmt"
2024-11-20 09:31:24 -06:00
"os"
"path/filepath"
"go.wit.com/log"
)
2024-11-28 00:02:27 -06:00
func (f *Forge) ConfigSave() error {
2024-12-05 12:37:07 -06:00
var err error
// backup the current config files
if e := backupConfig(); e != nil {
log.Info("forge.BackupConfig() error", e)
err = e
// continue here? notsure. could be bad either way
// out of disk space?
}
if f.Config != nil {
if e := f.Config.ConfigSave(); e != nil {
log.Info("forge.Config.ConfigSave() error", e)
err = e
}
2024-11-20 11:02:41 -06:00
}
2024-12-05 12:37:07 -06:00
if f.Repos != nil {
if e := f.Repos.ConfigSave(); e != nil {
log.Info("forge.Repos.ConfigSave() error", e)
err = e
}
2024-11-20 11:02:41 -06:00
}
2024-12-05 12:37:07 -06:00
return err
}
// write to ~/.config/forge/ unless ENV{FORGE_CONFIG} is set
func (f *ForgeConfigs) ConfigSave() error {
data, err := f.Marshal()
2024-11-20 09:31:24 -06:00
if err != nil {
log.Info("proto.Marshal() failed len", len(data), err)
return err
}
2024-12-01 10:44:29 -06:00
log.Info("forgepb.ConfigSave() proto.Marshal() worked len", len(data))
2024-11-20 09:31:24 -06:00
2024-12-05 12:37:07 -06:00
s := f.FormatTEXT()
2024-11-20 09:31:24 -06:00
configWrite("forge.text", []byte(s))
2024-11-20 11:02:41 -06:00
2024-12-05 12:37:07 -06:00
s = f.FormatJSON()
2024-11-20 11:02:41 -06:00
configWrite("forge.json", []byte(s))
2024-12-01 10:44:29 -06:00
2024-11-20 09:31:24 -06:00
return nil
}
2024-11-21 10:27:40 -06:00
// load the ~/.config/forge/ files
func (c *ForgeConfigs) ConfigLoad() error {
2024-12-05 12:37:07 -06:00
if os.Getenv("FORGE_CONFIG") == "" {
2024-11-20 12:11:13 -06:00
homeDir, _ := os.UserHomeDir()
fullpath := filepath.Join(homeDir, ".config/forge")
2024-12-05 12:37:07 -06:00
os.Setenv("FORGE_CONFIG", fullpath)
2024-11-20 12:11:13 -06:00
}
2024-12-05 12:37:07 -06:00
// var data []byte
// var err error
2024-11-21 10:27:40 -06:00
if c == nil {
2024-11-27 21:21:06 -06:00
// can't safely do c = new(ForgeConfig) if c is in a struct from the caller. notsure why
2025-01-05 01:20:12 -06:00
// TODO: recheck this. it might work now? It's probably still a bad idea(?)
2024-11-21 10:27:40 -06:00
return errors.New("It's not safe to run ConfigLoad() on a nil")
2024-11-20 09:31:24 -06:00
}
2025-01-05 01:20:12 -06:00
if err := c.loadText(); err == nil {
return nil
2024-11-20 12:11:13 -06:00
}
// forge.text doesn't exist. try forge.json
2024-12-27 14:37:20 -06:00
// this lets the user hand edit the JSON config
// probably just deprecate this
2024-12-05 12:37:07 -06:00
if data, err := loadFile("forge.json"); err != nil {
if data != nil {
// this means the forge.json file exists and was read
if len(data) != 0 {
if err = c.UnmarshalJSON(data); err == nil {
log.Info("forge.ConfigLoad()", len(c.ForgeConfigs), "entries in ~/.config/forge")
2025-01-05 01:20:12 -06:00
// forge.text file was broken. save on load right away
2024-12-05 12:37:07 -06:00
log.Info("attempting forge.ConfigSave()")
c.ConfigSave()
return nil
}
}
}
2024-11-20 12:11:13 -06:00
}
2024-12-05 12:37:07 -06:00
cpath := filepath.Join(os.Getenv("FORGE_CONFIG"), ".")
if _, err := os.Stat(cpath); err == nil {
log.Info("Something has gone wrong. Your", os.Getenv("FORGE_CONFIG"), "directory exists")
log.Info("However, the config files could not be loaded")
2024-11-20 10:31:25 -06:00
}
// first time user. make a template config file
2025-01-08 04:08:12 -06:00
c.sampleConfig()
2025-01-05 01:20:12 -06:00
return nil
}
2024-11-21 10:27:40 -06:00
2025-01-05 01:20:12 -06:00
func (c *ForgeConfigs) loadText() error {
// this lets the user hand edit the config
data, err := loadFile("forge.text")
if err != nil {
return err
}
if data == nil {
return fmt.Errorf("forge.text data was nil")
}
if len(data) == 0 {
return fmt.Errorf("forge.text was empty")
}
// attempt to marshal forge.text
if err := c.UnmarshalTEXT(data); err != nil {
return err
}
2025-01-08 04:08:12 -06:00
log.Log(INFO, "forge.ConfigLoad()", len(c.ForgeConfigs), "entries in ~/.config/forge")
2024-11-20 09:31:24 -06:00
return nil
}
func loadFile(filename string) ([]byte, error) {
2024-12-05 12:37:07 -06:00
fullname := filepath.Join(os.Getenv("FORGE_CONFIG"), filename)
2024-11-20 09:31:24 -06:00
data, err := os.ReadFile(fullname)
2024-11-20 10:31:25 -06:00
if errors.Is(err, os.ErrNotExist) {
// if file does not exist, just return nil. this
// will cause ConfigLoad() to try the next config file like "forge.text"
// because the user might want to edit the .config by hand
return nil, nil
}
2024-11-20 09:31:24 -06:00
if err != nil {
// log.Info("open config file :", err)
return nil, err
}
return data, nil
}
func configWrite(filename string, data []byte) error {
2024-12-05 12:37:07 -06:00
fullname := filepath.Join(os.Getenv("FORGE_CONFIG"), filename)
2024-11-20 11:02:41 -06:00
2024-12-11 13:55:21 -06:00
cfgfile, err := os.OpenFile(fullname, os.O_RDWR|os.O_CREATE|os.O_TRUNC, 0644)
2024-11-20 09:31:24 -06:00
defer cfgfile.Close()
if err != nil {
log.Warn("open config file :", err)
return err
}
2024-11-20 23:51:28 -06:00
if filename == "forge.text" {
// add header
cfgfile.Write([]byte("\n"))
cfgfile.Write([]byte("# this file is intended to be used to customize settings on what\n"))
cfgfile.Write([]byte("# git repos you have write access to. That is, where you can run 'git push'\n"))
2024-12-27 14:37:20 -06:00
cfgfile.Write([]byte("\n"))
}
if filename == "forge.json" {
// add header
cfgfile.Write([]byte("\n"))
cfgfile.Write([]byte("# this file is intended to be used to customize settings on what\n"))
cfgfile.Write([]byte("# git repos you have write access to. That is, where you can run 'git push'\n"))
cfgfile.Write([]byte("\n"))
cfgfile.Write([]byte("# this file is parsed only if forge.text is missing\n"))
cfgfile.Write([]byte("# also, these comment lines don't work in json files and have to be removed for Marshal() to work\n"))
cfgfile.Write([]byte("# probably, JSON syntax for this is just going to be deprecated for the TEXT syntax\n"))
cfgfile.Write([]byte("\n"))
2024-11-20 23:51:28 -06:00
}
2024-11-20 09:31:24 -06:00
cfgfile.Write(data)
return nil
}