| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162 |
- // 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,
- // You can obtain one at https://mozilla.org/MPL/2.0/.
- package dialer
- import (
- "net"
- "net/http"
- "net/url"
- "os"
- "strings"
- "time"
- "golang.org/x/net/proxy"
- "github.com/syncthing/syncthing/lib/logger"
- )
- var (
- l = logger.DefaultLogger.NewFacility("dialer", "Dialing connections")
- proxyDialer proxy.Dialer
- usingProxy bool
- noFallback = os.Getenv("ALL_PROXY_NO_FALLBACK") != ""
- )
- type dialFunc func(network, addr string) (net.Conn, error)
- func init() {
- l.SetDebug("dialer", strings.Contains(os.Getenv("STTRACE"), "dialer") || os.Getenv("STTRACE") == "all")
- proxy.RegisterDialerType("socks", socksDialerFunction)
- proxyDialer = getDialer(proxy.Direct)
- usingProxy = proxyDialer != proxy.Direct
- if usingProxy {
- http.DefaultTransport = &http.Transport{
- Dial: Dial,
- Proxy: http.ProxyFromEnvironment,
- TLSHandshakeTimeout: 10 * time.Second,
- }
- // Defer this, so that logging gets setup.
- go func() {
- time.Sleep(500 * time.Millisecond)
- l.Infoln("Proxy settings detected")
- if noFallback {
- l.Infoln("Proxy fallback disabled")
- }
- }()
- } else {
- go func() {
- time.Sleep(500 * time.Millisecond)
- l.Debugln("Dialer logging disabled, as no proxy was detected")
- }()
- }
- }
- func dialWithFallback(proxyDialFunc dialFunc, fallbackDialFunc dialFunc, network, addr string) (net.Conn, error) {
- conn, err := proxyDialFunc(network, addr)
- if err == nil {
- l.Debugf("Dialing %s address %s via proxy - success, %s -> %s", network, addr, conn.LocalAddr(), conn.RemoteAddr())
- SetTCPOptions(conn)
- return dialerConn{
- conn, newDialerAddr(network, addr),
- }, nil
- }
- l.Debugf("Dialing %s address %s via proxy - error %s", network, addr, err)
- if noFallback {
- return conn, err
- }
- conn, err = fallbackDialFunc(network, addr)
- if err == nil {
- l.Debugf("Dialing %s address %s via fallback - success, %s -> %s", network, addr, conn.LocalAddr(), conn.RemoteAddr())
- SetTCPOptions(conn)
- } else {
- l.Debugf("Dialing %s address %s via fallback - error %s", network, addr, err)
- }
- return conn, err
- }
- // This is a rip off of proxy.FromURL for "socks" URL scheme
- func socksDialerFunction(u *url.URL, forward proxy.Dialer) (proxy.Dialer, error) {
- var auth *proxy.Auth
- if u.User != nil {
- auth = new(proxy.Auth)
- auth.User = u.User.Username()
- if p, ok := u.User.Password(); ok {
- auth.Password = p
- }
- }
- return proxy.SOCKS5("tcp", u.Host, auth, forward)
- }
- // This is a rip off of proxy.FromEnvironment with a custom forward dialer
- func getDialer(forward proxy.Dialer) proxy.Dialer {
- allProxy := os.Getenv("all_proxy")
- if len(allProxy) == 0 {
- return forward
- }
- proxyURL, err := url.Parse(allProxy)
- if err != nil {
- return forward
- }
- prxy, err := proxy.FromURL(proxyURL, forward)
- if err != nil {
- return forward
- }
- noProxy := os.Getenv("no_proxy")
- if len(noProxy) == 0 {
- return prxy
- }
- perHost := proxy.NewPerHost(prxy, forward)
- perHost.AddFromString(noProxy)
- return perHost
- }
- type timeoutDirectDialer struct {
- timeout time.Duration
- }
- func (d *timeoutDirectDialer) Dial(network, addr string) (net.Conn, error) {
- return net.DialTimeout(network, addr, d.timeout)
- }
- type dialerConn struct {
- net.Conn
- addr net.Addr
- }
- func (c dialerConn) RemoteAddr() net.Addr {
- return c.addr
- }
- func newDialerAddr(network, addr string) net.Addr {
- netaddr, err := net.ResolveIPAddr(network, addr)
- if err == nil {
- return netaddr
- }
- return fallbackAddr{network, addr}
- }
- type fallbackAddr struct {
- network string
- addr string
- }
- func (a fallbackAddr) Network() string {
- return a.network
- }
- func (a fallbackAddr) String() string {
- return a.addr
- }
|