mirror of
https://github.com/makew0rld/amfora.git
synced 2024-12-04 14:46:29 -05:00
ea9c7f214a
Fixes #243
468 lines
14 KiB
Go
468 lines
14 KiB
Go
// Package config initializes all files required for Amfora, even those used by
|
|
// other packages. It also reads in the config file and initializes a Viper and
|
|
// the theme
|
|
//nolint:golint,goerr113
|
|
package config
|
|
|
|
import (
|
|
"fmt"
|
|
"os"
|
|
"path/filepath"
|
|
"runtime"
|
|
"strings"
|
|
|
|
"code.rocketnine.space/tslocum/cview"
|
|
"github.com/gdamore/tcell/v2"
|
|
"github.com/makeworld-the-better-one/amfora/cache"
|
|
homedir "github.com/mitchellh/go-homedir"
|
|
"github.com/muesli/termenv"
|
|
"github.com/rkoesters/xdg/basedir"
|
|
"github.com/rkoesters/xdg/userdirs"
|
|
"github.com/spf13/viper"
|
|
)
|
|
|
|
var amforaAppData string // Where amfora files are stored on Windows - cached here
|
|
var configDir string
|
|
var configPath string
|
|
|
|
var NewTabPath string
|
|
var CustomNewTab bool
|
|
|
|
var TofuStore = viper.New()
|
|
var tofuDBDir string
|
|
var tofuDBPath string
|
|
|
|
// Bookmarks
|
|
var BkmkStore = viper.New() // TOML API for old bookmarks file
|
|
var bkmkDir string
|
|
var OldBkmkPath string // Old bookmarks file that used TOML format
|
|
var BkmkPath string // New XBEL (XML) bookmarks file, see #68
|
|
|
|
var DownloadsDir string
|
|
var TempDownloadsDir string
|
|
|
|
// Subscriptions
|
|
var subscriptionDir string
|
|
var SubscriptionPath string
|
|
|
|
// Command for opening HTTP(S) URLs in the browser, from "a-general.http" in config.
|
|
var HTTPCommand []string
|
|
|
|
type MediaHandler struct {
|
|
Cmd []string
|
|
NoPrompt bool
|
|
Stream bool
|
|
}
|
|
|
|
var MediaHandlers = make(map[string]MediaHandler)
|
|
|
|
// Controlled by "a-general.scrollbar" in config
|
|
// Defaults to ScrollBarAuto on an invalid value
|
|
var ScrollBar cview.ScrollBarVisibility
|
|
|
|
// Whether the user's terminal is dark or light
|
|
// Defaults to dark, but is determined in Init()
|
|
// Used to prevent white text on a white background with the default theme
|
|
var hasDarkTerminalBackground bool
|
|
|
|
func Init() error {
|
|
|
|
// *** Set paths ***
|
|
// Windows uses paths under APPDATA, Unix systems use XDG paths
|
|
// Windows systems use XDG paths if variables are defined, see #255
|
|
|
|
home, err := homedir.Dir()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
// Store AppData path
|
|
if runtime.GOOS == "windows" { //nolint:goconst
|
|
appdata, ok := os.LookupEnv("APPDATA")
|
|
if ok {
|
|
amforaAppData = filepath.Join(appdata, "amfora")
|
|
} else {
|
|
amforaAppData = filepath.Join(home, filepath.FromSlash("AppData/Roaming/amfora/"))
|
|
}
|
|
}
|
|
|
|
// Store config directory and file paths
|
|
if runtime.GOOS == "windows" && os.Getenv("XDG_CONFIG_HOME") == "" {
|
|
configDir = amforaAppData
|
|
} else {
|
|
// Unix / POSIX system, or Windows with XDG_CONFIG_HOME defined
|
|
configDir = filepath.Join(basedir.ConfigHome, "amfora")
|
|
}
|
|
configPath = filepath.Join(configDir, "config.toml")
|
|
|
|
// Search for a custom new tab
|
|
NewTabPath = filepath.Join(configDir, "newtab.gmi")
|
|
CustomNewTab = false
|
|
if _, err := os.Stat(NewTabPath); err == nil {
|
|
CustomNewTab = true
|
|
}
|
|
|
|
// Store TOFU db directory and file paths
|
|
if runtime.GOOS == "windows" && os.Getenv("XDG_CACHE_HOME") == "" {
|
|
// Windows just stores it in APPDATA along with other stuff
|
|
tofuDBDir = amforaAppData
|
|
} else {
|
|
// XDG cache dir on POSIX systems
|
|
tofuDBDir = filepath.Join(basedir.CacheHome, "amfora")
|
|
}
|
|
tofuDBPath = filepath.Join(tofuDBDir, "tofu.toml")
|
|
|
|
// Store bookmarks dir and path
|
|
if runtime.GOOS == "windows" && os.Getenv("XDG_DATA_HOME") == "" {
|
|
// Windows just keeps it in APPDATA along with other Amfora files
|
|
bkmkDir = amforaAppData
|
|
} else {
|
|
// XDG data dir on POSIX systems
|
|
bkmkDir = filepath.Join(basedir.DataHome, "amfora")
|
|
}
|
|
OldBkmkPath = filepath.Join(bkmkDir, "bookmarks.toml")
|
|
BkmkPath = filepath.Join(bkmkDir, "bookmarks.xml")
|
|
|
|
// Feeds dir and path
|
|
if runtime.GOOS == "windows" && os.Getenv("XDG_DATA_HOME") == "" {
|
|
// In APPDATA beside other Amfora files
|
|
subscriptionDir = amforaAppData
|
|
} else {
|
|
// XDG data dir on POSIX systems
|
|
subscriptionDir = filepath.Join(basedir.DataHome, "amfora")
|
|
}
|
|
SubscriptionPath = filepath.Join(subscriptionDir, "subscriptions.json")
|
|
|
|
// *** Create necessary files and folders ***
|
|
|
|
// Config
|
|
err = os.MkdirAll(configDir, 0755)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
f, err := os.OpenFile(configPath, os.O_RDWR|os.O_CREATE|os.O_EXCL, 0666)
|
|
if err == nil {
|
|
// Config file doesn't exist yet, write the default one
|
|
_, err = f.Write(defaultConf)
|
|
if err != nil {
|
|
f.Close()
|
|
return err
|
|
}
|
|
f.Close()
|
|
}
|
|
// TOFU
|
|
err = os.MkdirAll(tofuDBDir, 0755)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
f, err = os.OpenFile(tofuDBPath, os.O_RDWR|os.O_CREATE|os.O_EXCL, 0666)
|
|
if err == nil {
|
|
f.Close()
|
|
}
|
|
// Bookmarks
|
|
err = os.MkdirAll(bkmkDir, 0755)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
// OldBkmkPath isn't created because it shouldn't be there anyway
|
|
|
|
// Feeds
|
|
err = os.MkdirAll(subscriptionDir, 0755)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// *** Setup vipers ***
|
|
|
|
TofuStore.SetConfigFile(tofuDBPath)
|
|
TofuStore.SetConfigType("toml")
|
|
err = TofuStore.ReadInConfig()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
BkmkStore.SetConfigFile(OldBkmkPath)
|
|
BkmkStore.SetConfigType("toml")
|
|
err = BkmkStore.ReadInConfig()
|
|
if err != nil {
|
|
// File doesn't exist, so remove the viper
|
|
BkmkStore = nil
|
|
}
|
|
|
|
// Setup main config
|
|
|
|
viper.SetDefault("a-general.home", "gemini://gemini.circumlunar.space")
|
|
viper.SetDefault("a-general.auto_redirect", false)
|
|
viper.SetDefault("a-general.http", "default")
|
|
viper.SetDefault("a-general.search", "gemini://geminispace.info/search")
|
|
viper.SetDefault("a-general.color", true)
|
|
viper.SetDefault("a-general.ansi", true)
|
|
viper.SetDefault("a-general.highlight_code", true)
|
|
viper.SetDefault("a-general.highlight_style", "monokai")
|
|
viper.SetDefault("a-general.bullets", true)
|
|
viper.SetDefault("a-general.show_link", false)
|
|
viper.SetDefault("a-general.max_width", 80)
|
|
viper.SetDefault("a-general.downloads", "")
|
|
viper.SetDefault("a-general.temp_downloads", "")
|
|
viper.SetDefault("a-general.page_max_size", 2097152)
|
|
viper.SetDefault("a-general.page_max_time", 10)
|
|
viper.SetDefault("a-general.scrollbar", "auto")
|
|
viper.SetDefault("a-general.underline", true)
|
|
viper.SetDefault("keybindings.bind_reload", []string{"R", "Ctrl-R"})
|
|
viper.SetDefault("keybindings.bind_home", "Backspace")
|
|
viper.SetDefault("keybindings.bind_bookmarks", "Ctrl-B")
|
|
viper.SetDefault("keybindings.bind_add_bookmark", "Ctrl-D")
|
|
viper.SetDefault("keybindings.bind_sub", "Ctrl-A")
|
|
viper.SetDefault("keybindings.bind_add_sub", "Ctrl-X")
|
|
viper.SetDefault("keybindings.bind_save", "Ctrl-S")
|
|
viper.SetDefault("keybindings.bind_moveup", "k")
|
|
viper.SetDefault("keybindings.bind_movedown", "j")
|
|
viper.SetDefault("keybindings.bind_moveleft", "h")
|
|
viper.SetDefault("keybindings.bind_moveright", "l")
|
|
viper.SetDefault("keybindings.bind_pgup", []string{"PgUp", "u"})
|
|
viper.SetDefault("keybindings.bind_pgdn", []string{"PgDn", "d"})
|
|
viper.SetDefault("keybindings.bind_bottom", "Space")
|
|
viper.SetDefault("keybindings.bind_edit", "e")
|
|
viper.SetDefault("keybindings.bind_back", []string{"b", "Alt-Left"})
|
|
viper.SetDefault("keybindings.bind_forward", []string{"f", "Alt-Right"})
|
|
viper.SetDefault("keybindings.bind_new_tab", "Ctrl-T")
|
|
viper.SetDefault("keybindings.bind_close_tab", "Ctrl-W")
|
|
viper.SetDefault("keybindings.bind_next_tab", "F2")
|
|
viper.SetDefault("keybindings.bind_prev_tab", "F1")
|
|
viper.SetDefault("keybindings.bind_quit", []string{"Ctrl-C", "Ctrl-Q", "Q"})
|
|
viper.SetDefault("keybindings.bind_help", "?")
|
|
viper.SetDefault("keybindings.bind_link1", "1")
|
|
viper.SetDefault("keybindings.bind_link2", "2")
|
|
viper.SetDefault("keybindings.bind_link3", "3")
|
|
viper.SetDefault("keybindings.bind_link4", "4")
|
|
viper.SetDefault("keybindings.bind_link5", "5")
|
|
viper.SetDefault("keybindings.bind_link6", "6")
|
|
viper.SetDefault("keybindings.bind_link7", "7")
|
|
viper.SetDefault("keybindings.bind_link8", "8")
|
|
viper.SetDefault("keybindings.bind_link9", "9")
|
|
viper.SetDefault("keybindings.bind_link0", "0")
|
|
viper.SetDefault("keybindings.bind_tab1", "!")
|
|
viper.SetDefault("keybindings.bind_tab2", "@")
|
|
viper.SetDefault("keybindings.bind_tab3", "#")
|
|
viper.SetDefault("keybindings.bind_tab4", "$")
|
|
viper.SetDefault("keybindings.bind_tab5", "%")
|
|
viper.SetDefault("keybindings.bind_tab6", "^")
|
|
viper.SetDefault("keybindings.bind_tab7", "&")
|
|
viper.SetDefault("keybindings.bind_tab8", "*")
|
|
viper.SetDefault("keybindings.bind_tab9", "(")
|
|
viper.SetDefault("keybindings.bind_tab0", ")")
|
|
viper.SetDefault("keybindings.bind_copy_page_url", "C")
|
|
viper.SetDefault("keybindings.bind_copy_target_url", "c")
|
|
viper.SetDefault("keybindings.bind_beginning", []string{"Home", "g"})
|
|
viper.SetDefault("keybindings.bind_end", []string{"End", "G"})
|
|
viper.SetDefault("keybindings.shift_numbers", "")
|
|
viper.SetDefault("keybindings.bind_url_handler_open", "Ctrl-U")
|
|
viper.SetDefault("url-handlers.other", "default")
|
|
viper.SetDefault("cache.max_size", 0)
|
|
viper.SetDefault("cache.max_pages", 20)
|
|
viper.SetDefault("cache.timeout", 1800)
|
|
viper.SetDefault("subscriptions.popup", true)
|
|
viper.SetDefault("subscriptions.update_interval", 1800)
|
|
viper.SetDefault("subscriptions.workers", 3)
|
|
viper.SetDefault("subscriptions.entries_per_page", 20)
|
|
viper.SetDefault("subscriptions.header", true)
|
|
|
|
viper.SetConfigFile(configPath)
|
|
viper.SetConfigType("toml")
|
|
err = viper.ReadInConfig()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// Setup the key bindings
|
|
KeyInit()
|
|
|
|
// *** Downloads paths, setup, and creation ***
|
|
|
|
// Setup downloads dir
|
|
if viper.GetString("a-general.downloads") == "" {
|
|
// Find default Downloads dir
|
|
if userdirs.Download == "" {
|
|
DownloadsDir = filepath.Join(home, "Downloads")
|
|
} else {
|
|
DownloadsDir = userdirs.Download
|
|
}
|
|
// Create it just in case
|
|
err = os.MkdirAll(DownloadsDir, 0755)
|
|
if err != nil {
|
|
return fmt.Errorf("downloads path could not be created: %s", DownloadsDir)
|
|
}
|
|
} else {
|
|
// Validate path
|
|
dDir := viper.GetString("a-general.downloads")
|
|
di, err := os.Stat(dDir)
|
|
if err == nil {
|
|
if !di.IsDir() {
|
|
return fmt.Errorf("downloads path specified is not a directory: %s", dDir)
|
|
}
|
|
} else if os.IsNotExist(err) {
|
|
// Try to create path
|
|
err = os.MkdirAll(dDir, 0755)
|
|
if err != nil {
|
|
return fmt.Errorf("downloads path could not be created: %s", dDir)
|
|
}
|
|
} else {
|
|
// Some other error
|
|
return fmt.Errorf("couldn't access downloads directory: %s", dDir)
|
|
}
|
|
DownloadsDir = dDir
|
|
}
|
|
|
|
// Setup temporary downloads dir
|
|
if viper.GetString("a-general.temp_downloads") == "" {
|
|
TempDownloadsDir = filepath.Join(os.TempDir(), "amfora_temp")
|
|
|
|
// Make sure it exists
|
|
err = os.MkdirAll(TempDownloadsDir, 0755)
|
|
if err != nil {
|
|
return fmt.Errorf("temp downloads path could not be created: %s", TempDownloadsDir)
|
|
}
|
|
} else {
|
|
// Validate path
|
|
dDir := viper.GetString("a-general.temp_downloads")
|
|
di, err := os.Stat(dDir)
|
|
if err == nil {
|
|
if !di.IsDir() {
|
|
return fmt.Errorf("temp downloads path specified is not a directory: %s", dDir)
|
|
}
|
|
} else if os.IsNotExist(err) {
|
|
// Try to create path
|
|
err = os.MkdirAll(dDir, 0755)
|
|
if err != nil {
|
|
return fmt.Errorf("temp downloads path could not be created: %s", dDir)
|
|
}
|
|
} else {
|
|
// Some other error
|
|
return fmt.Errorf("couldn't access temp downloads directory: %s", dDir)
|
|
}
|
|
TempDownloadsDir = dDir
|
|
}
|
|
|
|
// Setup cache from config
|
|
cache.SetMaxSize(viper.GetInt("cache.max_size"))
|
|
cache.SetMaxPages(viper.GetInt("cache.max_pages"))
|
|
cache.SetTimeout(viper.GetInt("cache.timeout"))
|
|
|
|
setColor := func(k string, colorStr string) error {
|
|
if k == "include" {
|
|
return nil
|
|
}
|
|
colorStr = strings.ToLower(colorStr)
|
|
var color tcell.Color
|
|
if colorStr == "default" {
|
|
if strings.HasSuffix(k, "bg") {
|
|
color = tcell.ColorDefault
|
|
} else {
|
|
return fmt.Errorf(`"default" is only valid for a background color (color ending in "bg"), not "%s"`, k)
|
|
}
|
|
} else {
|
|
color = tcell.GetColor(colorStr)
|
|
if color == tcell.ColorDefault {
|
|
return fmt.Errorf(`invalid color format for "%s": %s`, k, colorStr)
|
|
}
|
|
}
|
|
SetColor(k, color)
|
|
return nil
|
|
}
|
|
|
|
// Setup theme
|
|
configTheme := viper.Sub("theme")
|
|
if configTheme != nil {
|
|
// Include key comes first
|
|
if incPath := configTheme.GetString("include"); incPath != "" {
|
|
incViper := viper.New()
|
|
incViper.SetConfigFile(incPath)
|
|
incViper.SetConfigType("toml")
|
|
err = incViper.ReadInConfig()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
for k2, v2 := range incViper.AllSettings() {
|
|
colorStr, ok := v2.(string)
|
|
if !ok {
|
|
return fmt.Errorf(`include: value for "%s" is not a string: %v`, k2, v2)
|
|
}
|
|
if err := setColor(k2, colorStr); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
}
|
|
for k, v := range configTheme.AllSettings() {
|
|
colorStr, ok := v.(string)
|
|
if !ok {
|
|
return fmt.Errorf(`value for "%s" is not a string: %v`, k, v)
|
|
}
|
|
if err := setColor(k, colorStr); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
}
|
|
if viper.GetBool("a-general.color") {
|
|
cview.Styles.PrimitiveBackgroundColor = GetColor("bg")
|
|
} else {
|
|
// No colors allowed, set background to black instead of default
|
|
themeMu.Lock()
|
|
theme["bg"] = tcell.ColorBlack
|
|
cview.Styles.PrimitiveBackgroundColor = tcell.ColorBlack
|
|
themeMu.Unlock()
|
|
}
|
|
|
|
hasDarkTerminalBackground = termenv.HasDarkBackground()
|
|
|
|
// Parse HTTP command
|
|
HTTPCommand = viper.GetStringSlice("a-general.http")
|
|
if len(HTTPCommand) == 0 {
|
|
// Not a string array, interpret as a string instead
|
|
// Split on spaces to maintain compatibility with old versions
|
|
// The new better way to is to just define a string array in config
|
|
HTTPCommand = strings.Fields(viper.GetString("a-general.http"))
|
|
}
|
|
|
|
var rawMediaHandlers []struct {
|
|
Cmd []string `mapstructure:"cmd"`
|
|
Types []string `mapstructure:"types"`
|
|
NoPrompt bool `mapstructure:"no_prompt"`
|
|
Stream bool `mapstructure:"stream"`
|
|
}
|
|
err = viper.UnmarshalKey("mediatype-handlers", &rawMediaHandlers)
|
|
if err != nil {
|
|
return fmt.Errorf("couldn't parse mediatype-handlers section in config: %w", err)
|
|
}
|
|
for _, rawMediaHandler := range rawMediaHandlers {
|
|
if len(rawMediaHandler.Cmd) == 0 {
|
|
return fmt.Errorf("empty cmd array in mediatype-handlers section")
|
|
}
|
|
if len(rawMediaHandler.Types) == 0 {
|
|
return fmt.Errorf("empty types array in mediatype-handlers section")
|
|
}
|
|
|
|
for _, typ := range rawMediaHandler.Types {
|
|
if _, ok := MediaHandlers[typ]; ok {
|
|
return fmt.Errorf("multiple mediatype-handlers defined for %v", typ)
|
|
}
|
|
MediaHandlers[typ] = MediaHandler{
|
|
Cmd: rawMediaHandler.Cmd,
|
|
NoPrompt: rawMediaHandler.NoPrompt,
|
|
Stream: rawMediaHandler.Stream,
|
|
}
|
|
}
|
|
}
|
|
|
|
// Parse scrollbar options
|
|
switch viper.GetString("a-general.scrollbar") {
|
|
case "never":
|
|
ScrollBar = cview.ScrollBarNever
|
|
case "always":
|
|
ScrollBar = cview.ScrollBarAlways
|
|
default:
|
|
ScrollBar = cview.ScrollBarAuto
|
|
}
|
|
|
|
return nil
|
|
}
|