http3.go 1.7 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455
  1. //go:build with_quic
  2. package networkquality
  3. import (
  4. "context"
  5. "crypto/tls"
  6. "net"
  7. "net/http"
  8. "github.com/sagernet/quic-go"
  9. "github.com/sagernet/quic-go/http3"
  10. sBufio "github.com/sagernet/sing/common/bufio"
  11. M "github.com/sagernet/sing/common/metadata"
  12. N "github.com/sagernet/sing/common/network"
  13. )
  14. func NewHTTP3MeasurementClientFactory(dialer N.Dialer) (MeasurementClientFactory, error) {
  15. // singleConnection and disableKeepAlives are not applied:
  16. // HTTP/3 multiplexes streams over a single QUIC connection by default.
  17. return func(connectEndpoint string, _, _ bool, readCounters, writeCounters []N.CountFunc) (*http.Client, error) {
  18. transport := &http3.Transport{
  19. Dial: func(ctx context.Context, addr string, tlsCfg *tls.Config, cfg *quic.Config) (*quic.Conn, error) {
  20. dialAddr := addr
  21. if connectEndpoint != "" {
  22. dialAddr = rewriteDialAddress(addr, connectEndpoint)
  23. }
  24. destination := M.ParseSocksaddr(dialAddr)
  25. var udpConn net.Conn
  26. var dialErr error
  27. if dialer != nil {
  28. udpConn, dialErr = dialer.DialContext(ctx, N.NetworkUDP, destination)
  29. } else {
  30. var netDialer net.Dialer
  31. udpConn, dialErr = netDialer.DialContext(ctx, N.NetworkUDP, destination.String())
  32. }
  33. if dialErr != nil {
  34. return nil, dialErr
  35. }
  36. wrappedConn := udpConn
  37. if len(readCounters) > 0 || len(writeCounters) > 0 {
  38. wrappedConn = sBufio.NewCounterConn(udpConn, readCounters, writeCounters)
  39. }
  40. packetConn := sBufio.NewUnbindPacketConn(wrappedConn)
  41. quicConn, dialErr := quic.DialEarly(ctx, packetConn, udpConn.RemoteAddr(), tlsCfg, cfg)
  42. if dialErr != nil {
  43. udpConn.Close()
  44. return nil, dialErr
  45. }
  46. return quicConn, nil
  47. },
  48. }
  49. return &http.Client{Transport: transport}, nil
  50. }, nil
  51. }