mirror of https://github.com/gorilla/websocket.git
474 lines
13 KiB
Go
474 lines
13 KiB
Go
|
// Code generated by golang.org/x/tools/cmd/bundle. DO NOT EDIT.
|
||
|
//go:generate bundle -o x_net_proxy.go golang.org/x/net/proxy
|
||
|
|
||
|
// Package proxy provides support for a variety of protocols to proxy network
|
||
|
// data.
|
||
|
//
|
||
|
|
||
|
package websocket
|
||
|
|
||
|
import (
|
||
|
"errors"
|
||
|
"io"
|
||
|
"net"
|
||
|
"net/url"
|
||
|
"os"
|
||
|
"strconv"
|
||
|
"strings"
|
||
|
"sync"
|
||
|
)
|
||
|
|
||
|
type proxy_direct struct{}
|
||
|
|
||
|
// Direct is a direct proxy: one that makes network connections directly.
|
||
|
var proxy_Direct = proxy_direct{}
|
||
|
|
||
|
func (proxy_direct) Dial(network, addr string) (net.Conn, error) {
|
||
|
return net.Dial(network, addr)
|
||
|
}
|
||
|
|
||
|
// A PerHost directs connections to a default Dialer unless the host name
|
||
|
// requested matches one of a number of exceptions.
|
||
|
type proxy_PerHost struct {
|
||
|
def, bypass proxy_Dialer
|
||
|
|
||
|
bypassNetworks []*net.IPNet
|
||
|
bypassIPs []net.IP
|
||
|
bypassZones []string
|
||
|
bypassHosts []string
|
||
|
}
|
||
|
|
||
|
// NewPerHost returns a PerHost Dialer that directs connections to either
|
||
|
// defaultDialer or bypass, depending on whether the connection matches one of
|
||
|
// the configured rules.
|
||
|
func proxy_NewPerHost(defaultDialer, bypass proxy_Dialer) *proxy_PerHost {
|
||
|
return &proxy_PerHost{
|
||
|
def: defaultDialer,
|
||
|
bypass: bypass,
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Dial connects to the address addr on the given network through either
|
||
|
// defaultDialer or bypass.
|
||
|
func (p *proxy_PerHost) Dial(network, addr string) (c net.Conn, err error) {
|
||
|
host, _, err := net.SplitHostPort(addr)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
return p.dialerForRequest(host).Dial(network, addr)
|
||
|
}
|
||
|
|
||
|
func (p *proxy_PerHost) dialerForRequest(host string) proxy_Dialer {
|
||
|
if ip := net.ParseIP(host); ip != nil {
|
||
|
for _, net := range p.bypassNetworks {
|
||
|
if net.Contains(ip) {
|
||
|
return p.bypass
|
||
|
}
|
||
|
}
|
||
|
for _, bypassIP := range p.bypassIPs {
|
||
|
if bypassIP.Equal(ip) {
|
||
|
return p.bypass
|
||
|
}
|
||
|
}
|
||
|
return p.def
|
||
|
}
|
||
|
|
||
|
for _, zone := range p.bypassZones {
|
||
|
if strings.HasSuffix(host, zone) {
|
||
|
return p.bypass
|
||
|
}
|
||
|
if host == zone[1:] {
|
||
|
// For a zone ".example.com", we match "example.com"
|
||
|
// too.
|
||
|
return p.bypass
|
||
|
}
|
||
|
}
|
||
|
for _, bypassHost := range p.bypassHosts {
|
||
|
if bypassHost == host {
|
||
|
return p.bypass
|
||
|
}
|
||
|
}
|
||
|
return p.def
|
||
|
}
|
||
|
|
||
|
// AddFromString parses a string that contains comma-separated values
|
||
|
// specifying hosts that should use the bypass proxy. Each value is either an
|
||
|
// IP address, a CIDR range, a zone (*.example.com) or a host name
|
||
|
// (localhost). A best effort is made to parse the string and errors are
|
||
|
// ignored.
|
||
|
func (p *proxy_PerHost) AddFromString(s string) {
|
||
|
hosts := strings.Split(s, ",")
|
||
|
for _, host := range hosts {
|
||
|
host = strings.TrimSpace(host)
|
||
|
if len(host) == 0 {
|
||
|
continue
|
||
|
}
|
||
|
if strings.Contains(host, "/") {
|
||
|
// We assume that it's a CIDR address like 127.0.0.0/8
|
||
|
if _, net, err := net.ParseCIDR(host); err == nil {
|
||
|
p.AddNetwork(net)
|
||
|
}
|
||
|
continue
|
||
|
}
|
||
|
if ip := net.ParseIP(host); ip != nil {
|
||
|
p.AddIP(ip)
|
||
|
continue
|
||
|
}
|
||
|
if strings.HasPrefix(host, "*.") {
|
||
|
p.AddZone(host[1:])
|
||
|
continue
|
||
|
}
|
||
|
p.AddHost(host)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// AddIP specifies an IP address that will use the bypass proxy. Note that
|
||
|
// this will only take effect if a literal IP address is dialed. A connection
|
||
|
// to a named host will never match an IP.
|
||
|
func (p *proxy_PerHost) AddIP(ip net.IP) {
|
||
|
p.bypassIPs = append(p.bypassIPs, ip)
|
||
|
}
|
||
|
|
||
|
// AddNetwork specifies an IP range that will use the bypass proxy. Note that
|
||
|
// this will only take effect if a literal IP address is dialed. A connection
|
||
|
// to a named host will never match.
|
||
|
func (p *proxy_PerHost) AddNetwork(net *net.IPNet) {
|
||
|
p.bypassNetworks = append(p.bypassNetworks, net)
|
||
|
}
|
||
|
|
||
|
// AddZone specifies a DNS suffix that will use the bypass proxy. A zone of
|
||
|
// "example.com" matches "example.com" and all of its subdomains.
|
||
|
func (p *proxy_PerHost) AddZone(zone string) {
|
||
|
if strings.HasSuffix(zone, ".") {
|
||
|
zone = zone[:len(zone)-1]
|
||
|
}
|
||
|
if !strings.HasPrefix(zone, ".") {
|
||
|
zone = "." + zone
|
||
|
}
|
||
|
p.bypassZones = append(p.bypassZones, zone)
|
||
|
}
|
||
|
|
||
|
// AddHost specifies a host name that will use the bypass proxy.
|
||
|
func (p *proxy_PerHost) AddHost(host string) {
|
||
|
if strings.HasSuffix(host, ".") {
|
||
|
host = host[:len(host)-1]
|
||
|
}
|
||
|
p.bypassHosts = append(p.bypassHosts, host)
|
||
|
}
|
||
|
|
||
|
// A Dialer is a means to establish a connection.
|
||
|
type proxy_Dialer interface {
|
||
|
// Dial connects to the given address via the proxy.
|
||
|
Dial(network, addr string) (c net.Conn, err error)
|
||
|
}
|
||
|
|
||
|
// Auth contains authentication parameters that specific Dialers may require.
|
||
|
type proxy_Auth struct {
|
||
|
User, Password string
|
||
|
}
|
||
|
|
||
|
// FromEnvironment returns the dialer specified by the proxy related variables in
|
||
|
// the environment.
|
||
|
func proxy_FromEnvironment() proxy_Dialer {
|
||
|
allProxy := proxy_allProxyEnv.Get()
|
||
|
if len(allProxy) == 0 {
|
||
|
return proxy_Direct
|
||
|
}
|
||
|
|
||
|
proxyURL, err := url.Parse(allProxy)
|
||
|
if err != nil {
|
||
|
return proxy_Direct
|
||
|
}
|
||
|
proxy, err := proxy_FromURL(proxyURL, proxy_Direct)
|
||
|
if err != nil {
|
||
|
return proxy_Direct
|
||
|
}
|
||
|
|
||
|
noProxy := proxy_noProxyEnv.Get()
|
||
|
if len(noProxy) == 0 {
|
||
|
return proxy
|
||
|
}
|
||
|
|
||
|
perHost := proxy_NewPerHost(proxy, proxy_Direct)
|
||
|
perHost.AddFromString(noProxy)
|
||
|
return perHost
|
||
|
}
|
||
|
|
||
|
// proxySchemes is a map from URL schemes to a function that creates a Dialer
|
||
|
// from a URL with such a scheme.
|
||
|
var proxy_proxySchemes map[string]func(*url.URL, proxy_Dialer) (proxy_Dialer, error)
|
||
|
|
||
|
// RegisterDialerType takes a URL scheme and a function to generate Dialers from
|
||
|
// a URL with that scheme and a forwarding Dialer. Registered schemes are used
|
||
|
// by FromURL.
|
||
|
func proxy_RegisterDialerType(scheme string, f func(*url.URL, proxy_Dialer) (proxy_Dialer, error)) {
|
||
|
if proxy_proxySchemes == nil {
|
||
|
proxy_proxySchemes = make(map[string]func(*url.URL, proxy_Dialer) (proxy_Dialer, error))
|
||
|
}
|
||
|
proxy_proxySchemes[scheme] = f
|
||
|
}
|
||
|
|
||
|
// FromURL returns a Dialer given a URL specification and an underlying
|
||
|
// Dialer for it to make network requests.
|
||
|
func proxy_FromURL(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
|
||
|
}
|
||
|
}
|
||
|
|
||
|
switch u.Scheme {
|
||
|
case "socks5":
|
||
|
return proxy_SOCKS5("tcp", u.Host, auth, forward)
|
||
|
}
|
||
|
|
||
|
// If the scheme doesn't match any of the built-in schemes, see if it
|
||
|
// was registered by another package.
|
||
|
if proxy_proxySchemes != nil {
|
||
|
if f, ok := proxy_proxySchemes[u.Scheme]; ok {
|
||
|
return f(u, forward)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return nil, errors.New("proxy: unknown scheme: " + u.Scheme)
|
||
|
}
|
||
|
|
||
|
var (
|
||
|
proxy_allProxyEnv = &proxy_envOnce{
|
||
|
names: []string{"ALL_PROXY", "all_proxy"},
|
||
|
}
|
||
|
proxy_noProxyEnv = &proxy_envOnce{
|
||
|
names: []string{"NO_PROXY", "no_proxy"},
|
||
|
}
|
||
|
)
|
||
|
|
||
|
// envOnce looks up an environment variable (optionally by multiple
|
||
|
// names) once. It mitigates expensive lookups on some platforms
|
||
|
// (e.g. Windows).
|
||
|
// (Borrowed from net/http/transport.go)
|
||
|
type proxy_envOnce struct {
|
||
|
names []string
|
||
|
once sync.Once
|
||
|
val string
|
||
|
}
|
||
|
|
||
|
func (e *proxy_envOnce) Get() string {
|
||
|
e.once.Do(e.init)
|
||
|
return e.val
|
||
|
}
|
||
|
|
||
|
func (e *proxy_envOnce) init() {
|
||
|
for _, n := range e.names {
|
||
|
e.val = os.Getenv(n)
|
||
|
if e.val != "" {
|
||
|
return
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// SOCKS5 returns a Dialer that makes SOCKSv5 connections to the given address
|
||
|
// with an optional username and password. See RFC 1928 and RFC 1929.
|
||
|
func proxy_SOCKS5(network, addr string, auth *proxy_Auth, forward proxy_Dialer) (proxy_Dialer, error) {
|
||
|
s := &proxy_socks5{
|
||
|
network: network,
|
||
|
addr: addr,
|
||
|
forward: forward,
|
||
|
}
|
||
|
if auth != nil {
|
||
|
s.user = auth.User
|
||
|
s.password = auth.Password
|
||
|
}
|
||
|
|
||
|
return s, nil
|
||
|
}
|
||
|
|
||
|
type proxy_socks5 struct {
|
||
|
user, password string
|
||
|
network, addr string
|
||
|
forward proxy_Dialer
|
||
|
}
|
||
|
|
||
|
const proxy_socks5Version = 5
|
||
|
|
||
|
const (
|
||
|
proxy_socks5AuthNone = 0
|
||
|
proxy_socks5AuthPassword = 2
|
||
|
)
|
||
|
|
||
|
const proxy_socks5Connect = 1
|
||
|
|
||
|
const (
|
||
|
proxy_socks5IP4 = 1
|
||
|
proxy_socks5Domain = 3
|
||
|
proxy_socks5IP6 = 4
|
||
|
)
|
||
|
|
||
|
var proxy_socks5Errors = []string{
|
||
|
"",
|
||
|
"general failure",
|
||
|
"connection forbidden",
|
||
|
"network unreachable",
|
||
|
"host unreachable",
|
||
|
"connection refused",
|
||
|
"TTL expired",
|
||
|
"command not supported",
|
||
|
"address type not supported",
|
||
|
}
|
||
|
|
||
|
// Dial connects to the address addr on the given network via the SOCKS5 proxy.
|
||
|
func (s *proxy_socks5) Dial(network, addr string) (net.Conn, error) {
|
||
|
switch network {
|
||
|
case "tcp", "tcp6", "tcp4":
|
||
|
default:
|
||
|
return nil, errors.New("proxy: no support for SOCKS5 proxy connections of type " + network)
|
||
|
}
|
||
|
|
||
|
conn, err := s.forward.Dial(s.network, s.addr)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
if err := s.connect(conn, addr); err != nil {
|
||
|
conn.Close()
|
||
|
return nil, err
|
||
|
}
|
||
|
return conn, nil
|
||
|
}
|
||
|
|
||
|
// connect takes an existing connection to a socks5 proxy server,
|
||
|
// and commands the server to extend that connection to target,
|
||
|
// which must be a canonical address with a host and port.
|
||
|
func (s *proxy_socks5) connect(conn net.Conn, target string) error {
|
||
|
host, portStr, err := net.SplitHostPort(target)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
port, err := strconv.Atoi(portStr)
|
||
|
if err != nil {
|
||
|
return errors.New("proxy: failed to parse port number: " + portStr)
|
||
|
}
|
||
|
if port < 1 || port > 0xffff {
|
||
|
return errors.New("proxy: port number out of range: " + portStr)
|
||
|
}
|
||
|
|
||
|
// the size here is just an estimate
|
||
|
buf := make([]byte, 0, 6+len(host))
|
||
|
|
||
|
buf = append(buf, proxy_socks5Version)
|
||
|
if len(s.user) > 0 && len(s.user) < 256 && len(s.password) < 256 {
|
||
|
buf = append(buf, 2 /* num auth methods */, proxy_socks5AuthNone, proxy_socks5AuthPassword)
|
||
|
} else {
|
||
|
buf = append(buf, 1 /* num auth methods */, proxy_socks5AuthNone)
|
||
|
}
|
||
|
|
||
|
if _, err := conn.Write(buf); err != nil {
|
||
|
return errors.New("proxy: failed to write greeting to SOCKS5 proxy at " + s.addr + ": " + err.Error())
|
||
|
}
|
||
|
|
||
|
if _, err := io.ReadFull(conn, buf[:2]); err != nil {
|
||
|
return errors.New("proxy: failed to read greeting from SOCKS5 proxy at " + s.addr + ": " + err.Error())
|
||
|
}
|
||
|
if buf[0] != 5 {
|
||
|
return errors.New("proxy: SOCKS5 proxy at " + s.addr + " has unexpected version " + strconv.Itoa(int(buf[0])))
|
||
|
}
|
||
|
if buf[1] == 0xff {
|
||
|
return errors.New("proxy: SOCKS5 proxy at " + s.addr + " requires authentication")
|
||
|
}
|
||
|
|
||
|
// See RFC 1929
|
||
|
if buf[1] == proxy_socks5AuthPassword {
|
||
|
buf = buf[:0]
|
||
|
buf = append(buf, 1 /* password protocol version */)
|
||
|
buf = append(buf, uint8(len(s.user)))
|
||
|
buf = append(buf, s.user...)
|
||
|
buf = append(buf, uint8(len(s.password)))
|
||
|
buf = append(buf, s.password...)
|
||
|
|
||
|
if _, err := conn.Write(buf); err != nil {
|
||
|
return errors.New("proxy: failed to write authentication request to SOCKS5 proxy at " + s.addr + ": " + err.Error())
|
||
|
}
|
||
|
|
||
|
if _, err := io.ReadFull(conn, buf[:2]); err != nil {
|
||
|
return errors.New("proxy: failed to read authentication reply from SOCKS5 proxy at " + s.addr + ": " + err.Error())
|
||
|
}
|
||
|
|
||
|
if buf[1] != 0 {
|
||
|
return errors.New("proxy: SOCKS5 proxy at " + s.addr + " rejected username/password")
|
||
|
}
|
||
|
}
|
||
|
|
||
|
buf = buf[:0]
|
||
|
buf = append(buf, proxy_socks5Version, proxy_socks5Connect, 0 /* reserved */)
|
||
|
|
||
|
if ip := net.ParseIP(host); ip != nil {
|
||
|
if ip4 := ip.To4(); ip4 != nil {
|
||
|
buf = append(buf, proxy_socks5IP4)
|
||
|
ip = ip4
|
||
|
} else {
|
||
|
buf = append(buf, proxy_socks5IP6)
|
||
|
}
|
||
|
buf = append(buf, ip...)
|
||
|
} else {
|
||
|
if len(host) > 255 {
|
||
|
return errors.New("proxy: destination host name too long: " + host)
|
||
|
}
|
||
|
buf = append(buf, proxy_socks5Domain)
|
||
|
buf = append(buf, byte(len(host)))
|
||
|
buf = append(buf, host...)
|
||
|
}
|
||
|
buf = append(buf, byte(port>>8), byte(port))
|
||
|
|
||
|
if _, err := conn.Write(buf); err != nil {
|
||
|
return errors.New("proxy: failed to write connect request to SOCKS5 proxy at " + s.addr + ": " + err.Error())
|
||
|
}
|
||
|
|
||
|
if _, err := io.ReadFull(conn, buf[:4]); err != nil {
|
||
|
return errors.New("proxy: failed to read connect reply from SOCKS5 proxy at " + s.addr + ": " + err.Error())
|
||
|
}
|
||
|
|
||
|
failure := "unknown error"
|
||
|
if int(buf[1]) < len(proxy_socks5Errors) {
|
||
|
failure = proxy_socks5Errors[buf[1]]
|
||
|
}
|
||
|
|
||
|
if len(failure) > 0 {
|
||
|
return errors.New("proxy: SOCKS5 proxy at " + s.addr + " failed to connect: " + failure)
|
||
|
}
|
||
|
|
||
|
bytesToDiscard := 0
|
||
|
switch buf[3] {
|
||
|
case proxy_socks5IP4:
|
||
|
bytesToDiscard = net.IPv4len
|
||
|
case proxy_socks5IP6:
|
||
|
bytesToDiscard = net.IPv6len
|
||
|
case proxy_socks5Domain:
|
||
|
_, err := io.ReadFull(conn, buf[:1])
|
||
|
if err != nil {
|
||
|
return errors.New("proxy: failed to read domain length from SOCKS5 proxy at " + s.addr + ": " + err.Error())
|
||
|
}
|
||
|
bytesToDiscard = int(buf[0])
|
||
|
default:
|
||
|
return errors.New("proxy: got unknown address type " + strconv.Itoa(int(buf[3])) + " from SOCKS5 proxy at " + s.addr)
|
||
|
}
|
||
|
|
||
|
if cap(buf) < bytesToDiscard {
|
||
|
buf = make([]byte, bytesToDiscard)
|
||
|
} else {
|
||
|
buf = buf[:bytesToDiscard]
|
||
|
}
|
||
|
if _, err := io.ReadFull(conn, buf); err != nil {
|
||
|
return errors.New("proxy: failed to read address from SOCKS5 proxy at " + s.addr + ": " + err.Error())
|
||
|
}
|
||
|
|
||
|
// Also need to discard the port number
|
||
|
if _, err := io.ReadFull(conn, buf[:2]); err != nil {
|
||
|
return errors.New("proxy: failed to read port from SOCKS5 proxy at " + s.addr + ": " + err.Error())
|
||
|
}
|
||
|
|
||
|
return nil
|
||
|
}
|