1
0
mirror of https://github.com/jesseduffield/lazygit.git synced 2025-01-18 05:17:55 +02:00
lazygit/pkg/config/app_config.go

341 lines
8.2 KiB
Go
Raw Normal View History

package config
2018-08-15 21:34:25 +10:00
import (
"fmt"
2019-02-18 21:29:43 +11:00
"os"
2018-08-25 15:55:49 +10:00
"path/filepath"
"strings"
2018-08-15 21:34:25 +10:00
2020-10-03 14:54:55 +10:00
"github.com/OpenPeeDeeP/xdg"
"github.com/jesseduffield/lazygit/pkg/utils/yaml_utils"
2023-07-26 12:50:39 +03:00
yaml "github.com/jesseduffield/yaml"
2018-08-15 21:34:25 +10:00
)
// AppConfig contains the base configuration fields required for lazygit.
type AppConfig struct {
2021-07-27 22:03:37 +02:00
Debug bool `long:"debug" env:"DEBUG" default:"false"`
Version string `long:"version" env:"VERSION" default:"unversioned"`
BuildDate string `long:"build-date" env:"BUILD_DATE"`
Name string `long:"name" env:"NAME" default:"lazygit"`
BuildSource string `long:"build-source" env:"BUILD_SOURCE" default:""`
UserConfig *UserConfig
2021-10-16 12:18:49 +11:00
UserConfigPaths []string
2021-07-27 22:03:37 +02:00
DeafultConfFiles bool
UserConfigDir string
TempDir string
AppState *AppState
IsNewRepo bool
}
type AppConfigurer interface {
GetDebug() bool
2021-12-29 12:03:35 +11:00
// build info
GetVersion() string
GetName() string
2018-08-25 15:55:49 +10:00
GetBuildSource() string
2021-12-29 12:03:35 +11:00
2020-10-03 14:54:55 +10:00
GetUserConfig() *UserConfig
2021-10-16 12:18:49 +11:00
GetUserConfigPaths() []string
2019-09-15 11:19:39 +02:00
GetUserConfigDir() string
2021-12-29 12:03:35 +11:00
ReloadUserConfig() error
GetTempDir() string
2021-12-29 12:03:35 +11:00
2018-08-25 15:55:49 +10:00
GetAppState() *AppState
SaveAppState() error
2018-08-15 21:34:25 +10:00
}
// NewAppConfig makes a new app config
2022-05-06 23:00:44 +01:00
func NewAppConfig(
name string,
version,
commit,
date string,
buildSource string,
debuggingFlag bool,
tempDir string,
) (*AppConfig, error) {
2020-10-04 09:53:56 +11:00
configDir, err := findOrCreateConfigDir()
2021-07-27 22:03:37 +02:00
if err != nil && !os.IsPermission(err) {
2020-10-03 14:54:55 +10:00
return nil, err
}
2021-10-16 12:18:49 +11:00
var userConfigPaths []string
2021-10-16 12:07:24 +11:00
customConfigFiles := os.Getenv("LG_CONFIG_FILE")
if customConfigFiles != "" {
2021-07-27 22:03:37 +02:00
// Load user defined config files
2021-10-16 12:18:49 +11:00
userConfigPaths = strings.Split(customConfigFiles, ",")
2021-07-27 22:03:37 +02:00
} else {
// Load default config files
2021-10-16 12:18:49 +11:00
userConfigPaths = []string{filepath.Join(configDir, ConfigFilename)}
2021-07-27 22:03:37 +02:00
}
2021-10-16 12:18:49 +11:00
userConfig, err := loadUserConfigWithDefaults(userConfigPaths)
2018-08-15 21:34:25 +10:00
if err != nil {
2018-08-25 15:55:49 +10:00
return nil, err
2018-08-15 21:34:25 +10:00
}
appState, err := loadAppState()
if err != nil {
return nil, err
}
2018-08-15 21:34:25 +10:00
appConfig := &AppConfig{
2022-05-06 23:00:44 +01:00
Name: name,
2021-10-16 12:07:24 +11:00
Version: version,
BuildDate: date,
Debug: debuggingFlag,
BuildSource: buildSource,
UserConfig: userConfig,
2021-10-16 12:18:49 +11:00
UserConfigPaths: userConfigPaths,
2021-10-16 12:07:24 +11:00
UserConfigDir: configDir,
TempDir: tempDir,
AppState: appState,
IsNewRepo: false,
2018-08-25 15:55:49 +10:00
}
2018-08-15 21:34:25 +10:00
return appConfig, nil
}
2021-10-16 12:07:24 +11:00
func isCustomConfigFile(path string) bool {
return path != filepath.Join(ConfigDir(), ConfigFilename)
}
2020-11-22 14:51:48 +01:00
func ConfigDir() string {
legacyConfigDirectory := configDirForVendor("jesseduffield")
2020-11-21 16:17:35 +01:00
if _, err := os.Stat(legacyConfigDirectory); !os.IsNotExist(err) {
return legacyConfigDirectory
}
2020-11-22 14:51:48 +01:00
configDirectory := configDirForVendor("")
2020-11-21 16:17:35 +01:00
return configDirectory
}
2020-11-22 14:51:48 +01:00
func configDirForVendor(vendor string) string {
2020-10-04 22:05:39 +11:00
envConfigDir := os.Getenv("CONFIG_DIR")
if envConfigDir != "" {
return envConfigDir
}
configDirs := xdg.New(vendor, "lazygit")
2020-10-04 09:53:56 +11:00
return configDirs.ConfigHome()
}
func findOrCreateConfigDir() (string, error) {
2020-11-22 14:51:48 +01:00
folder := ConfigDir()
2022-03-19 09:38:49 +11:00
return folder, os.MkdirAll(folder, 0o755)
2020-10-03 14:54:55 +10:00
}
2021-10-16 12:07:24 +11:00
func loadUserConfigWithDefaults(configFiles []string) (*UserConfig, error) {
return loadUserConfig(configFiles, GetDefaultConfig())
2020-10-03 14:54:55 +10:00
}
2021-10-16 12:07:24 +11:00
func loadUserConfig(configFiles []string, base *UserConfig) (*UserConfig, error) {
for _, path := range configFiles {
if _, err := os.Stat(path); err != nil {
if !os.IsNotExist(err) {
return nil, err
2021-07-27 22:03:37 +02:00
}
2021-10-16 12:07:24 +11:00
// if use has supplied their own custom config file path(s), we assume
// the files have already been created, so we won't go and create them here.
if isCustomConfigFile(path) {
return nil, err
2021-07-27 22:03:37 +02:00
}
2020-10-03 14:54:55 +10:00
2021-10-16 12:07:24 +11:00
file, err := os.Create(path)
2020-10-03 14:54:55 +10:00
if err != nil {
2021-07-27 22:03:37 +02:00
if os.IsPermission(err) {
2021-10-16 12:07:24 +11:00
// apparently when people have read-only permissions they prefer us to fail silently
2021-07-27 22:03:37 +02:00
continue
}
2021-10-16 12:07:24 +11:00
return nil, err
2020-10-03 14:54:55 +10:00
}
file.Close()
}
2022-09-13 18:11:03 +08:00
content, err := os.ReadFile(path)
2021-10-16 12:07:24 +11:00
if err != nil {
return nil, err
}
content, err = migrateUserConfig(path, content)
if err != nil {
return nil, err
}
2021-07-27 22:03:37 +02:00
if err := yaml.Unmarshal(content, base); err != nil {
return nil, fmt.Errorf("The config at `%s` couldn't be parsed, please inspect it before opening up an issue.\n%w", path, err)
2021-07-27 22:03:37 +02:00
}
2020-10-03 14:54:55 +10:00
}
return base, nil
}
// Do any backward-compatibility migrations of things that have changed in the
// config over time; examples are renaming a key to a better name, moving a key
// from one container to another, or changing the type of a key (e.g. from bool
// to an enum).
func migrateUserConfig(path string, content []byte) ([]byte, error) {
changedContent, err := yaml_utils.RenameYamlKey(content, []string{"gui", "skipUnstageLineWarning"},
"skipDiscardChangeWarning")
if err != nil {
return nil, fmt.Errorf("Couldn't migrate config file at `%s`: %s", path, err)
}
// Add more migrations here...
// Write config back if changed
if string(changedContent) != string(content) {
if err := os.WriteFile(path, changedContent, 0o644); err != nil {
return nil, fmt.Errorf("Couldn't write migrated config back to `%s`: %s", path, err)
}
return changedContent, nil
}
return content, nil
}
func (c *AppConfig) GetDebug() bool {
return c.Debug
}
func (c *AppConfig) GetVersion() string {
return c.Version
}
func (c *AppConfig) GetName() string {
return c.Name
}
2018-08-15 21:34:25 +10:00
2018-08-25 15:55:49 +10:00
// GetBuildSource returns the source of the build. For builds from goreleaser
// this will be binaryBuild
func (c *AppConfig) GetBuildSource() string {
return c.BuildSource
}
2018-08-15 21:34:25 +10:00
// GetUserConfig returns the user config
2020-10-03 14:54:55 +10:00
func (c *AppConfig) GetUserConfig() *UserConfig {
2018-08-15 21:34:25 +10:00
return c.UserConfig
}
2018-08-25 15:55:49 +10:00
// GetAppState returns the app state
func (c *AppConfig) GetAppState() *AppState {
return c.AppState
}
2021-10-16 12:18:49 +11:00
func (c *AppConfig) GetUserConfigPaths() []string {
return c.UserConfigPaths
2021-07-27 22:03:37 +02:00
}
2019-09-15 11:19:39 +02:00
func (c *AppConfig) GetUserConfigDir() string {
return c.UserConfigDir
}
func (c *AppConfig) ReloadUserConfig() error {
2021-10-16 12:18:49 +11:00
userConfig, err := loadUserConfigWithDefaults(c.UserConfigPaths)
if err != nil {
return err
}
c.UserConfig = userConfig
return nil
}
func (c *AppConfig) GetTempDir() string {
return c.TempDir
}
2020-10-03 14:54:55 +10:00
func configFilePath(filename string) (string, error) {
2020-10-04 09:53:56 +11:00
folder, err := findOrCreateConfigDir()
if err != nil {
2020-10-03 14:54:55 +10:00
return "", err
2018-08-18 13:22:05 +10:00
}
2020-10-03 14:54:55 +10:00
return filepath.Join(folder, filename), nil
2018-08-18 13:22:05 +10:00
}
2021-07-27 22:03:37 +02:00
var ConfigFilename = "config.yml"
2022-04-03 15:19:15 -05:00
// ConfigFilename returns the filename of the default config file
2020-10-03 14:54:55 +10:00
func (c *AppConfig) ConfigFilename() string {
2021-07-27 22:03:37 +02:00
return filepath.Join(c.UserConfigDir, ConfigFilename)
2018-08-15 21:34:25 +10:00
}
2019-05-22 21:34:29 +08:00
// SaveAppState marshalls the AppState struct and writes it to the disk
2018-08-25 15:55:49 +10:00
func (c *AppConfig) SaveAppState() error {
marshalledAppState, err := yaml.Marshal(c.AppState)
if err != nil {
return err
}
2020-10-03 14:54:55 +10:00
filepath, err := configFilePath("state.yml")
2018-08-25 15:55:49 +10:00
if err != nil {
return err
}
2022-09-13 18:11:03 +08:00
err = os.WriteFile(filepath, marshalledAppState, 0o644)
2021-10-16 12:07:24 +11:00
if err != nil && os.IsPermission(err) {
// apparently when people have read-only permissions they prefer us to fail silently
return nil
}
return err
2018-08-25 15:55:49 +10:00
}
// loadAppState loads recorded AppState from file
func loadAppState() (*AppState, error) {
2020-10-03 14:54:55 +10:00
filepath, err := configFilePath("state.yml")
2018-08-25 15:55:49 +10:00
if err != nil {
2021-07-27 22:03:37 +02:00
if os.IsPermission(err) {
2021-10-16 12:09:22 +11:00
// apparently when people have read-only permissions they prefer us to fail silently
2021-07-27 22:03:37 +02:00
return getDefaultAppState(), nil
}
return nil, err
2018-08-25 15:55:49 +10:00
}
2022-09-13 18:11:03 +08:00
appStateBytes, err := os.ReadFile(filepath)
2020-10-04 22:05:39 +11:00
if err != nil && !os.IsNotExist(err) {
return nil, err
2018-08-25 15:55:49 +10:00
}
2018-08-25 15:55:49 +10:00
if len(appStateBytes) == 0 {
return getDefaultAppState(), nil
}
appState := &AppState{}
err = yaml.Unmarshal(appStateBytes, appState)
if err != nil {
return nil, err
2018-08-25 15:55:49 +10:00
}
return appState, nil
2018-08-25 15:55:49 +10:00
}
// AppState stores data between runs of the app like when the last update check
// was performed and which other repos have been checked out
type AppState struct {
LastUpdateCheck int64
RecentRepos []string
StartupPopupVersion int
2021-10-23 11:25:37 +11:00
// these are for custom commands typed in directly, not for custom commands in the lazygit config
CustomCommandsHistory []string
HideCommandLog bool
IgnoreWhitespaceInDiffView bool
2018-08-25 15:55:49 +10:00
}
func getDefaultAppState() *AppState {
return &AppState{
LastUpdateCheck: 0,
RecentRepos: []string{},
StartupPopupVersion: 0,
}
2018-08-15 21:34:25 +10:00
}
2020-10-03 14:54:55 +10:00
func LogPath() (string, error) {
if os.Getenv("LAZYGIT_LOG_PATH") != "" {
return os.Getenv("LAZYGIT_LOG_PATH"), nil
}
2020-10-03 14:54:55 +10:00
return configFilePath("development.log")
}