Create & Init Project...

This commit is contained in:
2019-04-22 18:49:16 +08:00
commit fc4fa37393
25440 changed files with 4054998 additions and 0 deletions

View File

@@ -0,0 +1,38 @@
package(default_visibility = ["//visibility:public"])
load(
"@io_bazel_rules_go//go:def.bzl",
"go_test",
"go_library",
)
go_test(
name = "go_default_test",
srcs = ["deliver_test.go"],
embed = [":go_default_library"],
rundir = ".",
tags = ["automanaged"],
)
go_library(
name = "go_default_library",
srcs = ["deliver.go"],
importpath = "go-common/app/service/main/dapper/pkg/deliver",
tags = ["automanaged"],
visibility = ["//visibility:public"],
deps = ["//library/log:go_default_library"],
)
filegroup(
name = "package-srcs",
srcs = glob(["**"]),
tags = ["automanaged"],
visibility = ["//visibility:private"],
)
filegroup(
name = "all-srcs",
srcs = [":package-srcs"],
tags = ["automanaged"],
visibility = ["//visibility:public"],
)

View File

@@ -0,0 +1,155 @@
package deliver
import (
"encoding/binary"
"fmt"
"math/rand"
"net"
"sync"
"time"
"go-common/library/log"
)
var (
_magicBuf = []byte{0xAC, 0xBE}
_bufpool sync.Pool
)
func init() {
rand.Seed(time.Now().UnixNano())
_bufpool = sync.Pool{New: func() interface{} {
return make([]byte, 0, 4096)
}}
}
func freeBuf(buf []byte) {
buf = buf[:0]
_bufpool.Put(buf)
}
func getBuf() []byte {
return _bufpool.Get().([]byte)
}
// Deliver deliver span to dapper-service through tcp
type Deliver struct {
servers []string
readFn func() ([]byte, error)
conn *net.TCPConn
dataCh chan []byte
closeCh chan struct{}
closed bool
}
// New Deliver
func New(servers []string, readFn func() ([]byte, error)) (*Deliver, error) {
if len(servers) == 0 {
return nil, fmt.Errorf("no server provide")
}
d := &Deliver{
servers: servers,
readFn: readFn,
closeCh: make(chan struct{}, 1),
dataCh: make(chan []byte),
}
return d, d.start()
}
func (d *Deliver) start() error {
if err := d.dial(); err != nil {
return err
}
go d.fetch()
go d.loop()
return nil
}
func (d *Deliver) fetch() {
for {
if d.closed {
return
}
data, err := d.readFn()
if err != nil {
log.Error("deliver read data error: %s", err)
continue
}
d.dataCh <- data
}
}
func (d *Deliver) loop() {
for {
select {
case <-d.closeCh:
return
case data := <-d.dataCh:
data = warpData(data)
send:
_, err := d.conn.Write(data)
if err == nil {
freeBuf(data)
continue
}
d.reDial()
goto send
}
}
}
// Close deliver
func (d *Deliver) Close() error {
if d.closed {
return fmt.Errorf("already closed")
}
d.closed = true
d.closeCh <- struct{}{}
timer := time.NewTimer(50 * time.Millisecond)
select {
case data := <-d.dataCh:
// write last data to conn
_, err := d.conn.Write(data)
return fmt.Errorf("write last data error: %s", err)
case <-timer.C:
return nil
}
return nil
}
func (d *Deliver) reDial() {
if d.conn != nil {
d.conn.Close()
}
for {
if err := d.dial(); err != nil {
log.Error("redial error: %s, retry after second", err)
time.Sleep(time.Second)
}
break
}
}
func (d *Deliver) dial() error {
server := chioceServer(d.servers)
conn, err := net.Dial("tcp", server)
if err != nil {
return fmt.Errorf("dial tcp://%s error: %s", server, err)
}
d.conn = conn.(*net.TCPConn)
d.conn.SetKeepAlive(true)
return nil
}
func chioceServer(servers []string) string {
return servers[rand.Intn(len(servers))]
}
func warpData(data []byte) []byte {
buf := getBuf()
buf = append(buf, _magicBuf...)
buf = append(buf, []byte{0, 0, 0, 0, 0, 0}...)
binary.BigEndian.PutUint32(buf[2:6], uint32(len(data)+2))
buf = append(buf, data...)
return buf
}

View File

@@ -0,0 +1,45 @@
package deliver
import (
"bytes"
"encoding/binary"
"io"
"net"
"testing"
"time"
)
func TestDeliver(t *testing.T) {
buf := &bytes.Buffer{}
lis, err := net.Listen("tcp", "127.0.0.1:12233")
if err != nil {
t.Fatal(err)
}
go func() {
conn, err := lis.Accept()
if err != nil {
t.Fatal(err)
}
io.Copy(buf, conn)
}()
data := []byte("hello world")
readed := make(chan bool, 1)
d, err := New([]string{"127.0.0.1:12233"}, func() ([]byte, error) {
readed <- true
return data, nil
})
if err != nil {
t.Fatal(err)
}
time.Sleep(500 * time.Millisecond)
if !bytes.Equal(buf.Bytes()[0:2], _magicBuf) {
t.Error("invalid data, wrong magic header")
}
if int(binary.BigEndian.Uint32(buf.Bytes()[2:6])) != len(data) {
t.Error("wrong data length")
}
if !bytes.Equal(buf.Bytes()[6:], data) {
t.Errorf("invalid content %s", buf.Bytes()[6:])
}
d.Close()
}