diff options
-rw-r--r-- | obfs4.go | 15 | ||||
-rw-r--r-- | obfs4proxy/obfs4proxy.go | 29 | ||||
-rw-r--r-- | obfs4proxy/proxy_extras.go | 51 |
3 files changed, 83 insertions, 12 deletions
@@ -533,10 +533,21 @@ func (c *Obfs4Conn) SetWriteDeadline(t time.Time) error { return syscall.ENOTSUP } +// DialFn is a function pointer to a dial routine that matches the +// net.Dialer.Dial routine. +type DialFn func(string, string) (net.Conn, error) + // DialObfs4 connects to the remote address on the network, and handshakes with // the peer's obfs4 Node ID and Identity Public Key. nodeID and publicKey are // expected as strings containing the Base64 encoded values. func DialObfs4(network, address, nodeID, publicKey string, iatObfuscation bool) (*Obfs4Conn, error) { + + return DialObfs4DialFn(net.Dial, network, address, nodeID, publicKey, iatObfuscation) +} + +// DialObfs4DialFn connects to the remote address on the network via DialFn, +// and handshakes with the peers' obfs4 Node ID and Identity Public Key. +func DialObfs4DialFn(dialFn DialFn, network, address, nodeID, publicKey string, iatObfuscation bool) (*Obfs4Conn, error) { // Decode the node_id/public_key. pub, err := ntor.PublicKeyFromBase64(publicKey) if err != nil { @@ -553,13 +564,13 @@ func DialObfs4(network, address, nodeID, publicKey string, iatObfuscation bool) return nil, err } - // Connect to the peer. + // Generate the Obfs4Conn. c := new(Obfs4Conn) c.lenProbDist = newWDist(seed, 0, framing.MaximumSegmentLength) if iatObfuscation { c.iatProbDist = newWDist(seed, 0, maxIatDelay) } - c.conn, err = net.Dial(network, address) + c.conn, err = dialFn(network, address) if err != nil { return nil, err } diff --git a/obfs4proxy/obfs4proxy.go b/obfs4proxy/obfs4proxy.go index 2e8a011..c00ee88 100644 --- a/obfs4proxy/obfs4proxy.go +++ b/obfs4proxy/obfs4proxy.go @@ -53,6 +53,7 @@ import ( "io/ioutil" "log" "net" + "net/url" "os" "os/signal" "path" @@ -222,7 +223,7 @@ func serverSetup() (launched bool) { return } -func clientHandler(conn *pt.SocksConn) error { +func clientHandler(conn *pt.SocksConn, proxyURI *url.URL) error { defer conn.Close() var addr string @@ -254,8 +255,13 @@ func clientHandler(conn *pt.SocksConn) error { }() defer logAndRecover(nil) - remote, err := obfs4.DialObfs4("tcp", conn.Req.Target, nodeID, publicKey, - iatObfuscation) + dialFn, err := getProxyDialer(proxyURI) + if err != nil { + log.Printf("[ERROR] client: failed to get proxy dialer: %s", err) + conn.Reject() + return err + } + remote, err := obfs4.DialObfs4DialFn(dialFn, "tcp", conn.Req.Target, nodeID, publicKey, iatObfuscation) if err != nil { log.Printf("[ERROR] client: %p: Handshake failed: %s", remote, err) conn.Reject() @@ -272,7 +278,7 @@ func clientHandler(conn *pt.SocksConn) error { return nil } -func clientAcceptLoop(ln *pt.SocksListener) error { +func clientAcceptLoop(ln *pt.SocksListener, proxyURI *url.URL) error { defer ln.Close() for { conn, err := ln.AcceptSocks() @@ -282,7 +288,7 @@ func clientAcceptLoop(ln *pt.SocksListener) error { } continue } - go clientHandler(conn) + go clientHandler(conn, proxyURI) } } @@ -296,17 +302,20 @@ func clientSetup() (launched bool) { ptClientInfo, err := pt.ClientSetup([]string{obfs4Method}) if err != nil { log.Fatal(err) - return } ptClientProxy, err := ptGetProxy() if err != nil { log.Fatal(err) - return } if ptClientProxy != nil { - // XXX: Remove this once done. - ptProxyError("proxy are not supported yet") + // XXX: Limit this to SOCKS5 for now. + if ptClientProxy.Scheme != "socks5" { + ptProxyError(fmt.Sprintf("proxy scheme not supported: %s", + ptClientProxy.Scheme)) + return + } + ptProxyDone() } for _, methodName := range ptClientInfo.MethodNames { @@ -317,7 +326,7 @@ func clientSetup() (launched bool) { pt.CmethodError(methodName, err.Error()) break } - go clientAcceptLoop(ln) + go clientAcceptLoop(ln, ptClientProxy) pt.Cmethod(methodName, ln.Version(), ln.Addr()) ptListeners = append(ptListeners, ln) launched = true diff --git a/obfs4proxy/proxy_extras.go b/obfs4proxy/proxy_extras.go new file mode 100644 index 0000000..27b638b --- /dev/null +++ b/obfs4proxy/proxy_extras.go @@ -0,0 +1,51 @@ +/* + * Copyright (c) 2014, Yawning Angel <yawning at torproject dot org> + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions are met: + * + * * Redistributions of source code must retain the above copyright notice, + * this list of conditions and the following disclaimer. + * + * * Redistributions in binary form must reproduce the above copyright notice, + * this list of conditions and the following disclaimer in the documentation + * and/or other materials provided with the distribution. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" + * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE + * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE + * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE + * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR + * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF + * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS + * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN + * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + * POSSIBILITY OF SUCH DAMAGE. + */ + +package main + +import ( + "net/url" + + "code.google.com/p/go.net/proxy" + + "github.com/yawning/obfs4" +) + +// getProxyDialer is a trival wrapper around the go.net/proxy package to avoid +// having it as a dependency for anything else. +func getProxyDialer(uri *url.URL) (obfs4.DialFn, error) { + if uri == nil { + return proxy.Direct.Dial, nil + } + + dialer, err := proxy.FromURL(uri, proxy.Direct) + if err != nil { + return nil, err + } + + return dialer.Dial, nil +} |