2021-04-19 21:58:05 +00:00
|
|
|
package paths
|
|
|
|
|
|
|
|
import (
|
|
|
|
"os"
|
|
|
|
"os/user"
|
|
|
|
"path/filepath"
|
|
|
|
"runtime"
|
|
|
|
"strings"
|
2023-02-22 12:38:22 +00:00
|
|
|
|
|
|
|
"github.com/ledgerwatch/erigon/params/networkname"
|
|
|
|
"github.com/ledgerwatch/log/v3"
|
2021-04-19 21:58:05 +00:00
|
|
|
)
|
|
|
|
|
2021-05-26 10:35:39 +00:00
|
|
|
const dirname = "Erigon"
|
2021-04-19 21:58:05 +00:00
|
|
|
|
|
|
|
// 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 ""
|
|
|
|
}
|
2023-02-22 12:38:22 +00:00
|
|
|
|
|
|
|
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.RinkebyChainName:
|
|
|
|
return networkDataDirCheckingLegacy(datadir, "rinkeby")
|
|
|
|
case networkname.GoerliChainName:
|
|
|
|
return networkDataDirCheckingLegacy(datadir, "goerli")
|
|
|
|
case networkname.SokolChainName:
|
|
|
|
return networkDataDirCheckingLegacy(datadir, "sokol")
|
|
|
|
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
|
|
|
|
}
|