Utility to interact with a stream of newline delimited JSON
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 

76 lines
1.6 KiB

package jsonstream
import (
"bufio"
"encoding/json"
"errors"
"io"
)
// Reader is used for reading newline delimited JSON objects from an input io.Reader
type Reader struct {
buffer *bufio.Reader
keys map[string]bool
}
// NewReader will create a new Reader from the provided io.Reader and keys
// k, when not an empty slice, will tell the Reader to only include the provided keys from the input JSON object
func NewReader(r io.Reader, k []string) *Reader {
var keys map[string]bool
keys = make(map[string]bool, 0)
for _, key := range k {
keys[key] = true
}
return &Reader{
buffer: bufio.NewReader(r),
keys: keys,
}
}
func (reader *Reader) processData(data interface{}) (processed map[string]interface{}, err error) {
var fields map[string]interface{}
fields, err = getAsMap(data)
if err != nil {
return processed, err
}
processed = make(map[string]interface{})
for key, value := range fields {
if _, ok := reader.keys[key]; ok {
processed[key] = value
}
}
return processed, err
}
// ReadLine will read the next JSON object from the input io.Reader
func (reader *Reader) ReadLine() (data interface{}, err error) {
var line []byte
var isPrefix bool
line, isPrefix, err = reader.buffer.ReadLine()
if isPrefix {
err = errors.New("Line exceeds the length of the buffer")
}
if err != nil {
return data, err
}
// skip empty lines, we'll fail at processing them anyways
if len(line) == 0 {
return reader.ReadLine()
}
err = json.Unmarshal(line, &data)
if err != nil {
return data, err
}
if len(reader.keys) > 0 {
data, err = reader.processData(data)
}
return data, err
}