odps/tunnel/protoc_stream_writer.go (96 lines of code) (raw):
// Licensed to the Apache Software Foundation (ASF) under one
// or more contributor license agreements. See the NOTICE file
// distributed with this work for additional information
// regarding copyright ownership. The ASF licenses this file
// to you under the Apache License, Version 2.0 (the
// "License"); you may not use this file except in compliance
// with the License. You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package tunnel
import (
"io"
"math"
"sync"
"google.golang.org/protobuf/encoding/protowire"
)
var bufPool = &sync.Pool{
New: func() interface{} {
v := make([]byte, 0, 10)
return &v
},
}
func releaseToPool(b *[]byte) {
*b = (*b)[:0]
bufPool.Put(b)
}
type ProtocStreamWriter struct {
inner io.Writer
}
func NewProtocStreamWriter(w io.Writer) *ProtocStreamWriter {
return &ProtocStreamWriter{
inner: w,
}
}
func (r *ProtocStreamWriter) WriteTag(num protowire.Number, typ protowire.Type) error {
return r.WriteVarint(protowire.EncodeTag(num, typ))
}
func (r *ProtocStreamWriter) WriteVarint(v uint64) error {
b := bufPool.Get().(*[]byte)
*b = protowire.AppendVarint(*b, v)
err := writeFull(r.inner, *b)
releaseToPool(b)
return err
}
func (r *ProtocStreamWriter) WriteFixed32(val uint32) error {
b := bufPool.Get().(*[]byte)
*b = protowire.AppendFixed32(*b, val)
err := writeFull(r.inner, *b)
releaseToPool(b)
return err
}
func (r *ProtocStreamWriter) WriteFixed64(val uint64) error {
b := bufPool.Get().(*[]byte)
*b = protowire.AppendFixed64(*b, val)
err := writeFull(r.inner, *b)
releaseToPool(b)
return err
}
func (r *ProtocStreamWriter) WriteBytes(data []byte) error {
if err := r.WriteVarint(uint64(len(data))); err != nil {
return err
}
err := writeFull(r.inner, data)
return err
}
func writeFull(w io.Writer, data []byte) error {
for len(data) > 0 {
n, err := w.Write(data)
if err != nil {
return err
}
data = data[n:]
}
return nil
}
func (r *ProtocStreamWriter) WriteBool(val bool) error {
// true: 1, false: 0
u := protowire.EncodeBool(val)
return r.WriteVarint(u)
}
func (r *ProtocStreamWriter) WriteInt32(val int32) error {
return r.WriteVarint(uint64(val))
}
func (r *ProtocStreamWriter) WriteSInt32(val int32) error {
u := protowire.EncodeZigZag(int64(val))
return r.WriteVarint(u)
}
func (r *ProtocStreamWriter) WriteInt64(val int64) error {
return r.WriteVarint(uint64(val))
}
func (r *ProtocStreamWriter) WriteSInt64(val int64) error {
u := protowire.EncodeZigZag(val)
return r.WriteVarint(u)
}
func (r *ProtocStreamWriter) WriteUInt32(val uint32) error {
return r.WriteVarint(uint64(val))
}
func (r *ProtocStreamWriter) WriteUInt64(val uint64) error {
return r.WriteVarint(val)
}
func (r *ProtocStreamWriter) WriteFloat32(val float32) error {
return r.WriteFixed32(math.Float32bits(val))
}
func (r *ProtocStreamWriter) WriteFloat64(val float64) error {
return r.WriteFixed64(math.Float64bits(val))
}