/* Copyright © 2022 Seednode */ package cmd import ( "errors" "fmt" "math/rand" "os" "path/filepath" "regexp" "strconv" "strings" "sync" "sync/atomic" "time" "github.com/h2non/filetype" ) var ( ErrNoImagesFound = fmt.Errorf("no supported image formats found") ) type Files struct { Mutex sync.Mutex List map[string][]string } type Stats struct { FilesMatched uint64 FilesSkipped uint64 DirectoriesMatched uint64 } func (s *Stats) GetFilesTotal() uint64 { return atomic.LoadUint64(&s.FilesMatched) + atomic.LoadUint64(&s.FilesSkipped) } func (s *Stats) IncrementFilesMatched() { atomic.AddUint64(&s.FilesMatched, 1) } func (s *Stats) GetFilesMatched() uint64 { return atomic.LoadUint64(&s.FilesMatched) } func (s *Stats) IncrementFilesSkipped() { atomic.AddUint64(&s.FilesSkipped, 1) } func (s *Stats) GetFilesSkipped() uint64 { return atomic.LoadUint64(&s.FilesSkipped) } func (s *Stats) IncrementDirectoriesMatched() { atomic.AddUint64(&s.DirectoriesMatched, 1) } func (s *Stats) GetDirectoriesMatched() uint64 { return atomic.LoadUint64(&s.DirectoriesMatched) } type Path struct { Base string Number int Extension string } func (p *Path) Increment() { p.Number = p.Number + 1 } func (p *Path) Decrement() { p.Number = p.Number - 1 } func appendPath(directory, path string, files *Files, stats *Stats) { files.Mutex.Lock() files.List[directory] = append(files.List[directory], path) files.Mutex.Unlock() stats.IncrementFilesMatched() } func appendPaths(path string, files *Files, filters *Filters, stats *Stats) error { absolutePath, err := filepath.Abs(path) if err != nil { return err } directory, filename := filepath.Split(absolutePath) filename = strings.ToLower(filename) if filters.HasExcludes() { for i := 0; i < len(filters.Excludes); i++ { if strings.Contains( filename, filters.Excludes[i], ) { stats.IncrementFilesSkipped() return nil } } } if filters.HasIncludes() { for i := 0; i < len(filters.Includes); i++ { if strings.Contains( filename, filters.Includes[i], ) { appendPath(directory, path, files, stats) return nil } } stats.IncrementFilesSkipped() return nil } appendPath(directory, path, files, stats) return nil } func getFirstFile(p *Path) (string, error) { p.Number = 1 fileName, err := tryExtensions(p) if err != nil { return "", err } return fileName, nil } func getLastFile(p *Path) (string, error) { var fileName string var err error p.Number = 1 for { p.Increment() fileName, err = tryExtensions(p) if err != nil { return "", err } if fileName == "" { p.Decrement() fileName, err = tryExtensions(p) if err != nil { return "", err } break } } return fileName, nil } func getNextFile(p *Path) (string, error) { p.Increment() fileName, err := tryExtensions(p) if err != nil { return "", err } return fileName, err } func getPreviousFile(p *Path) (string, error) { p.Decrement() fileName, err := tryExtensions(p) if err != nil { return "", err } return fileName, err } func splitPath(path string) (*Path, error) { p := Path{} var err error re := regexp.MustCompile(`(.+)([0-9]{3})(\..+)`) split := re.FindAllStringSubmatch(path, -1) if len(split) < 1 || len(split[0]) < 3 { return &Path{}, nil } p.Base = split[0][1] p.Number, err = strconv.Atoi(split[0][2]) if err != nil { return &Path{}, err } p.Extension = split[0][3] return &p, nil } func tryExtensions(p *Path) (string, error) { extensions := [6]string{p.Extension, ".jpg", ".jpeg", ".png", ".gif", ".webp"} var fileName string for _, i := range extensions { fileName = fmt.Sprintf("%v%.3d%v", p.Base, p.Number, i) exists, err := fileExists(fileName) if err != nil { return "", err } if exists { return fileName, nil } } return "", nil } func fileExists(path string) (bool, error) { _, err := os.Stat(path) switch { case err == nil: return true, nil case errors.Is(err, os.ErrNotExist): return false, nil default: return false, err } } func pathIsValid(filePath string, paths []string) bool { var matchesPrefix = false for i := 0; i < len(paths); i++ { if strings.HasPrefix(filePath, paths[i]) { matchesPrefix = true } } if Verbose && !matchesPrefix { fmt.Printf("%v | Error: Failed to serve file outside specified path(s): %v\n", time.Now().Format(LOGDATE), filePath, ) return false } return true } func isImage(path string) (bool, error) { file, err := os.Open(path) if err != nil { return false, err } defer file.Close() head := make([]byte, 261) file.Read(head) return filetype.IsImage(head), nil } func getFiles(path string, files *Files, filters *Filters, stats *Stats, concurrency *Concurrency) error { var wg sync.WaitGroup err := filepath.WalkDir(path, func(p string, info os.DirEntry, err error) error { if err != nil { return err } switch { case !Recursive && info.IsDir() && p != path: return filepath.SkipDir case !info.IsDir(): wg.Add(1) concurrency.FileScans <- 1 go func() { defer func() { <-concurrency.FileScans wg.Done() }() err = appendPaths(p, files, filters, stats) if err != nil { fmt.Println(err) } }() case info.IsDir(): stats.IncrementDirectoriesMatched() } return err }) wg.Wait() if err != nil { return err } return nil } func getFileList(paths []string, files *Files, filters *Filters, stats *Stats, concurrency *Concurrency) { var wg sync.WaitGroup for i := 0; i < len(paths); i++ { wg.Add(1) concurrency.DirectoryScans <- 1 go func(i int) { defer func() { <-concurrency.DirectoryScans wg.Done() }() err := getFiles(paths[i], files, filters, stats, concurrency) if err != nil { fmt.Println(err) } }(i) } wg.Wait() } func cleanFilename(filename string) string { return filename[:len(filename)-(len(filepath.Ext(filename))+3)] } func prepareDirectory(directory []string) []string { _, first := filepath.Split(directory[0]) first = cleanFilename(first) _, last := filepath.Split(directory[len(directory)-1]) last = cleanFilename(last) if first == last { return append([]string{}, directory[0]) } else { return directory } } func prepareDirectories(files *Files, sort string) []string { directories := []string{} keys := make([]string, len(files.List)) i := 0 for k := range files.List { keys[i] = k i++ } if sort == "asc" || sort == "desc" { for i := 0; i < len(keys); i++ { directories = append(directories, prepareDirectory(files.List[keys[i]])...) } } else { for i := 0; i < len(keys); i++ { directories = append(directories, files.List[keys[i]]...) } } return directories } func pickFile(args []string, filters *Filters, sort string) (string, error) { stats := Stats{} files := Files{} files.List = make(map[string][]string) concurrency := Concurrency{} concurrency.DirectoryScans = make(chan int, maxDirectoryScans) concurrency.FileScans = make(chan int, maxFileScans) startTime := time.Now() getFileList(args, &files, filters, &stats, &concurrency) runTime := time.Since(startTime) if Verbose { fmt.Printf("%v | Scanned %v/%v files across %v directories in %v\n", time.Now().Format(LOGDATE), stats.GetFilesMatched(), stats.GetFilesTotal(), stats.GetDirectoriesMatched(), runTime, ) } fileList := prepareDirectories(&files, sort) rand.Seed(time.Now().UnixNano()) rand.Shuffle(len(fileList), func(i, j int) { fileList[i], fileList[j] = fileList[j], fileList[i] }) for i := 0; i < len(fileList); i++ { filePath := fileList[i] isImage, err := isImage(filePath) if err != nil { return "", err } if isImage { return filePath, nil } } return "", ErrNoImagesFound } func normalizePaths(args []string) ([]string, error) { var paths []string for i := 0; i < len(args); i++ { absolutePath, err := filepath.Abs(args[i]) if err != nil { return nil, err } paths = append(paths, absolutePath) } return paths, nil }