@ -2,6 +2,7 @@ package fsutil
import (
import (
"context"
"context"
gofs "io/fs"
"os"
"os"
"path/filepath"
"path/filepath"
"strings"
"strings"
@ -47,11 +48,11 @@ func Walk(ctx context.Context, p string, opt *WalkOpt, fn filepath.WalkFunc) err
if err != nil {
if err != nil {
return errors . WithStack ( & os . PathError { Op : "resolve" , Path : root , Err : err } )
return errors . WithStack ( & os . PathError { Op : "resolve" , Path : root , Err : err } )
}
}
fi , err := os . Stat ( root )
rootFI , err := os . Stat ( root )
if err != nil {
if err != nil {
return errors . WithStack ( err )
return errors . WithStack ( err )
}
}
if ! fi . IsDir ( ) {
if ! rootFI . IsDir ( ) {
return errors . WithStack ( & os . PathError { Op : "walk" , Path : root , Err : syscall . ENOTDIR } )
return errors . WithStack ( & os . PathError { Op : "walk" , Path : root , Err : syscall . ENOTDIR } )
}
}
@ -126,7 +127,7 @@ func Walk(ctx context.Context, p string, opt *WalkOpt, fn filepath.WalkFunc) err
var parentDirs [ ] visitedDir
var parentDirs [ ] visitedDir
seenFiles := make ( map [ uint64 ] string )
seenFiles := make ( map [ uint64 ] string )
return filepath . Walk ( root , func ( path string , fi os . FileInfo , walkErr error ) ( retErr error ) {
return filepath . Walk Dir ( root , func ( path string , dirEntry gofs . DirEntry , walkErr error ) ( retErr error ) {
defer func ( ) {
defer func ( ) {
if retErr != nil && isNotExist ( retErr ) {
if retErr != nil && isNotExist ( retErr ) {
retErr = filepath . SkipDir
retErr = filepath . SkipDir
@ -146,9 +147,10 @@ func Walk(ctx context.Context, p string, opt *WalkOpt, fn filepath.WalkFunc) err
var (
var (
dir visitedDir
dir visitedDir
isDir bool
isDir bool
fi gofs . FileInfo
)
)
if fi != nil {
if dirEntry != nil {
isDir = fi . IsDir ( )
isDir = dirEntry . IsDir ( )
}
}
if includeMatcher != nil || excludeMatcher != nil {
if includeMatcher != nil || excludeMatcher != nil {
@ -161,6 +163,11 @@ func Walk(ctx context.Context, p string, opt *WalkOpt, fn filepath.WalkFunc) err
}
}
if isDir {
if isDir {
fi , err = dirEntry . Info ( )
if err != nil {
return err
}
dir = visitedDir {
dir = visitedDir {
fi : fi ,
fi : fi ,
path : path ,
path : path ,
@ -268,6 +275,14 @@ func Walk(ctx context.Context, p string, opt *WalkOpt, fn filepath.WalkFunc) err
dir . calledFn = true
dir . calledFn = true
// The FileInfo might have already been read further up.
if fi == nil {
fi , err = dirEntry . Info ( )
if err != nil {
return err
}
}
stat , err := mkstat ( origpath , path , fi , seenFiles )
stat , err := mkstat ( origpath , path , fi , seenFiles )
if err != nil {
if err != nil {
return err
return err