2025-12-08 09:49:07 +01:00
|
|
|
package config
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"fmt"
|
|
|
|
|
"os"
|
|
|
|
|
"path/filepath"
|
2026-03-23 12:39:51 -06:00
|
|
|
"strings"
|
2025-12-08 09:49:07 +01:00
|
|
|
|
|
|
|
|
"github.com/spf13/viper"
|
|
|
|
|
"gopkg.in/yaml.v3"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
type Config struct {
|
|
|
|
|
Hosts map[string]HostConfig `yaml:"hosts"`
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
type HostConfig struct {
|
2026-03-23 12:39:51 -06:00
|
|
|
Hostname string `yaml:"hostname"`
|
|
|
|
|
Token string `yaml:"token"`
|
|
|
|
|
User string `yaml:"user,omitempty"`
|
|
|
|
|
GitProtocol string `yaml:"git_protocol,omitempty"`
|
|
|
|
|
MatchDirs []string `yaml:"match_dirs,omitempty"`
|
2025-12-08 09:49:07 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func GetConfigDir() (string, error) {
|
2026-03-12 15:44:24 +01:00
|
|
|
if xdgConfigHome := os.Getenv("XDG_CONFIG_HOME"); xdgConfigHome != "" {
|
|
|
|
|
return filepath.Join(xdgConfigHome, "fgj"), nil
|
|
|
|
|
}
|
2025-12-08 09:49:07 +01:00
|
|
|
home, err := os.UserHomeDir()
|
|
|
|
|
if err != nil {
|
|
|
|
|
return "", err
|
|
|
|
|
}
|
|
|
|
|
return filepath.Join(home, ".config", "fgj"), nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func GetConfigPath() (string, error) {
|
|
|
|
|
dir, err := GetConfigDir()
|
|
|
|
|
if err != nil {
|
|
|
|
|
return "", err
|
|
|
|
|
}
|
|
|
|
|
return filepath.Join(dir, "config.yaml"), nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func Load() (*Config, error) {
|
|
|
|
|
path, err := GetConfigPath()
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
2025-12-08 10:14:47 +01:00
|
|
|
return LoadFromPath(path)
|
|
|
|
|
}
|
2025-12-08 09:49:07 +01:00
|
|
|
|
2025-12-08 10:14:47 +01:00
|
|
|
func LoadFromPath(path string) (*Config, error) {
|
2025-12-08 09:49:07 +01:00
|
|
|
data, err := os.ReadFile(path)
|
|
|
|
|
if err != nil {
|
|
|
|
|
if os.IsNotExist(err) {
|
|
|
|
|
return &Config{Hosts: make(map[string]HostConfig)}, nil
|
|
|
|
|
}
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var cfg Config
|
|
|
|
|
if err := yaml.Unmarshal(data, &cfg); err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if cfg.Hosts == nil {
|
|
|
|
|
cfg.Hosts = make(map[string]HostConfig)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return &cfg, nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (c *Config) Save() error {
|
|
|
|
|
path, err := GetConfigPath()
|
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
2025-12-08 10:14:47 +01:00
|
|
|
return c.SaveToPath(path)
|
|
|
|
|
}
|
2025-12-08 09:49:07 +01:00
|
|
|
|
2025-12-08 10:14:47 +01:00
|
|
|
func (c *Config) SaveToPath(path string) error {
|
2025-12-08 09:49:07 +01:00
|
|
|
dir := filepath.Dir(path)
|
|
|
|
|
if err := os.MkdirAll(dir, 0755); err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
data, err := yaml.Marshal(c)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return os.WriteFile(path, data, 0600)
|
|
|
|
|
}
|
|
|
|
|
|
2026-01-05 12:47:28 +01:00
|
|
|
// GetHost resolves the hostname to use for API client creation.
|
|
|
|
|
// Priority order:
|
|
|
|
|
// 1. Explicitly provided hostname parameter
|
|
|
|
|
// 2. CLI flag (--hostname)
|
|
|
|
|
// 3. Environment variable (FGJ_HOST)
|
|
|
|
|
// 4. Auto-detected hostname from git remote
|
2026-03-23 12:39:51 -06:00
|
|
|
// 5. match_dirs lookup (longest prefix match)
|
|
|
|
|
// 6. Default to codeberg.org
|
|
|
|
|
func (c *Config) GetHost(hostname string, detectedHost string, cwd string) (HostConfig, error) {
|
2025-12-08 09:49:07 +01:00
|
|
|
if hostname == "" {
|
|
|
|
|
hostname = viper.GetString("hostname")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if hostname == "" {
|
|
|
|
|
hostname = os.Getenv("FGJ_HOST")
|
|
|
|
|
}
|
|
|
|
|
|
2026-01-05 12:47:28 +01:00
|
|
|
if hostname == "" {
|
|
|
|
|
hostname = detectedHost
|
|
|
|
|
}
|
|
|
|
|
|
2026-03-23 12:39:51 -06:00
|
|
|
if hostname == "" {
|
|
|
|
|
hostname = c.ResolveHostByPath(cwd)
|
|
|
|
|
}
|
|
|
|
|
|
2025-12-08 09:49:07 +01:00
|
|
|
if hostname == "" {
|
|
|
|
|
hostname = "codeberg.org"
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
host, ok := c.Hosts[hostname]
|
|
|
|
|
if !ok {
|
|
|
|
|
return HostConfig{}, fmt.Errorf("no configuration found for host %s", hostname)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return host, nil
|
|
|
|
|
}
|
|
|
|
|
|
2026-03-23 12:39:51 -06:00
|
|
|
// ResolveHostByPath finds the host whose match_dirs entry is the longest
|
|
|
|
|
// prefix of cwd. Returns "" if no match is found.
|
|
|
|
|
// Both cwd and match_dirs entries are resolved through filepath.EvalSymlinks
|
|
|
|
|
// to handle symlinks (e.g. macOS /tmp → /private/tmp).
|
|
|
|
|
func (c *Config) ResolveHostByPath(cwd string) string {
|
|
|
|
|
if cwd == "" {
|
|
|
|
|
return ""
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Resolve symlinks in cwd so /tmp becomes /private/tmp on macOS, etc.
|
|
|
|
|
if resolved, err := filepath.EvalSymlinks(cwd); err == nil {
|
|
|
|
|
cwd = resolved
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bestHost := ""
|
|
|
|
|
bestLen := 0
|
|
|
|
|
|
|
|
|
|
for hostname, host := range c.Hosts {
|
|
|
|
|
for _, dir := range host.MatchDirs {
|
|
|
|
|
if dir == "" {
|
|
|
|
|
continue
|
|
|
|
|
}
|
2026-03-23 12:50:49 -06:00
|
|
|
// Expand ~ to home directory
|
|
|
|
|
dir = expandHome(dir)
|
2026-03-23 12:39:51 -06:00
|
|
|
// Resolve symlinks in the configured dir as well
|
|
|
|
|
if resolved, err := filepath.EvalSymlinks(dir); err == nil {
|
|
|
|
|
dir = resolved
|
|
|
|
|
}
|
|
|
|
|
// Normalize: ensure trailing slash for prefix matching
|
|
|
|
|
prefix := dir
|
|
|
|
|
if !strings.HasSuffix(prefix, "/") {
|
|
|
|
|
prefix += "/"
|
|
|
|
|
}
|
|
|
|
|
// Match if cwd equals dir exactly or is under it
|
|
|
|
|
if cwd == dir || strings.HasPrefix(cwd, prefix) {
|
|
|
|
|
if len(dir) > bestLen {
|
|
|
|
|
bestLen = len(dir)
|
|
|
|
|
bestHost = hostname
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return bestHost
|
|
|
|
|
}
|
|
|
|
|
|
2026-03-23 12:50:49 -06:00
|
|
|
// expandHome replaces a leading ~ with the user's home directory.
|
|
|
|
|
func expandHome(path string) string {
|
|
|
|
|
if path == "~" || strings.HasPrefix(path, "~/") {
|
|
|
|
|
home, err := os.UserHomeDir()
|
|
|
|
|
if err != nil {
|
|
|
|
|
return path
|
|
|
|
|
}
|
|
|
|
|
return filepath.Join(home, path[1:])
|
|
|
|
|
}
|
|
|
|
|
return path
|
|
|
|
|
}
|
|
|
|
|
|
2025-12-08 09:49:07 +01:00
|
|
|
func (c *Config) SetHost(hostname string, host HostConfig) {
|
|
|
|
|
if c.Hosts == nil {
|
|
|
|
|
c.Hosts = make(map[string]HostConfig)
|
|
|
|
|
}
|
|
|
|
|
c.Hosts[hostname] = host
|
|
|
|
|
}
|