2022-01-17 09:57:51 +00:00
|
|
|
//go:build !windows
|
2017-08-05 16:23:55 +00:00
|
|
|
|
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2017-08-06 08:05:46 +00:00
|
|
|
"log"
|
2017-08-05 16:23:55 +00:00
|
|
|
"os"
|
|
|
|
"os/exec"
|
2017-08-06 08:05:46 +00:00
|
|
|
"os/user"
|
2017-08-05 16:23:55 +00:00
|
|
|
"path/filepath"
|
2018-03-02 20:15:36 +00:00
|
|
|
"runtime"
|
2019-03-17 17:16:19 +00:00
|
|
|
"strings"
|
2020-07-03 21:16:16 +00:00
|
|
|
"syscall"
|
2018-03-02 20:15:36 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
2020-12-04 20:59:12 +00:00
|
|
|
envOpener = os.Getenv("OPENER")
|
|
|
|
envEditor = os.Getenv("EDITOR")
|
|
|
|
envPager = os.Getenv("PAGER")
|
|
|
|
envShell = os.Getenv("SHELL")
|
2017-08-05 16:23:55 +00:00
|
|
|
)
|
|
|
|
|
2017-08-06 08:05:46 +00:00
|
|
|
var (
|
2018-02-17 17:04:30 +00:00
|
|
|
gDefaultShell = "sh"
|
2021-04-04 15:23:14 +00:00
|
|
|
gDefaultShellFlag = "-c"
|
2017-08-06 08:05:46 +00:00
|
|
|
gDefaultSocketProt = "unix"
|
|
|
|
gDefaultSocketPath string
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
2018-06-28 16:37:19 +00:00
|
|
|
gUser *user.User
|
|
|
|
gConfigPaths []string
|
2021-05-19 16:54:50 +00:00
|
|
|
gFilesPath string
|
2018-07-11 17:09:26 +00:00
|
|
|
gMarksPath string
|
2018-07-11 17:49:55 +00:00
|
|
|
gHistoryPath string
|
2017-08-06 08:05:46 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
func init() {
|
2018-06-28 18:51:24 +00:00
|
|
|
if envOpener == "" {
|
|
|
|
if runtime.GOOS == "darwin" {
|
|
|
|
envOpener = "open"
|
|
|
|
} else {
|
|
|
|
envOpener = "xdg-open"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if envEditor == "" {
|
|
|
|
envEditor = "vi"
|
|
|
|
}
|
|
|
|
|
|
|
|
if envPager == "" {
|
|
|
|
envPager = "less"
|
|
|
|
}
|
|
|
|
|
|
|
|
if envShell == "" {
|
|
|
|
envShell = "sh"
|
|
|
|
}
|
|
|
|
|
2017-11-19 18:55:13 +00:00
|
|
|
u, err := user.Current()
|
2017-08-06 08:05:46 +00:00
|
|
|
if err != nil {
|
|
|
|
log.Printf("user: %s", err)
|
2018-08-27 16:36:40 +00:00
|
|
|
if os.Getenv("HOME") == "" {
|
|
|
|
log.Print("$HOME variable is empty or not set")
|
|
|
|
}
|
|
|
|
if os.Getenv("USER") == "" {
|
|
|
|
log.Print("$USER variable is empty or not set")
|
|
|
|
}
|
2017-08-06 08:05:46 +00:00
|
|
|
}
|
2017-11-19 18:55:13 +00:00
|
|
|
gUser = u
|
2017-08-06 08:05:46 +00:00
|
|
|
|
|
|
|
config := os.Getenv("XDG_CONFIG_HOME")
|
|
|
|
if config == "" {
|
|
|
|
config = filepath.Join(gUser.HomeDir, ".config")
|
|
|
|
}
|
|
|
|
|
2018-06-28 16:37:19 +00:00
|
|
|
gConfigPaths = []string{
|
2018-07-13 19:46:37 +00:00
|
|
|
filepath.Join("/etc", "lf", "lfrc"),
|
2018-06-28 16:37:19 +00:00
|
|
|
filepath.Join(config, "lf", "lfrc"),
|
|
|
|
}
|
2017-08-06 08:05:46 +00:00
|
|
|
|
2018-07-11 17:09:26 +00:00
|
|
|
data := os.Getenv("XDG_DATA_HOME")
|
|
|
|
if data == "" {
|
|
|
|
data = filepath.Join(gUser.HomeDir, ".local", "share")
|
|
|
|
}
|
|
|
|
|
2021-05-19 16:54:50 +00:00
|
|
|
gFilesPath = filepath.Join(data, "lf", "files")
|
2018-07-11 17:09:26 +00:00
|
|
|
gMarksPath = filepath.Join(data, "lf", "marks")
|
2018-07-11 17:49:55 +00:00
|
|
|
gHistoryPath = filepath.Join(data, "lf", "history")
|
2018-07-11 17:09:26 +00:00
|
|
|
|
2017-08-06 08:05:46 +00:00
|
|
|
gDefaultSocketPath = filepath.Join(os.TempDir(), fmt.Sprintf("lf.%s.sock", gUser.Username))
|
|
|
|
}
|
2017-08-05 16:23:55 +00:00
|
|
|
|
2018-07-31 21:21:55 +00:00
|
|
|
func detachedCommand(name string, arg ...string) *exec.Cmd {
|
2020-12-11 13:34:11 +00:00
|
|
|
cmd := exec.Command(name, arg...)
|
|
|
|
cmd.SysProcAttr = &syscall.SysProcAttr{Setsid: true}
|
|
|
|
return cmd
|
2018-07-31 21:21:55 +00:00
|
|
|
}
|
|
|
|
|
2017-08-05 16:23:55 +00:00
|
|
|
func shellCommand(s string, args []string) *exec.Cmd {
|
|
|
|
if len(gOpts.ifs) != 0 {
|
|
|
|
s = fmt.Sprintf("IFS='%s'; %s", gOpts.ifs, s)
|
|
|
|
}
|
|
|
|
|
2021-04-04 15:23:14 +00:00
|
|
|
args = append([]string{gOpts.shellflag, s, "--"}, args...)
|
2017-08-05 16:23:55 +00:00
|
|
|
|
2018-06-26 18:14:55 +00:00
|
|
|
args = append(gOpts.shellopts, args...)
|
|
|
|
|
2017-08-05 16:23:55 +00:00
|
|
|
return exec.Command(gOpts.shell, args...)
|
|
|
|
}
|
|
|
|
|
2018-03-02 20:15:36 +00:00
|
|
|
func setDefaults() {
|
2018-06-28 18:51:24 +00:00
|
|
|
gOpts.cmds["open"] = &execExpr{"&", `$OPENER "$f"`}
|
|
|
|
gOpts.keys["e"] = &execExpr{"$", `$EDITOR "$f"`}
|
|
|
|
gOpts.keys["i"] = &execExpr{"$", `$PAGER "$f"`}
|
|
|
|
gOpts.keys["w"] = &execExpr{"$", "$SHELL"}
|
2018-07-07 14:50:47 +00:00
|
|
|
|
|
|
|
gOpts.cmds["doc"] = &execExpr{"$", "lf -doc | $PAGER"}
|
|
|
|
gOpts.keys["<f-1>"] = &callExpr{"doc", nil, 1}
|
2018-03-02 20:15:36 +00:00
|
|
|
}
|
|
|
|
|
2018-06-28 21:12:54 +00:00
|
|
|
func isExecutable(f os.FileInfo) bool {
|
|
|
|
return f.Mode()&0111 != 0
|
|
|
|
}
|
2019-03-17 17:16:19 +00:00
|
|
|
|
2020-11-18 19:46:59 +00:00
|
|
|
func isHidden(f os.FileInfo, path string, hiddenfiles []string) bool {
|
2020-06-11 01:11:40 +00:00
|
|
|
hidden := false
|
2020-11-18 19:46:59 +00:00
|
|
|
for _, pattern := range hiddenfiles {
|
2020-06-10 23:52:15 +00:00
|
|
|
matched := matchPattern(strings.TrimPrefix(pattern, "!"), f.Name(), path)
|
2020-06-11 01:11:40 +00:00
|
|
|
if strings.HasPrefix(pattern, "!") && matched {
|
2020-06-10 23:52:15 +00:00
|
|
|
hidden = false
|
|
|
|
} else if matched {
|
|
|
|
hidden = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return hidden
|
|
|
|
}
|
|
|
|
|
2021-12-25 17:49:30 +00:00
|
|
|
func isDirty(f os.FileInfo, path string, dirtyfiles []string) bool {
|
|
|
|
dirty := false
|
|
|
|
for _, pattern := range dirtyfiles {
|
|
|
|
matched := matchPattern(strings.TrimPrefix(pattern, "!"), f.Name(), path)
|
|
|
|
if strings.HasPrefix(pattern, "!") && matched {
|
|
|
|
dirty = false
|
|
|
|
} else if matched {
|
|
|
|
dirty = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return dirty
|
|
|
|
}
|
|
|
|
|
2021-02-12 13:01:54 +00:00
|
|
|
func userName(f os.FileInfo) string {
|
|
|
|
if stat, ok := f.Sys().(*syscall.Stat_t); ok {
|
|
|
|
if u, err := user.LookupId(fmt.Sprint(stat.Uid)); err == nil {
|
|
|
|
return fmt.Sprintf("%v ", u.Username)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
func groupName(f os.FileInfo) string {
|
|
|
|
if stat, ok := f.Sys().(*syscall.Stat_t); ok {
|
|
|
|
if g, err := user.LookupGroupId(fmt.Sprint(stat.Gid)); err == nil {
|
|
|
|
return fmt.Sprintf("%v ", g.Name)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
func linkCount(f os.FileInfo) string {
|
|
|
|
if stat, ok := f.Sys().(*syscall.Stat_t); ok {
|
|
|
|
return fmt.Sprintf("%v ", stat.Nlink)
|
|
|
|
}
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
2020-06-10 23:52:15 +00:00
|
|
|
func matchPattern(pattern, name, path string) bool {
|
2020-06-11 01:11:40 +00:00
|
|
|
s := name
|
2020-06-10 23:52:15 +00:00
|
|
|
|
2020-06-11 01:11:40 +00:00
|
|
|
pattern = replaceTilde(pattern)
|
|
|
|
|
|
|
|
if filepath.IsAbs(pattern) {
|
|
|
|
s = filepath.Join(path, name)
|
2020-06-10 23:52:15 +00:00
|
|
|
}
|
|
|
|
|
2020-06-11 00:14:45 +00:00
|
|
|
// pattern errors are checked when 'hiddenfiles' option is set
|
2020-06-11 01:11:40 +00:00
|
|
|
matched, _ := filepath.Match(pattern, s)
|
2020-06-11 00:14:45 +00:00
|
|
|
|
|
|
|
return matched
|
2019-03-31 20:39:41 +00:00
|
|
|
}
|
|
|
|
|
2020-07-03 21:16:16 +00:00
|
|
|
func errCrossDevice(err error) bool {
|
|
|
|
return err.(*os.LinkError).Err.(syscall.Errno) == syscall.EXDEV
|
|
|
|
}
|
|
|
|
|
2021-03-19 10:58:10 +00:00
|
|
|
func exportFiles(f string, fs []string, pwd string) {
|
2019-03-17 17:16:19 +00:00
|
|
|
envFile := f
|
|
|
|
envFiles := strings.Join(fs, gOpts.filesep)
|
|
|
|
|
|
|
|
os.Setenv("f", envFile)
|
|
|
|
os.Setenv("fs", envFiles)
|
2021-03-19 10:58:10 +00:00
|
|
|
os.Setenv("PWD", pwd)
|
2019-03-17 17:16:19 +00:00
|
|
|
|
|
|
|
if len(fs) == 0 {
|
|
|
|
os.Setenv("fx", envFile)
|
|
|
|
} else {
|
|
|
|
os.Setenv("fx", envFiles)
|
|
|
|
}
|
|
|
|
}
|