gost/pkg/dialer/http3/dialer.go
2022-01-18 23:54:59 +08:00

113 lines
2.2 KiB
Go

package http3
import (
"context"
"errors"
"fmt"
"io"
"net"
"net/http"
"net/http/httputil"
"strings"
"time"
"github.com/go-gost/gost/pkg/dialer"
"github.com/go-gost/gost/pkg/logger"
md "github.com/go-gost/gost/pkg/metadata"
"github.com/go-gost/gost/pkg/registry"
"github.com/lucas-clemente/quic-go/http3"
)
func init() {
registry.RegisterDialer("http3", NewDialer)
}
type http3Dialer struct {
client *http.Client
md metadata
logger logger.Logger
options dialer.Options
}
func NewDialer(opts ...dialer.Option) dialer.Dialer {
options := dialer.Options{}
for _, opt := range opts {
opt(&options)
}
tr := &http3.RoundTripper{
TLSClientConfig: options.TLSConfig,
}
client := &http.Client{
Timeout: 60 * time.Second,
Transport: tr,
}
return &http3Dialer{
client: client,
logger: options.Logger,
options: options,
}
}
func (d *http3Dialer) Init(md md.Metadata) (err error) {
return d.parseMetadata(md)
}
func (d *http3Dialer) Dial(ctx context.Context, addr string, opts ...dialer.DialOption) (net.Conn, error) {
token, err := d.authorize(ctx, addr)
if err != nil {
d.logger.Error(err)
return nil, err
}
c := &conn{
cid: token,
addr: addr,
client: d.client,
rxc: make(chan []byte, 128),
closed: make(chan struct{}),
md: d.md,
logger: d.logger,
}
go c.readLoop()
return c, nil
}
func (d *http3Dialer) authorize(ctx context.Context, addr string) (token string, err error) {
url := fmt.Sprintf("https://%s%s", addr, d.md.authorizePath)
r, err := http.NewRequestWithContext(ctx, http.MethodGet, url, nil)
if err != nil {
return
}
if d.logger.IsLevelEnabled(logger.DebugLevel) {
dump, _ := httputil.DumpRequest(r, false)
d.logger.Debug(string(dump))
}
resp, err := d.client.Do(r)
if err != nil {
return
}
defer resp.Body.Close()
if d.logger.IsLevelEnabled(logger.DebugLevel) {
dump, _ := httputil.DumpResponse(resp, false)
d.logger.Debug(string(dump))
}
data, err := io.ReadAll(resp.Body)
if err != nil {
return
}
if strings.HasPrefix(string(data), "token=") {
token = strings.TrimPrefix(string(data), "token=")
}
if token == "" {
err = errors.New("authorize failed")
}
return
}