// +build !windows package main import ( "fmt" "log" "os" "os/exec" "os/user" "path/filepath" "runtime" "strings" "syscall" ) var ( envOpener = os.Getenv("OPENER") envEditor = os.Getenv("EDITOR") envPager = os.Getenv("PAGER") envShell = os.Getenv("SHELL") ) var ( gDefaultShell = "sh" gDefaultSocketProt = "unix" gDefaultSocketPath string ) var ( gUser *user.User gConfigPaths []string gMarksPath string gHistoryPath string ) func init() { if envOpener == "" { if runtime.GOOS == "darwin" { envOpener = "open" } else { envOpener = "xdg-open" } } if envEditor == "" { envEditor = "vi" } if envPager == "" { envPager = "less" } if envShell == "" { envShell = "sh" } u, err := user.Current() if err != nil { log.Printf("user: %s", err) 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") } } gUser = u config := os.Getenv("XDG_CONFIG_HOME") if config == "" { config = filepath.Join(gUser.HomeDir, ".config") } gConfigPaths = []string{ filepath.Join("/etc", "lf", "lfrc"), filepath.Join(config, "lf", "lfrc"), } data := os.Getenv("XDG_DATA_HOME") if data == "" { data = filepath.Join(gUser.HomeDir, ".local", "share") } gMarksPath = filepath.Join(data, "lf", "marks") gHistoryPath = filepath.Join(data, "lf", "history") gDefaultSocketPath = filepath.Join(os.TempDir(), fmt.Sprintf("lf.%s.sock", gUser.Username)) } func detachedCommand(name string, arg ...string) *exec.Cmd { cmd := exec.Command(name, arg...) cmd.SysProcAttr = &syscall.SysProcAttr{Setsid: true} return cmd } func shellCommand(s string, args []string) *exec.Cmd { if len(gOpts.ifs) != 0 { s = fmt.Sprintf("IFS='%s'; %s", gOpts.ifs, s) } args = append([]string{"-c", s, "--"}, args...) args = append(gOpts.shellopts, args...) return exec.Command(gOpts.shell, args...) } func setDefaults() { gOpts.cmds["open"] = &execExpr{"&", `$OPENER "$f"`} gOpts.keys["e"] = &execExpr{"$", `$EDITOR "$f"`} gOpts.keys["i"] = &execExpr{"$", `$PAGER "$f"`} gOpts.keys["w"] = &execExpr{"$", "$SHELL"} gOpts.cmds["doc"] = &execExpr{"$", "lf -doc | $PAGER"} gOpts.keys[""] = &callExpr{"doc", nil, 1} } func isExecutable(f os.FileInfo) bool { return f.Mode()&0111 != 0 } func isHidden(f os.FileInfo, path string, hiddenfiles []string) bool { hidden := false for _, pattern := range hiddenfiles { matched := matchPattern(strings.TrimPrefix(pattern, "!"), f.Name(), path) if strings.HasPrefix(pattern, "!") && matched { hidden = false } else if matched { hidden = true } } return hidden } 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 "" } func matchPattern(pattern, name, path string) bool { s := name pattern = replaceTilde(pattern) if filepath.IsAbs(pattern) { s = filepath.Join(path, name) } // pattern errors are checked when 'hiddenfiles' option is set matched, _ := filepath.Match(pattern, s) return matched } func errCrossDevice(err error) bool { return err.(*os.LinkError).Err.(syscall.Errno) == syscall.EXDEV } func exportFiles(f string, fs []string, pwd string) { envFile := f envFiles := strings.Join(fs, gOpts.filesep) os.Setenv("f", envFile) os.Setenv("fs", envFiles) os.Setenv("PWD", pwd) if len(fs) == 0 { os.Setenv("fx", envFile) } else { os.Setenv("fx", envFiles) } }