X-Git-Url: https://git.arvados.org/arvados.git/blobdiff_plain/66c19e11db2626bd82eb755ea6552ce5caec69af..7930d7abaabf2fd1f3432eca10f26b821e0ef94f:/services/keepstore/volume_unix_test.go diff --git a/services/keepstore/volume_unix_test.go b/services/keepstore/volume_unix_test.go index 6bafa7c1ca..6ccc865b11 100644 --- a/services/keepstore/volume_unix_test.go +++ b/services/keepstore/volume_unix_test.go @@ -2,9 +2,14 @@ package main import ( "bytes" + "errors" "fmt" + "io" "io/ioutil" "os" + "regexp" + "sort" + "strings" "syscall" "testing" "time" @@ -74,6 +79,42 @@ func TestGetNotFound(t *testing.T) { } } +func TestIndexTo(t *testing.T) { + v := TempUnixVolume(t, false, false) + defer _teardown(v) + + _store(t, v, TEST_HASH, TEST_BLOCK) + _store(t, v, TEST_HASH_2, TEST_BLOCK_2) + _store(t, v, TEST_HASH_3, TEST_BLOCK_3) + + buf := new(bytes.Buffer) + v.IndexTo("", buf) + index_rows := strings.Split(string(buf.Bytes()), "\n") + sort.Strings(index_rows) + sorted_index := strings.Join(index_rows, "\n") + m, err := regexp.MatchString( + `^\n`+TEST_HASH+`\+\d+ \d+\n`+ + TEST_HASH_3+`\+\d+ \d+\n`+ + TEST_HASH_2+`\+\d+ \d+$`, + sorted_index) + if err != nil { + t.Error(err) + } else if !m { + t.Errorf("Got index %q for empty prefix", sorted_index) + } + + for _, prefix := range []string{"f", "f15", "f15ac"} { + buf = new(bytes.Buffer) + v.IndexTo(prefix, buf) + m, err := regexp.MatchString(`^`+TEST_HASH_2+`\+\d+ \d+\n$`, string(buf.Bytes())) + if err != nil { + t.Error(err) + } else if !m { + t.Errorf("Got index %q for prefix %q", string(buf.Bytes()), prefix) + } + } +} + func TestPut(t *testing.T) { v := TempUnixVolume(t, false, false) defer _teardown(v) @@ -277,7 +318,7 @@ func TestPutSerialized(t *testing.T) { }(sem) // Wait for all goroutines to finish - for done := 0; done < 2; { + for done := 0; done < 3; { done += <-sem } @@ -346,3 +387,83 @@ func TestNodeStatus(t *testing.T) { t.Errorf("uninitialized bytes_used in %v", volinfo) } } + +func TestUnixVolumeGetFuncWorkerError(t *testing.T) { + v := TempUnixVolume(t, false, false) + defer _teardown(v) + + v.Put(TEST_HASH, TEST_BLOCK) + mockErr := errors.New("Mock error") + err := v.getFunc(v.blockPath(TEST_HASH), func(rdr io.Reader) error { + return mockErr + }) + if err != mockErr { + t.Errorf("Got %v, expected %v", err, mockErr) + } +} + +func TestUnixVolumeGetFuncFileError(t *testing.T) { + v := TempUnixVolume(t, false, false) + defer _teardown(v) + + funcCalled := false + err := v.getFunc(v.blockPath(TEST_HASH), func(rdr io.Reader) error { + funcCalled = true + return nil + }) + if err == nil { + t.Errorf("Expected error opening non-existent file") + } + if funcCalled { + t.Errorf("Worker func should not have been called") + } +} + +func TestUnixVolumeGetFuncWorkerWaitsOnMutex(t *testing.T) { + v := TempUnixVolume(t, true, false) + defer _teardown(v) + + v.mutex.Lock() + locked := true + go func() { + // TODO(TC): Don't rely on Sleep. Mock the mutex instead? + time.Sleep(10 * time.Millisecond) + locked = false + v.mutex.Unlock() + }() + v.getFunc(v.blockPath(TEST_HASH), func(rdr io.Reader) error { + if locked { + t.Errorf("Worker func called before serialize lock was obtained") + } + return nil + }) +} + +func TestUnixVolumeCompare(t *testing.T) { + v := TempUnixVolume(t, false, false) + defer _teardown(v) + + v.Put(TEST_HASH, TEST_BLOCK) + err := v.Compare(TEST_HASH, TEST_BLOCK) + if err != nil { + t.Errorf("Got err %q, expected nil", err) + } + + err = v.Compare(TEST_HASH, []byte("baddata")) + if err != CollisionError { + t.Errorf("Got err %q, expected %q", err, CollisionError) + } + + _store(t, v, TEST_HASH, []byte("baddata")) + err = v.Compare(TEST_HASH, TEST_BLOCK) + if err != DiskHashError { + t.Errorf("Got err %q, expected %q", err, DiskHashError) + } + + p := fmt.Sprintf("%s/%s/%s", v.root, TEST_HASH[:3], TEST_HASH) + os.Chmod(p, 000) + err = v.Compare(TEST_HASH, TEST_BLOCK) + if err == nil || strings.Index(err.Error(), "permission denied") < 0 { + t.Errorf("Got err %q, expected %q", err, "permission denied") + } +}