mirror of
https://github.com/jesseduffield/lazygit.git
synced 2025-01-10 04:07:18 +02:00
348 lines
8.4 KiB
Go
348 lines
8.4 KiB
Go
package config
|
|
|
|
import (
|
|
"fmt"
|
|
"os"
|
|
"path/filepath"
|
|
"strings"
|
|
|
|
"github.com/OpenPeeDeeP/xdg"
|
|
"github.com/jesseduffield/lazygit/pkg/utils/yaml_utils"
|
|
yaml "github.com/jesseduffield/yaml"
|
|
)
|
|
|
|
// AppConfig contains the base configuration fields required for lazygit.
|
|
type AppConfig struct {
|
|
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
|
|
UserConfigPaths []string
|
|
DeafultConfFiles bool
|
|
UserConfigDir string
|
|
TempDir string
|
|
AppState *AppState
|
|
IsNewRepo bool
|
|
}
|
|
|
|
type AppConfigurer interface {
|
|
GetDebug() bool
|
|
|
|
// build info
|
|
GetVersion() string
|
|
GetName() string
|
|
GetBuildSource() string
|
|
|
|
GetUserConfig() *UserConfig
|
|
GetUserConfigPaths() []string
|
|
GetUserConfigDir() string
|
|
ReloadUserConfig() error
|
|
GetTempDir() string
|
|
|
|
GetAppState() *AppState
|
|
SaveAppState() error
|
|
}
|
|
|
|
// NewAppConfig makes a new app config
|
|
func NewAppConfig(
|
|
name string,
|
|
version,
|
|
commit,
|
|
date string,
|
|
buildSource string,
|
|
debuggingFlag bool,
|
|
tempDir string,
|
|
) (*AppConfig, error) {
|
|
configDir, err := findOrCreateConfigDir()
|
|
if err != nil && !os.IsPermission(err) {
|
|
return nil, err
|
|
}
|
|
|
|
var userConfigPaths []string
|
|
customConfigFiles := os.Getenv("LG_CONFIG_FILE")
|
|
if customConfigFiles != "" {
|
|
// Load user defined config files
|
|
userConfigPaths = strings.Split(customConfigFiles, ",")
|
|
} else {
|
|
// Load default config files
|
|
userConfigPaths = []string{filepath.Join(configDir, ConfigFilename)}
|
|
}
|
|
|
|
userConfig, err := loadUserConfigWithDefaults(userConfigPaths)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
appState, err := loadAppState()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
appConfig := &AppConfig{
|
|
Name: name,
|
|
Version: version,
|
|
BuildDate: date,
|
|
Debug: debuggingFlag,
|
|
BuildSource: buildSource,
|
|
UserConfig: userConfig,
|
|
UserConfigPaths: userConfigPaths,
|
|
UserConfigDir: configDir,
|
|
TempDir: tempDir,
|
|
AppState: appState,
|
|
IsNewRepo: false,
|
|
}
|
|
|
|
return appConfig, nil
|
|
}
|
|
|
|
func isCustomConfigFile(path string) bool {
|
|
return path != filepath.Join(ConfigDir(), ConfigFilename)
|
|
}
|
|
|
|
func ConfigDir() string {
|
|
legacyConfigDirectory := configDirForVendor("jesseduffield")
|
|
if _, err := os.Stat(legacyConfigDirectory); !os.IsNotExist(err) {
|
|
return legacyConfigDirectory
|
|
}
|
|
configDirectory := configDirForVendor("")
|
|
return configDirectory
|
|
}
|
|
|
|
func configDirForVendor(vendor string) string {
|
|
envConfigDir := os.Getenv("CONFIG_DIR")
|
|
if envConfigDir != "" {
|
|
return envConfigDir
|
|
}
|
|
configDirs := xdg.New(vendor, "lazygit")
|
|
return configDirs.ConfigHome()
|
|
}
|
|
|
|
func findOrCreateConfigDir() (string, error) {
|
|
folder := ConfigDir()
|
|
return folder, os.MkdirAll(folder, 0o755)
|
|
}
|
|
|
|
func loadUserConfigWithDefaults(configFiles []string) (*UserConfig, error) {
|
|
return loadUserConfig(configFiles, GetDefaultConfig())
|
|
}
|
|
|
|
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
|
|
}
|
|
|
|
// 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
|
|
}
|
|
|
|
file, err := os.Create(path)
|
|
if err != nil {
|
|
if os.IsPermission(err) {
|
|
// apparently when people have read-only permissions they prefer us to fail silently
|
|
continue
|
|
}
|
|
return nil, err
|
|
}
|
|
file.Close()
|
|
}
|
|
|
|
content, err := os.ReadFile(path)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
content, err = migrateUserConfig(path, content)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
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)
|
|
}
|
|
}
|
|
|
|
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
|
|
}
|
|
|
|
// GetBuildSource returns the source of the build. For builds from goreleaser
|
|
// this will be binaryBuild
|
|
func (c *AppConfig) GetBuildSource() string {
|
|
return c.BuildSource
|
|
}
|
|
|
|
// GetUserConfig returns the user config
|
|
func (c *AppConfig) GetUserConfig() *UserConfig {
|
|
return c.UserConfig
|
|
}
|
|
|
|
// GetAppState returns the app state
|
|
func (c *AppConfig) GetAppState() *AppState {
|
|
return c.AppState
|
|
}
|
|
|
|
func (c *AppConfig) GetUserConfigPaths() []string {
|
|
return c.UserConfigPaths
|
|
}
|
|
|
|
func (c *AppConfig) GetUserConfigDir() string {
|
|
return c.UserConfigDir
|
|
}
|
|
|
|
func (c *AppConfig) ReloadUserConfig() error {
|
|
userConfig, err := loadUserConfigWithDefaults(c.UserConfigPaths)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
c.UserConfig = userConfig
|
|
return nil
|
|
}
|
|
|
|
func (c *AppConfig) GetTempDir() string {
|
|
return c.TempDir
|
|
}
|
|
|
|
func configFilePath(filename string) (string, error) {
|
|
folder, err := findOrCreateConfigDir()
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
return filepath.Join(folder, filename), nil
|
|
}
|
|
|
|
var ConfigFilename = "config.yml"
|
|
|
|
// ConfigFilename returns the filename of the default config file
|
|
func (c *AppConfig) ConfigFilename() string {
|
|
return filepath.Join(c.UserConfigDir, ConfigFilename)
|
|
}
|
|
|
|
// SaveAppState marshalls the AppState struct and writes it to the disk
|
|
func (c *AppConfig) SaveAppState() error {
|
|
marshalledAppState, err := yaml.Marshal(c.AppState)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
filepath, err := configFilePath("state.yml")
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
err = os.WriteFile(filepath, marshalledAppState, 0o644)
|
|
if err != nil && os.IsPermission(err) {
|
|
// apparently when people have read-only permissions they prefer us to fail silently
|
|
return nil
|
|
}
|
|
|
|
return err
|
|
}
|
|
|
|
// loadAppState loads recorded AppState from file
|
|
func loadAppState() (*AppState, error) {
|
|
appState := getDefaultAppState()
|
|
|
|
filepath, err := configFilePath("state.yml")
|
|
if err != nil {
|
|
if os.IsPermission(err) {
|
|
// apparently when people have read-only permissions they prefer us to fail silently
|
|
return appState, nil
|
|
}
|
|
return nil, err
|
|
}
|
|
|
|
appStateBytes, err := os.ReadFile(filepath)
|
|
if err != nil && !os.IsNotExist(err) {
|
|
return nil, err
|
|
}
|
|
|
|
if len(appStateBytes) == 0 {
|
|
return appState, nil
|
|
}
|
|
|
|
err = yaml.Unmarshal(appStateBytes, appState)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return appState, nil
|
|
}
|
|
|
|
// 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
|
|
|
|
// these are for custom commands typed in directly, not for custom commands in the lazygit config
|
|
CustomCommandsHistory []string
|
|
HideCommandLog bool
|
|
IgnoreWhitespaceInDiffView bool
|
|
DiffContextSize int
|
|
LocalBranchSortOrder string
|
|
RemoteBranchSortOrder string
|
|
}
|
|
|
|
func getDefaultAppState() *AppState {
|
|
return &AppState{
|
|
LastUpdateCheck: 0,
|
|
RecentRepos: []string{},
|
|
StartupPopupVersion: 0,
|
|
DiffContextSize: 3,
|
|
LocalBranchSortOrder: "recency",
|
|
RemoteBranchSortOrder: "alphabetical",
|
|
}
|
|
}
|
|
|
|
func LogPath() (string, error) {
|
|
if os.Getenv("LAZYGIT_LOG_PATH") != "" {
|
|
return os.Getenv("LAZYGIT_LOG_PATH"), nil
|
|
}
|
|
|
|
return configFilePath("development.log")
|
|
}
|