blob: 171b29337f97943644c12ae34e70c19cc25a1fe7 [file] [log] [blame]
package impl_test
import (
"io/ioutil"
"os"
"path"
"testing"
"veyron/services/mgmt/logreader/impl"
"veyron2/ipc"
"veyron2/naming"
"veyron2/rt"
"veyron2/security"
"veyron2/services/mgmt/logreader"
"veyron2/verror"
)
type logFileDispatcher struct {
root string
}
func (d *logFileDispatcher) Lookup(suffix, _ string) (ipc.Invoker, security.Authorizer, error) {
invoker := ipc.ReflectInvoker(logreader.NewServerLogFile(impl.NewLogFileInvoker(d.root, suffix)))
return invoker, nil, nil
}
func writeAndSync(t *testing.T, w *os.File, s string) {
if _, err := w.WriteString(s); err != nil {
t.Fatalf("w.WriteString failed: %v", err)
}
if err := w.Sync(); err != nil {
t.Fatalf("w.Sync failed: %v", err)
}
}
func TestReadLogImplNoFollow(t *testing.T) {
runtime := rt.Init()
workdir, err := ioutil.TempDir("", "logreadertest")
if err != nil {
t.Fatalf("ioutil.TempDir: %v", err)
}
defer os.RemoveAll(workdir)
server, endpoint, err := startServer(t, &logFileDispatcher{workdir})
if err != nil {
t.Fatalf("startServer failed: %v", err)
}
defer stopServer(t, server)
const testFile = "mylogfile.INFO"
writer, err := os.Create(path.Join(workdir, testFile))
if err != nil {
t.Fatalf("Create failed: %v", err)
}
tests := []string{
"Hello World!",
"Life is too short",
"Have fun",
"Play hard",
"Break something",
"Fix it later",
}
for _, s := range tests {
writeAndSync(t, writer, s+"\n")
}
// Try to access a file that doesn't exist.
lf, err := logreader.BindLogFile(naming.JoinAddressName(endpoint, "//doesntexist"))
if err != nil {
t.Errorf("BindLogFile: %v", err)
}
_, err = lf.Size(runtime.NewContext())
if expected := verror.NotFound; !verror.Is(err, expected) {
t.Errorf("unexpected error value, got %v, want: %v", err, expected)
}
// Try to access a file that does exist.
lf, err = logreader.BindLogFile(naming.JoinAddressName(endpoint, "//"+testFile))
if err != nil {
t.Errorf("BindLogFile: %v", err)
}
_, err = lf.Size(runtime.NewContext())
if err != nil {
t.Errorf("Size failed: %v", err)
}
// Read without follow.
stream, err := lf.ReadLog(runtime.NewContext(), 0, logreader.AllEntries, false)
if err != nil {
t.Errorf("ReadLog failed: %v", err)
}
rStream := stream.RecvStream()
expectedPosition := int64(0)
for count := 0; rStream.Advance(); count++ {
entry := rStream.Value()
if entry.Position != expectedPosition {
t.Errorf("unexpected position. Got %v, want %v", entry.Position, expectedPosition)
}
if expected := tests[count]; entry.Line != expected {
t.Errorf("unexpected content. Got %q, want %q", entry.Line, expected)
}
expectedPosition += int64(len(entry.Line)) + 1
}
if err := rStream.Err(); err != nil {
t.Errorf("unexpected stream error: %v", rStream.Err())
}
offset, err := stream.Finish()
if err != nil {
t.Errorf("Finish failed: %v", err)
}
if offset != expectedPosition {
t.Errorf("unexpected offset. Got %q, want %q", offset, expectedPosition)
}
// Read with follow from EOF (where the previous read ended).
stream, err = lf.ReadLog(runtime.NewContext(), offset, logreader.AllEntries, false)
if err != nil {
t.Errorf("ReadLog failed: %v", err)
}
_, err = stream.Finish()
if !verror.Is(err, logreader.EOF) {
t.Errorf("unexpected error, got %#v, want EOF", err)
}
}
func TestReadLogImplWithFollow(t *testing.T) {
runtime := rt.Init()
workdir, err := ioutil.TempDir("", "logreadertest")
if err != nil {
t.Fatalf("ioutil.TempDir: %v", err)
}
defer os.RemoveAll(workdir)
server, endpoint, err := startServer(t, &logFileDispatcher{workdir})
if err != nil {
t.Fatalf("startServer failed: %v", err)
}
defer stopServer(t, server)
const testFile = "mylogfile.INFO"
writer, err := os.Create(path.Join(workdir, testFile))
if err != nil {
t.Fatalf("Create failed: %v", err)
}
tests := []string{
"Hello World!",
"Life is too short",
"Have fun",
"Play hard",
"Break something",
"Fix it later",
}
lf, err := logreader.BindLogFile(naming.JoinAddressName(endpoint, "//"+testFile))
if err != nil {
t.Errorf("BindLogFile: %v", err)
}
_, err = lf.Size(runtime.NewContext())
if err != nil {
t.Errorf("Size failed: %v", err)
}
// Read with follow.
stream, err := lf.ReadLog(runtime.NewContext(), 0, int32(len(tests)), true)
if err != nil {
t.Errorf("ReadLog failed: %v", err)
}
rStream := stream.RecvStream()
writeAndSync(t, writer, tests[0]+"\n")
for count, pos := 0, int64(0); rStream.Advance(); count++ {
entry := rStream.Value()
if entry.Position != pos {
t.Errorf("unexpected position. Got %v, want %v", entry.Position, pos)
}
if expected := tests[count]; entry.Line != expected {
t.Errorf("unexpected content. Got %q, want %q", entry.Line, expected)
}
pos += int64(len(entry.Line)) + 1
if count+1 < len(tests) {
writeAndSync(t, writer, tests[count+1]+"\n")
}
}
if err := rStream.Err(); err != nil {
t.Errorf("unexpected stream error: %v", rStream.Err())
}
_, err = stream.Finish()
if err != nil {
t.Errorf("Finish failed: %v", err)
}
}