2019-05-24 18:39:26 +01:00
|
|
|
// Package quic provides a QUIC based transport
|
|
|
|
package quic
|
|
|
|
|
|
|
|
import (
|
2019-07-11 10:47:02 +01:00
|
|
|
"context"
|
2019-05-24 18:39:26 +01:00
|
|
|
"crypto/tls"
|
|
|
|
"encoding/gob"
|
2019-09-12 16:22:43 -07:00
|
|
|
"time"
|
2019-05-24 18:39:26 +01:00
|
|
|
|
2019-12-03 22:59:44 +03:00
|
|
|
quic "github.com/lucas-clemente/quic-go"
|
2020-08-19 17:47:17 +03:00
|
|
|
"github.com/unistack-org/micro/v3/transport"
|
|
|
|
utls "github.com/unistack-org/micro/v3/util/tls"
|
2019-05-24 18:39:26 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
type quicSocket struct {
|
|
|
|
s quic.Session
|
|
|
|
st quic.Stream
|
|
|
|
enc *gob.Encoder
|
|
|
|
dec *gob.Decoder
|
|
|
|
}
|
|
|
|
|
|
|
|
type quicTransport struct {
|
|
|
|
opts transport.Options
|
|
|
|
}
|
|
|
|
|
|
|
|
type quicClient struct {
|
|
|
|
*quicSocket
|
|
|
|
t *quicTransport
|
|
|
|
opts transport.DialOptions
|
|
|
|
}
|
|
|
|
|
|
|
|
type quicListener struct {
|
|
|
|
l quic.Listener
|
|
|
|
t *quicTransport
|
|
|
|
opts transport.ListenOptions
|
|
|
|
}
|
|
|
|
|
|
|
|
func (q *quicClient) Close() error {
|
|
|
|
return q.quicSocket.st.Close()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (q *quicSocket) Recv(m *transport.Message) error {
|
|
|
|
return q.dec.Decode(&m)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (q *quicSocket) Send(m *transport.Message) error {
|
2019-09-12 16:22:43 -07:00
|
|
|
// set the write deadline
|
|
|
|
q.st.SetWriteDeadline(time.Now().Add(time.Second * 10))
|
|
|
|
// send the data
|
2019-05-24 18:39:26 +01:00
|
|
|
return q.enc.Encode(m)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (q *quicSocket) Close() error {
|
2020-08-20 12:37:40 +03:00
|
|
|
return q.s.CloseWithError(0, "")
|
2019-05-24 18:39:26 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func (q *quicSocket) Local() string {
|
|
|
|
return q.s.LocalAddr().String()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (q *quicSocket) Remote() string {
|
|
|
|
return q.s.RemoteAddr().String()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (q *quicListener) Addr() string {
|
|
|
|
return q.l.Addr().String()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (q *quicListener) Close() error {
|
|
|
|
return q.l.Close()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (q *quicListener) Accept(fn func(transport.Socket)) error {
|
|
|
|
for {
|
2019-07-11 10:47:02 +01:00
|
|
|
s, err := q.l.Accept(context.TODO())
|
2019-05-24 18:39:26 +01:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2019-07-11 10:47:02 +01:00
|
|
|
stream, err := s.AcceptStream(context.TODO())
|
2019-05-24 18:39:26 +01:00
|
|
|
if err != nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
fn(&quicSocket{
|
|
|
|
s: s,
|
|
|
|
st: stream,
|
|
|
|
enc: gob.NewEncoder(stream),
|
|
|
|
dec: gob.NewDecoder(stream),
|
|
|
|
})
|
|
|
|
}()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (q *quicTransport) Init(opts ...transport.Option) error {
|
|
|
|
for _, o := range opts {
|
|
|
|
o(&q.opts)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (q *quicTransport) Options() transport.Options {
|
|
|
|
return q.opts
|
|
|
|
}
|
|
|
|
|
|
|
|
func (q *quicTransport) Dial(addr string, opts ...transport.DialOption) (transport.Client, error) {
|
|
|
|
var options transport.DialOptions
|
|
|
|
for _, o := range opts {
|
|
|
|
o(&options)
|
|
|
|
}
|
|
|
|
|
|
|
|
config := q.opts.TLSConfig
|
|
|
|
if config == nil {
|
|
|
|
config = &tls.Config{
|
|
|
|
InsecureSkipVerify: true,
|
2019-07-11 09:38:20 +01:00
|
|
|
NextProtos: []string{"http/1.1"},
|
2019-05-24 18:39:26 +01:00
|
|
|
}
|
|
|
|
}
|
2019-09-12 16:22:43 -07:00
|
|
|
s, err := quic.DialAddr(addr, config, &quic.Config{
|
2020-08-20 12:37:40 +03:00
|
|
|
MaxIdleTimeout: time.Minute * 2,
|
|
|
|
KeepAlive: true,
|
2019-09-12 16:22:43 -07:00
|
|
|
})
|
2019-05-24 18:39:26 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2019-07-11 10:47:02 +01:00
|
|
|
st, err := s.OpenStreamSync(context.TODO())
|
2019-05-24 18:39:26 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
enc := gob.NewEncoder(st)
|
|
|
|
dec := gob.NewDecoder(st)
|
|
|
|
|
|
|
|
return &quicClient{
|
|
|
|
&quicSocket{
|
|
|
|
s: s,
|
|
|
|
st: st,
|
|
|
|
enc: enc,
|
|
|
|
dec: dec,
|
|
|
|
},
|
|
|
|
q,
|
|
|
|
options,
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (q *quicTransport) Listen(addr string, opts ...transport.ListenOption) (transport.Listener, error) {
|
|
|
|
var options transport.ListenOptions
|
|
|
|
for _, o := range opts {
|
|
|
|
o(&options)
|
|
|
|
}
|
|
|
|
|
|
|
|
config := q.opts.TLSConfig
|
|
|
|
if config == nil {
|
|
|
|
cfg, err := utls.Certificate(addr)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
config = &tls.Config{
|
|
|
|
Certificates: []tls.Certificate{cfg},
|
2019-07-11 09:38:20 +01:00
|
|
|
NextProtos: []string{"http/1.1"},
|
2019-05-24 18:39:26 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-08-11 16:12:31 +01:00
|
|
|
l, err := quic.ListenAddr(addr, config, &quic.Config{KeepAlive: true})
|
2019-05-24 18:39:26 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &quicListener{
|
|
|
|
l: l,
|
|
|
|
t: q,
|
|
|
|
opts: options,
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (q *quicTransport) String() string {
|
|
|
|
return "quic"
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewTransport(opts ...transport.Option) transport.Transport {
|
|
|
|
options := transport.Options{}
|
|
|
|
|
|
|
|
for _, o := range opts {
|
|
|
|
o(&options)
|
|
|
|
}
|
|
|
|
|
|
|
|
return &quicTransport{
|
|
|
|
opts: options,
|
|
|
|
}
|
|
|
|
}
|