-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathhikari.go
89 lines (70 loc) · 1.96 KB
/
hikari.go
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
package hikari
import (
"bufio"
"encoding/json"
"errors"
"fmt"
"io"
"github.com/olehvolynets/hikari/config"
)
type Hikari struct {
sink io.Writer
refHandlers []*ReferenceHandler
eventHandlers []*EventHandler
}
func NewHikari(out io.Writer, cfg *config.Config) (*Hikari, error) {
s := Hikari{
sink: out,
refHandlers: make([]*ReferenceHandler, len(cfg.Types)),
eventHandlers: make([]*EventHandler, len(cfg.Events)),
}
for idx, t := range cfg.Types {
s.refHandlers[idx] = NewReferenceHandler(t)
}
for idx, evt := range cfg.Events {
s.eventHandlers[idx] = NewEventHandler(evt, s.refHandlers)
}
return &s, nil
}
func (app *Hikari) Start(r io.Reader) error {
// Has to be bufio.Reader to be able to read till the end of the line.
buffReader := bufio.NewReader(r)
decoder := json.NewDecoder(buffReader)
for {
entry := make(Entry)
if err := decoder.Decode(&entry); err != nil {
if errors.Is(err, io.EOF) {
// All readers are exhausted at this point.
break
}
// json.Decoder internally buffers 4kB read from the reader
// so to process input need to aggregate remaining buffered data
// with the primary reader.
mr := io.MultiReader(decoder.Buffered(), buffReader)
buffReader = bufio.NewReader(mr)
line, err := buffReader.ReadString('\n')
if err != nil {
return err
}
fmt.Fprint(app.sink, line)
// Recreate decoder after processing the line that caused
// the previous one to fail. It doesn't mean that the next line
// will be parsed successfully, but in that case it will
// be handled in the same maner.
decoder = json.NewDecoder(buffReader)
continue
}
handler := app.MatchEvent(entry)
ctx := Context{W: app.sink, IndentChar: "\t"}
handler.Render(&ctx, entry)
}
return nil
}
func (app *Hikari) MatchEvent(entry Entry) Handler {
for _, handler := range app.eventHandlers {
if handler.Event.Match(entry) {
return handler
}
}
return DefaultEventHandler
}