jdk/src/share/classes/java/net/SocksSocketImpl.java
author chegar
Tue, 12 May 2009 16:32:34 +0100
changeset 3450 2f08a8bb9b83
parent 715 f16baef3a20e
child 3464 4ef40ee318ef
permissions -rw-r--r--
6801071: Remote sites can compromise user privacy and possibly hijack web sessions Reviewed-by: jccollet, hawtin
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
     1
/*
715
f16baef3a20e 6719955: Update copyright year
xdono
parents: 51
diff changeset
     2
 * Copyright 2000-2008 Sun Microsystems, Inc.  All Rights Reserved.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
     3
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
90ce3da70b43 Initial load
duke
parents:
diff changeset
     4
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
     5
 * This code is free software; you can redistribute it and/or modify it
90ce3da70b43 Initial load
duke
parents:
diff changeset
     6
 * under the terms of the GNU General Public License version 2 only, as
90ce3da70b43 Initial load
duke
parents:
diff changeset
     7
 * published by the Free Software Foundation.  Sun designates this
90ce3da70b43 Initial load
duke
parents:
diff changeset
     8
 * particular file as subject to the "Classpath" exception as provided
90ce3da70b43 Initial load
duke
parents:
diff changeset
     9
 * by Sun in the LICENSE file that accompanied this code.
90ce3da70b43 Initial load
duke
parents:
diff changeset
    10
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
    11
 * This code is distributed in the hope that it will be useful, but WITHOUT
90ce3da70b43 Initial load
duke
parents:
diff changeset
    12
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
90ce3da70b43 Initial load
duke
parents:
diff changeset
    13
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
90ce3da70b43 Initial load
duke
parents:
diff changeset
    14
 * version 2 for more details (a copy is included in the LICENSE file that
90ce3da70b43 Initial load
duke
parents:
diff changeset
    15
 * accompanied this code).
90ce3da70b43 Initial load
duke
parents:
diff changeset
    16
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
    17
 * You should have received a copy of the GNU General Public License version
90ce3da70b43 Initial load
duke
parents:
diff changeset
    18
 * 2 along with this work; if not, write to the Free Software Foundation,
90ce3da70b43 Initial load
duke
parents:
diff changeset
    19
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
90ce3da70b43 Initial load
duke
parents:
diff changeset
    20
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
    21
 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
90ce3da70b43 Initial load
duke
parents:
diff changeset
    22
 * CA 95054 USA or visit www.sun.com if you need additional information or
90ce3da70b43 Initial load
duke
parents:
diff changeset
    23
 * have any questions.
90ce3da70b43 Initial load
duke
parents:
diff changeset
    24
 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
    25
package java.net;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    26
import java.io.IOException;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    27
import java.io.InputStream;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    28
import java.io.OutputStream;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    29
import java.io.BufferedOutputStream;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    30
import java.security.AccessController;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    31
import java.security.PrivilegedExceptionAction;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    32
import java.util.prefs.Preferences;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    33
import sun.net.www.ParseUtil;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    34
/* import org.ietf.jgss.*; */
90ce3da70b43 Initial load
duke
parents:
diff changeset
    35
90ce3da70b43 Initial load
duke
parents:
diff changeset
    36
/**
90ce3da70b43 Initial load
duke
parents:
diff changeset
    37
 * SOCKS (V4 & V5) TCP socket implementation (RFC 1928).
90ce3da70b43 Initial load
duke
parents:
diff changeset
    38
 * This is a subclass of PlainSocketImpl.
90ce3da70b43 Initial load
duke
parents:
diff changeset
    39
 * Note this class should <b>NOT</b> be public.
90ce3da70b43 Initial load
duke
parents:
diff changeset
    40
 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
    41
90ce3da70b43 Initial load
duke
parents:
diff changeset
    42
class SocksSocketImpl extends PlainSocketImpl implements SocksConsts {
90ce3da70b43 Initial load
duke
parents:
diff changeset
    43
    private String server = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    44
    private int port = DEFAULT_PORT;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    45
    private InetSocketAddress external_address;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    46
    private boolean useV4 = false;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    47
    private Socket cmdsock = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    48
    private InputStream cmdIn = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    49
    private OutputStream cmdOut = null;
3450
2f08a8bb9b83 6801071: Remote sites can compromise user privacy and possibly hijack web sessions
chegar
parents: 715
diff changeset
    50
    /* true if the Proxy has been set programatically */
2f08a8bb9b83 6801071: Remote sites can compromise user privacy and possibly hijack web sessions
chegar
parents: 715
diff changeset
    51
    private boolean applicationSetProxy;  /* false */
