erigon-pulse/common/paths/paths.go

125 lines
3.5 KiB
Go

package paths
import (
"os"
"os/user"
"path/filepath"
"runtime"
"strings"
"github.com/ledgerwatch/erigon-lib/chain/networkname"
"github.com/ledgerwatch/log/v3"
)
const dirname = "Erigon"
// DefaultDataDir is the default data directory to use for the databases and other
// persistence requirements.
func DefaultDataDir() string {
// Try to place the data folder in the user's home dir
home := homeDir()
if home != "" {
switch runtime.GOOS {
case "darwin":
return filepath.Join(home, "Library", dirname)
case "windows":
// We used to put everything in %HOME%\AppData\Roaming, but this caused
// problems with non-typical setups. If this fallback location exists and
// is non-empty, use it, otherwise DTRT and check %LOCALAPPDATA%.
fallback := filepath.Join(home, "AppData", "Roaming", dirname)
appdata := windowsAppData()
if appdata == "" || isNonEmptyDir(fallback) {
return fallback
}
return filepath.Join(appdata, dirname)
default:
if xdgDataDir := os.Getenv("XDG_DATA_HOME"); xdgDataDir != "" {
return filepath.Join(xdgDataDir, strings.ToLower(dirname))
}
return filepath.Join(home, ".local/share", strings.ToLower(dirname))
}
}
// As we cannot guess a stable location, return empty and handle later
return ""
}
func windowsAppData() string {
v := os.Getenv("LOCALAPPDATA")
if v == "" {
// Windows XP and below don't have LocalAppData. Crash here because
// we don't support Windows XP and undefining the variable will cause
// other issues.
panic("environment variable LocalAppData is undefined")
}
return v
}
func isNonEmptyDir(dir string) bool {
f, err := os.Open(dir)
if err != nil {
return false
}
names, _ := f.Readdir(1)
f.Close()
return len(names) > 0
}
func homeDir() string {
if home := os.Getenv("HOME"); home != "" {
return home
}
if usr, err := user.Current(); err == nil {
return usr.HomeDir
}
return ""
}
func DataDirForNetwork(datadir string, network string) string {
if datadir != DefaultDataDir() {
return datadir
}
switch network {
case networkname.DevChainName:
return "" // unless explicitly requested, use memory databases
case networkname.HoleskyChainName:
return networkDataDirCheckingLegacy(datadir, "holesky")
case networkname.GoerliChainName:
return networkDataDirCheckingLegacy(datadir, "goerli")
case networkname.MumbaiChainName:
return networkDataDirCheckingLegacy(datadir, "mumbai")
case networkname.BorMainnetChainName:
return networkDataDirCheckingLegacy(datadir, "bor-mainnet")
case networkname.BorDevnetChainName:
return networkDataDirCheckingLegacy(datadir, "bor-devnet")
case networkname.SepoliaChainName:
return networkDataDirCheckingLegacy(datadir, "sepolia")
case networkname.GnosisChainName:
return networkDataDirCheckingLegacy(datadir, "gnosis")
case networkname.ChiadoChainName:
return networkDataDirCheckingLegacy(datadir, "chiado")
default:
return datadir
}
}
// networkDataDirCheckingLegacy checks if the datadir for the network already exists and uses that if found.
// if not checks for a LOCK file at the root of the datadir and uses this if found
// or by default assume a fresh node and to use the nested directory for the network
func networkDataDirCheckingLegacy(datadir, network string) string {
anticipated := filepath.Join(datadir, network)
if _, err := os.Stat(anticipated); !os.IsNotExist(err) {
return anticipated
}
legacyLockFile := filepath.Join(datadir, "LOCK")
if _, err := os.Stat(legacyLockFile); !os.IsNotExist(err) {
log.Info("Using legacy datadir")
return datadir
}
return anticipated
}