update
Some checks failed
Pipeline: Test, Lint, Build / Get version info (push) Has been cancelled
Pipeline: Test, Lint, Build / Lint Go code (push) Has been cancelled
Pipeline: Test, Lint, Build / Test Go code (push) Has been cancelled
Pipeline: Test, Lint, Build / Test JS code (push) Has been cancelled
Pipeline: Test, Lint, Build / Lint i18n files (push) Has been cancelled
Pipeline: Test, Lint, Build / Check Docker configuration (push) Has been cancelled
Pipeline: Test, Lint, Build / Build (darwin/amd64) (push) Has been cancelled
Pipeline: Test, Lint, Build / Build (darwin/arm64) (push) Has been cancelled
Pipeline: Test, Lint, Build / Build (linux/386) (push) Has been cancelled
Pipeline: Test, Lint, Build / Build (linux/amd64) (push) Has been cancelled
Pipeline: Test, Lint, Build / Build (linux/arm/v5) (push) Has been cancelled
Pipeline: Test, Lint, Build / Build (linux/arm/v6) (push) Has been cancelled
Pipeline: Test, Lint, Build / Build (linux/arm/v7) (push) Has been cancelled
Pipeline: Test, Lint, Build / Build (linux/arm64) (push) Has been cancelled
Pipeline: Test, Lint, Build / Build (windows/386) (push) Has been cancelled
Pipeline: Test, Lint, Build / Build (windows/amd64) (push) Has been cancelled
Pipeline: Test, Lint, Build / Push to GHCR (push) Has been cancelled
Pipeline: Test, Lint, Build / Push to Docker Hub (push) Has been cancelled
Pipeline: Test, Lint, Build / Cleanup digest artifacts (push) Has been cancelled
Pipeline: Test, Lint, Build / Build Windows installers (push) Has been cancelled
Pipeline: Test, Lint, Build / Package/Release (push) Has been cancelled
Pipeline: Test, Lint, Build / Upload Linux PKG (push) Has been cancelled
Close stale issues and PRs / stale (push) Has been cancelled
POEditor import / update-translations (push) Has been cancelled
Some checks failed
Pipeline: Test, Lint, Build / Get version info (push) Has been cancelled
Pipeline: Test, Lint, Build / Lint Go code (push) Has been cancelled
Pipeline: Test, Lint, Build / Test Go code (push) Has been cancelled
Pipeline: Test, Lint, Build / Test JS code (push) Has been cancelled
Pipeline: Test, Lint, Build / Lint i18n files (push) Has been cancelled
Pipeline: Test, Lint, Build / Check Docker configuration (push) Has been cancelled
Pipeline: Test, Lint, Build / Build (darwin/amd64) (push) Has been cancelled
Pipeline: Test, Lint, Build / Build (darwin/arm64) (push) Has been cancelled
Pipeline: Test, Lint, Build / Build (linux/386) (push) Has been cancelled
Pipeline: Test, Lint, Build / Build (linux/amd64) (push) Has been cancelled
Pipeline: Test, Lint, Build / Build (linux/arm/v5) (push) Has been cancelled
Pipeline: Test, Lint, Build / Build (linux/arm/v6) (push) Has been cancelled
Pipeline: Test, Lint, Build / Build (linux/arm/v7) (push) Has been cancelled
Pipeline: Test, Lint, Build / Build (linux/arm64) (push) Has been cancelled
Pipeline: Test, Lint, Build / Build (windows/386) (push) Has been cancelled
Pipeline: Test, Lint, Build / Build (windows/amd64) (push) Has been cancelled
Pipeline: Test, Lint, Build / Push to GHCR (push) Has been cancelled
Pipeline: Test, Lint, Build / Push to Docker Hub (push) Has been cancelled
Pipeline: Test, Lint, Build / Cleanup digest artifacts (push) Has been cancelled
Pipeline: Test, Lint, Build / Build Windows installers (push) Has been cancelled
Pipeline: Test, Lint, Build / Package/Release (push) Has been cancelled
Pipeline: Test, Lint, Build / Upload Linux PKG (push) Has been cancelled
Close stale issues and PRs / stale (push) Has been cancelled
POEditor import / update-translations (push) Has been cancelled
This commit is contained in:
323
core/storage/storagetest/fake_storage.go
Normal file
323
core/storage/storagetest/fake_storage.go
Normal file
@@ -0,0 +1,323 @@
|
||||
//nolint:unused
|
||||
package storagetest
|
||||
|
||||
import (
|
||||
"encoding/json"
|
||||
"errors"
|
||||
"fmt"
|
||||
"io/fs"
|
||||
"net/url"
|
||||
"path"
|
||||
"testing/fstest"
|
||||
"time"
|
||||
|
||||
"github.com/navidrome/navidrome/core/storage"
|
||||
"github.com/navidrome/navidrome/log"
|
||||
"github.com/navidrome/navidrome/model/metadata"
|
||||
"github.com/navidrome/navidrome/utils/random"
|
||||
)
|
||||
|
||||
// FakeStorage is a fake storage that provides a FakeFS.
|
||||
// It is used for testing purposes.
|
||||
type FakeStorage struct{ fs *FakeFS }
|
||||
|
||||
// Register registers the FakeStorage for the given scheme. To use it, set the model.Library's Path to "fake:///music",
|
||||
// and register a FakeFS with schema = "fake". The storage registered will always return the same FakeFS instance.
|
||||
func Register(schema string, fs *FakeFS) {
|
||||
storage.Register(schema, func(url url.URL) storage.Storage { return &FakeStorage{fs: fs} })
|
||||
}
|
||||
|
||||
func (s FakeStorage) FS() (storage.MusicFS, error) {
|
||||
return s.fs, nil
|
||||
}
|
||||
|
||||
// FakeFS is a fake filesystem that can be used for testing purposes.
|
||||
// It implements the storage.MusicFS interface and keeps all files in memory, by using a fstest.MapFS internally.
|
||||
// You must NOT add files directly in the MapFS property, but use SetFiles and its other methods instead.
|
||||
// This is because the FakeFS keeps track of the latest modification time of directories, simulating the
|
||||
// behavior of a real filesystem, and you should not bypass this logic.
|
||||
type FakeFS struct {
|
||||
fstest.MapFS
|
||||
properInit bool
|
||||
}
|
||||
|
||||
func (ffs *FakeFS) SetFiles(files fstest.MapFS) {
|
||||
ffs.properInit = true
|
||||
ffs.MapFS = files
|
||||
ffs.createDirTimestamps()
|
||||
}
|
||||
|
||||
func (ffs *FakeFS) Add(filePath string, file *fstest.MapFile, when ...time.Time) {
|
||||
if len(when) == 0 {
|
||||
when = append(when, time.Now())
|
||||
}
|
||||
ffs.MapFS[filePath] = file
|
||||
ffs.touchContainingFolder(filePath, when[0])
|
||||
ffs.createDirTimestamps()
|
||||
}
|
||||
|
||||
func (ffs *FakeFS) Remove(filePath string, when ...time.Time) *fstest.MapFile {
|
||||
filePath = path.Clean(filePath)
|
||||
if len(when) == 0 {
|
||||
when = append(when, time.Now())
|
||||
}
|
||||
if f, ok := ffs.MapFS[filePath]; ok {
|
||||
ffs.touchContainingFolder(filePath, when[0])
|
||||
delete(ffs.MapFS, filePath)
|
||||
return f
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func (ffs *FakeFS) Move(srcPath string, destPath string, when ...time.Time) {
|
||||
if len(when) == 0 {
|
||||
when = append(when, time.Now())
|
||||
}
|
||||
srcPath = path.Clean(srcPath)
|
||||
destPath = path.Clean(destPath)
|
||||
ffs.MapFS[destPath] = ffs.MapFS[srcPath]
|
||||
ffs.touchContainingFolder(destPath, when[0])
|
||||
ffs.Remove(srcPath, when...)
|
||||
}
|
||||
|
||||
// Touch sets the modification time of a file.
|
||||
func (ffs *FakeFS) Touch(filePath string, when ...time.Time) {
|
||||
if len(when) == 0 {
|
||||
when = append(when, time.Now())
|
||||
}
|
||||
filePath = path.Clean(filePath)
|
||||
file, ok := ffs.MapFS[filePath]
|
||||
if ok {
|
||||
file.ModTime = when[0]
|
||||
} else {
|
||||
ffs.MapFS[filePath] = &fstest.MapFile{ModTime: when[0]}
|
||||
}
|
||||
ffs.touchContainingFolder(filePath, file.ModTime)
|
||||
}
|
||||
|
||||
func (ffs *FakeFS) touchContainingFolder(filePath string, ts time.Time) {
|
||||
dir := path.Dir(filePath)
|
||||
dirFile, ok := ffs.MapFS[dir]
|
||||
if !ok {
|
||||
log.Fatal("Directory not found. Forgot to call SetFiles?", "file", filePath)
|
||||
}
|
||||
if dirFile.ModTime.Before(ts) {
|
||||
dirFile.ModTime = ts
|
||||
}
|
||||
}
|
||||
|
||||
// SetError sets an error that will be returned when trying to read the file.
|
||||
func (ffs *FakeFS) SetError(filePath string, err error) {
|
||||
filePath = path.Clean(filePath)
|
||||
if ffs.MapFS[filePath] == nil {
|
||||
ffs.MapFS[filePath] = &fstest.MapFile{Data: []byte{}}
|
||||
}
|
||||
ffs.MapFS[filePath].Sys = err
|
||||
ffs.Touch(filePath)
|
||||
}
|
||||
|
||||
// ClearError clears the error set by SetError.
|
||||
func (ffs *FakeFS) ClearError(filePath string) {
|
||||
filePath = path.Clean(filePath)
|
||||
if file := ffs.MapFS[filePath]; file != nil {
|
||||
file.Sys = nil
|
||||
}
|
||||
ffs.Touch(filePath)
|
||||
}
|
||||
|
||||
func (ffs *FakeFS) UpdateTags(filePath string, newTags map[string]any, when ...time.Time) {
|
||||
f, ok := ffs.MapFS[filePath]
|
||||
if !ok {
|
||||
panic(fmt.Errorf("file %s not found", filePath))
|
||||
}
|
||||
var tags map[string]any
|
||||
err := json.Unmarshal(f.Data, &tags)
|
||||
if err != nil {
|
||||
panic(err)
|
||||
}
|
||||
for k, v := range newTags {
|
||||
tags[k] = v
|
||||
}
|
||||
data, _ := json.Marshal(tags)
|
||||
f.Data = data
|
||||
ffs.Touch(filePath, when...)
|
||||
}
|
||||
|
||||
// createDirTimestamps loops through all entries and create/updates directories entries in the map with the
|
||||
// latest ModTime from any children of that directory.
|
||||
func (ffs *FakeFS) createDirTimestamps() bool {
|
||||
var changed bool
|
||||
for filePath, file := range ffs.MapFS {
|
||||
dir := path.Dir(filePath)
|
||||
dirFile, ok := ffs.MapFS[dir]
|
||||
if !ok {
|
||||
dirFile = &fstest.MapFile{Mode: fs.ModeDir}
|
||||
ffs.MapFS[dir] = dirFile
|
||||
}
|
||||
if dirFile.ModTime.IsZero() {
|
||||
dirFile.ModTime = file.ModTime
|
||||
changed = true
|
||||
}
|
||||
}
|
||||
if changed {
|
||||
// If we updated any directory, we need to re-run the loop to create any parent directories
|
||||
ffs.createDirTimestamps()
|
||||
}
|
||||
return changed
|
||||
}
|
||||
|
||||
func ModTime(ts string) map[string]any { return map[string]any{fakeFileInfoModTime: ts} }
|
||||
func BirthTime(ts string) map[string]any { return map[string]any{fakeFileInfoBirthTime: ts} }
|
||||
|
||||
func Template(t ...map[string]any) func(...map[string]any) *fstest.MapFile {
|
||||
return func(tags ...map[string]any) *fstest.MapFile {
|
||||
return MP3(append(t, tags...)...)
|
||||
}
|
||||
}
|
||||
|
||||
func Track(num int, title string, tags ...map[string]any) map[string]any {
|
||||
ts := audioProperties("mp3", 320)
|
||||
ts["title"] = title
|
||||
ts["track"] = num
|
||||
for _, t := range tags {
|
||||
for k, v := range t {
|
||||
ts[k] = v
|
||||
}
|
||||
}
|
||||
return ts
|
||||
}
|
||||
|
||||
func MP3(tags ...map[string]any) *fstest.MapFile {
|
||||
ts := audioProperties("mp3", 320)
|
||||
if _, ok := ts[fakeFileInfoSize]; !ok {
|
||||
duration := ts["duration"].(int64)
|
||||
bitrate := ts["bitrate"].(int)
|
||||
ts[fakeFileInfoSize] = duration * int64(bitrate) / 8 * 1000
|
||||
}
|
||||
return File(append([]map[string]any{ts}, tags...)...)
|
||||
}
|
||||
|
||||
func File(tags ...map[string]any) *fstest.MapFile {
|
||||
ts := map[string]any{}
|
||||
for _, t := range tags {
|
||||
for k, v := range t {
|
||||
ts[k] = v
|
||||
}
|
||||
}
|
||||
modTime := time.Now()
|
||||
if mt, ok := ts[fakeFileInfoModTime]; !ok {
|
||||
ts[fakeFileInfoModTime] = time.Now().Format(time.RFC3339)
|
||||
} else {
|
||||
modTime, _ = time.Parse(time.RFC3339, mt.(string))
|
||||
}
|
||||
if _, ok := ts[fakeFileInfoBirthTime]; !ok {
|
||||
ts[fakeFileInfoBirthTime] = time.Now().Format(time.RFC3339)
|
||||
}
|
||||
if _, ok := ts[fakeFileInfoMode]; !ok {
|
||||
ts[fakeFileInfoMode] = fs.ModePerm
|
||||
}
|
||||
data, _ := json.Marshal(ts)
|
||||
if _, ok := ts[fakeFileInfoSize]; !ok {
|
||||
ts[fakeFileInfoSize] = int64(len(data))
|
||||
}
|
||||
return &fstest.MapFile{Data: data, ModTime: modTime, Mode: ts[fakeFileInfoMode].(fs.FileMode)}
|
||||
}
|
||||
|
||||
func audioProperties(suffix string, bitrate int) map[string]any {
|
||||
duration := random.Int64N(300) + 120
|
||||
return map[string]any{
|
||||
"suffix": suffix,
|
||||
"bitrate": bitrate,
|
||||
"duration": duration,
|
||||
"samplerate": 44100,
|
||||
"bitdepth": 16,
|
||||
"channels": 2,
|
||||
}
|
||||
}
|
||||
|
||||
func (ffs *FakeFS) ReadTags(paths ...string) (map[string]metadata.Info, error) {
|
||||
if !ffs.properInit {
|
||||
log.Fatal("FakeFS not initialized properly. Use SetFiles")
|
||||
}
|
||||
result := make(map[string]metadata.Info)
|
||||
var errs []error
|
||||
for _, file := range paths {
|
||||
p, err := ffs.parseFile(file)
|
||||
if err != nil {
|
||||
log.Warn("Error reading metadata from file", "file", file, "err", err)
|
||||
errs = append(errs, err)
|
||||
} else {
|
||||
result[file] = *p
|
||||
}
|
||||
}
|
||||
if len(errs) > 0 {
|
||||
return result, fmt.Errorf("errors reading metadata: %w", errors.Join(errs...))
|
||||
}
|
||||
return result, nil
|
||||
}
|
||||
|
||||
func (ffs *FakeFS) parseFile(filePath string) (*metadata.Info, error) {
|
||||
// Check if it should throw an error when reading this file
|
||||
stat, err := ffs.Stat(filePath)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
if stat.Sys() != nil {
|
||||
return nil, stat.Sys().(error)
|
||||
}
|
||||
|
||||
// Read the file contents and parse the tags
|
||||
contents, err := fs.ReadFile(ffs, filePath)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
data := map[string]any{}
|
||||
err = json.Unmarshal(contents, &data)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
p := metadata.Info{
|
||||
Tags: map[string][]string{},
|
||||
AudioProperties: metadata.AudioProperties{},
|
||||
HasPicture: data["has_picture"] == "true",
|
||||
}
|
||||
if d, ok := data["duration"].(float64); ok {
|
||||
p.AudioProperties.Duration = time.Duration(d) * time.Second
|
||||
}
|
||||
getInt := func(key string) int { v, _ := data[key].(float64); return int(v) }
|
||||
p.AudioProperties.BitRate = getInt("bitrate")
|
||||
p.AudioProperties.BitDepth = getInt("bitdepth")
|
||||
p.AudioProperties.SampleRate = getInt("samplerate")
|
||||
p.AudioProperties.Channels = getInt("channels")
|
||||
for k, v := range data {
|
||||
p.Tags[k] = []string{fmt.Sprintf("%v", v)}
|
||||
}
|
||||
file := ffs.MapFS[filePath]
|
||||
p.FileInfo = &fakeFileInfo{path: filePath, tags: data, file: file}
|
||||
return &p, nil
|
||||
}
|
||||
|
||||
const (
|
||||
fakeFileInfoMode = "_mode"
|
||||
fakeFileInfoSize = "_size"
|
||||
fakeFileInfoModTime = "_modtime"
|
||||
fakeFileInfoBirthTime = "_birthtime"
|
||||
)
|
||||
|
||||
type fakeFileInfo struct {
|
||||
path string
|
||||
file *fstest.MapFile
|
||||
tags map[string]any
|
||||
}
|
||||
|
||||
func (ffi *fakeFileInfo) Name() string { return path.Base(ffi.path) }
|
||||
func (ffi *fakeFileInfo) Size() int64 { v, _ := ffi.tags[fakeFileInfoSize].(float64); return int64(v) }
|
||||
func (ffi *fakeFileInfo) Mode() fs.FileMode { return ffi.file.Mode }
|
||||
func (ffi *fakeFileInfo) IsDir() bool { return false }
|
||||
func (ffi *fakeFileInfo) Sys() any { return nil }
|
||||
func (ffi *fakeFileInfo) ModTime() time.Time { return ffi.file.ModTime }
|
||||
func (ffi *fakeFileInfo) BirthTime() time.Time { return ffi.parseTime(fakeFileInfoBirthTime) }
|
||||
func (ffi *fakeFileInfo) parseTime(key string) time.Time {
|
||||
t, _ := time.Parse(time.RFC3339, ffi.tags[key].(string))
|
||||
return t
|
||||
}
|
||||
139
core/storage/storagetest/fake_storage_test.go
Normal file
139
core/storage/storagetest/fake_storage_test.go
Normal file
@@ -0,0 +1,139 @@
|
||||
//nolint:unused
|
||||
package storagetest_test
|
||||
|
||||
import (
|
||||
"io/fs"
|
||||
"testing"
|
||||
"testing/fstest"
|
||||
"time"
|
||||
|
||||
. "github.com/navidrome/navidrome/core/storage/storagetest"
|
||||
. "github.com/onsi/ginkgo/v2"
|
||||
. "github.com/onsi/gomega"
|
||||
)
|
||||
|
||||
type _t = map[string]any
|
||||
|
||||
func TestFakeStorage(t *testing.T) {
|
||||
RegisterFailHandler(Fail)
|
||||
RunSpecs(t, "Fake Storage Test Suite")
|
||||
}
|
||||
|
||||
var _ = Describe("FakeFS", func() {
|
||||
var ffs FakeFS
|
||||
var startTime time.Time
|
||||
|
||||
BeforeEach(func() {
|
||||
startTime = time.Now().Add(-time.Hour)
|
||||
boy := Template(_t{"albumartist": "U2", "album": "Boy", "year": 1980, "genre": "Rock"})
|
||||
files := fstest.MapFS{
|
||||
"U2/Boy/I Will Follow.mp3": boy(Track(1, "I Will Follow")),
|
||||
"U2/Boy/Twilight.mp3": boy(Track(2, "Twilight")),
|
||||
"U2/Boy/An Cat Dubh.mp3": boy(Track(3, "An Cat Dubh")),
|
||||
}
|
||||
ffs.SetFiles(files)
|
||||
})
|
||||
|
||||
It("should implement a fs.FS", func() {
|
||||
Expect(fstest.TestFS(ffs, "U2/Boy/I Will Follow.mp3")).To(Succeed())
|
||||
})
|
||||
|
||||
It("should read file info", func() {
|
||||
props, err := ffs.ReadTags("U2/Boy/I Will Follow.mp3", "U2/Boy/Twilight.mp3")
|
||||
Expect(err).ToNot(HaveOccurred())
|
||||
|
||||
prop := props["U2/Boy/Twilight.mp3"]
|
||||
Expect(prop).ToNot(BeNil())
|
||||
Expect(prop.AudioProperties.Channels).To(Equal(2))
|
||||
Expect(prop.AudioProperties.BitRate).To(Equal(320))
|
||||
Expect(prop.FileInfo.Name()).To(Equal("Twilight.mp3"))
|
||||
Expect(prop.Tags["albumartist"]).To(ConsistOf("U2"))
|
||||
Expect(prop.FileInfo.ModTime()).To(BeTemporally(">=", startTime))
|
||||
|
||||
prop = props["U2/Boy/I Will Follow.mp3"]
|
||||
Expect(prop).ToNot(BeNil())
|
||||
Expect(prop.FileInfo.Name()).To(Equal("I Will Follow.mp3"))
|
||||
})
|
||||
|
||||
It("should return ModTime for directories", func() {
|
||||
root := ffs.MapFS["."]
|
||||
dirInfo1, err := ffs.Stat("U2")
|
||||
Expect(err).ToNot(HaveOccurred())
|
||||
dirInfo2, err := ffs.Stat("U2/Boy")
|
||||
Expect(err).ToNot(HaveOccurred())
|
||||
Expect(dirInfo1.ModTime()).To(Equal(root.ModTime))
|
||||
Expect(dirInfo1.ModTime()).To(BeTemporally(">=", startTime))
|
||||
Expect(dirInfo1.ModTime()).To(Equal(dirInfo2.ModTime()))
|
||||
})
|
||||
|
||||
When("the file is touched", func() {
|
||||
It("should only update the file and the file's directory ModTime", func() {
|
||||
root, _ := ffs.Stat(".")
|
||||
u2Dir, _ := ffs.Stat("U2")
|
||||
boyDir, _ := ffs.Stat("U2/Boy")
|
||||
previousTime := root.ModTime()
|
||||
|
||||
aTimeStamp := previousTime.Add(time.Hour)
|
||||
ffs.Touch("U2/./Boy/Twilight.mp3", aTimeStamp)
|
||||
|
||||
twilightFile, err := ffs.Stat("U2/Boy/Twilight.mp3")
|
||||
Expect(err).ToNot(HaveOccurred())
|
||||
Expect(twilightFile.ModTime()).To(Equal(aTimeStamp))
|
||||
|
||||
Expect(root.ModTime()).To(Equal(previousTime))
|
||||
Expect(u2Dir.ModTime()).To(Equal(previousTime))
|
||||
Expect(boyDir.ModTime()).To(Equal(aTimeStamp))
|
||||
})
|
||||
})
|
||||
|
||||
When("adding/removing files", func() {
|
||||
It("should keep the timestamps correct", func() {
|
||||
root, _ := ffs.Stat(".")
|
||||
u2Dir, _ := ffs.Stat("U2")
|
||||
boyDir, _ := ffs.Stat("U2/Boy")
|
||||
previousTime := root.ModTime()
|
||||
aTimeStamp := previousTime.Add(time.Hour)
|
||||
|
||||
ffs.Add("U2/Boy/../Boy/Another.mp3", &fstest.MapFile{ModTime: aTimeStamp}, aTimeStamp)
|
||||
Expect(u2Dir.ModTime()).To(Equal(previousTime))
|
||||
Expect(boyDir.ModTime()).To(Equal(aTimeStamp))
|
||||
|
||||
aTimeStamp = aTimeStamp.Add(time.Hour)
|
||||
ffs.Remove("U2/./Boy/Twilight.mp3", aTimeStamp)
|
||||
|
||||
_, err := ffs.Stat("U2/Boy/Twilight.mp3")
|
||||
Expect(err).To(MatchError(fs.ErrNotExist))
|
||||
Expect(u2Dir.ModTime()).To(Equal(previousTime))
|
||||
Expect(boyDir.ModTime()).To(Equal(aTimeStamp))
|
||||
})
|
||||
})
|
||||
|
||||
When("moving files", func() {
|
||||
It("should allow relative paths", func() {
|
||||
ffs.Move("U2/../U2/Boy/Twilight.mp3", "./Twilight.mp3")
|
||||
Expect(ffs.MapFS).To(HaveKey("Twilight.mp3"))
|
||||
file, err := ffs.Stat("Twilight.mp3")
|
||||
Expect(err).ToNot(HaveOccurred())
|
||||
Expect(file.Name()).To(Equal("Twilight.mp3"))
|
||||
})
|
||||
It("should keep the timestamps correct", func() {
|
||||
root, _ := ffs.Stat(".")
|
||||
u2Dir, _ := ffs.Stat("U2")
|
||||
boyDir, _ := ffs.Stat("U2/Boy")
|
||||
previousTime := root.ModTime()
|
||||
twilightFile, _ := ffs.Stat("U2/Boy/Twilight.mp3")
|
||||
filePreviousTime := twilightFile.ModTime()
|
||||
aTimeStamp := previousTime.Add(time.Hour)
|
||||
|
||||
ffs.Move("U2/Boy/Twilight.mp3", "Twilight.mp3", aTimeStamp)
|
||||
|
||||
Expect(root.ModTime()).To(Equal(aTimeStamp))
|
||||
Expect(u2Dir.ModTime()).To(Equal(previousTime))
|
||||
Expect(boyDir.ModTime()).To(Equal(aTimeStamp))
|
||||
|
||||
Expect(ffs.MapFS).ToNot(HaveKey("U2/Boy/Twilight.mp3"))
|
||||
twilight := ffs.MapFS["Twilight.mp3"]
|
||||
Expect(twilight.ModTime).To(Equal(filePreviousTime))
|
||||
})
|
||||
})
|
||||
})
|
||||
Reference in New Issue
Block a user