2f08a8bb9b83 6801071: Remote sites can compromise user privacy and possibly hijack web sessions
chegar
parents: 715
diff changeset
    52
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    53
90ce3da70b43 Initial load
duke
parents:
diff changeset
    54
    SocksSocketImpl() {
90ce3da70b43 Initial load
duke
parents:
diff changeset
    55
        // Nothing needed
90ce3da70b43 Initial load
duke
parents:
diff changeset
    56
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
    57
90ce3da70b43 Initial load
duke
parents:
diff changeset
    58
    SocksSocketImpl(String server, int port) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
    59
        this.server = server;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    60
        this.port = (port == -1 ? DEFAULT_PORT : port);
90ce3da70b43 Initial load
duke
parents:
diff changeset
    61
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
    62
90ce3da70b43 Initial load
duke
parents:
diff changeset
    63
    SocksSocketImpl(Proxy proxy) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
    64
        SocketAddress a = proxy.address();
90ce3da70b43 Initial load
duke
parents:
diff changeset
    65
        if (a instanceof InetSocketAddress) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
    66
            InetSocketAddress ad = (InetSocketAddress) a;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    67
            // Use getHostString() to avoid reverse lookups
90ce3da70b43 Initial load
duke
parents:
diff changeset
    68
            server = ad.getHostString();
90ce3da70b43 Initial load
duke
parents:
diff changeset
    69
            port = ad.getPort();
3450
2f08a8bb9b83 6801071: Remote sites can compromise user privacy and possibly hijack web sessions
chegar
parents: 715
diff changeset
    70
            applicationSetProxy = true;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    71
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
    72
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
    73
90ce3da70b43 Initial load
duke
parents:
diff changeset
    74
    void setV4() {
90ce3da70b43 Initial load
duke
parents:
diff changeset
    75
        useV4 = true;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    76
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
    77
90ce3da70b43 Initial load
duke
parents:
diff changeset
    78
    private synchronized void privilegedConnect(final String host,
90ce3da70b43 Initial load
duke
parents:
diff changeset
    79
                                              final int port,
90ce3da70b43 Initial load
duke
parents:
diff changeset
    80
                                              final int timeout)
90ce3da70b43 Initial load
duke
parents:
diff changeset
    81
         throws IOException
90ce3da70b43 Initial load
duke
parents:
diff changeset
    82
    {
90ce3da70b43 Initial load
duke
parents:
diff changeset
    83
        try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
    84
            AccessController.doPrivileged(
51
6fe31bc95bbc 6600143: Remove another 450 unnecessary casts
martin
parents: 2
diff changeset
    85
                new java.security.PrivilegedExceptionAction<Void>() {
6fe31bc95bbc 6600143: Remove another 450 unnecessary casts
martin
parents: 2
diff changeset
    86
                    public Void run() throws IOException {
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    87
                              superConnectServer(host, port, timeout);
90ce3da70b43 Initial load
duke
parents:
diff changeset
    88
                              cmdIn = getInputStream();
90ce3da70b43 Initial load
duke
parents:
diff changeset
    89
                              cmdOut = getOutputStream();
90ce3da70b43 Initial load
duke
parents:
diff changeset
    90
                              return null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    91
                          }
90ce3da70b43 Initial load
duke
parents:
diff changeset
    92
                      });
90ce3da70b43 Initial load
duke
parents:
diff changeset
    93
        } catch (java.security.PrivilegedActionException pae) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
    94
            throw (IOException) pae.getException();
90ce3da70b43 Initial load
duke
parents:
diff changeset
    95
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
    96
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
    97
90ce3da70b43 Initial load
duke
parents:
diff changeset
    98
    private void superConnectServer(String host, int port,
90ce3da70b43 Initial load
duke
parents:
diff changeset
    99
                                    int timeout) throws IOException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   100
        super.connect(new InetSocketAddress(host, port), timeout);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   101
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   102
90ce3da70b43 Initial load
duke
parents:
diff changeset
   103
    private int readSocksReply(InputStream in, byte[] data) throws IOException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   104
        int len = data.length;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   105
        int received = 0;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   106
        for (int attempts = 0; received < len && attempts < 3; attempts++) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   107
            int count = in.read(data, received, len - received);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   108
            if (count < 0)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   109
                throw new SocketException("Malformed reply from SOCKS server");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   110
            received += count;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   111
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   112
        return received;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   113
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   114
90ce3da70b43 Initial load
duke
parents:
diff changeset
   115
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   116
     * Provides the authentication machanism required by the proxy.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   117
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   118
    private boolean authenticate(byte method, InputStream in,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   119
                                 BufferedOutputStream out) throws IOException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   120
        byte[] data = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   121
        int i;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   122
        // No Authentication required. We're done then!
90ce3da70b43 Initial load
duke
parents:
diff changeset
   123
        if (method == NO_AUTH)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   124
            return true;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   125
        /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   126
         * User/Password authentication. Try, in that order :
90ce3da70b43 Initial load
duke
parents:
diff changeset
   127
         * - The application provided Authenticator, if any
90ce3da70b43 Initial load
duke
parents:
diff changeset
   128
         * - The user preferences java.net.socks.username &
90ce3da70b43 Initial load
duke
parents:
diff changeset
   129
         *   java.net.socks.password
90ce3da70b43 Initial load
duke
parents:
diff changeset
   130
         * - the user.name & no password (backward compatibility behavior).
90ce3da70b43 Initial load
duke
parents:
diff changeset
   131
         */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   132
        if (method == USER_PASSW) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   133
            String userName;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   134
            String password = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   135
            final InetAddress addr = InetAddress.getByName(server);
51
6fe31bc95bbc 6600143: Remove another 450 unnecessary casts
martin
parents: 2
diff changeset
   136
            PasswordAuthentication pw =
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   137
                java.security.AccessController.doPrivileged(
51
6fe31bc95bbc 6600143: Remove another 450 unnecessary casts
martin
parents: 2
diff changeset
   138
                    new java.security.PrivilegedAction<PasswordAuthentication>() {
6fe31bc95bbc 6600143: Remove another 450 unnecessary casts
martin
parents: 2
diff changeset
   139
                        public PasswordAuthentication run() {
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   140
                                return Authenticator.requestPasswordAuthentication(
90ce3da70b43 Initial load
duke
parents:
diff changeset
   141
                                       server, addr, port, "SOCKS5", "SOCKS authentication", null);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   142
                            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   143
                        });
90ce3da70b43 Initial load
duke
parents:
diff changeset
   144
            if (pw != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   145
                userName = pw.getUserName();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   146
                password = new String(pw.getPassword());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   147
            } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   148
                final Preferences prefs = Preferences.userRoot().node("/java/net/socks");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   149
                try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   150
                    userName = AccessController.doPrivileged(
90ce3da70b43 Initial load
duke
parents:
diff changeset
   151
                        new java.security.PrivilegedExceptionAction<String>() {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   152
                            public String run() throws IOException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   153
                                return prefs.get("username", null);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   154
                            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   155
                        });
90ce3da70b43 Initial load
duke
parents:
diff changeset
   156
                } catch (java.security.PrivilegedActionException pae) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   157
                    throw (IOException) pae.getException();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   158
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   159
90ce3da70b43 Initial load
duke
parents:
diff changeset
   160
                if (userName != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   161
                    try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   162
                        password = AccessController.doPrivileged(
90ce3da70b43 Initial load
duke
parents:
diff changeset
   163
                            new java.security.PrivilegedExceptionAction<String>() {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   164
                                public String run() throws IOException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   165
                                    return prefs.get("password", null);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   166
                                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   167
                            });
90ce3da70b43 Initial load
duke
parents:
diff changeset
   168
                    } catch (java.security.PrivilegedActionException pae) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   169
                        throw (IOException) pae.getException();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   170
                    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   171
                } else {
3450
2f08a8bb9b83 6801071: Remote sites can compromise user privacy and possibly hijack web sessions
chegar
parents: 715
diff changeset
   172
                    userName = getUserName();
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   173
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   174
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   175
            if (userName == null)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   176
                return false;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   177
            out.write(1);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   178
            out.write(userName.length());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   179
            try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   180
                out.write(userName.getBytes("ISO-8859-1"));
90ce3da70b43 Initial load
duke
parents:
diff changeset
   181
            } catch (java.io.UnsupportedEncodingException uee) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   182
                assert false;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   183
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   184
            if (password != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   185
                out.write(password.length());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   186
                try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   187
                    out.write(password.getBytes("ISO-8859-1"));
90ce3da70b43 Initial load
duke
parents:
diff changeset
   188
                } catch (java.io.UnsupportedEncodingException uee) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   189
                    assert false;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   190
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   191
            } else
90ce3da70b43 Initial load
duke
parents:
diff changeset
   192
                out.write(0);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   193
            out.flush();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   194
            data = new byte[2];
