| | |
| | |
| | |
| |
|
| | package main |
| |
|
| | import ( |
| | "bytes" |
| | "encoding/json" |
| | "errors" |
| | "fmt" |
| | "io" |
| | "sync" |
| | "time" |
| | ) |
| |
|
| | |
| | type lockedWriter struct { |
| | lock sync.Mutex |
| | w io.Writer |
| | } |
| |
|
| | func (w *lockedWriter) Write(b []byte) (int, error) { |
| | w.lock.Lock() |
| | defer w.lock.Unlock() |
| | return w.w.Write(b) |
| | } |
| |
|
| | |
| | |
| | type testJSONFilter struct { |
| | w io.Writer |
| | variant string |
| |
|
| | lineBuf bytes.Buffer |
| | } |
| |
|
| | func (f *testJSONFilter) Write(b []byte) (int, error) { |
| | bn := len(b) |
| |
|
| | |
| | for len(b) > 0 { |
| | nl := bytes.IndexByte(b, '\n') |
| | if nl < 0 { |
| | f.lineBuf.Write(b) |
| | break |
| | } |
| | var line []byte |
| | if f.lineBuf.Len() > 0 { |
| | |
| | |
| | f.lineBuf.Write(b[:nl+1]) |
| | line = f.lineBuf.Bytes() |
| | } else { |
| | |
| | line = b[:nl+1] |
| | } |
| | b = b[nl+1:] |
| | f.process(line) |
| | f.lineBuf.Reset() |
| | } |
| |
|
| | return bn, nil |
| | } |
| |
|
| | func (f *testJSONFilter) Flush() { |
| | |
| | if f.lineBuf.Len() > 0 { |
| | f.w.Write(f.lineBuf.Bytes()) |
| | f.lineBuf.Reset() |
| | } |
| | } |
| |
|
| | func (f *testJSONFilter) process(line []byte) { |
| | if len(line) > 0 && line[0] == '{' { |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | dec := json.NewDecoder(bytes.NewBuffer(line)) |
| | dec.UseNumber() |
| | val, err := decodeJSONValue(dec) |
| | if err == nil && val.atom == json.Delim('{') { |
| | |
| | found := false |
| | for i := 0; i < len(val.seq); i += 2 { |
| | if val.seq[i].atom == "Package" { |
| | if pkg, ok := val.seq[i+1].atom.(string); ok { |
| | val.seq[i+1].atom = pkg + ":" + f.variant |
| | found = true |
| | break |
| | } |
| | } |
| | } |
| | if found { |
| | data, err := json.Marshal(val) |
| | if err != nil { |
| | |
| | panic(fmt.Sprintf("failed to round-trip JSON %q: %s", line, err)) |
| | } |
| | f.w.Write(data) |
| | |
| | |
| | io.Copy(f.w, dec.Buffered()) |
| | return |
| | } |
| | } |
| | } |
| |
|
| | |
| | f.w.Write(line) |
| | } |
| |
|
| | type jsonValue struct { |
| | atom json.Token |
| | seq []jsonValue |
| | } |
| |
|
| | var jsonPop = errors.New("end of JSON sequence") |
| |
|
| | func decodeJSONValue(dec *json.Decoder) (jsonValue, error) { |
| | t, err := dec.Token() |
| | if err != nil { |
| | if err == io.EOF { |
| | err = io.ErrUnexpectedEOF |
| | } |
| | return jsonValue{}, err |
| | } |
| |
|
| | switch t := t.(type) { |
| | case json.Delim: |
| | if t == '}' || t == ']' { |
| | return jsonValue{}, jsonPop |
| | } |
| |
|
| | var seq []jsonValue |
| | for { |
| | val, err := decodeJSONValue(dec) |
| | if err == jsonPop { |
| | break |
| | } else if err != nil { |
| | return jsonValue{}, err |
| | } |
| | seq = append(seq, val) |
| | } |
| | return jsonValue{t, seq}, nil |
| | default: |
| | return jsonValue{t, nil}, nil |
| | } |
| | } |
| |
|
| | func (v jsonValue) MarshalJSON() ([]byte, error) { |
| | var buf bytes.Buffer |
| | var marshal1 func(v jsonValue) error |
| | marshal1 = func(v jsonValue) error { |
| | if t, ok := v.atom.(json.Delim); ok { |
| | buf.WriteRune(rune(t)) |
| | for i, v2 := range v.seq { |
| | if t == '{' && i%2 == 1 { |
| | buf.WriteByte(':') |
| | } else if i > 0 { |
| | buf.WriteByte(',') |
| | } |
| | if err := marshal1(v2); err != nil { |
| | return err |
| | } |
| | } |
| | if t == '{' { |
| | buf.WriteByte('}') |
| | } else { |
| | buf.WriteByte(']') |
| | } |
| | return nil |
| | } |
| | bytes, err := json.Marshal(v.atom) |
| | if err != nil { |
| | return err |
| | } |
| | buf.Write(bytes) |
| | return nil |
| | } |
| | err := marshal1(v) |
| | return buf.Bytes(), err |
| | } |
| |
|
| | func synthesizeSkipEvent(enc *json.Encoder, pkg, msg string) { |
| | type event struct { |
| | Time time.Time |
| | Action string |
| | Package string |
| | Output string `json:",omitempty"` |
| | } |
| | ev := event{Time: time.Now(), Package: pkg, Action: "start"} |
| | enc.Encode(ev) |
| | ev.Action = "output" |
| | ev.Output = msg |
| | enc.Encode(ev) |
| | ev.Action = "skip" |
| | ev.Output = "" |
| | enc.Encode(ev) |
| | } |
| |
|