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
|
package toml
import (
"fmt"
"io"
toml "github.com/komkom/toml/internal"
"github.com/pkg/errors"
)
type Reader struct {
filter *toml.Filter
filterDone bool
reader io.Reader
readerDone bool
}
// New wraps an io.Reader around an io.Reader.
// Reading data from this Reader reads data from
// its underlying wrapped io.Reader, parses and
// encodes it as a JSON stream.
func New(reader io.Reader) *Reader {
return &Reader{
filter: toml.NewFilter(),
reader: reader,
}
}
func (r *Reader) Read(p []byte) (int, error) {
if !r.readerDone {
for r.filter.State.Buf.Len() < len(p) {
n, err := r.reader.Read(p)
_, err = r.filter.Write(p[:n])
if err != nil {
return 0, err
}
if errors.Is(err, io.EOF) || n == 0 {
r.readerDone = true
break
}
if err != nil {
return 0, err
}
}
}
if r.readerDone && !r.filterDone {
r.filterDone = true
r.filter.WriteRune('\n')
r.filter.WriteRune(toml.EOF)
r.filter.Close()
}
if r.readerDone && r.filterDone && len(r.filter.State.Buf.Bytes()) == 0 {
if len(r.filter.State.Scopes) != 0 {
return 0, fmt.Errorf(`invalid EOF`)
}
return 0, io.EOF
}
ln := len(p)
if len(r.filter.State.Buf.Bytes()) < ln {
ln = len(r.filter.State.Buf.Bytes())
}
n, err := r.filter.State.Buf.Read(p)
r.filter.State.Buf.Truncate(len(r.filter.State.Buf.Bytes()))
return n, err
}
|