implement in order of creation for oldFiles
All checks were successful
go/folderwatcher/pipeline/head This commit looks good
All checks were successful
go/folderwatcher/pipeline/head This commit looks good
This commit is contained in:
parent
479ed620e3
commit
097c4b3d84
4 changed files with 324 additions and 179 deletions
102
folderwatcher.go
102
folderwatcher.go
|
@ -5,6 +5,7 @@ import (
|
|||
"os"
|
||||
"path/filepath"
|
||||
"regexp"
|
||||
"slices"
|
||||
"sync"
|
||||
"time"
|
||||
|
||||
|
@ -22,8 +23,10 @@ type FolderWatcher struct {
|
|||
quit chan struct{}
|
||||
mutex sync.Mutex
|
||||
wg sync.WaitGroup
|
||||
debounceWg sync.WaitGroup
|
||||
debounceMap map[string]*time.Timer
|
||||
debouncedChan chan string
|
||||
existingList []string
|
||||
enqueuedFiles int
|
||||
}
|
||||
|
||||
|
@ -71,6 +74,7 @@ func NewFolderWatcher(conf Config, includeExisting bool, quit chan struct{}) (*F
|
|||
quit: quit,
|
||||
debounceMap: make(map[string]*time.Timer),
|
||||
debouncedChan: make(chan string),
|
||||
existingList: []string{},
|
||||
}
|
||||
|
||||
if includeExisting {
|
||||
|
@ -81,10 +85,18 @@ func NewFolderWatcher(conf Config, includeExisting bool, quit chan struct{}) (*F
|
|||
}
|
||||
for _, entry := range entries {
|
||||
if !entry.IsDir() {
|
||||
folderWatcher.foundFileEvent(filepath.Join(path, entry.Name()))
|
||||
fullFilePath := filepath.Join(path, entry.Name())
|
||||
if folderWatcher.matchesPattern(fullFilePath) {
|
||||
folderWatcher.existingList = append(folderWatcher.existingList, fullFilePath)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
slices.SortFunc(folderWatcher.existingList, func(a, b string) int {
|
||||
infoA, _ := os.Stat(a)
|
||||
infoB, _ := os.Stat(b)
|
||||
return infoA.ModTime().Compare(infoB.ModTime())
|
||||
})
|
||||
}
|
||||
|
||||
return folderWatcher, nil
|
||||
|
@ -95,21 +107,14 @@ type HandlerCallback func(path string, err error)
|
|||
|
||||
func (w *FolderWatcher) Watch(handler FileHandler, callback HandlerCallback) {
|
||||
defer w.watcher.Close()
|
||||
for stopped := false; !stopped; {
|
||||
select {
|
||||
case <-w.quit:
|
||||
logger.Info("Stopping watcher")
|
||||
stopped = true
|
||||
case event, ok := <-w.watcher.Events:
|
||||
if !ok {
|
||||
stopped = true
|
||||
break
|
||||
}
|
||||
logger.Debug("event:", event)
|
||||
if event.Has(fsnotify.Write) || event.Has(fsnotify.Create) {
|
||||
w.foundFileEvent(event.Name)
|
||||
}
|
||||
case file, _ := <-w.debouncedChan:
|
||||
initialWait := sync.WaitGroup{}
|
||||
|
||||
handlerWg := sync.WaitGroup{}
|
||||
handlerWg.Add(1)
|
||||
// Start File Handling
|
||||
go func() {
|
||||
defer handlerWg.Done()
|
||||
for file := range w.debouncedChan {
|
||||
triggerCallback, err := handler(file)
|
||||
if err != nil {
|
||||
logger.Warn("error during handle:", err)
|
||||
|
@ -117,7 +122,42 @@ func (w *FolderWatcher) Watch(handler FileHandler, callback HandlerCallback) {
|
|||
if triggerCallback {
|
||||
callback(file, err)
|
||||
}
|
||||
w.wg.Done()
|
||||
w.wg.Done() // mark enqueued file as done
|
||||
}
|
||||
}()
|
||||
|
||||
if len(w.existingList) > 0 {
|
||||
logger.Infof("got %d existing files", len(w.existingList))
|
||||
initialWait.Add(1)
|
||||
go func() {
|
||||
defer initialWait.Done()
|
||||
for _, path := range w.existingList {
|
||||
w.enqueue(path)
|
||||
}
|
||||
}()
|
||||
}
|
||||
|
||||
// Start Event Handling
|
||||
for stopped := false; !stopped; {
|
||||
select {
|
||||
case <-w.quit:
|
||||
logger.Info("Stopping watcher")
|
||||
w.debounceWg.Wait() // wait for debounce to settle
|
||||
w.wg.Wait() // wait for existing files to be handled
|
||||
close(w.debouncedChan) //close the debounced chan to mark it as done
|
||||
stopped = true
|
||||
case event, ok := <-w.watcher.Events:
|
||||
initialWait.Wait() // wait that all initial are enqueued, before handling of first event
|
||||
if !ok {
|
||||
stopped = true
|
||||
break
|
||||
}
|
||||
logger.Debug("event:", event)
|
||||
if event.Has(fsnotify.Write) || event.Has(fsnotify.Create) {
|
||||
if w.matchesPattern(event.Name) {
|
||||
w.enqueueDebounce(event.Name)
|
||||
}
|
||||
}
|
||||
case err, ok := <-w.watcher.Errors:
|
||||
if !ok {
|
||||
stopped = true
|
||||
|
@ -126,31 +166,39 @@ func (w *FolderWatcher) Watch(handler FileHandler, callback HandlerCallback) {
|
|||
logger.Warn("error:", err)
|
||||
}
|
||||
}
|
||||
w.wg.Wait()
|
||||
handlerWg.Wait() // wait for filehandler to be closed, implies debouncedChan is closed
|
||||
logger.Infof("watcher handled %d files", w.enqueuedFiles)
|
||||
}
|
||||
|
||||
func (w *FolderWatcher) foundFileEvent(fileName string) {
|
||||
func (w *FolderWatcher) matchesPattern(fileName string) bool {
|
||||
baseName := filepath.Base(fileName)
|
||||
if w.regex == nil || w.regex.MatchString(baseName) {
|
||||
w.enqueueDebounce(fileName)
|
||||
return w.regex == nil || w.regex.MatchString(baseName)
|
||||
}
|
||||
|
||||
func (w *FolderWatcher) enqueue(f string) {
|
||||
w.wg.Add(1)
|
||||
w.mutex.Lock()
|
||||
defer w.mutex.Unlock()
|
||||
if timer, ok := w.debounceMap[f]; ok && timer != nil {
|
||||
timer.Stop()
|
||||
}
|
||||
delete(w.debounceMap, f)
|
||||
w.enqueuedFiles = w.enqueuedFiles + 1
|
||||
w.debouncedChan <- f
|
||||
}
|
||||
|
||||
func (w *FolderWatcher) enqueueDebounce(f string) {
|
||||
logger.Infof("Debounced Enqueue of %s", f)
|
||||
w.mutex.Lock()
|
||||
defer w.mutex.Unlock()
|
||||
|
||||
if timer, ok := w.debounceMap[f]; ok && timer != nil {
|
||||
timer.Stop()
|
||||
} else {
|
||||
w.wg.Add(1)
|
||||
w.debounceWg.Add(1)
|
||||
}
|
||||
w.debounceMap[f] = time.AfterFunc(DebounceTime, func() {
|
||||
w.mutex.Lock()
|
||||
defer w.mutex.Unlock()
|
||||
delete(w.debounceMap, f)
|
||||
w.enqueuedFiles = w.enqueuedFiles + 1
|
||||
w.debouncedChan <- f
|
||||
defer w.debounceWg.Done()
|
||||
w.enqueue(f)
|
||||
})
|
||||
}
|
||||
|
|
162
folderwatcher_globbing_test.go
Normal file
162
folderwatcher_globbing_test.go
Normal file
|
@ -0,0 +1,162 @@
|
|||
package folderwatcher
|
||||
|
||||
import (
|
||||
"context"
|
||||
"os"
|
||||
"path/filepath"
|
||||
"strconv"
|
||||
"sync"
|
||||
"testing"
|
||||
"time"
|
||||
)
|
||||
|
||||
func TestGlobbingN1(t *testing.T) {
|
||||
tmpPath, err := os.MkdirTemp("", ".folderwatcher_test")
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
defer os.RemoveAll(tmpPath)
|
||||
quitChan := make(chan struct{})
|
||||
|
||||
_ = os.MkdirAll(filepath.Join(tmpPath, "g0"), os.ModePerm)
|
||||
_ = os.MkdirAll(filepath.Join(tmpPath, "g1"), os.ModePerm)
|
||||
_ = os.MkdirAll(filepath.Join(tmpPath, "g2"), os.ModePerm)
|
||||
|
||||
filesIn := []string{"A", "B", "C"}
|
||||
var filesOut []string
|
||||
|
||||
wg := sync.WaitGroup{}
|
||||
|
||||
conf := Config{
|
||||
Folder: filepath.Join(tmpPath, "*"),
|
||||
}
|
||||
|
||||
watcher, err := NewFolderWatcher(conf, true, quitChan)
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
|
||||
go watcher.Watch(func(filePath string) (bool, error) {
|
||||
if !filepath.IsAbs(filePath) {
|
||||
t.Errorf("file %s is not an absolute path", filePath)
|
||||
}
|
||||
rel, err := filepath.Rel(tmpPath, filePath)
|
||||
if err != nil {
|
||||
return true, err
|
||||
}
|
||||
filesOut = append(filesOut, rel)
|
||||
wg.Done()
|
||||
return true, nil
|
||||
}, func(s string, err error) {
|
||||
})
|
||||
|
||||
for i, f := range filesIn {
|
||||
wg.Add(1)
|
||||
_ = os.WriteFile(filepath.Join(tmpPath, "g"+strconv.Itoa(i), f), []byte{0}, os.ModePerm)
|
||||
}
|
||||
|
||||
finishedChan := make(chan struct{})
|
||||
go func() {
|
||||
wg.Wait()
|
||||
finishedChan <- struct{}{}
|
||||
}()
|
||||
|
||||
ctx, ctxCancel := context.WithTimeout(context.Background(), 10*time.Second)
|
||||
defer ctxCancel()
|
||||
|
||||
select {
|
||||
case <-finishedChan:
|
||||
case <-ctx.Done():
|
||||
t.Error(ctx.Err())
|
||||
}
|
||||
quitChan <- struct{}{}
|
||||
|
||||
if len(filesOut) != len(filesIn) {
|
||||
t.Errorf("filesOut length %d != %d", len(filesOut), len(filesIn))
|
||||
}
|
||||
for i, f := range filesIn {
|
||||
if !Contains(filesOut, filepath.Join("g"+strconv.Itoa(i), f)) {
|
||||
t.Errorf("File %s not found in %s", f, filesOut)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func TestGlobbingN2(t *testing.T) {
|
||||
tmpPath, err := os.MkdirTemp("", ".folderwatcher_test")
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
defer os.RemoveAll(tmpPath)
|
||||
quitChan := make(chan struct{})
|
||||
|
||||
_ = os.MkdirAll(filepath.Join(tmpPath, "g0", "s0"), os.ModePerm)
|
||||
_ = os.MkdirAll(filepath.Join(tmpPath, "g0", "s2"), os.ModePerm)
|
||||
_ = os.MkdirAll(filepath.Join(tmpPath, "g1", "s1"), os.ModePerm)
|
||||
_ = os.MkdirAll(filepath.Join(tmpPath, "g1", "s3"), os.ModePerm)
|
||||
_ = os.MkdirAll(filepath.Join(tmpPath, "gX"), os.ModePerm)
|
||||
|
||||
filesIn := []string{"A", "B", "C", "D"}
|
||||
var filesOut []string
|
||||
|
||||
wg := sync.WaitGroup{}
|
||||
|
||||
conf := Config{
|
||||
Folder: filepath.Join(tmpPath, "*", "*"),
|
||||
}
|
||||
|
||||
watcher, err := NewFolderWatcher(conf, true, quitChan)
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
|
||||
go watcher.Watch(func(filePath string) (bool, error) {
|
||||
if !filepath.IsAbs(filePath) {
|
||||
t.Errorf("file %s is not an absolute path", filePath)
|
||||
}
|
||||
rel, err := filepath.Rel(tmpPath, filePath)
|
||||
if err != nil {
|
||||
return true, err
|
||||
}
|
||||
filesOut = append(filesOut, rel)
|
||||
wg.Done()
|
||||
return true, nil
|
||||
}, func(s string, err error) {
|
||||
})
|
||||
|
||||
for i, f := range filesIn {
|
||||
wg.Add(1)
|
||||
_ = os.WriteFile(filepath.Join(tmpPath, "g"+strconv.Itoa(i%2), "s"+strconv.Itoa(i), f), []byte{0}, os.ModePerm)
|
||||
}
|
||||
|
||||
wg.Add(1)
|
||||
_ = os.WriteFile(filepath.Join(tmpPath, "gX", "foo"), []byte{0}, os.ModePerm)
|
||||
|
||||
finishedChan := make(chan struct{})
|
||||
go func() {
|
||||
wg.Wait()
|
||||
finishedChan <- struct{}{}
|
||||
}()
|
||||
|
||||
ctx, ctxCancel := context.WithTimeout(context.Background(), 5*time.Second)
|
||||
defer ctxCancel()
|
||||
|
||||
select {
|
||||
case <-finishedChan:
|
||||
case <-ctx.Done():
|
||||
wg.Done() // ctx Will Timeout, because gX/foo will always be remaining
|
||||
// t.Error(ctx.Err())
|
||||
}
|
||||
quitChan <- struct{}{}
|
||||
|
||||
if len(filesOut) != len(filesIn) {
|
||||
t.Errorf("filesOut length %d != %d", len(filesOut), len(filesIn))
|
||||
}
|
||||
for i, f := range filesIn {
|
||||
if !Contains(filesOut, filepath.Join("g"+strconv.Itoa(i%2), "s"+strconv.Itoa(i), f)) {
|
||||
t.Errorf("File %s not found in %s", f, filesOut)
|
||||
}
|
||||
}
|
||||
if Contains(filesOut, filepath.Join("gX", "foo")) {
|
||||
t.Errorf("File %s found in %s", filepath.Join("gX", "foo"), filesOut)
|
||||
}
|
||||
}
|
87
folderwatcher_oldFilesInOrder_test.go
Normal file
87
folderwatcher_oldFilesInOrder_test.go
Normal file
|
@ -0,0 +1,87 @@
|
|||
package folderwatcher
|
||||
|
||||
import (
|
||||
"context"
|
||||
"fmt"
|
||||
"os"
|
||||
"path/filepath"
|
||||
"sync"
|
||||
"testing"
|
||||
"time"
|
||||
)
|
||||
|
||||
func TestOldFilesInCreationOrder(t *testing.T) {
|
||||
tmpPath, err := os.MkdirTemp("", ".folderwatcher_test")
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
defer os.RemoveAll(tmpPath)
|
||||
quitChan := make(chan struct{})
|
||||
|
||||
filesIn := []string{"2", "C", "B", "A", "1"}
|
||||
makeAbsFilePaths(t, tmpPath, &filesIn)
|
||||
var filesOut []string
|
||||
|
||||
wg := sync.WaitGroup{}
|
||||
|
||||
conf := Config{
|
||||
Folder: tmpPath,
|
||||
}
|
||||
|
||||
for _, f := range filesIn {
|
||||
wg.Add(1)
|
||||
_ = os.WriteFile(f, []byte{0}, os.ModePerm)
|
||||
time.Sleep(10 * time.Millisecond)
|
||||
}
|
||||
|
||||
watcher, err := NewFolderWatcher(conf, true, quitChan)
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
|
||||
go watcher.Watch(func(filePath string) (bool, error) {
|
||||
if !filepath.IsAbs(filePath) {
|
||||
t.Errorf("file %s is not an absolute path", filePath)
|
||||
}
|
||||
filesOut = append(filesOut, filePath)
|
||||
wg.Done()
|
||||
return true, nil
|
||||
}, func(s string, err error) {
|
||||
})
|
||||
|
||||
finishedChan := make(chan struct{})
|
||||
go func() {
|
||||
wg.Wait()
|
||||
finishedChan <- struct{}{}
|
||||
}()
|
||||
|
||||
ctx, ctxCancel := context.WithTimeout(context.Background(), 10*time.Second)
|
||||
defer ctxCancel()
|
||||
|
||||
select {
|
||||
case <-finishedChan:
|
||||
case <-ctx.Done():
|
||||
t.Error(ctx.Err())
|
||||
}
|
||||
quitChan <- struct{}{}
|
||||
|
||||
if len(filesOut) != len(filesIn) {
|
||||
t.Errorf("filesOut length %d != %d", len(filesOut), len(filesIn))
|
||||
}
|
||||
|
||||
filesInS := fmt.Sprintf("%s", filesIn)
|
||||
filesOutS := fmt.Sprintf("%s", filesOut)
|
||||
if filesInS != filesOutS {
|
||||
t.Errorf("filesIn != filesOut %s != %s", filesIn, filesOut)
|
||||
}
|
||||
}
|
||||
|
||||
func makeAbsFilePaths(t *testing.T, rootPath string, filesIn *[]string) {
|
||||
t.Helper()
|
||||
for i, f := range *filesIn { //remap filesIn to Full Path
|
||||
(*filesIn)[i] = filepath.Join(rootPath, f)
|
||||
if !filepath.IsAbs((*filesIn)[i]) {
|
||||
t.Fatalf("file %s is not an absolute path", (*filesIn)[i])
|
||||
}
|
||||
}
|
||||
}
|
|
@ -4,7 +4,6 @@ import (
|
|||
"context"
|
||||
"os"
|
||||
"path/filepath"
|
||||
"strconv"
|
||||
"sync"
|
||||
"testing"
|
||||
"time"
|
||||
|
@ -314,157 +313,6 @@ func TestOldAndNewFileHandlingWithoutExisting(t *testing.T) {
|
|||
}
|
||||
}
|
||||
|
||||
func TestGlobbingN1(t *testing.T) {
|
||||
tmpPath, err := os.MkdirTemp("", ".folderwatcher_test")
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
defer os.RemoveAll(tmpPath)
|
||||
quitChan := make(chan struct{})
|
||||
|
||||
_ = os.MkdirAll(filepath.Join(tmpPath, "g0"), os.ModePerm)
|
||||
_ = os.MkdirAll(filepath.Join(tmpPath, "g1"), os.ModePerm)
|
||||
_ = os.MkdirAll(filepath.Join(tmpPath, "g2"), os.ModePerm)
|
||||
|
||||
filesIn := []string{"A", "B", "C"}
|
||||
var filesOut []string
|
||||
|
||||
wg := sync.WaitGroup{}
|
||||
|
||||
conf := Config{
|
||||
Folder: filepath.Join(tmpPath, "*"),
|
||||
}
|
||||
|
||||
watcher, err := NewFolderWatcher(conf, true, quitChan)
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
|
||||
go watcher.Watch(func(filePath string) (bool, error) {
|
||||
if !filepath.IsAbs(filePath) {
|
||||
t.Errorf("file %s is not an absolute path", filePath)
|
||||
}
|
||||
rel, err := filepath.Rel(tmpPath, filePath)
|
||||
if err != nil {
|
||||
return true, err
|
||||
}
|
||||
filesOut = append(filesOut, rel)
|
||||
wg.Done()
|
||||
return true, nil
|
||||
}, func(s string, err error) {
|
||||
})
|
||||
|
||||
for i, f := range filesIn {
|
||||
wg.Add(1)
|
||||
_ = os.WriteFile(filepath.Join(tmpPath, "g"+strconv.Itoa(i), f), []byte{0}, os.ModePerm)
|
||||
}
|
||||
|
||||
finishedChan := make(chan struct{})
|
||||
go func() {
|
||||
wg.Wait()
|
||||
finishedChan <- struct{}{}
|
||||
}()
|
||||
|
||||
ctx, ctxCancel := context.WithTimeout(context.Background(), 10*time.Second)
|
||||
defer ctxCancel()
|
||||
|
||||
select {
|
||||
case <-finishedChan:
|
||||
case <-ctx.Done():
|
||||
t.Error(ctx.Err())
|
||||
}
|
||||
quitChan <- struct{}{}
|
||||
|
||||
if len(filesOut) != len(filesIn) {
|
||||
t.Errorf("filesOut length %d != %d", len(filesOut), len(filesIn))
|
||||
}
|
||||
for i, f := range filesIn {
|
||||
if !Contains(filesOut, filepath.Join("g"+strconv.Itoa(i), f)) {
|
||||
t.Errorf("File %s not found in %s", f, filesOut)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func TestGlobbingN2(t *testing.T) {
|
||||
tmpPath, err := os.MkdirTemp("", ".folderwatcher_test")
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
defer os.RemoveAll(tmpPath)
|
||||
quitChan := make(chan struct{})
|
||||
|
||||
_ = os.MkdirAll(filepath.Join(tmpPath, "g0", "s0"), os.ModePerm)
|
||||
_ = os.MkdirAll(filepath.Join(tmpPath, "g0", "s2"), os.ModePerm)
|
||||
_ = os.MkdirAll(filepath.Join(tmpPath, "g1", "s1"), os.ModePerm)
|
||||
_ = os.MkdirAll(filepath.Join(tmpPath, "g1", "s3"), os.ModePerm)
|
||||
_ = os.MkdirAll(filepath.Join(tmpPath, "gX"), os.ModePerm)
|
||||
|
||||
filesIn := []string{"A", "B", "C", "D"}
|
||||
var filesOut []string
|
||||
|
||||
wg := sync.WaitGroup{}
|
||||
|
||||
conf := Config{
|
||||
Folder: filepath.Join(tmpPath, "*", "*"),
|
||||
}
|
||||
|
||||
watcher, err := NewFolderWatcher(conf, true, quitChan)
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
|
||||
go watcher.Watch(func(filePath string) (bool, error) {
|
||||
if !filepath.IsAbs(filePath) {
|
||||
t.Errorf("file %s is not an absolute path", filePath)
|
||||
}
|
||||
rel, err := filepath.Rel(tmpPath, filePath)
|
||||
if err != nil {
|
||||
return true, err
|
||||
}
|
||||
filesOut = append(filesOut, rel)
|
||||
wg.Done()
|
||||
return true, nil
|
||||
}, func(s string, err error) {
|
||||
})
|
||||
|
||||
for i, f := range filesIn {
|
||||
wg.Add(1)
|
||||
_ = os.WriteFile(filepath.Join(tmpPath, "g"+strconv.Itoa(i%2), "s"+strconv.Itoa(i), f), []byte{0}, os.ModePerm)
|
||||
}
|
||||
|
||||
wg.Add(1)
|
||||
_ = os.WriteFile(filepath.Join(tmpPath, "gX", "foo"), []byte{0}, os.ModePerm)
|
||||
|
||||
finishedChan := make(chan struct{})
|
||||
go func() {
|
||||
wg.Wait()
|
||||
finishedChan <- struct{}{}
|
||||
}()
|
||||
|
||||
ctx, ctxCancel := context.WithTimeout(context.Background(), 5*time.Second)
|
||||
defer ctxCancel()
|
||||
|
||||
select {
|
||||
case <-finishedChan:
|
||||
case <-ctx.Done():
|
||||
wg.Done() // ctx Will Timeout, because gX/foo will always be remaining
|
||||
// t.Error(ctx.Err())
|
||||
}
|
||||
quitChan <- struct{}{}
|
||||
|
||||
if len(filesOut) != len(filesIn) {
|
||||
t.Errorf("filesOut length %d != %d", len(filesOut), len(filesIn))
|
||||
}
|
||||
for i, f := range filesIn {
|
||||
if !Contains(filesOut, filepath.Join("g"+strconv.Itoa(i%2), "s"+strconv.Itoa(i), f)) {
|
||||
t.Errorf("File %s not found in %s", f, filesOut)
|
||||
}
|
||||
}
|
||||
if Contains(filesOut, filepath.Join("gX", "foo")) {
|
||||
t.Errorf("File %s found in %s", filepath.Join("gX", "foo"), filesOut)
|
||||
}
|
||||
}
|
||||
|
||||
func Contains[T comparable](slice []T, item T) bool {
|
||||
for _, v := range slice {
|
||||
if v == item {
|
||||
|
|
Loading…
Reference in a new issue