summaryrefslogtreecommitdiffstats
path: root/main.go
blob: bc74af071ff864cf7b4776e97f7085be74227ac7 (plain) (blame)
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
package gopkgserver

import (
	"context"
	_ "embed"
	"fmt"
	"io/fs"
	"strconv"

	"log/slog"
	"net/http"
	"os"

	"github.com/pelletier/go-toml/v2"
	"go.sudomsg.com/gopkgserver/repo"
	httpServer "go.sudomsg.com/kit/http"
	"go.sudomsg.com/kit/logging"
	logHandler "go.sudomsg.com/kit/logging/http"
	"go.sudomsg.com/kit/runner"
)

type ServerConfig struct {
	Network httpServer.NetType `toml:"network"`
	Address string             `toml:"address"`
	Mode    string             `toml:"mode"`
}

func RawServerConfig(rawCfg []ServerConfig) ([]httpServer.ServerConfig, error) {
	cfg := make([]httpServer.ServerConfig, 0, len(rawCfg))
	for _, srv := range rawCfg {
		var mode uint64
		if srv.Mode != "" {
			var err error
			mode, err = strconv.ParseUint(srv.Mode, 8, 32)
			if err != nil {
				return nil, fmt.Errorf("failed to convert mode %s: %w", srv.Mode, err)
			}
		}
		cfg = append(cfg, httpServer.ServerConfig{
			Network: srv.Network,
			Address: srv.Address,
			Mode:    fs.FileMode(mode),
		})
	}
	return cfg, nil
}

type Config struct {
	Server  []ServerConfig       `toml:"server"`
	Logging logging.LogConfig    `toml:"logging"`
	Repos   map[string]repo.Repo `toml:"repo"`
}

func Run(ctx context.Context, fs *runner.CommandSet, args []string) error {
	var cfgFile string
	fs.StringVar(&cfgFile, "config", "config.toml", "Path to config file")
	
	if err := fs.Parse(args); err != nil{
		return err
	}

	cfg, err := LoadConfig(cfgFile)
	if err != nil {
		return err
	}

	if err := logging.Setup(cfg.Logging); err != nil {
		return err
	}
	ctx = logging.WithLogger(ctx, slog.Default())

	h, err := repo.New(cfg.Repos)
	if err != nil {
		return fmt.Errorf(": %v", err)
	}

	mux := http.NewServeMux()
	mux.Handle("GET /robots.txt", Robot())
	mux.Handle("GET /", h)

	handler := logHandler.New(mux)

	srvCfg, err := RawServerConfig(cfg.Server)
	if err != nil {
		return err
	}

	lns, err := httpServer.OpenListeners(ctx, srvCfg)
	if err != nil {
		return err
	}
	defer lns.CloseAll()

	return httpServer.RunHTTPServers(ctx, lns, handler)
}

func LoadConfig(cfgFile string) (Config, error) {
	data, err := os.ReadFile(cfgFile)
	if err != nil {
		return Config{}, fmt.Errorf("read config: %w", err)
	}

	var cfg Config
	if err := toml.Unmarshal(data, &cfg); err != nil {
		return Config{}, fmt.Errorf("parse config: %w", err)
	}

	return cfg, nil
}