| // Copyright 2012 The Go Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style |
| // license that can be found in the LICENSE file. |
| |
| package bytes_test |
| |
| import ( |
| . "bytes" |
| "fmt" |
| "io" |
| "io/ioutil" |
| "os" |
| "testing" |
| ) |
| |
| func TestReader(t *testing.T) { |
| r := NewReader([]byte("0123456789")) |
| tests := []struct { |
| off int64 |
| seek int |
| n int |
| want string |
| wantpos int64 |
| seekerr string |
| }{ |
| {seek: os.SEEK_SET, off: 0, n: 20, want: "0123456789"}, |
| {seek: os.SEEK_SET, off: 1, n: 1, want: "1"}, |
| {seek: os.SEEK_CUR, off: 1, wantpos: 3, n: 2, want: "34"}, |
| {seek: os.SEEK_SET, off: -1, seekerr: "bytes: negative position"}, |
| {seek: os.SEEK_SET, off: 1<<31 - 1}, |
| {seek: os.SEEK_CUR, off: 1, seekerr: "bytes: position out of range"}, |
| {seek: os.SEEK_SET, n: 5, want: "01234"}, |
| {seek: os.SEEK_CUR, n: 5, want: "56789"}, |
| {seek: os.SEEK_END, off: -1, n: 1, wantpos: 9, want: "9"}, |
| } |
| |
| for i, tt := range tests { |
| pos, err := r.Seek(tt.off, tt.seek) |
| if err == nil && tt.seekerr != "" { |
| t.Errorf("%d. want seek error %q", i, tt.seekerr) |
| continue |
| } |
| if err != nil && err.Error() != tt.seekerr { |
| t.Errorf("%d. seek error = %q; want %q", i, err.Error(), tt.seekerr) |
| continue |
| } |
| if tt.wantpos != 0 && tt.wantpos != pos { |
| t.Errorf("%d. pos = %d, want %d", i, pos, tt.wantpos) |
| } |
| buf := make([]byte, tt.n) |
| n, err := r.Read(buf) |
| if err != nil { |
| t.Errorf("%d. read = %v", i, err) |
| continue |
| } |
| got := string(buf[:n]) |
| if got != tt.want { |
| t.Errorf("%d. got %q; want %q", i, got, tt.want) |
| } |
| } |
| } |
| |
| func TestReaderAt(t *testing.T) { |
| r := NewReader([]byte("0123456789")) |
| tests := []struct { |
| off int64 |
| n int |
| want string |
| wanterr interface{} |
| }{ |
| {0, 10, "0123456789", nil}, |
| {1, 10, "123456789", io.EOF}, |
| {1, 9, "123456789", nil}, |
| {11, 10, "", io.EOF}, |
| {0, 0, "", nil}, |
| {-1, 0, "", "bytes: invalid offset"}, |
| } |
| for i, tt := range tests { |
| b := make([]byte, tt.n) |
| rn, err := r.ReadAt(b, tt.off) |
| got := string(b[:rn]) |
| if got != tt.want { |
| t.Errorf("%d. got %q; want %q", i, got, tt.want) |
| } |
| if fmt.Sprintf("%v", err) != fmt.Sprintf("%v", tt.wanterr) { |
| t.Errorf("%d. got error = %v; want %v", i, err, tt.wanterr) |
| } |
| } |
| } |
| |
| func TestReaderWriteTo(t *testing.T) { |
| for i := 0; i < 30; i += 3 { |
| var l int |
| if i > 0 { |
| l = len(data) / i |
| } |
| s := data[:l] |
| r := NewReader(testBytes[:l]) |
| var b Buffer |
| n, err := r.WriteTo(&b) |
| if expect := int64(len(s)); n != expect { |
| t.Errorf("got %v; want %v", n, expect) |
| } |
| if err != nil { |
| t.Errorf("for length %d: got error = %v; want nil", l, err) |
| } |
| if b.String() != s { |
| t.Errorf("got string %q; want %q", b.String(), s) |
| } |
| if r.Len() != 0 { |
| t.Errorf("reader contains %v bytes; want 0", r.Len()) |
| } |
| } |
| } |
| |
| // verify that copying from an empty reader always has the same results, |
| // regardless of the presence of a WriteTo method. |
| func TestReaderCopyNothing(t *testing.T) { |
| type nErr struct { |
| n int64 |
| err error |
| } |
| type justReader struct { |
| io.Reader |
| } |
| type justWriter struct { |
| io.Writer |
| } |
| discard := justWriter{ioutil.Discard} // hide ReadFrom |
| |
| var with, withOut nErr |
| with.n, with.err = io.Copy(discard, NewReader(nil)) |
| withOut.n, withOut.err = io.Copy(discard, justReader{NewReader(nil)}) |
| if with != withOut { |
| t.Errorf("behavior differs: with = %#v; without: %#v", with, withOut) |
| } |
| } |