mirror of https://github.com/spf13/afero.git
310 lines
5.4 KiB
Go
310 lines
5.4 KiB
Go
package mem
|
|
|
|
import (
|
|
"bytes"
|
|
"io"
|
|
"testing"
|
|
"time"
|
|
)
|
|
|
|
func TestFileDataNameRace(t *testing.T) {
|
|
t.Parallel()
|
|
const someName = "someName"
|
|
const someOtherName = "someOtherName"
|
|
d := FileData{
|
|
name: someName,
|
|
}
|
|
|
|
if d.Name() != someName {
|
|
t.Errorf("Failed to read correct Name, was %v", d.Name())
|
|
}
|
|
|
|
ChangeFileName(&d, someOtherName)
|
|
if d.Name() != someOtherName {
|
|
t.Errorf("Failed to set Name, was %v", d.Name())
|
|
}
|
|
|
|
go func() {
|
|
ChangeFileName(&d, someName)
|
|
}()
|
|
|
|
if d.Name() != someName && d.Name() != someOtherName {
|
|
t.Errorf("Failed to read either Name, was %v", d.Name())
|
|
}
|
|
}
|
|
|
|
func TestFileDataModTimeRace(t *testing.T) {
|
|
t.Parallel()
|
|
someTime := time.Now()
|
|
someOtherTime := someTime.Add(1 * time.Minute)
|
|
|
|
d := FileData{
|
|
modtime: someTime,
|
|
}
|
|
|
|
s := FileInfo{
|
|
FileData: &d,
|
|
}
|
|
|
|
if s.ModTime() != someTime {
|
|
t.Errorf("Failed to read correct value, was %v", s.ModTime())
|
|
}
|
|
|
|
SetModTime(&d, someOtherTime)
|
|
if s.ModTime() != someOtherTime {
|
|
t.Errorf("Failed to set ModTime, was %v", s.ModTime())
|
|
}
|
|
|
|
go func() {
|
|
SetModTime(&d, someTime)
|
|
}()
|
|
|
|
if s.ModTime() != someTime && s.ModTime() != someOtherTime {
|
|
t.Errorf("Failed to read either modtime, was %v", s.ModTime())
|
|
}
|
|
}
|
|
|
|
func TestFileDataModeRace(t *testing.T) {
|
|
t.Parallel()
|
|
const someMode = 0777
|
|
const someOtherMode = 0660
|
|
|
|
d := FileData{
|
|
mode: someMode,
|
|
}
|
|
|
|
s := FileInfo{
|
|
FileData: &d,
|
|
}
|
|
|
|
if s.Mode() != someMode {
|
|
t.Errorf("Failed to read correct value, was %v", s.Mode())
|
|
}
|
|
|
|
SetMode(&d, someOtherMode)
|
|
if s.Mode() != someOtherMode {
|
|
t.Errorf("Failed to set Mode, was %v", s.Mode())
|
|
}
|
|
|
|
go func() {
|
|
SetMode(&d, someMode)
|
|
}()
|
|
|
|
if s.Mode() != someMode && s.Mode() != someOtherMode {
|
|
t.Errorf("Failed to read either mode, was %v", s.Mode())
|
|
}
|
|
}
|
|
|
|
// See https://github.com/spf13/afero/issues/286.
|
|
func TestFileWriteAt(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
data := CreateFile("abc.txt")
|
|
f := NewFileHandle(data)
|
|
|
|
testData := []byte{1, 2, 3, 4, 5}
|
|
offset := len(testData)
|
|
|
|
// 5 zeros + testdata
|
|
_, err := f.WriteAt(testData, int64(offset))
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
// 2 * testdata
|
|
_, err = f.WriteAt(testData, 0)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
// 3 * testdata
|
|
_, err = f.WriteAt(testData, int64(offset*2))
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
// 3 * testdata + 5 zeros + testdata
|
|
_, err = f.WriteAt(testData, int64(offset*4))
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
// 5 * testdata
|
|
_, err = f.WriteAt(testData, int64(offset*3))
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
err = f.Close()
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
expected := bytes.Repeat(testData, 5)
|
|
if !bytes.Equal(expected, data.data) {
|
|
t.Fatalf("expected: %v, got: %v", expected, data.data)
|
|
}
|
|
}
|
|
|
|
func TestFileDataIsDirRace(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
d := FileData{
|
|
dir: true,
|
|
}
|
|
|
|
s := FileInfo{
|
|
FileData: &d,
|
|
}
|
|
|
|
if s.IsDir() != true {
|
|
t.Errorf("Failed to read correct value, was %v", s.IsDir())
|
|
}
|
|
|
|
go func() {
|
|
s.Lock()
|
|
d.dir = false
|
|
s.Unlock()
|
|
}()
|
|
|
|
//just logging the value to trigger a read:
|
|
t.Logf("Value is %v", s.IsDir())
|
|
}
|
|
|
|
func TestFileDataSizeRace(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
const someData = "Hello"
|
|
const someOtherDataSize = "Hello World"
|
|
|
|
d := FileData{
|
|
data: []byte(someData),
|
|
dir: false,
|
|
}
|
|
|
|
s := FileInfo{
|
|
FileData: &d,
|
|
}
|
|
|
|
if s.Size() != int64(len(someData)) {
|
|
t.Errorf("Failed to read correct value, was %v", s.Size())
|
|
}
|
|
|
|
go func() {
|
|
s.Lock()
|
|
d.data = []byte(someOtherDataSize)
|
|
s.Unlock()
|
|
}()
|
|
|
|
//just logging the value to trigger a read:
|
|
t.Logf("Value is %v", s.Size())
|
|
|
|
//Testing the Dir size case
|
|
d.dir = true
|
|
if s.Size() != int64(42) {
|
|
t.Errorf("Failed to read correct value for dir, was %v", s.Size())
|
|
}
|
|
}
|
|
|
|
func TestFileReadAtSeekOffset(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
fd := CreateFile("foo")
|
|
f := NewFileHandle(fd)
|
|
|
|
_, err := f.WriteString("TEST")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
offset, err := f.Seek(0, io.SeekStart)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if offset != 0 {
|
|
t.Fail()
|
|
}
|
|
|
|
offsetBeforeReadAt, err := f.Seek(0, io.SeekCurrent)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if offsetBeforeReadAt != 0 {
|
|
t.Fatal("expected 0")
|
|
}
|
|
|
|
b := make([]byte, 4, 5)
|
|
n, err := f.ReadAt(b, 0)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if n != 4 {
|
|
t.Fail()
|
|
}
|
|
if string(b) != "TEST" {
|
|
t.Fail()
|
|
}
|
|
|
|
n, err = f.ReadAt(b[:cap(b)], 0)
|
|
if err != io.EOF {
|
|
t.Fatal(err)
|
|
}
|
|
if n != 4 {
|
|
t.Fail()
|
|
}
|
|
if string(b) != "TEST" {
|
|
t.Fail()
|
|
}
|
|
|
|
offsetAfterReadAt, err := f.Seek(0, io.SeekCurrent)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if offsetAfterReadAt != offsetBeforeReadAt {
|
|
t.Fatal("ReadAt should not affect offset")
|
|
}
|
|
|
|
err = f.Close()
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
}
|
|
|
|
func TestFileWriteAndSeek(t *testing.T) {
|
|
fd := CreateFile("foo")
|
|
f := NewFileHandle(fd)
|
|
|
|
assert := func(expected bool, v ...interface{}) {
|
|
if !expected {
|
|
t.Helper()
|
|
t.Fatal(v...)
|
|
}
|
|
}
|
|
|
|
data4 := []byte{0, 1, 2, 3}
|
|
data20 := bytes.Repeat(data4, 5)
|
|
var off int64
|
|
|
|
for i := 0; i < 100; i++ {
|
|
// write 20 bytes
|
|
n, err := f.Write(data20)
|
|
assert(err == nil, err)
|
|
off += int64(n)
|
|
assert(n == len(data20), n)
|
|
assert(off == int64((i+1)*len(data20)), off)
|
|
|
|
// rewind to start and write 4 bytes there
|
|
cur, err := f.Seek(-off, io.SeekCurrent)
|
|
assert(err == nil, err)
|
|
assert(cur == 0, cur)
|
|
|
|
n, err = f.Write(data4)
|
|
assert(err == nil, err)
|
|
assert(n == len(data4), n)
|
|
|
|
// back at the end
|
|
cur, err = f.Seek(off-int64(n), io.SeekCurrent)
|
|
assert(err == nil, err)
|
|
assert(cur == off, cur, off)
|
|
}
|
|
}
|