2020-03-08 03:24:41 +03:00
|
|
|
package ls
|
|
|
|
import(
|
|
|
|
"os"
|
|
|
|
"fmt"
|
|
|
|
"flag"
|
|
|
|
"strings"
|
2020-05-06 03:57:54 +03:00
|
|
|
"regexp"
|
2020-11-25 02:00:35 +03:00
|
|
|
"path"
|
|
|
|
)
|
|
|
|
|
|
|
|
var(
|
|
|
|
listHidden bool
|
2020-12-31 17:39:03 +03:00
|
|
|
args []string
|
|
|
|
arg0 string
|
|
|
|
flagSet *flag.FlagSet
|
2020-03-08 03:24:41 +03:00
|
|
|
)
|
|
|
|
|
2020-12-31 17:39:03 +03:00
|
|
|
var slashRegexp = regexp.MustCompile("/+")
|
|
|
|
|
2020-03-08 03:24:41 +03:00
|
|
|
func IsDir(p string) (bool, error) {
|
|
|
|
finfo, e := os.Stat(p)
|
|
|
|
if e != nil {
|
|
|
|
return false, e
|
|
|
|
}
|
|
|
|
return finfo.IsDir(), e
|
|
|
|
}
|
|
|
|
|
|
|
|
func ReadDir(p string) ([]os.FileInfo, error) {
|
|
|
|
f, e := os.Open(p)
|
|
|
|
if e != nil {
|
|
|
|
return nil, e
|
|
|
|
}
|
|
|
|
l, e := f.Readdir(-1)
|
|
|
|
if e != nil {
|
|
|
|
return nil, e
|
|
|
|
}
|
|
|
|
f.Close()
|
|
|
|
|
|
|
|
return l, nil
|
|
|
|
}
|
|
|
|
|
2020-12-31 17:39:03 +03:00
|
|
|
func
|
|
|
|
Stat(p string) (os.FileInfo, error) {
|
2020-03-08 03:24:41 +03:00
|
|
|
f, e := os.Open(p)
|
|
|
|
if e != nil {
|
|
|
|
return nil, e
|
|
|
|
}
|
|
|
|
s, e := f.Stat()
|
|
|
|
f.Close()
|
|
|
|
return s, nil
|
|
|
|
}
|
|
|
|
|
2020-12-31 17:39:03 +03:00
|
|
|
func findString(slice []string, val string) (int, bool) {
|
|
|
|
for i, v := range slice {
|
|
|
|
if v == val {
|
|
|
|
return i, true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return -1, false
|
|
|
|
}
|
|
|
|
|
|
|
|
func
|
|
|
|
isHidden(p string) bool {
|
|
|
|
if path.Base(p)[0] == '.' {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2020-11-25 02:00:35 +03:00
|
|
|
func shouldList(p string) bool {
|
2020-12-31 17:39:03 +03:00
|
|
|
if _, found := findString(args, p) ; found {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
if !listHidden && isHidden(p) {
|
2020-11-25 02:00:35 +03:00
|
|
|
return false
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
2020-12-31 17:39:03 +03:00
|
|
|
func
|
|
|
|
deleteExceedSlashes(p string) string {
|
|
|
|
p = slashRegexp.ReplaceAllString(p, "/")
|
|
|
|
if p != "/" { // Do not trim if it is root dir.
|
|
|
|
p = strings.TrimRight(p, "/")
|
|
|
|
}
|
|
|
|
return p
|
|
|
|
}
|
|
|
|
|
|
|
|
func
|
|
|
|
ls(p string, fold int) error {
|
2020-11-25 02:00:35 +03:00
|
|
|
if !shouldList(p) {
|
|
|
|
return nil
|
|
|
|
}
|
2020-12-31 17:39:03 +03:00
|
|
|
|
|
|
|
p = deleteExceedSlashes(p)
|
|
|
|
|
2020-03-08 03:24:41 +03:00
|
|
|
isDir, e := IsDir(p)
|
|
|
|
if e != nil {
|
|
|
|
return e
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-05-06 03:36:41 +03:00
|
|
|
if !isDir {
|
2020-05-06 03:57:54 +03:00
|
|
|
fmt.Println(p)
|
2020-05-06 03:36:41 +03:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
if fold>0 {
|
|
|
|
/* It's a directory and it can be ls-ed. */
|
2020-05-06 03:57:54 +03:00
|
|
|
l, e := ReadDir(p)
|
2020-04-23 10:41:19 +03:00
|
|
|
if e!=nil {
|
2020-03-08 03:24:41 +03:00
|
|
|
return e
|
|
|
|
}
|
|
|
|
for _, f := range l {
|
2020-05-06 03:57:54 +03:00
|
|
|
s := p+"/"+f.Name()
|
2020-05-06 03:36:41 +03:00
|
|
|
if b, _ := IsDir(s) ; b && fold!=1 {
|
|
|
|
fmt.Println(s)
|
2020-04-23 10:41:19 +03:00
|
|
|
}
|
2020-05-06 03:36:41 +03:00
|
|
|
ls(s, fold-1)
|
2020-03-08 03:24:41 +03:00
|
|
|
}
|
2020-05-05 09:04:29 +03:00
|
|
|
} else {
|
2020-05-06 03:36:41 +03:00
|
|
|
/* It's finish directory. Fold==0 or fold<0. */
|
2020-05-06 03:57:54 +03:00
|
|
|
fmt.Println(p)
|
2020-03-08 03:24:41 +03:00
|
|
|
}
|
2020-04-23 10:41:19 +03:00
|
|
|
|
2020-03-08 03:24:41 +03:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-12-31 17:39:03 +03:00
|
|
|
func Run(argv []string) int {
|
2020-03-08 03:24:41 +03:00
|
|
|
status := 0
|
2020-12-31 17:39:03 +03:00
|
|
|
arg0 = argv[0]
|
|
|
|
args = argv[1:]
|
|
|
|
flagSet = flag.NewFlagSet(arg0, flag.ExitOnError)
|
2020-05-05 09:04:29 +03:00
|
|
|
var foldLvl int
|
2020-04-01 22:25:52 +03:00
|
|
|
flagSet.IntVar(&foldLvl, "r", 1, "List recursively with choosing deepness, can't be negative or zero.")
|
2020-11-25 02:00:35 +03:00
|
|
|
flagSet.BoolVar(&listHidden, "a", false, "List hidden files.")
|
2020-03-08 03:24:41 +03:00
|
|
|
flagSet.Usage = func() {
|
2020-12-31 17:39:03 +03:00
|
|
|
fmt.Fprintf(os.Stderr, "usage: %s [files]\n", arg0, arg0)
|
2020-03-08 03:24:41 +03:00
|
|
|
flagSet.PrintDefaults()
|
|
|
|
}
|
|
|
|
flagSet.Parse(args)
|
2020-05-05 09:04:29 +03:00
|
|
|
args = flagSet.Args()
|
|
|
|
|
|
|
|
if foldLvl<0 {
|
2020-04-01 22:25:52 +03:00
|
|
|
flagSet.Usage()
|
|
|
|
return 1
|
|
|
|
}
|
2020-05-05 09:04:29 +03:00
|
|
|
|
|
|
|
if foldLvl==0 && len(args)==0 {
|
2020-04-01 22:25:52 +03:00
|
|
|
flagSet.Usage()
|
|
|
|
return 1
|
|
|
|
}
|
2020-05-05 09:04:29 +03:00
|
|
|
|
2020-03-08 03:24:41 +03:00
|
|
|
if len(args) == 0 {
|
2020-05-05 09:04:29 +03:00
|
|
|
foldLvl -= 1
|
|
|
|
if l, e := ReadDir(".") ; e != nil {
|
2020-03-08 03:24:41 +03:00
|
|
|
status = 1
|
|
|
|
fmt.Fprintf(os.Stderr, "%s: %s.\n", arg0, e)
|
|
|
|
} else {
|
|
|
|
for _, f := range l {
|
2020-11-25 02:00:35 +03:00
|
|
|
if !shouldList(f.Name()) {
|
|
|
|
continue
|
|
|
|
}
|
2020-05-05 09:04:29 +03:00
|
|
|
isDir, _ := IsDir(f.Name())
|
2020-05-06 03:36:41 +03:00
|
|
|
|
|
|
|
fmt.Println(f.Name())
|
2020-05-05 09:04:29 +03:00
|
|
|
if isDir && foldLvl>0 {
|
|
|
|
e := ls(f.Name(), foldLvl)
|
|
|
|
if e!=nil {
|
|
|
|
status = 1
|
2020-12-31 17:39:03 +03:00
|
|
|
fmt.Fprintf(os.Stderr,
|
|
|
|
"%s: %s\n",
|
|
|
|
arg0, e)
|
2020-05-05 09:04:29 +03:00
|
|
|
}
|
2020-04-22 19:44:39 +03:00
|
|
|
}
|
2020-03-08 03:24:41 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
for _, p := range args {
|
|
|
|
e := ls(p, foldLvl)
|
|
|
|
if e != nil {
|
|
|
|
status = 1
|
2020-12-31 17:39:03 +03:00
|
|
|
fmt.Fprintf(os.Stderr, "%s: %s\n", arg0, e)
|
2020-03-08 03:24:41 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return status
|
|
|
|
}
|