package jsoniter import ( "io" ) type Stream struct { cfg *frozenConfig out io.Writer buf []byte n int Error error indention int } func NewStream(cfg *frozenConfig, out io.Writer, bufSize int) *Stream { return &Stream{ cfg: cfg, out: out, buf: make([]byte, bufSize), n: 0, Error: nil, indention: 0, } } func (b *Stream) Reset(out io.Writer) { b.out = out b.n = 0 } // Available returns how many bytes are unused in the buffer. func (b *Stream) Available() int { return len(b.buf) - b.n } // Buffered returns the number of bytes that have been written into the current buffer. func (b *Stream) Buffered() int { return b.n } func (b *Stream) Buffer() []byte { return b.buf[:b.n] } // Write writes the contents of p into the buffer. // It returns the number of bytes written. // If nn < len(p), it also returns an error explaining // why the write is short. func (b *Stream) Write(p []byte) (nn int, err error) { for len(p) > b.Available() && b.Error == nil { if b.out == nil { b.growAtLeast(len(p)) } else { var n int if b.Buffered() == 0 { // Large write, empty buffer. // Write directly from p to avoid copy. n, b.Error = b.out.Write(p) } else { n = copy(b.buf[b.n:], p) b.n += n b.Flush() } nn += n p = p[n:] } } if b.Error != nil { return nn, b.Error } n := copy(b.buf[b.n:], p) b.n += n nn += n return nn, nil } // WriteByte writes a single byte. func (b *Stream) writeByte(c byte) { if b.Error != nil { return } if b.Available() < 1 { b.growAtLeast(1) } b.buf[b.n] = c b.n++ } func (b *Stream) writeTwoBytes(c1 byte, c2 byte) { if b.Error != nil { return } if b.Available() < 2 { b.growAtLeast(2) } b.buf[b.n] = c1 b.buf[b.n+1] = c2 b.n += 2 } func (b *Stream) writeThreeBytes(c1 byte, c2 byte, c3 byte) { if b.Error != nil { return } if b.Available() < 3 { b.growAtLeast(3) } b.buf[b.n] = c1 b.buf[b.n+1] = c2 b.buf[b.n+2] = c3 b.n += 3 } func (b *Stream) writeFourBytes(c1 byte, c2 byte, c3 byte, c4 byte) { if b.Error != nil { return } if b.Available() < 4 { b.growAtLeast(4) } b.buf[b.n] = c1 b.buf[b.n+1] = c2 b.buf[b.n+2] = c3 b.buf[b.n+3] = c4 b.n += 4 } func (b *Stream) writeFiveBytes(c1 byte, c2 byte, c3 byte, c4 byte, c5 byte) { if b.Error != nil { return } if b.Available() < 5 { b.growAtLeast(5) } b.buf[b.n] = c1 b.buf[b.n+1] = c2 b.buf[b.n+2] = c3 b.buf[b.n+3] = c4 b.buf[b.n+4] = c5 b.n += 5 } // Flush writes any buffered data to the underlying io.Writer. func (b *Stream) Flush() error { if b.out == nil { return nil } if b.Error != nil { return b.Error } if b.n == 0 { return nil } n, err := b.out.Write(b.buf[0:b.n]) if n < b.n && err == nil { err = io.ErrShortWrite } if err != nil { if n > 0 && n < b.n { copy(b.buf[0:b.n-n], b.buf[n:b.n]) } b.n -= n b.Error = err return err } b.n = 0 return nil } func (b *Stream) ensure(minimal int) { available := b.Available() if available < minimal { if b.n > 1024 { b.Flush() } b.growAtLeast(minimal) } } func (b *Stream) growAtLeast(minimal int) { toGrow := len(b.buf) if toGrow < minimal { toGrow = minimal } newBuf := make([]byte, len(b.buf)+toGrow) copy(newBuf, b.Buffer()) b.buf = newBuf } func (b *Stream) WriteRaw(s string) { b.ensure(len(s)) if b.Error != nil { return } n := copy(b.buf[b.n:], s) b.n += n } func (stream *Stream) WriteNil() { stream.writeFourBytes('n', 'u', 'l', 'l') } func (stream *Stream) WriteTrue() { stream.writeFourBytes('t', 'r', 'u', 'e') } func (stream *Stream) WriteFalse() { stream.writeFiveBytes('f', 'a', 'l', 's', 'e') } func (stream *Stream) WriteBool(val bool) { if val { stream.WriteTrue() } else { stream.WriteFalse() } } func (stream *Stream) WriteObjectStart() { stream.indention += stream.cfg.indentionStep stream.writeByte('{') stream.writeIndention(0) } func (stream *Stream) WriteObjectField(field string) { stream.WriteString(field) if stream.indention > 0 { stream.writeTwoBytes(':', ' ') } else { stream.writeByte(':') } } func (stream *Stream) WriteObjectEnd() { stream.writeIndention(stream.cfg.indentionStep) stream.indention -= stream.cfg.indentionStep stream.writeByte('}') } func (stream *Stream) WriteEmptyObject() { stream.writeByte('{') stream.writeByte('}') } func (stream *Stream) WriteMore() { stream.writeByte(',') stream.writeIndention(0) } func (stream *Stream) WriteArrayStart() { stream.indention += stream.cfg.indentionStep stream.writeByte('[') stream.writeIndention(0) } func (stream *Stream) WriteEmptyArray() { stream.writeByte('[') stream.writeByte(']') } func (stream *Stream) WriteArrayEnd() { stream.writeIndention(stream.cfg.indentionStep) stream.indention -= stream.cfg.indentionStep stream.writeByte(']') } func (stream *Stream) writeIndention(delta int) { if stream.indention == 0 { return } stream.writeByte('\n') toWrite := stream.indention - delta stream.ensure(toWrite) for i := 0; i < toWrite && stream.n < len(stream.buf); i++ { stream.buf[stream.n] = ' ' stream.n++ } }