2015-11-26 23:31:37 +00:00
|
|
|
// Copyright (C) 2015 The Syncthing Authors.
|
|
|
|
//
|
|
|
|
// This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
// License, v. 2.0. If a copy of the MPL was not distributed with this file,
|
2017-02-09 07:52:18 +01:00
|
|
|
// You can obtain one at https://mozilla.org/MPL/2.0/.
|
2015-11-26 23:31:37 +00:00
|
|
|
|
|
|
|
package dialer
|
|
|
|
|
|
|
|
import (
|
2019-11-26 08:39:51 +01:00
|
|
|
"context"
|
|
|
|
"errors"
|
2016-05-31 08:11:57 +00:00
|
|
|
"fmt"
|
2015-11-26 23:31:37 +00:00
|
|
|
"net"
|
|
|
|
"time"
|
2016-12-20 20:14:52 +00:00
|
|
|
|
|
|
|
"golang.org/x/net/ipv4"
|
|
|
|
"golang.org/x/net/ipv6"
|
2019-11-26 08:39:51 +01:00
|
|
|
"golang.org/x/net/proxy"
|
2015-11-26 23:31:37 +00:00
|
|
|
)
|
|
|
|
|
2019-11-26 08:39:51 +01:00
|
|
|
var errUnexpectedInterfaceType = errors.New("unexpected interface type")
|
2016-01-30 01:52:32 +00:00
|
|
|
|
2016-05-31 08:11:57 +00:00
|
|
|
// SetTCPOptions sets our default TCP options on a TCP connection, possibly
|
|
|
|
// digging through dialerConn to extract the *net.TCPConn
|
|
|
|
func SetTCPOptions(conn net.Conn) error {
|
|
|
|
switch conn := conn.(type) {
|
|
|
|
case *net.TCPConn:
|
|
|
|
var err error
|
|
|
|
if err = conn.SetLinger(0); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err = conn.SetNoDelay(false); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err = conn.SetKeepAlivePeriod(60 * time.Second); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err = conn.SetKeepAlive(true); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
default:
|
|
|
|
return fmt.Errorf("unknown connection type %T", conn)
|
2016-01-30 01:52:32 +00:00
|
|
|
}
|
|
|
|
}
|
2016-12-20 20:14:52 +00:00
|
|
|
|
|
|
|
func SetTrafficClass(conn net.Conn, class int) error {
|
|
|
|
switch conn := conn.(type) {
|
|
|
|
case *net.TCPConn:
|
|
|
|
e1 := ipv4.NewConn(conn).SetTOS(class)
|
|
|
|
e2 := ipv6.NewConn(conn).SetTrafficClass(class)
|
|
|
|
|
|
|
|
if e1 != nil {
|
|
|
|
return e1
|
|
|
|
}
|
|
|
|
return e2
|
|
|
|
default:
|
|
|
|
return fmt.Errorf("unknown connection type %T", conn)
|
|
|
|
}
|
|
|
|
}
|
2019-11-26 08:39:51 +01:00
|
|
|
|
|
|
|
func dialContextWithFallback(ctx context.Context, fallback proxy.ContextDialer, network, addr string) (net.Conn, error) {
|
|
|
|
dialer, ok := proxy.FromEnvironment().(proxy.ContextDialer)
|
|
|
|
if !ok {
|
|
|
|
return nil, errUnexpectedInterfaceType
|
|
|
|
}
|
|
|
|
if dialer == proxy.Direct {
|
|
|
|
return fallback.DialContext(ctx, network, addr)
|
|
|
|
}
|
|
|
|
if noFallback {
|
|
|
|
return dialer.DialContext(ctx, network, addr)
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx, cancel := context.WithCancel(ctx)
|
|
|
|
defer cancel()
|
|
|
|
var proxyConn, fallbackConn net.Conn
|
|
|
|
var proxyErr, fallbackErr error
|
|
|
|
proxyDone := make(chan struct{})
|
|
|
|
fallbackDone := make(chan struct{})
|
|
|
|
go func() {
|
|
|
|
proxyConn, proxyErr = dialer.DialContext(ctx, network, addr)
|
|
|
|
close(proxyDone)
|
|
|
|
}()
|
|
|
|
go func() {
|
|
|
|
fallbackConn, fallbackErr = fallback.DialContext(ctx, network, addr)
|
|
|
|
close(fallbackDone)
|
|
|
|
}()
|
|
|
|
<-proxyDone
|
|
|
|
if proxyErr == nil {
|
|
|
|
go func() {
|
|
|
|
<-fallbackDone
|
|
|
|
if fallbackErr == nil {
|
|
|
|
fallbackConn.Close()
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
return proxyConn, nil
|
|
|
|
}
|
|
|
|
<-fallbackDone
|
|
|
|
return fallbackConn, fallbackErr
|
|
|
|
}
|
|
|
|
|
|
|
|
// DialContext dials via context and/or directly, depending on how it is configured.
|
|
|
|
// If dialing via proxy and allowing fallback, dialing for both happens simultaneously
|
|
|
|
// and the proxy connection is returned if successful.
|
|
|
|
func DialContext(ctx context.Context, network, addr string) (net.Conn, error) {
|
|
|
|
return dialContextWithFallback(ctx, proxy.Direct, network, addr)
|
|
|
|
}
|