summaryrefslogtreecommitdiff
path: root/config.go
blob: 99ba64d591f15a4ddffac54f2c11d131a2173f89 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
package gitpb

// functions to import and export the protobuf
// data to and from config files

import (
	"errors"
	"os"
	"path/filepath"

	"go.wit.com/log"
)

// write to ~/.config/forge/ unless ENV{FORGE_CONFIG} is set
func (all *Repos) ConfigSave() error {
	if os.Getenv("FORGE_CONFIG") == "" {
		homeDir, _ := os.UserHomeDir()
		fullpath := filepath.Join(homeDir, ".config/forge")
		os.Setenv("FORGE_CONFIG", fullpath)
	}
	if all == nil {
		log.Warn("gitpb all == nil")
		return errors.New("gitpb.ConfigSave() all == nil")
	}

	data, err := all.Marshal()
	if err != nil {
		log.Info("gitpb proto.Marshal() failed len", len(data), err)
		return err
	}
	log.Info("gitpb.ConfigSave() repos.Marshal() worked len", len(data))
	configWrite("repos.pb", data)
	return nil
}

// load the ~/.config/forge/ files
func (all *Repos) ConfigLoad() error {
	if os.Getenv("FORGE_CONFIG") == "" {
		homeDir, _ := os.UserHomeDir()
		fullpath := filepath.Join(homeDir, ".config/forge")
		os.Setenv("FORGE_CONFIG", fullpath)
	}
	var data []byte
	var err error

	if data, err = loadFile("repos.pb"); err != nil {
		// something went wrong loading the file
		return err
	}
	if data != nil {
		// this means the forge.pb file exists and was read
		if len(data) == 0 {
			// todo: error out if the file is empty?
			// try forge.text & forge.json?
			log.Warn("gitpb.ConfigLoad() repos.pb is empty")
			return errors.New("gitpb.ConfigLoad() repos.pb is empty")
		}
		if all.Repos == nil {
			log.Warn("gitpb.ConfigLoad() all.Repos == nil")
		} else {
			log.Warn("gitpb.ConfigLoad() all.Repos.Len()", all.Len())
		}
		if err = all.Unmarshal(data); err != nil {
			log.Warn("gitpb.ConfigLoad() failed", err)
			if all.Repos == nil {
				log.Warn("gitpb.ConfigLoad() all.Repos == nil")
			} else {
				log.Warn("gitpb.ConfigLoad() all.Repos.Len()", all.Len())
				log.Warn("gitpb.ConfigLoad() trying to resave the file")
				all.ConfigSave()
			}
			return err
		}
		log.Info("gitpb.Init()", len(all.Repos), "repos in ~/.config/forge/repos.pb")
		return nil
	}
	return nil
}

func loadFile(filename string) ([]byte, error) {
	fullname := filepath.Join(os.Getenv("FORGE_CONFIG"), filename)
	data, err := os.ReadFile(fullname)
	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
	}
	if err != nil {
		// log.Info("open config file :", err)
		return nil, err
	}
	return data, nil
}

func configWrite(filename string, data []byte) error {
	fullname := filepath.Join(os.Getenv("FORGE_CONFIG"), filename)

	cfgfile, err := os.OpenFile(fullname, os.O_RDWR|os.O_CREATE, 0666)
	defer cfgfile.Close()
	if err != nil {
		log.Warn("open config file :", err)
		return err
	}
	if filename == "forge.text" {
		// add header
		cfgfile.Write([]byte("# this file is automatically re-generated from forge.pb, however,\n"))
		cfgfile.Write([]byte("# if you want to edit it by hand, you can:\n"))
		cfgfile.Write([]byte("# stop forge; remove forge.pb; edit forge.text; start forge\n"))
		cfgfile.Write([]byte("# this will cause the default behavior to fallback to parsing this file for the config\n"))
		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(data)
	return nil
}