1package config
2
3import (
4 "errors"
5 "fmt"
6 "io"
7 "os"
8 "path"
9 "path/filepath"
10 "strconv"
11
12 "git.gabrielgio.me/cerrado/pkg/u"
13 "git.sr.ht/~emersion/go-scfg"
14)
15
16var (
17 ScanPathErr = errors.New("Scan path does not exist")
18 RepoPathErr = errors.New("Repository path does not exist")
19 InvalidPropertyErr = errors.New("Invalid property")
20)
21
22type (
23
24 // scan represents piece of the scan from the configuration file.
25 scan struct {
26 Path string
27 Public bool
28 }
29
30 // configuration represents file configuration.
31 // fields needs to be exported to cmp to work
32 configuration struct {
33 Scan *scan
34 RootReadme string
35 ListenAddr string
36 Repositories []*GitRepositoryConfiguration
37 }
38
39 // This is a per repository configuration.
40 GitRepositoryConfiguration struct {
41 Name string
42 Path string
43 Description string
44 Public bool
45 }
46
47 // ConfigurationRepository represents the configuration repository (as in
48 // database repositories).
49 // This holds all the function necessary to ask for configuration
50 // information.
51 ConfigurationRepository struct {
52 rootReadme string
53 listenAddr string
54 repositories []*GitRepositoryConfiguration
55 }
56)
57
58func LoadConfigurationRepository(configPath string) (*ConfigurationRepository, error) {
59 f, err := os.Open(configPath)
60 if err != nil {
61 return nil, err
62 }
63
64 config, err := parse(f)
65 if err != nil {
66 return nil, err
67 }
68
69 repo := &ConfigurationRepository{
70 rootReadme: config.RootReadme,
71 listenAddr: config.ListenAddr,
72 repositories: config.Repositories,
73 }
74
75 if config.Scan.Path != "" {
76 err = repo.expandOnScanPath(config.Scan.Path, config.Scan.Public)
77 if err != nil {
78 return nil, err
79 }
80 }
81
82 return repo, nil
83
84}
85
86// GetRootReadme returns root read path
87func (c *ConfigurationRepository) GetRootReadme() string {
88 return c.rootReadme
89}
90
91func (c *ConfigurationRepository) GetListenAddr() string {
92 return c.listenAddr
93}
94
95// GetByName returns configuration of repository for a given name.
96// It returns nil if there is not match for it.
97func (c *ConfigurationRepository) GetByName(name string) *GitRepositoryConfiguration {
98 for _, r := range c.repositories {
99 if r.Name == name {
100 return r
101 }
102 }
103 return nil
104}
105
106// List returns all the configuration for all repositories.
107func (c *ConfigurationRepository) List() []*GitRepositoryConfiguration {
108 return c.repositories
109}
110
111// expandOnScanPath scans the scanPath for folders taking them as repositories
112// and applying them default configuration.
113func (c *ConfigurationRepository) expandOnScanPath(scanPath string, public bool) error {
114 if !u.FileExist(scanPath) {
115 return ScanPathErr
116 }
117
118 entries, err := os.ReadDir(scanPath)
119 if err != nil {
120 return err
121 }
122
123 for _, e := range entries {
124 if !e.IsDir() {
125 continue
126 }
127
128 fullPath := path.Join(scanPath, e.Name())
129 if !c.repoExits(fullPath) {
130 c.repositories = append(c.repositories, &GitRepositoryConfiguration{
131 Name: e.Name(),
132 Path: fullPath,
133 Public: public,
134 })
135 }
136 }
137 return nil
138}
139
140func (c *ConfigurationRepository) repoExits(path string) bool {
141 for _, r := range c.repositories {
142 if path == r.Path {
143 return true
144 }
145 }
146 return false
147}
148
149func parse(r io.Reader) (*configuration, error) {
150 block, err := scfg.Read(r)
151 if err != nil {
152 return nil, err
153 }
154
155 config := defaultConfiguration()
156
157 err = setScan(block, config.Scan)
158 if err != nil {
159 return nil, err
160 }
161
162 err = setRootReadme(block, &config.RootReadme)
163 if err != nil {
164 return nil, err
165 }
166
167 err = setListenAddr(block, &config.ListenAddr)
168 if err != nil {
169 return nil, err
170 }
171
172 err = setRepositories(block, &config.Repositories)
173 if err != nil {
174 return nil, err
175 }
176
177 return config, nil
178}
179
180func setRepositories(block scfg.Block, repositories *[]*GitRepositoryConfiguration) error {
181 blocks := block.GetAll("repository")
182
183 for _, r := range blocks {
184 if len(r.Params) != 1 {
185 return fmt.Errorf(
186 "Invlid number of params for repository: %w",
187 InvalidPropertyErr,
188 )
189 }
190
191 path := u.FirstOrZero(r.Params)
192 repository := defaultRepisotryConfiguration(path)
193
194 for _, d := range r.Children {
195 // under repository there is only single param properties
196 if len(d.Params) != 1 {
197 return fmt.Errorf(
198 "Invlid number of params for %s: %w",
199 d.Name,
200 InvalidPropertyErr,
201 )
202 }
203
204 switch d.Name {
205 case "name":
206 if err := setString(d, &repository.Name); err != nil {
207 return err
208 }
209 case "description":
210 if err := setString(d, &repository.Description); err != nil {
211 return err
212 }
213 case "public":
214 if err := setBool(d, &repository.Public); err != nil {
215 return err
216 }
217 }
218 }
219
220 *repositories = append(*repositories, repository)
221 }
222
223 return nil
224}
225
226func defaultConfiguration() *configuration {
227 return &configuration{
228 Scan: defaultScan(),
229 RootReadme: "",
230 ListenAddr: "http//0.0.0.0:8080",
231 Repositories: make([]*GitRepositoryConfiguration, 0),
232 }
233}
234
235func defaultScan() *scan {
236 return &scan{
237 Public: false,
238 Path: "",
239 }
240
241}
242
243func defaultRepisotryConfiguration(path string) *GitRepositoryConfiguration {
244 return &GitRepositoryConfiguration{
245 Path: path,
246 Name: filepath.Base(path),
247 Description: "",
248 Public: false,
249 }
250}
251
252func setRootReadme(block scfg.Block, readme *string) error {
253 scanDir := block.Get("root-readme")
254 return setString(scanDir, readme)
255}
256
257func setListenAddr(block scfg.Block, listenAddr *string) error {
258 scanDir := block.Get("listen-addr")
259 return setString(scanDir, listenAddr)
260}
261
262func setScan(block scfg.Block, scan *scan) error {
263 scanDir := block.Get("scan")
264 if scanDir == nil {
265 return nil
266 }
267 err := setString(scanDir, &scan.Path)
268 if err != nil {
269 return err
270 }
271
272 public := scanDir.Children.Get("public")
273 return setBool(public, &scan.Public)
274}
275
276func setBool(dir *scfg.Directive, field *bool) error {
277 if dir != nil {
278
279 p1, _ := u.First(dir.Params)
280 v, err := strconv.ParseBool(p1)
281 if err != nil {
282 return fmt.Errorf("Error parsing bool param of %s: %w", dir.Name, err)
283 }
284 *field = v
285 }
286 return nil
287}
288
289func setString(dir *scfg.Directive, field *string) error {
290 if dir != nil {
291 *field = u.FirstOrZero(dir.Params)
292 }
293 return nil
294}