90ce3da70b43 Initial load
duke
parents:
diff changeset
   195
            i = readSocksReply(in, data);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   196
            if (i != 2 || data[1] != 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   197
                /* RFC 1929 specifies that the connection MUST be closed if
90ce3da70b43 Initial load
duke
parents:
diff changeset
   198
                   authentication fails */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   199
                out.close();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   200
                in.close();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   201
                return false;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   202
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   203
            /* Authentication succeeded */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   204
            return true;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   205
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   206
        /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   207
         * GSSAPI authentication mechanism.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   208
         * Unfortunately the RFC seems out of sync with the Reference
90ce3da70b43 Initial load
duke
parents:
diff changeset
   209
         * implementation. I'll leave this in for future completion.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   210
         */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   211
//      if (method == GSSAPI) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   212
//          try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   213
//              GSSManager manager = GSSManager.getInstance();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   214
//              GSSName name = manager.createName("SERVICE:socks@"+server,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   215
//                                                   null);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   216
//              GSSContext context = manager.createContext(name, null, null,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   217
//                                                         GSSContext.DEFAULT_LIFETIME);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   218
//              context.requestMutualAuth(true);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   219
//              context.requestReplayDet(true);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   220
//              context.requestSequenceDet(true);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   221
//              context.requestCredDeleg(true);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   222
//              byte []inToken = new byte[0];
90ce3da70b43 Initial load
duke
parents:
diff changeset
   223
//              while (!context.isEstablished()) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   224
//                  byte[] outToken
90ce3da70b43 Initial load
duke
parents:
diff changeset
   225
//                      = context.initSecContext(inToken, 0, inToken.length);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   226
//                  // send the output token if generated
90ce3da70b43 Initial load
duke
parents:
diff changeset
   227
//                  if (outToken != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   228
//                      out.write(1);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   229
//                      out.write(1);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   230
//                      out.writeShort(outToken.length);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   231
//                      out.write(outToken);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   232
//                      out.flush();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   233
//                      data = new byte[2];
90ce3da70b43 Initial load
duke
parents:
diff changeset
   234
//                      i = readSocksReply(in, data);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   235
//                      if (i != 2 || data[1] == 0xff) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   236
//                          in.close();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   237
//                          out.close();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   238
//                          return false;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   239
//                      }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   240
//                      i = readSocksReply(in, data);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   241
//                      int len = 0;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   242
//                      len = ((int)data[0] & 0xff) << 8;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   243
//                      len += data[1];
90ce3da70b43 Initial load
duke
parents:
diff changeset
   244
//                      data = new byte[len];
90ce3da70b43 Initial load
duke
parents:
diff changeset
   245
//                      i = readSocksReply(in, data);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   246
//                      if (i == len)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   247
//                          return true;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   248
//                      in.close();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   249
//                      out.close();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   250
//                  }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   251
//              }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   252
//          } catch (GSSException e) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   253
//              /* RFC 1961 states that if Context initialisation fails the connection
90ce3da70b43 Initial load
duke
parents:
diff changeset
   254
//                 MUST be closed */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   255
//              e.printStackTrace();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   256
//              in.close();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   257
//              out.close();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   258
//          }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   259
//      }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   260
        return false;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   261
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   262
90ce3da70b43 Initial load
duke
parents:
diff changeset
   263
    private void connectV4(InputStream in, OutputStream out,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   264
                           InetSocketAddress endpoint) throws IOException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   265
        if (!(endpoint.getAddress() instanceof Inet4Address)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   266
            throw new SocketException("SOCKS V4 requires IPv4 only addresses");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   267
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   268
        out.write(PROTO_VERS4);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   269
        out.write(CONNECT);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   270
        out.write((endpoint.getPort() >> 8) & 0xff);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   271
        out.write((endpoint.getPort() >> 0) & 0xff);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   272
        out.write(endpoint.getAddress().getAddress());
3450
2f08a8bb9b83 6801071: Remote sites can compromise user privacy and possibly hijack web sessions
chegar
parents: 715
diff changeset
   273
        String userName = getUserName();
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   274
        try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   275
            out.write(userName.getBytes("ISO-8859-1"));
90ce3da70b43 Initial load
duke
parents:
diff changeset
   276
        } catch (java.io.UnsupportedEncodingException uee) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   277
            assert false;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   278
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   279
        out.write(0);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   280
        out.flush();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   281
        byte[] data = new byte[8];
90ce3da70b43 Initial load
duke
parents:
diff changeset
   282
        int n = readSocksReply(in, data);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   283
        if (n != 8)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   284
            throw new SocketException("Reply from SOCKS server has bad length: " + n);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   285
        if (data[0] != 0 && data[0] != 4)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   286
            throw new SocketException("Reply from SOCKS server has bad version");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   287
        SocketException ex = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   288
        switch (data[1]) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   289
        case 90:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   290
            // Success!
90ce3da70b43 Initial load
duke
parents:
diff changeset
   291
            external_address = endpoint;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   292
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   293
        case 91:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   294
            ex = new SocketException("SOCKS request rejected");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   295
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   296
        case 92:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   297
            ex = new SocketException("SOCKS server couldn't reach destination");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   298
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   299
        case 93:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   300
            ex = new SocketException("SOCKS authentication failed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   301
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   302
        default:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   303
            ex = new SocketException("Reply from SOCKS server contains bad status");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   304
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   305
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   306
        if (ex != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   307
            in.close();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   308
            out.close();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   309
            throw ex;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   310
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   311
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   312
90ce3da70b43 Initial load
duke
parents:
diff changeset
   313
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   314
     * Connects the Socks Socket to the specified endpoint. It will first
90ce3da70b43 Initial load
duke
parents:
diff changeset
   315
     * connect to the SOCKS proxy and negotiate the access. If the proxy
90ce3da70b43 Initial load
duke
parents:
diff changeset
   316
     * grants the connections, then the connect is successful and all
90ce3da70b43 Initial load
duke
parents:
diff changeset
   317
     * further traffic will go to the "real" endpoint.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   318
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   319
     * @param   endpoint        the <code>SocketAddress</code> to connect to.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   320
     * @param   timeout         the timeout value in milliseconds
90ce3da70b43 Initial load
duke
parents:
diff changeset
   321
     * @throws  IOException     if the connection can't be established.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   322
     * @throws  SecurityException if there is a security manager and it
90ce3da70b43 Initial load
duke
parents:
diff changeset
   323
     *                          doesn't allow the connection
90ce3da70b43 Initial load
duke
parents:
diff changeset
   324
     * @throws  IllegalArgumentException if endpoint is null or a
90ce3da70b43 Initial load
duke
parents:
diff changeset
   325
     *          SocketAddress subclass not supported by this socket
90ce3da70b43 Initial load
duke
parents:
diff changeset
   326
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   327
    protected void connect(SocketAddress endpoint, int timeout) throws IOException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   328
        SecurityManager security = System.getSecurityManager();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   329
        if (endpoint == null || !(endpoint instanceof InetSocketAddress))
90ce3da70b43 Initial load
duke
parents:
diff changeset
   330
            throw new IllegalArgumentException("Unsupported address type");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   331
        InetSocketAddress epoint = (InetSocketAddress) endpoint;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   332
        if (security != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   333
            if (epoint.isUnresolved())
90ce3da70b43 Initial load
duke
parents:
diff changeset
   334
                security.checkConnect(epoint.getHostName(),
90ce3da70b43 Initial load
duke
parents:
diff changeset
   335
                                      epoint.getPort());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   336
            else
90ce3da70b43 Initial load
duke
parents:
diff changeset
   337
                security.checkConnect(epoint.getAddress().getHostAddress(),
90ce3da70b43 Initial load
duke
parents:
diff changeset
   338
                                      epoint.getPort());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   339
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   340
        if (server == null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   341
            // This is the general case
90ce3da70b43 Initial load
duke
parents:
diff changeset
   342
            // server is not null only when the socket was created with a
90ce3da70b43 Initial load
duke
parents:
diff changeset
   343
            // specified proxy in which case it does bypass the ProxySelector
51
6fe31bc95bbc 6600143: Remove another 450 unnecessary casts
martin
parents: 2
diff changeset
   344
            ProxySelector sel = java.security.AccessController.doPrivileged(
6fe31bc95bbc 6600143: Remove another 450 unnecessary casts
martin
parents: 2
diff changeset
   345
                new java.security.PrivilegedAction<ProxySelector>() {
6fe31bc95bbc 6600143: Remove another 450 unnecessary casts
martin
parents: 2
diff changeset
   346
                    public ProxySelector run() {
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   347
                            return ProxySelector.getDefault();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   348
                        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   349
                    });
90ce3da70b43 Initial load
duke
parents:
diff changeset
   350
            if (sel == null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   351
                /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   352
                 * No default proxySelector --> direct connection
90ce3da70b43 Initial load
duke
parents:
diff changeset
   353
                 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   354
                super.connect(epoint, timeout);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   355
                return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   356
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   357
            URI uri = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   358
            // Use getHostString() to avoid reverse lookups
90ce3da70b43 Initial load
duke
parents:
diff changeset
   359
            String host = epoint.getHostString();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   360
            // IPv6 litteral?
90ce3da70b43 Initial load
duke
parents:
diff changeset
   361
            if (epoint.getAddress() instanceof Inet6Address &&
90ce3da70b43 Initial load
duke
parents:
diff changeset
   362
                (!host.startsWith("[")) && (host.indexOf(":") >= 0)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   363
                host = "[" + host + "]";
90ce3da70b43 Initial load
duke
parents:
diff changeset
   364
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   365
            try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   366
                uri = new URI("socket://" + ParseUtil.encodePath(host) + ":"+ epoint.getPort());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   367
            } catch (URISyntaxException e) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   368
                // This shouldn't happen
90ce3da70b43 Initial load
duke
parents:
diff changeset
   369
                assert false : e;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   370
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   371
            Proxy p = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   372
            IOException savedExc = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   373
            java.util.Iterator<Proxy> iProxy = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   374
            iProxy = sel.select(uri).iterator();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   375
            if (iProxy == null || !(iProxy.hasNext())) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   376
                super.connect(epoint, timeout);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   377
                return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   378
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   379
            while (iProxy.hasNext()) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   380
                p = iProxy.next();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   381
                if (p == null || p == Proxy.NO_PROXY) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   382
                    super.connect(epoint, timeout);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   383
                    return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   384
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   385
                if (p.type() != Proxy.Type.SOCKS)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   386
                    throw new SocketException("Unknown proxy type : " + p.type());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   387
                if (!(p.address() instanceof InetSocketAddress))
90ce3da70b43 Initial load
duke
parents:
diff changeset
   388
                    throw new SocketException("Unknow address type for proxy: " + p);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   389
                // Use getHostString() to avoid reverse lookups
90ce3da70b43 Initial load
duke
parents:
diff changeset
   390
                server = ((InetSocketAddress) p.address()).getHostString();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   391
                port = ((InetSocketAddress) p.address()).getPort();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   392
90ce3da70b43 Initial load
duke
parents:
diff changeset
   393
                // Connects to the SOCKS server
90ce3da70b43 Initial load
duke
parents:
diff changeset
   394
                try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   395
                    privilegedConnect(server, port, timeout);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   396
                    // Worked, let's get outta here
90ce3da70b43 Initial load
duke
parents:
diff changeset
   397
                    break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   398
                } catch (IOException e) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   399
                    // Ooops, let's notify the ProxySelector
90ce3da70b43 Initial load
duke
parents:
diff changeset
   400
                    sel.connectFailed(uri,p.address(),e);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   401
                    server = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   402
                    port = -1;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   403
                    savedExc = e;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   404
                    // Will continue the while loop and try the next proxy
90ce3da70b43 Initial load
duke
parents:
diff changeset
   405
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   406
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   407
90ce3da70b43 Initial load
duke
parents:
diff changeset
   408
            /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   409
             * If server is still null at this point, none of the proxy
90ce3da70b43 Initial load
duke
parents:
diff changeset
   410
             * worked
90ce3da70b43 Initial load
duke
parents:
diff changeset
   411
             */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   412
            if (server == null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   413
                throw new SocketException("Can't connect to SOCKS proxy:"
90ce3da70b43 Initial load
duke
parents:
diff changeset
   414
                                          + savedExc.getMessage());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   415
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   416
        } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   417
            // Connects to the SOCKS server
90ce3da70b43 Initial load
duke
parents:
diff changeset
   418
            try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   419
                privilegedConnect(server, port, timeout);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   420
            } catch (IOException e) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   421
                throw new SocketException(e.getMessage());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   422
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   423
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   424
90ce3da70b43 Initial load
duke
parents:
diff changeset
   425
        // cmdIn & cmdOut were intialized during the privilegedConnect() call
90ce3da70b43 Initial load
duke
parents:
diff changeset
   426
        BufferedOutputStream out = new BufferedOutputStream(cmdOut, 512);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   427
        InputStream in = cmdIn;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   428
90ce3da70b43 Initial load
duke
parents:
diff changeset
   429
        if (useV4) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   430
            // SOCKS Protocol version 4 doesn't know how to deal with
90ce3da70b43 Initial load
duke
parents:
diff changeset
   431
            // DOMAIN type of addresses (unresolved addresses here)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   432
            if (epoint.isUnresolved())
90ce3da70b43 Initial load
duke
parents:
diff changeset
   433
                throw new UnknownHostException(epoint.toString());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   434
            connectV4(in, out, epoint);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   435
            return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   436
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   437
90ce3da70b43 Initial load
duke
parents:
diff changeset
   438
        // This is SOCKS V5
90ce3da70b43 Initial load
duke
parents:
diff changeset
   439
        out.write(PROTO_VERS);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   440
        out.write(2);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   441
        out.write(NO_AUTH);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   442
        out.write(USER_PASSW);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   443
        out.flush();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   444
        byte[] data = new byte[2];
90ce3da70b43 Initial load
duke
parents:
diff changeset
   445
        int i = readSocksReply(in, data);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   446
        if (i != 2 || ((int)data[0]) != PROTO_VERS) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   447
            // Maybe it's not a V5 sever after all
90ce3da70b43 Initial load
duke
parents:
diff changeset
   448
            // Let's try V4 before we give up
90ce3da70b43 Initial load
duke
parents:
diff changeset
   449
            // SOCKS Protocol version 4 doesn't know how to deal with
90ce3da70b43 Initial load
duke
parents:
diff changeset
   450
            // DOMAIN type of addresses (unresolved addresses here)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   451
            if (epoint.isUnresolved())
90ce3da70b43 Initial load
duke
parents:
diff changeset
   452
                throw new UnknownHostException(epoint.toString());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   453
            connectV4(in, out, epoint);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   454
            return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   455
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   456
        if (((int)data[1]) == NO_METHODS)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   457
            throw new SocketException("SOCKS : No acceptable methods");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   458
        if (!authenticate(data[1], in, out)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   459
            throw new SocketException("SOCKS : authentication failed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   460
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   461
        out.write(PROTO_VERS);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   462
        out.write(CONNECT);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   463
        out.write(0);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   464
        /* Test for IPV4/IPV6/Unresolved */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   465
        if (epoint.isUnresolved()) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   466
            out.write(DOMAIN_NAME);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   467
            out.write(epoint.getHostName().length());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   468
            try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   469
                out.write(epoint.getHostName().getBytes("ISO-8859-1"));
90ce3da70b43 Initial load
duke
parents:
diff changeset
   470
            } catch (java.io.UnsupportedEncodingException uee) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   471
                assert false;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   472
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   473
            out.write((epoint.getPort() >> 8) & 0xff);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   474
            out.write((epoint.getPort() >> 0) & 0xff);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   475
        } else if (epoint.getAddress() instanceof Inet6Address) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   476
            out.write(IPV6);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   477
            out.write(epoint.getAddress().getAddress());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   478
            out.write((epoint.getPort() >> 8) & 0xff);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   479
            out.write((epoint.getPort() >> 0) & 0xff);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   480
        } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   481
            out.write(IPV4);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   482
            out.write(epoint.getAddress().getAddress());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   483
            out.write((epoint.getPort() >> 8) & 0xff);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   484
            out.write((epoint.getPort() >> 0) & 0xff);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   485
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   486
        out.flush();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   487
        data = new byte[4];
90ce3da70b43 Initial load
duke
parents:
diff changeset
   488
        i = readSocksReply(in, data);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   489
        if (i != 4)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   490
            throw new SocketException("Reply from SOCKS server has bad length");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   491
        SocketException ex = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   492
        int nport, len;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   493
        byte[] addr;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   494
        switch (data[1]) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   495
        case REQUEST_OK:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   496
            // success!
90ce3da70b43 Initial load
duke
parents:
diff changeset
   497
            switch(data[3]) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   498
            case IPV4:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   499
                addr = new byte[4];
90ce3da70b43 Initial load
duke
parents:
diff changeset
   500
                i = readSocksReply(in, addr);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   501
                if (i != 4)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   502
                    throw new SocketException("Reply from SOCKS server badly formatted");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   503
                data = new byte[2];
90ce3da70b43 Initial load
duke
parents:
diff changeset
   504
                i = readSocksReply(in, data);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   505
                if (i != 2)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   506
                    throw new SocketException("Reply from SOCKS server badly formatted");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   507
                nport = ((int)data[0] & 0xff) << 8;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   508
                nport += ((int)data[1] & 0xff);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   509
                break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   510
            case DOMAIN_NAME:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   511
                len = data[1];
90ce3da70b43 Initial load
duke
parents:
diff changeset
   512
                byte[] host = new byte[len];
90ce3da70b43 Initial load
duke
parents:
diff changeset
   513
                i = readSocksReply(in, host);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   514
                if (i != len)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   515
                    throw new SocketException("Reply from SOCKS server badly formatted");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   516
                data = new byte[2];
90ce3da70b43 Initial load
duke
parents:
diff changeset
   517
                i = readSocksReply(in, data);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   518
                if (i != 2)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   519
                    throw new SocketException("Reply from SOCKS server badly formatted");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   520
                nport = ((int)data[0] & 0xff) << 8;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   521
                nport += ((int)data[1] & 0xff);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   522
                break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   523
            case IPV6:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   524
                len = data[1];
90ce3da70b43 Initial load
duke
parents:
diff changeset
   525
                addr = new byte[len];
90ce3da70b43 Initial load
duke
parents:
diff changeset
   526
                i = readSocksReply(in, addr);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   527
                if (i != len)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   528
                    throw new SocketException("Reply from SOCKS server badly formatted");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   529
                data = new byte[2];
90ce3da70b43 Initial load
duke
parents:
diff changeset
   530
                i = readSocksReply(in, data);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   531
                if (i != 2)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   532
                    throw new SocketException("Reply from SOCKS server badly formatted");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   533
                nport = ((int)data[0] & 0xff) << 8;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   534
                nport += ((int)data[1] & 0xff);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   535
                break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   536
            default:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   537
                ex = new SocketException("Reply from SOCKS server contains wrong code");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   538
                break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   539
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   540
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   541
        case GENERAL_FAILURE:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   542
            ex = new SocketException("SOCKS server general failure");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   543
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   544
        case NOT_ALLOWED:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   545
            ex = new SocketException("SOCKS: Connection not allowed by ruleset");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   546
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   547
        case NET_UNREACHABLE:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   548
            ex = new SocketException("SOCKS: Network unreachable");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   549
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   550
        case HOST_UNREACHABLE:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   551
            ex = new SocketException("SOCKS: Host unreachable");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   552
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   553
        case CONN_REFUSED:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   554
            ex = new SocketException("SOCKS: Connection refused");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   555
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   556
        case TTL_EXPIRED:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   557
            ex =  new SocketException("SOCKS: TTL expired");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   558
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   559
        case CMD_NOT_SUPPORTED:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   560
            ex = new SocketException("SOCKS: Command not supported");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   561
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   562
        case ADDR_TYPE_NOT_SUP:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   563
            ex = new SocketException("SOCKS: address type not supported");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   564
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   565
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   566
        if (ex != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   567
            in.close();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   568
            out.close();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   569
            throw ex;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   570
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   571
        external_address = epoint;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   572
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   573
90ce3da70b43 Initial load
duke
parents:
diff changeset
   574
    private void bindV4(InputStream in, OutputStream out,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   575
                        InetAddress baddr,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   576
                        int lport) throws IOException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   577
        if (!(baddr instanceof Inet4Address)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   578
            throw new SocketException("SOCKS V4 requires IPv4 only addresses");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   579
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   580
        super.bind(baddr, lport);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   581
        byte[] addr1 = baddr.getAddress();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   582
        /* Test for AnyLocal */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   583
        InetAddress naddr = baddr;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   584
        if (naddr.isAnyLocalAddress()) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   585
            naddr = cmdsock.getLocalAddress();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   586
            addr1 = naddr.getAddress();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   587
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   588
        out.write(PROTO_VERS4);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   589
        out.write(BIND);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   590
        out.write((super.getLocalPort() >> 8) & 0xff);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   591
        out.write((super.getLocalPort() >> 0) & 0xff);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   592
        out.write(addr1);
3450
2f08a8bb9b83 6801071: Remote sites can compromise user privacy and possibly hijack web sessions
chegar
parents: 715
diff changeset
   593
        String userName = getUserName();
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   594
        try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   595
            out.write(userName.getBytes("ISO-8859-1"));
90ce3da70b43 Initial load
duke
parents:
diff changeset
   596
        } catch (java.io.UnsupportedEncodingException uee) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   597
            assert false;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   598
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   599
        out.write(0);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   600
        out.flush();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   601
        byte[] data = new byte[8];
90ce3da70b43 Initial load
duke
parents:
diff changeset
   602
        int n = readSocksReply(in, data);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   603
        if (n != 8)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   604
            throw new SocketException("Reply from SOCKS server has bad length: " + n);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   605
        if (data[0] != 0 && data[0] != 4)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   606
            throw new SocketException("Reply from SOCKS server has bad version");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   607
        SocketException ex = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   608
        switch (data[1]) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   609
        case 90:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   610
            // Success!
90ce3da70b43 Initial load
duke
parents:
diff changeset
   611
            external_address = new InetSocketAddress(baddr, lport);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   612
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   613
        case 91:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   614
            ex = new SocketException("SOCKS request rejected");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   615
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   616
        case 92:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   617
            ex = new SocketException("SOCKS server couldn't reach destination");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   618
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   619
        case 93:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   620
            ex = new SocketException("SOCKS authentication failed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   621
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   622
        default:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   623
            ex = new SocketException("Reply from SOCKS server contains bad status");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   624
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   625
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   626
        if (ex != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   627
            in.close();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   628
            out.close();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   629
            throw ex;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   630
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   631
90ce3da70b43 Initial load
duke
parents:
diff changeset
   632
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   633
90ce3da70b43 Initial load
duke
parents:
diff changeset
   634
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   635
     * Sends the Bind request to the SOCKS proxy. In the SOCKS protocol, bind
90ce3da70b43 Initial load
duke
parents:
diff changeset
   636
     * means "accept incoming connection from", so the SocketAddress is the
90ce3da70b43 Initial load
duke
parents:
diff changeset
   637
     * the one of the host we do accept connection from.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   638
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   639
     * @param      addr   the Socket address of the remote host.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   640
     * @exception  IOException  if an I/O error occurs when binding this socket.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   641
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   642
    protected synchronized void socksBind(InetSocketAddress saddr) throws IOException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   643
        if (socket != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   644
            // this is a client socket, not a server socket, don't
90ce3da70b43 Initial load
duke
parents:
diff changeset
   645
            // call the SOCKS proxy for a bind!
90ce3da70b43 Initial load
duke
parents:
diff changeset
   646
            return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   647
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   648
90ce3da70b43 Initial load
duke
parents:
diff changeset
   649
        // Connects to the SOCKS server
90ce3da70b43 Initial load
duke
parents:
diff changeset
   650
90ce3da70b43 Initial load
duke
parents:
diff changeset
   651
        if (server == null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   652
            // This is the general case
90ce3da70b43 Initial load
duke
parents:
diff changeset
   653
            // server is not null only when the socket was created with a
90ce3da70b43 Initial load
duke
parents:
diff changeset
   654
            // specified proxy in which case it does bypass the ProxySelector
51
6fe31bc95bbc 6600143: Remove another 450 unnecessary casts
martin
parents: 2
diff changeset
   655
            ProxySelector sel = java.security.AccessController.doPrivileged(
6fe31bc95bbc 6600143: Remove another 450 unnecessary casts
martin
parents: 2
diff changeset
   656
                new java.security.PrivilegedAction<ProxySelector>() {
6fe31bc95bbc 6600143: Remove another 450 unnecessary casts
martin
parents: 2
diff changeset
   657
                    public ProxySelector run() {
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   658
                            return ProxySelector.getDefault();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   659
                        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   660
                    });
90ce3da70b43 Initial load
duke
parents:
diff changeset
   661
            if (sel == null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   662
                /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   663
                 * No default proxySelector --> direct connection
90ce3da70b43 Initial load
duke
parents:
diff changeset
   664
                 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   665
                return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   666
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   667
            URI uri = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   668
            // Use getHostString() to avoid reverse lookups
90ce3da70b43 Initial load
duke
parents:
diff changeset
   669
            String host = saddr.getHostString();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   670
            // IPv6 litteral?
90ce3da70b43 Initial load
duke
parents:
diff changeset
   671
            if (saddr.getAddress() instanceof Inet6Address &&
90ce3da70b43 Initial load
duke
parents:
diff changeset
   672
                (!host.startsWith("[")) && (host.indexOf(":") >= 0)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   673
                host = "[" + host + "]";
90ce3da70b43 Initial load
duke
parents:
diff changeset
   674
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   675
            try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   676
                uri = new URI("serversocket://" + ParseUtil.encodePath(host) + ":"+ saddr.getPort());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   677
            } catch (URISyntaxException e) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   678
                // This shouldn't happen
90ce3da70b43 Initial load
duke
parents:
diff changeset
   679
                assert false : e;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   680
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   681
            Proxy p = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   682
            Exception savedExc = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   683
            java.util.Iterator<Proxy> iProxy = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   684
            iProxy = sel.select(uri).iterator();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   685
            if (iProxy == null || !(iProxy.hasNext())) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   686
                return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   687
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   688
            while (iProxy.hasNext()) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   689
                p = iProxy.next();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   690
                if (p == null || p == Proxy.NO_PROXY) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   691
                    return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   692
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   693
                if (p.type() != Proxy.Type.SOCKS)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   694
                    throw new SocketException("Unknown proxy type : " + p.type());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   695
                if (!(p.address() instanceof InetSocketAddress))
90ce3da70b43 Initial load
duke
parents:
diff changeset
   696
                    throw new SocketException("Unknow address type for proxy: " + p);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   697
                // Use getHostString() to avoid reverse lookups
90ce3da70b43 Initial load
duke
parents:
diff changeset
   698
                server = ((InetSocketAddress) p.address()).getHostString();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   699
                port = ((InetSocketAddress) p.address()).getPort();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   700
90ce3da70b43 Initial load
duke
parents:
diff changeset
   701
                // Connects to the SOCKS server
90ce3da70b43 Initial load
duke
parents:
diff changeset
   702
                try {
51
6fe31bc95bbc 6600143: Remove another 450 unnecessary casts
martin
parents: 2
diff changeset
   703
                    AccessController.doPrivileged(
6fe31bc95bbc 6600143: Remove another 450 unnecessary casts
martin
parents: 2
diff changeset
   704
                        new PrivilegedExceptionAction<Void>() {
6fe31bc95bbc 6600143: Remove another 450 unnecessary casts
martin
parents: 2
diff changeset
   705
                            public Void run() throws Exception {
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   706
                                cmdsock = new Socket(new PlainSocketImpl());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   707
                                cmdsock.connect(new InetSocketAddress(server, port));
90ce3da70b43 Initial load
duke
parents:
diff changeset
   708
                                cmdIn = cmdsock.getInputStream();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   709
                                cmdOut = cmdsock.getOutputStream();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   710
                                return null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   711
                            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   712
                        });
90ce3da70b43 Initial load
duke
parents:
diff changeset
   713
                } catch (Exception e) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   714
                    // Ooops, let's notify the ProxySelector
90ce3da70b43 Initial load
duke
parents:
diff changeset
   715
                    sel.connectFailed(uri,p.address(),new SocketException(e.getMessage()));
90ce3da70b43 Initial load
duke
parents:
diff changeset
   716
                    server = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   717
                    port = -1;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   718
                    cmdsock = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   719
                    savedExc = e;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   720
                    // Will continue the while loop and try the next proxy
90ce3da70b43 Initial load
duke
parents:
diff changeset
   721
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   722
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   723
90ce3da70b43 Initial load
duke
parents:
diff changeset
   724
            /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   725
             * If server is still null at this point, none of the proxy
90ce3da70b43 Initial load
duke
parents:
diff changeset
   726
             * worked
90ce3da70b43 Initial load
duke
parents:
diff changeset
   727
             */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   728
            if (server == null || cmdsock == null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   729
                throw new SocketException("Can't connect to SOCKS proxy:"
90ce3da70b43 Initial load
duke
parents:
diff changeset
   730
                                          + savedExc.getMessage());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   731
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   732
        } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   733
            try {
51
6fe31bc95bbc 6600143: Remove another 450 unnecessary casts
martin
parents: 2
diff changeset
   734
                AccessController.doPrivileged(
6fe31bc95bbc 6600143: Remove another 450 unnecessary casts
martin
parents: 2
diff changeset
   735
                    new PrivilegedExceptionAction<Void>() {
6fe31bc95bbc 6600143: Remove another 450 unnecessary casts
martin
parents: 2
diff changeset
   736
                        public Void run() throws Exception {
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   737
                            cmdsock = new Socket(new PlainSocketImpl());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   738
                            cmdsock.connect(new InetSocketAddress(server, port));
90ce3da70b43 Initial load
duke
parents:
diff changeset
   739
                            cmdIn = cmdsock.getInputStream();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   740
                            cmdOut = cmdsock.getOutputStream();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   741
                            return null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   742
                        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   743
                    });
90ce3da70b43 Initial load
duke
parents:
diff changeset
   744
            } catch (Exception e) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   745
                throw new SocketException(e.getMessage());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   746
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   747
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   748
        BufferedOutputStream out = new BufferedOutputStream(cmdOut, 512);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   749
        InputStream in = cmdIn;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   750
        if (useV4) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   751
            bindV4(in, out, saddr.getAddress(), saddr.getPort());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   752
            return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   753
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   754
        out.write(PROTO_VERS);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   755
        out.write(2);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   756
        out.write(NO_AUTH);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   757
        out.write(USER_PASSW);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   758
        out.flush();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   759
        byte[] data = new byte[2];
90ce3da70b43 Initial load
duke
parents:
diff changeset
   760
        int i = readSocksReply(in, data);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   761
        if (i != 2 || ((int)data[0]) != PROTO_VERS) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   762
            // Maybe it's not a V5 sever after all
90ce3da70b43 Initial load
duke
parents:
diff changeset
   763
            // Let's try V4 before we give up
90ce3da70b43 Initial load
duke
parents:
diff changeset
   764
            bindV4(in, out, saddr.getAddress(), saddr.getPort());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   765
            return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   766
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   767
        if (((int)data[1]) == NO_METHODS)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   768
            throw new SocketException("SOCKS : No acceptable methods");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   769
        if (!authenticate(data[1], in, out)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   770
            throw new SocketException("SOCKS : authentication failed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   771
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   772
        // We're OK. Let's issue the BIND command.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   773
        out.write(PROTO_VERS);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   774
        out.write(BIND);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   775
        out.write(0);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   776
        int lport = saddr.getPort();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   777
        if (saddr.isUnresolved()) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   778
            out.write(DOMAIN_NAME);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   779
            out.write(saddr.getHostName().length());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   780
            try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   781
                out.write(saddr.getHostName().getBytes("ISO-8859-1"));
90ce3da70b43 Initial load
duke
parents:
diff changeset
   782
            } catch (java.io.UnsupportedEncodingException uee) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   783
                assert false;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   784
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   785
            out.write((lport >> 8) & 0xff);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   786
            out.write((lport >> 0) & 0xff);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   787
        } else if (saddr.getAddress() instanceof Inet4Address) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   788
            byte[] addr1 = saddr.getAddress().getAddress();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   789
            out.write(IPV4);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   790
            out.write(addr1);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   791
            out.write((lport >> 8) & 0xff);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   792
            out.write((lport >> 0) & 0xff);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   793
            out.flush();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   794
        } else if (saddr.getAddress() instanceof Inet6Address) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   795
            byte[] addr1 = saddr.getAddress().getAddress();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   796
            out.write(IPV6);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   797
            out.write(addr1);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   798
            out.write((lport >> 8) & 0xff);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   799
            out.write((lport >> 0) & 0xff);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   800
            out.flush();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   801
        } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   802
            cmdsock.close();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   803
            throw new SocketException("unsupported address type : " + saddr);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   804
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   805
        data = new byte[4];
90ce3da70b43 Initial load
duke
parents:
diff changeset
   806
        i = readSocksReply(in, data);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   807
        SocketException ex = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   808
        int len, nport;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   809
        byte[] addr;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   810
        switch (data[1]) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   811
        case REQUEST_OK:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   812
            // success!
90ce3da70b43 Initial load
duke
parents:
diff changeset
   813
            InetSocketAddress real_end = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   814
            switch(data[3]) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   815
            case IPV4:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   816
                addr = new byte[4];
90ce3da70b43 Initial load
duke
parents:
diff changeset
   817
                i = readSocksReply(in, addr);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   818
                if (i != 4)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   819
                    throw new SocketException("Reply from SOCKS server badly formatted");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   820
                data = new byte[2];
90ce3da70b43 Initial load
duke
parents:
diff changeset
   821
                i = readSocksReply(in, data);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   822
                if (i != 2)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   823
                    throw new SocketException("Reply from SOCKS server badly formatted");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   824
                nport = ((int)data[0] & 0xff) << 8;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   825
                nport += ((int)data[1] & 0xff);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   826
                external_address =
90ce3da70b43 Initial load
duke
parents:
diff changeset
   827
                    new InetSocketAddress(new Inet4Address("", addr) , nport);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   828
                break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   829
            case DOMAIN_NAME:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   830
                len = data[1];
90ce3da70b43 Initial load
duke
parents:
diff changeset
   831
                byte[] host = new byte[len];
90ce3da70b43 Initial load
duke
parents:
diff changeset
   832
                i = readSocksReply(in, host);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   833
                if (i != len)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   834
                    throw new SocketException("Reply from SOCKS server badly formatted");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   835
                data = new byte[2];
90ce3da70b43 Initial load
duke
parents:
diff changeset
   836
                i = readSocksReply(in, data);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   837
                if (i != 2)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   838
                    throw new SocketException("Reply from SOCKS server badly formatted");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   839
                nport = ((int)data[0] & 0xff) << 8;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   840
                nport += ((int)data[1] & 0xff);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   841
                external_address = new InetSocketAddress(new String(host), nport);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   842
                break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   843
            case IPV6:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   844
                len = data[1];
90ce3da70b43 Initial load
duke
parents:
diff changeset
   845
                addr = new byte[len];
90ce3da70b43 Initial load
duke
parents:
diff changeset
   846
                i = readSocksReply(in, addr);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   847
                if (i != len)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   848
                    throw new SocketException("Reply from SOCKS server badly formatted");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   849
                data = new byte[2];
90ce3da70b43 Initial load
duke
parents:
diff changeset
   850
                i = readSocksReply(in, data);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   851
                if (i != 2)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   852
                    throw new SocketException("Reply from SOCKS server badly formatted");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   853
                nport = ((int)data[0] & 0xff) << 8;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   854
                nport += ((int)data[1] & 0xff);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   855
                external_address =
90ce3da70b43 Initial load
duke
parents:
diff changeset
   856
                    new InetSocketAddress(new Inet6Address("", addr), nport);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   857
                break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   858
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   859
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   860
        case GENERAL_FAILURE:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   861
            ex = new SocketException("SOCKS server general failure");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   862
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   863
        case NOT_ALLOWED:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   864
            ex = new SocketException("SOCKS: Bind not allowed by ruleset");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   865
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   866
        case NET_UNREACHABLE:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   867
            ex = new SocketException("SOCKS: Network unreachable");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   868
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   869
        case HOST_UNREACHABLE:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   870
            ex = new SocketException("SOCKS: Host unreachable");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   871
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   872
        case CONN_REFUSED:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   873
            ex = new SocketException("SOCKS: Connection refused");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   874
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   875
        case TTL_EXPIRED:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   876
            ex =  new SocketException("SOCKS: TTL expired");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   877
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   878
        case CMD_NOT_SUPPORTED:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   879
            ex = new SocketException("SOCKS: Command not supported");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   880
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   881
        case ADDR_TYPE_NOT_SUP:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   882
            ex = new SocketException("SOCKS: address type not supported");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   883
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   884
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   885
        if (ex != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   886
            in.close();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   887
            out.close();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   888
            cmdsock.close();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   889
            cmdsock = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   890
            throw ex;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   891
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   892
        cmdIn = in;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   893
        cmdOut = out;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   894
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   895
90ce3da70b43 Initial load
duke
parents:
diff changeset
   896
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   897
     * Accepts a connection from a specific host.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   898
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   899
     * @param      s   the accepted connection.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   900
     * @param      saddr the socket address of the host we do accept
90ce3da70b43 Initial load
duke
parents:
diff changeset
   901
     *               connection from
90ce3da70b43 Initial load
duke
parents:
diff changeset
   902
     * @exception  IOException  if an I/O error occurs when accepting the
90ce3da70b43 Initial load
duke
parents:
diff changeset
   903
     *               connection.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   904
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   905
    protected void acceptFrom(SocketImpl s, InetSocketAddress saddr) throws IOException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   906
        if (cmdsock == null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   907
            // Not a Socks ServerSocket.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   908
            return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   909
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   910
        InputStream in = cmdIn;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   911
        // Sends the "SOCKS BIND" request.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   912
        socksBind(saddr);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   913
        in.read();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   914
        int i = in.read();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   915
        in.read();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   916
        SocketException ex = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   917
        int nport;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   918
        byte[] addr;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   919
        InetSocketAddress real_end = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   920
        switch (i) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   921
        case REQUEST_OK:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   922
            // success!
90ce3da70b43 Initial load
duke
parents:
diff changeset
   923
            i = in.read();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   924
            switch(i) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   925
            case IPV4:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   926
                addr = new byte[4];
90ce3da70b43 Initial load
duke
parents:
diff changeset
   927
                readSocksReply(in, addr);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   928
                nport = in.read() << 8;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   929
                nport += in.read();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   930
                real_end =
90ce3da70b43 Initial load
duke
parents:
diff changeset
   931
                    new InetSocketAddress(new Inet4Address("", addr) , nport);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   932
                break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   933
            case DOMAIN_NAME:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   934
                int len = in.read();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   935
                addr = new byte[len];
90ce3da70b43 Initial load
duke
parents:
diff changeset
   936
                readSocksReply(in, addr);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   937
                nport = in.read() << 8;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   938
                nport += in.read();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   939
                real_end = new InetSocketAddress(new String(addr), nport);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   940
                break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   941
            case IPV6:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   942
                addr = new byte[16];
90ce3da70b43 Initial load
duke
parents:
diff changeset
   943
                readSocksReply(in, addr);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   944
                nport = in.read() << 8;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   945
                nport += in.read();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   946
                real_end =
90ce3da70b43 Initial load
duke
parents:
diff changeset
   947
                    new InetSocketAddress(new Inet6Address("", addr), nport);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   948
                break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   949
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   950
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   951
        case GENERAL_FAILURE:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   952
            ex = new SocketException("SOCKS server general failure");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   953
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   954
        case NOT_ALLOWED:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   955
            ex = new SocketException("SOCKS: Accept not allowed by ruleset");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   956
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   957
        case NET_UNREACHABLE:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   958
            ex = new SocketException("SOCKS: Network unreachable");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   959
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   960
        case HOST_UNREACHABLE:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   961
            ex = new SocketException("SOCKS: Host unreachable");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   962
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   963
        case CONN_REFUSED:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   964
            ex = new SocketException("SOCKS: Connection refused");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   965
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   966
        case TTL_EXPIRED:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   967
            ex =  new SocketException("SOCKS: TTL expired");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   968
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   969
        case CMD_NOT_SUPPORTED:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   970
            ex = new SocketException("SOCKS: Command not supported");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   971
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   972
        case ADDR_TYPE_NOT_SUP:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   973
            ex = new SocketException("SOCKS: address type not supported");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   974
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   975
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   976
        if (ex != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   977
            cmdIn.close();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   978
            cmdOut.close();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   979
            cmdsock.close();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   980
            cmdsock = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   981
            throw ex;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   982
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   983
90ce3da70b43 Initial load
duke
parents:
diff changeset
   984
        /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   985
         * This is where we have to do some fancy stuff.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   986
         * The datastream from the socket "accepted" by the proxy will
90ce3da70b43 Initial load
duke
parents:
diff changeset
   987
         * come through the cmdSocket. So we have to swap the socketImpls
90ce3da70b43 Initial load
duke
parents:
diff changeset
   988
         */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   989
        if (s instanceof SocksSocketImpl) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   990
            ((SocksSocketImpl)s).external_address = real_end;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   991
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   992
        if (s instanceof PlainSocketImpl) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   993
            PlainSocketImpl psi = (PlainSocketImpl) s;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   994
            psi.setInputStream((SocketInputStream) in);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   995
            psi.setFileDescriptor(cmdsock.getImpl().getFileDescriptor());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   996
            psi.setAddress(cmdsock.getImpl().getInetAddress());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   997
            psi.setPort(cmdsock.getImpl().getPort());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   998
            psi.setLocalPort(cmdsock.getImpl().getLocalPort());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   999
        } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1000
            s.fd = cmdsock.getImpl().fd;
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1001
            s.address = cmdsock.getImpl().address;
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1002
            s.port = cmdsock.getImpl().port;
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1003
            s.localport = cmdsock.getImpl().localport;
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1004
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1005
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1006
        // Need to do that so that the socket won't be closed
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1007
        // when the ServerSocket is closed by the user.
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1008
        // It kinds of detaches the Socket because it is now
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1009
        // used elsewhere.
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1010
        cmdsock = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1011
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1012
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1013
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1014
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1015
     * Returns the value of this socket's <code>address</code> field.
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1016
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1017
     * @return  the value of this socket's <code>address</code> field.
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1018
     * @see     java.net.SocketImpl#address
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1019
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1020
    protected InetAddress getInetAddress() {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1021
        if (external_address != null)
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1022
            return external_address.getAddress();
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1023
        else
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1024
            return super.getInetAddress();
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1025
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1026
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1027
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1028
     * Returns the value of this socket's <code>port</code> field.
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1029
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1030
     * @return  the value of this socket's <code>port</code> field.
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1031
     * @see     java.net.SocketImpl#port
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1032
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1033
    protected int getPort() {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1034
        if (external_address != null)
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1035
            return external_address.getPort();
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1036
        else
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1037
            return super.getPort();
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1038
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1039
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1040
    protected int getLocalPort() {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1041
        if (socket != null)
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1042
            return super.getLocalPort();
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1043
        if (external_address != null)
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1044
            return external_address.getPort();
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1045
        else
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1046
            return super.getLocalPort();
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1047
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1048
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1049
    protected void close() throws IOException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1050
        if (cmdsock != null)
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1051
            cmdsock.close();
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1052
        cmdsock = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1053
        super.close();
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1054
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1055
3450
2f08a8bb9b83 6801071: Remote sites can compromise user privacy and possibly hijack web sessions
chegar
parents: 715
diff changeset
  1056
    private String getUserName() {
2f08a8bb9b83 6801071: Remote sites can compromise user privacy and possibly hijack web sessions
chegar
parents: 715
diff changeset
  1057
        String userName = "";
2f08a8bb9b83 6801071: Remote sites can compromise user privacy and possibly hijack web sessions
chegar
parents: 715
diff changeset
  1058
        if (applicationSetProxy) {
2f08a8bb9b83 6801071: Remote sites can compromise user privacy and possibly hijack web sessions
chegar
parents: 715
diff changeset
  1059
            try {
2f08a8bb9b83 6801071: Remote sites can compromise user privacy and possibly hijack web sessions
chegar
parents: 715
diff changeset
  1060
                userName = System.getProperty("user.name");
2f08a8bb9b83 6801071: Remote sites can compromise user privacy and possibly hijack web sessions
chegar
parents: 715
diff changeset
  1061
            } catch (SecurityException se) { /* swallow Exception */ }
2f08a8bb9b83 6801071: Remote sites can compromise user privacy and possibly hijack web sessions
chegar
parents: 715
diff changeset
  1062
        } else {
2f08a8bb9b83 6801071: Remote sites can compromise user privacy and possibly hijack web sessions
chegar
parents: 715
diff changeset
  1063
            userName = java.security.AccessController.doPrivileged(
2f08a8bb9b83 6801071: Remote sites can compromise user privacy and possibly hijack web sessions
chegar
parents: 715
diff changeset
  1064
                new sun.security.action.GetPropertyAction("user.name"));
2f08a8bb9b83 6801071: Remote sites can compromise user privacy and possibly hijack web sessions
chegar
parents: 715
diff changeset
  1065
        }
2f08a8bb9b83 6801071: Remote sites can compromise user privacy and possibly hijack web sessions
chegar
parents: 715
diff changeset
  1066
        return userName;
2f08a8bb9b83 6801071: Remote sites can compromise user privacy and possibly hijack web sessions
chegar
parents: 715
diff changeset
  1067
    }
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1068
}