File: stream_test.go

package info (click to toggle)
relic 7.6.1-2
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid, trixie
  • size: 3,108 kB
  • sloc: sh: 230; makefile: 10
file content (107 lines) | stat: -rw-r--r-- 2,919 bytes parent folder | download
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
package comdoc_test

import (
	"bytes"
	"fmt"
	"io"
	"os"
	"testing"
	"testing/iotest"

	"github.com/sassoftware/relic/v7/lib/comdoc"
	"github.com/stretchr/testify/assert"
	"github.com/stretchr/testify/require"
)

func TestStreamRead(t *testing.T) {
	f, err := comdoc.ReadPath("../../functest/packages/dummy.msi")
	require.NoError(t, err)
	files, err := f.ListDir(nil)
	require.NoError(t, err)
	// go through all the files once and collect the content
	contents := make([][]byte, len(files))
	for i, ff := range files {
		reader, err := f.ReadStream(ff)
		require.NoError(t, err)
		buf := make([]byte, ff.StreamSize)
		_, err = io.ReadFull(reader, buf)
		require.NoError(t, err)
		contents[i] = buf
	}
	t.Run("OddSectors", func(t *testing.T) {
		b := new(bytes.Buffer)
		// ensure a mix of full and partial sectors is used
		buf := make([]byte, f.SectorSize+1)
		for i, ff := range files {
			reader, err := f.ReadStream(ff)
			require.NoError(t, err)
			b.Reset()
			n, err := io.CopyBuffer(nonReaderFrom{w: b}, reader, buf)
			require.NoError(t, err)
			assert.Equal(t, int64(ff.StreamSize), n)
			assert.Equal(t, contents[i], b.Bytes())
		}
	})
	t.Run("SmallReads", func(t *testing.T) {
		for i, ff := range files {
			reader, err := f.ReadStream(ff)
			require.NoError(t, err)
			assert.NoError(t, iotest.TestReader(reader, contents[i]))
		}
	})
	require.NoError(t, f.Close())
}

// hide ReadFrom method so CopyBuffer uses the provided buffer
type nonReaderFrom struct{ w io.Writer }

func (w nonReaderFrom) Write(d []byte) (int, error) { return w.w.Write(d) }

func TestTruncated(t *testing.T) {
	f, err := os.Open("../../functest/packages/dummy.msi")
	require.NoError(t, err)
	defer f.Close()
	info, err := f.Stat()
	require.NoError(t, err)
	// vary how many bytes are chopped off
	for _, shortenBy := range []int64{1, 4095, 4097} {
		// exercise both full and partial sector errors
		for _, bufSize := range []int{4096, 1023} {
			t.Run(fmt.Sprintf("%d-%d", shortenBy, bufSize), func(t *testing.T) {
				buf := make([]byte, bufSize)
				trunc := truncatedReaderAt{r: f, size: info.Size() - shortenBy}
				cdf, err := comdoc.ReadFile(trunc)
				require.NoError(t, err)
				files, err := cdf.ListDir(nil)
				require.NoError(t, err)
				var someError error
				for _, ff := range files {
					reader, err := cdf.ReadStream(ff)
					require.NoError(t, err)
					if _, err := io.CopyBuffer(nonReaderFrom{w: io.Discard}, reader, buf); err != nil {
						someError = err
					}
				}
				assert.ErrorContains(t, someError, "short read")
			})
		}
	}
}

type truncatedReaderAt struct {
	r    io.ReaderAt
	size int64
}

// pretend the file is shorter
func (r truncatedReaderAt) ReadAt(d []byte, offset int64) (int, error) {
	dlen := int64(len(d))
	overage := offset + dlen - r.size
	if overage > dlen {
		return 0, io.EOF
	} else if overage > 0 {
		dlen -= overage
		d = d[:int(dlen)]
	}
	return r.r.ReadAt(d, offset)
}