src/java.base/share/classes/java/net/SocksSocketImpl.java
author chegar
Thu, 17 Oct 2019 20:54:25 +0100
branchdatagramsocketimpl-branch
changeset 58679 9c3209ff7550
parent 58678 9cf78a70fa4f
parent 57964 6bee0a3d2a3a
permissions -rw-r--r--
datagramsocketimpl-branch: merge with default
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
     1
/*
54155
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
     2
 * Copyright (c) 2000, 2019, Oracle and/or its affiliates. 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
5506
202f599c92aa 6943119: Rebrand source copyright notices
ohair
parents: 5147
diff changeset
     7
 * published by the Free Software Foundation.  Oracle designates this
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
     8
 * particular file as subject to the "Classpath" exception as provided
5506
202f599c92aa 6943119: Rebrand source copyright notices
ohair
parents: 5147
diff changeset
     9
 * by Oracle in the LICENSE file that accompanied this code.
2
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
 *
5506
202f599c92aa 6943119: Rebrand source copyright notices
ohair
parents: 5147
diff changeset
    21
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
202f599c92aa 6943119: Rebrand source copyright notices
ohair
parents: 5147
diff changeset
    22
 * or visit www.oracle.com if you need additional information or have any
202f599c92aa 6943119: Rebrand source copyright notices
ohair
parents: 5147
diff changeset
    23
 * questions.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    24
 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
    25
package java.net;
54155
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
    26
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    27
import java.io.IOException;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    28
import java.io.InputStream;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    29
import java.io.OutputStream;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    30
import java.io.BufferedOutputStream;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    31
import java.security.AccessController;
57964
6bee0a3d2a3a 8230310: SocksSocketImpl should handle the IllegalArgumentException thrown by ProxySelector.select usage
jpai
parents: 54689
diff changeset
    32
import java.util.Iterator;
50817
fa1e04811ff6 8066709: Make some JDK system properties read only
rriggs
parents: 47478
diff changeset
    33
fa1e04811ff6 8066709: Make some JDK system properties read only
rriggs
parents: 47478
diff changeset
    34
import jdk.internal.util.StaticProperty;
7982
65f5328a67a2 6964547: Impossible to set useV4 in SocksSocketImpl
chegar
parents: 7668
diff changeset
    35
import sun.net.SocksProxy;
31529
31d7d82b39ff 8129444: socksProxyVersion system property ignored for Socket(Proxy)
asmotrak
parents: 29986
diff changeset
    36
import sun.net.spi.DefaultProxySelector;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    37
import sun.net.www.ParseUtil;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    38
90ce3da70b43 Initial load
duke
parents:
diff changeset
    39
/**
90ce3da70b43 Initial load
duke
parents:
diff changeset
    40
 * SOCKS (V4 & V5) TCP socket implementation (RFC 1928).
90ce3da70b43 Initial load
duke
parents:
diff changeset
    41
 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
    42
54155
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
    43
class SocksSocketImpl extends DelegatingSocketImpl implements SocksConsts {
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    44
    private String server = null;
3051
9481bd560a57 6852108: Remove Preferences dependance from SocksSocketImpl
jccollet
parents: 715
diff changeset
    45
    private int serverPort = DEFAULT_PORT;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    46
    private InetSocketAddress external_address;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    47
    private boolean useV4 = false;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    48
    private Socket cmdsock = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    49
    private InputStream cmdIn = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    50
    private OutputStream cmdOut = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    51
54155
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
    52
    SocksSocketImpl(SocketImpl delegate) {
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
    53
        super(delegate);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    54
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
    55
54155
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
    56
    SocksSocketImpl(Proxy proxy, SocketImpl delegate) {
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
    57
        super(delegate);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    58
        SocketAddress a = proxy.address();
90ce3da70b43 Initial load
duke
parents:
diff changeset
    59
        if (a instanceof InetSocketAddress) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
    60
            InetSocketAddress ad = (InetSocketAddress) a;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    61
            // Use getHostString() to avoid reverse lookups
90ce3da70b43 Initial load
duke
parents:
diff changeset
    62
            server = ad.getHostString();
3051
9481bd560a57 6852108: Remove Preferences dependance from SocksSocketImpl
jccollet
parents: 715
diff changeset
    63
            serverPort = ad.getPort();
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    64
        }
31529
31d7d82b39ff 8129444: socksProxyVersion system property ignored for Socket(Proxy)
asmotrak
parents: 29986
diff changeset
    65
        useV4 = useV4(proxy);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    66
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
    67
31529
31d7d82b39ff 8129444: socksProxyVersion system property ignored for Socket(Proxy)
asmotrak
parents: 29986
diff changeset
    68
    private static boolean useV4(Proxy proxy) {
31d7d82b39ff 8129444: socksProxyVersion system property ignored for Socket(Proxy)
asmotrak
parents: 29986
diff changeset
    69
        if (proxy instanceof SocksProxy
31d7d82b39ff 8129444: socksProxyVersion system property ignored for Socket(Proxy)
asmotrak
parents: 29986
diff changeset
    70
            && ((SocksProxy)proxy).protocolVersion() == 4) {
31d7d82b39ff 8129444: socksProxyVersion system property ignored for Socket(Proxy)
asmotrak
parents: 29986
diff changeset
    71
            return true;
31d7d82b39ff 8129444: socksProxyVersion system property ignored for Socket(Proxy)
asmotrak
parents: 29986
diff changeset
    72
        }
31d7d82b39ff 8129444: socksProxyVersion system property ignored for Socket(Proxy)
asmotrak
parents: 29986
diff changeset
    73
        return DefaultProxySelector.socksProxyVersion() == 4;
31d7d82b39ff 8129444: socksProxyVersion system property ignored for Socket(Proxy)
asmotrak
parents: 29986
diff changeset
    74
    }
31d7d82b39ff 8129444: socksProxyVersion system property ignored for Socket(Proxy)
asmotrak
parents: 29986
diff changeset
    75
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    76
    private synchronized void privilegedConnect(final String host,
90ce3da70b43 Initial load
duke
parents:
diff changeset
    77
                                              final int port,
90ce3da70b43 Initial load
duke
parents:
diff changeset
    78
                                              final int timeout)
54155
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
    79
        throws IOException
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    80
    {
90ce3da70b43 Initial load
duke
parents:
diff changeset
    81
        try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
    82
            AccessController.doPrivileged(
29986
97167d851fc4 8078467: Update core libraries to use diamond with anonymous classes
darcy
parents: 29112
diff changeset
    83
                new java.security.PrivilegedExceptionAction<>() {
51
6fe31bc95bbc 6600143: Remove another 450 unnecessary casts
martin
parents: 2
diff changeset
    84
                    public Void run() throws IOException {
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    85
                              superConnectServer(host, port, timeout);
90ce3da70b43 Initial load
duke
parents:
diff changeset
    86
                              cmdIn = getInputStream();
90ce3da70b43 Initial load
duke
parents:
diff changeset
    87
                              cmdOut = getOutputStream();
90ce3da70b43 Initial load
duke
parents:
diff changeset
    88
                              return null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    89
                          }
90ce3da70b43 Initial load
duke
parents:
diff changeset
    90
                      });
90ce3da70b43 Initial load
duke
parents:
diff changeset
    91
        } catch (java.security.PrivilegedActionException pae) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
    92
            throw (IOException) pae.getException();
90ce3da70b43 Initial load
duke
parents:
diff changeset
    93
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
    94
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
    95
90ce3da70b43 Initial load
duke
parents:
diff changeset
    96
    private void superConnectServer(String host, int port,
90ce3da70b43 Initial load
duke
parents:
diff changeset
    97
                                    int timeout) throws IOException {
54155
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
    98
        delegate.connect(new InetSocketAddress(host, port), timeout);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    99
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   100
5147
96642e83ad41 6223635: Code hangs at connect call even when Timeout is specified when using a socks proxy
chegar
parents: 3464
diff changeset
   101
    private static int remainingMillis(long deadlineMillis) throws IOException {
96642e83ad41 6223635: Code hangs at connect call even when Timeout is specified when using a socks proxy
chegar
parents: 3464
diff changeset
   102
        if (deadlineMillis == 0L)
96642e83ad41 6223635: Code hangs at connect call even when Timeout is specified when using a socks proxy
chegar
parents: 3464
diff changeset
   103
            return 0;
96642e83ad41 6223635: Code hangs at connect call even when Timeout is specified when using a socks proxy
chegar
parents: 3464
diff changeset
   104
96642e83ad41 6223635: Code hangs at connect call even when Timeout is specified when using a socks proxy
chegar
parents: 3464
diff changeset
   105
        final long remaining = deadlineMillis - System.currentTimeMillis();
96642e83ad41 6223635: Code hangs at connect call even when Timeout is specified when using a socks proxy
chegar
parents: 3464
diff changeset
   106
        if (remaining > 0)
96642e83ad41 6223635: Code hangs at connect call even when Timeout is specified when using a socks proxy
chegar
parents: 3464
diff changeset
   107
            return (int) remaining;
96642e83ad41 6223635: Code hangs at connect call even when Timeout is specified when using a socks proxy
chegar
parents: 3464
diff changeset
   108
96642e83ad41 6223635: Code hangs at connect call even when Timeout is specified when using a socks proxy
chegar
parents: 3464
diff changeset
   109
        throw new SocketTimeoutException();
96642e83ad41 6223635: Code hangs at connect call even when Timeout is specified when using a socks proxy
chegar
parents: 3464
diff changeset
   110
    }
96642e83ad41 6223635: Code hangs at connect call even when Timeout is specified when using a socks proxy
chegar
parents: 3464
diff changeset
   111
96642e83ad41 6223635: Code hangs at connect call even when Timeout is specified when using a socks proxy
chegar
parents: 3464
diff changeset
   112
    private int readSocksReply(InputStream in, byte[] data, long deadlineMillis) throws IOException {
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   113
        int len = data.length;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   114
        int received = 0;
54155
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
   115
        int originalTimeout = (int) getOption(SocketOptions.SO_TIMEOUT);
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
   116
        try {
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
   117
            while (received < len) {
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
   118
                int count;
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
   119
                int remaining = remainingMillis(deadlineMillis);
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
   120
                setOption(SocketOptions.SO_TIMEOUT, remaining);
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
   121
                try {
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
   122
                    count = in.read(data, received, len - received);
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
   123
                } catch (SocketTimeoutException e) {
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
   124
                    throw new SocketTimeoutException("Connect timed out");
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
   125
                }
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
   126
                if (count < 0)
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
   127
                    throw new SocketException("Malformed reply from SOCKS server");
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
   128
                received += count;
5147
96642e83ad41 6223635: Code hangs at connect call even when Timeout is specified when using a socks proxy
chegar
parents: 3464
diff changeset
   129
            }
54155
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
   130
        } finally {
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
   131
            setOption(SocketOptions.SO_TIMEOUT, originalTimeout);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   132
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   133
        return received;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   134
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   135
5147
96642e83ad41 6223635: Code hangs at connect call even when Timeout is specified when using a socks proxy
chegar
parents: 3464
diff changeset
   136
    private boolean authenticate(byte method, InputStream in,
96642e83ad41 6223635: Code hangs at connect call even when Timeout is specified when using a socks proxy
chegar
parents: 3464
diff changeset
   137
                                 BufferedOutputStream out,
96642e83ad41 6223635: Code hangs at connect call even when Timeout is specified when using a socks proxy
chegar
parents: 3464
diff changeset
   138
                                 long deadlineMillis) throws IOException {
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   139
        // No Authentication required. We're done then!
90ce3da70b43 Initial load
duke
parents:
diff changeset
   140
        if (method == NO_AUTH)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   141
            return true;
52499
768b1c612100 8213490: Networking area typos and inconsistencies cleanup
prappo
parents: 50817
diff changeset
   142
        /*
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   143
         * User/Password authentication. Try, in that order :
90ce3da70b43 Initial load
duke
parents:
diff changeset
   144
         * - The application provided Authenticator, if any
90ce3da70b43 Initial load
duke
parents:
diff changeset
   145
         * - the user.name & no password (backward compatibility behavior).
90ce3da70b43 Initial load
duke
parents:
diff changeset
   146
         */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   147
        if (method == USER_PASSW) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   148
            String userName;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   149
            String password = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   150
            final InetAddress addr = InetAddress.getByName(server);
51
6fe31bc95bbc 6600143: Remove another 450 unnecessary casts
martin
parents: 2
diff changeset
   151
            PasswordAuthentication pw =
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   152
                java.security.AccessController.doPrivileged(
29986
97167d851fc4 8078467: Update core libraries to use diamond with anonymous classes
darcy
parents: 29112
diff changeset
   153
                    new java.security.PrivilegedAction<>() {
51
6fe31bc95bbc 6600143: Remove another 450 unnecessary casts
martin
parents: 2
diff changeset
   154
                        public PasswordAuthentication run() {
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   155
                                return Authenticator.requestPasswordAuthentication(
3051
9481bd560a57 6852108: Remove Preferences dependance from SocksSocketImpl
jccollet
parents: 715
diff changeset
   156
                                       server, addr, serverPort, "SOCKS5", "SOCKS authentication", null);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   157
                            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   158
                        });
90ce3da70b43 Initial load
duke
parents:
diff changeset
   159
            if (pw != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   160
                userName = pw.getUserName();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   161
                password = new String(pw.getPassword());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   162
            } else {
50817
fa1e04811ff6 8066709: Make some JDK system properties read only
rriggs
parents: 47478
diff changeset
   163
                userName = StaticProperty.userName();
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   164
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   165
            if (userName == null)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   166
                return false;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   167
            out.write(1);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   168
            out.write(userName.length());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   169
            try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   170
                out.write(userName.getBytes("ISO-8859-1"));
90ce3da70b43 Initial load
duke
parents:
diff changeset
   171
            } catch (java.io.UnsupportedEncodingException uee) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   172
                assert false;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   173
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   174
            if (password != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   175
                out.write(password.length());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   176
                try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   177
                    out.write(password.getBytes("ISO-8859-1"));
90ce3da70b43 Initial load
duke
parents:
diff changeset
   178
                } catch (java.io.UnsupportedEncodingException uee) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   179
                    assert false;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   180
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   181
            } else
90ce3da70b43 Initial load
duke
parents:
diff changeset
   182
                out.write(0);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   183
            out.flush();
3051
9481bd560a57 6852108: Remove Preferences dependance from SocksSocketImpl
jccollet
parents: 715
diff changeset
   184
            byte[] data = new byte[2];
5147
96642e83ad41 6223635: Code hangs at connect call even when Timeout is specified when using a socks proxy
chegar
parents: 3464
diff changeset
   185
            int i = readSocksReply(in, data, deadlineMillis);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   186
            if (i != 2 || data[1] != 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   187
                /* RFC 1929 specifies that the connection MUST be closed if
90ce3da70b43 Initial load
duke
parents:
diff changeset
   188
                   authentication fails */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   189
                out.close();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   190
                in.close();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   191
                return false;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   192
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   193
            /* Authentication succeeded */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   194
            return true;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   195
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   196
        return false;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   197
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   198
90ce3da70b43 Initial load
duke
parents:
diff changeset
   199
    private void connectV4(InputStream in, OutputStream out,
5147
96642e83ad41 6223635: Code hangs at connect call even when Timeout is specified when using a socks proxy
chegar
parents: 3464
diff changeset
   200
                           InetSocketAddress endpoint,
96642e83ad41 6223635: Code hangs at connect call even when Timeout is specified when using a socks proxy
chegar
parents: 3464
diff changeset
   201
                           long deadlineMillis) throws IOException {
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   202
        if (!(endpoint.getAddress() instanceof Inet4Address)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   203
            throw new SocketException("SOCKS V4 requires IPv4 only addresses");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   204
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   205
        out.write(PROTO_VERS4);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   206
        out.write(CONNECT);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   207
        out.write((endpoint.getPort() >> 8) & 0xff);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   208
        out.write((endpoint.getPort() >> 0) & 0xff);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   209
        out.write(endpoint.getAddress().getAddress());
3450
2f08a8bb9b83 6801071: Remote sites can compromise user privacy and possibly hijack web sessions
chegar
parents: 715
diff changeset
   210
        String userName = getUserName();
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   211
        try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   212
            out.write(userName.getBytes("ISO-8859-1"));
90ce3da70b43 Initial load
duke
parents:
diff changeset
   213
        } catch (java.io.UnsupportedEncodingException uee) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   214
            assert false;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   215
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   216
        out.write(0);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   217
        out.flush();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   218
        byte[] data = new byte[8];
5147
96642e83ad41 6223635: Code hangs at connect call even when Timeout is specified when using a socks proxy
chegar
parents: 3464
diff changeset
   219
        int n = readSocksReply(in, data, deadlineMillis);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   220
        if (n != 8)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   221
            throw new SocketException("Reply from SOCKS server has bad length: " + n);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   222
        if (data[0] != 0 && data[0] != 4)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   223
            throw new SocketException("Reply from SOCKS server has bad version");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   224
        SocketException ex = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   225
        switch (data[1]) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   226
        case 90:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   227
            // Success!
90ce3da70b43 Initial load
duke
parents:
diff changeset
   228
            external_address = endpoint;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   229
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   230
        case 91:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   231
            ex = new SocketException("SOCKS request rejected");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   232
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   233
        case 92:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   234
            ex = new SocketException("SOCKS server couldn't reach destination");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   235
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   236
        case 93:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   237
            ex = new SocketException("SOCKS authentication failed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   238
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   239
        default:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   240
            ex = new SocketException("Reply from SOCKS server contains bad status");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   241
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   242
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   243
        if (ex != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   244
            in.close();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   245
            out.close();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   246
            throw ex;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   247
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   248
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   249
54155
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
   250
    @Override
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
   251
    protected void connect(String host, int port) throws IOException {
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
   252
        connect(new InetSocketAddress(host, port), 0);
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
   253
    }
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
   254
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
   255
    @Override
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
   256
    protected void connect(InetAddress address, int port) throws IOException {
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
   257
        connect(new InetSocketAddress(address, port), 0);
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
   258
    }
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
   259
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   260
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   261
     * Connects the Socks Socket to the specified endpoint. It will first
90ce3da70b43 Initial load
duke
parents:
diff changeset
   262
     * connect to the SOCKS proxy and negotiate the access. If the proxy
90ce3da70b43 Initial load
duke
parents:
diff changeset
   263
     * grants the connections, then the connect is successful and all
90ce3da70b43 Initial load
duke
parents:
diff changeset
   264
     * further traffic will go to the "real" endpoint.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   265
     *
19069
1d9cb0d080e3 8021833: javadoc cleanup in java.net
juh
parents: 18212
diff changeset
   266
     * @param   endpoint        the {@code SocketAddress} to connect to.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   267
     * @param   timeout         the timeout value in milliseconds
90ce3da70b43 Initial load
duke
parents:
diff changeset
   268
     * @throws  IOException     if the connection can't be established.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   269
     * @throws  SecurityException if there is a security manager and it
90ce3da70b43 Initial load
duke
parents:
diff changeset
   270
     *                          doesn't allow the connection
90ce3da70b43 Initial load
duke
parents:
diff changeset
   271
     * @throws  IllegalArgumentException if endpoint is null or a
90ce3da70b43 Initial load
duke
parents:
diff changeset
   272
     *          SocketAddress subclass not supported by this socket
90ce3da70b43 Initial load
duke
parents:
diff changeset
   273
     */
3051
9481bd560a57 6852108: Remove Preferences dependance from SocksSocketImpl
jccollet
parents: 715
diff changeset
   274
    @Override
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   275
    protected void connect(SocketAddress endpoint, int timeout) throws IOException {
5147
96642e83ad41 6223635: Code hangs at connect call even when Timeout is specified when using a socks proxy
chegar
parents: 3464
diff changeset
   276
        final long deadlineMillis;
96642e83ad41 6223635: Code hangs at connect call even when Timeout is specified when using a socks proxy
chegar
parents: 3464
diff changeset
   277
96642e83ad41 6223635: Code hangs at connect call even when Timeout is specified when using a socks proxy
chegar
parents: 3464
diff changeset
   278
        if (timeout == 0) {
96642e83ad41 6223635: Code hangs at connect call even when Timeout is specified when using a socks proxy
chegar
parents: 3464
diff changeset
   279
            deadlineMillis = 0L;
96642e83ad41 6223635: Code hangs at connect call even when Timeout is specified when using a socks proxy
chegar
parents: 3464
diff changeset
   280
        } else {
96642e83ad41 6223635: Code hangs at connect call even when Timeout is specified when using a socks proxy
chegar
parents: 3464
diff changeset
   281
            long finish = System.currentTimeMillis() + timeout;
96642e83ad41 6223635: Code hangs at connect call even when Timeout is specified when using a socks proxy
chegar
parents: 3464
diff changeset
   282
            deadlineMillis = finish < 0 ? Long.MAX_VALUE : finish;
96642e83ad41 6223635: Code hangs at connect call even when Timeout is specified when using a socks proxy
chegar
parents: 3464
diff changeset
   283
        }
96642e83ad41 6223635: Code hangs at connect call even when Timeout is specified when using a socks proxy
chegar
parents: 3464
diff changeset
   284
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   285
        SecurityManager security = System.getSecurityManager();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   286
        if (endpoint == null || !(endpoint instanceof InetSocketAddress))
90ce3da70b43 Initial load
duke
parents:
diff changeset
   287
            throw new IllegalArgumentException("Unsupported address type");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   288
        InetSocketAddress epoint = (InetSocketAddress) endpoint;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   289
        if (security != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   290
            if (epoint.isUnresolved())
90ce3da70b43 Initial load
duke
parents:
diff changeset
   291
                security.checkConnect(epoint.getHostName(),
90ce3da70b43 Initial load
duke
parents:
diff changeset
   292
                                      epoint.getPort());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   293
            else
90ce3da70b43 Initial load
duke
parents:
diff changeset
   294
                security.checkConnect(epoint.getAddress().getHostAddress(),
90ce3da70b43 Initial load
duke
parents:
diff changeset
   295
                                      epoint.getPort());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   296
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   297
        if (server == null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   298
            // This is the general case
90ce3da70b43 Initial load
duke
parents:
diff changeset
   299
            // server is not null only when the socket was created with a
90ce3da70b43 Initial load
duke
parents:
diff changeset
   300
            // specified proxy in which case it does bypass the ProxySelector
51
6fe31bc95bbc 6600143: Remove another 450 unnecessary casts
martin
parents: 2
diff changeset
   301
            ProxySelector sel = java.security.AccessController.doPrivileged(
29986
97167d851fc4 8078467: Update core libraries to use diamond with anonymous classes
darcy
parents: 29112
diff changeset
   302
                new java.security.PrivilegedAction<>() {
51
6fe31bc95bbc 6600143: Remove another 450 unnecessary casts
martin
parents: 2
diff changeset
   303
                    public ProxySelector run() {
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   304
                            return ProxySelector.getDefault();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   305
                        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   306
                    });
90ce3da70b43 Initial load
duke
parents:
diff changeset
   307
            if (sel == null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   308
                /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   309
                 * No default proxySelector --> direct connection
90ce3da70b43 Initial load
duke
parents:
diff changeset
   310
                 */
54155
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
   311
                delegate.connect(epoint, remainingMillis(deadlineMillis));
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   312
                return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   313
            }
3051
9481bd560a57 6852108: Remove Preferences dependance from SocksSocketImpl
jccollet
parents: 715
diff changeset
   314
            URI uri;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   315
            // Use getHostString() to avoid reverse lookups
90ce3da70b43 Initial load
duke
parents:
diff changeset
   316
            String host = epoint.getHostString();
52499
768b1c612100 8213490: Networking area typos and inconsistencies cleanup
prappo
parents: 50817
diff changeset
   317
            // IPv6 literal?
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   318
            if (epoint.getAddress() instanceof Inet6Address &&
24685
215fa91e1b4c 8044461: Cleanup new Boolean and single character strings
rriggs
parents: 22276
diff changeset
   319
                (!host.startsWith("[")) && (host.indexOf(':') >= 0)) {
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   320
                host = "[" + host + "]";
90ce3da70b43 Initial load
duke
parents:
diff changeset
   321
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   322
            try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   323
                uri = new URI("socket://" + ParseUtil.encodePath(host) + ":"+ epoint.getPort());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   324
            } catch (URISyntaxException e) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   325
                // This shouldn't happen
90ce3da70b43 Initial load
duke
parents:
diff changeset
   326
                assert false : e;
3051
9481bd560a57 6852108: Remove Preferences dependance from SocksSocketImpl
jccollet
parents: 715
diff changeset
   327
                uri = null;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   328
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   329
            Proxy p = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   330
            IOException savedExc = null;
57964
6bee0a3d2a3a 8230310: SocksSocketImpl should handle the IllegalArgumentException thrown by ProxySelector.select usage
jpai
parents: 54689
diff changeset
   331
            final Iterator<Proxy> iProxy;
6bee0a3d2a3a 8230310: SocksSocketImpl should handle the IllegalArgumentException thrown by ProxySelector.select usage
jpai
parents: 54689
diff changeset
   332
            try {
6bee0a3d2a3a 8230310: SocksSocketImpl should handle the IllegalArgumentException thrown by ProxySelector.select usage
jpai
parents: 54689
diff changeset
   333
                iProxy = sel.select(uri).iterator();
6bee0a3d2a3a 8230310: SocksSocketImpl should handle the IllegalArgumentException thrown by ProxySelector.select usage
jpai
parents: 54689
diff changeset
   334
            } catch (IllegalArgumentException iae) {
6bee0a3d2a3a 8230310: SocksSocketImpl should handle the IllegalArgumentException thrown by ProxySelector.select usage
jpai
parents: 54689
diff changeset
   335
                throw new IOException("Failed to select a proxy", iae);
6bee0a3d2a3a 8230310: SocksSocketImpl should handle the IllegalArgumentException thrown by ProxySelector.select usage
jpai
parents: 54689
diff changeset
   336
            }
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   337
            if (iProxy == null || !(iProxy.hasNext())) {
54155
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
   338
                delegate.connect(epoint, remainingMillis(deadlineMillis));
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   339
                return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   340
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   341
            while (iProxy.hasNext()) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   342
                p = iProxy.next();
29112
df7e4edb6566 7178362: Socket impls should ignore unsupported proxy types rather than throwing
coffeys
parents: 25859
diff changeset
   343
                if (p == null || p.type() != Proxy.Type.SOCKS) {
54155
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
   344
                    delegate.connect(epoint, remainingMillis(deadlineMillis));
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   345
                    return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   346
                }
29112
df7e4edb6566 7178362: Socket impls should ignore unsupported proxy types rather than throwing
coffeys
parents: 25859
diff changeset
   347
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   348
                if (!(p.address() instanceof InetSocketAddress))
29112
df7e4edb6566 7178362: Socket impls should ignore unsupported proxy types rather than throwing
coffeys
parents: 25859
diff changeset
   349
                    throw new SocketException("Unknown address type for proxy: " + p);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   350
                // Use getHostString() to avoid reverse lookups
90ce3da70b43 Initial load
duke
parents:
diff changeset
   351
                server = ((InetSocketAddress) p.address()).getHostString();
3051
9481bd560a57 6852108: Remove Preferences dependance from SocksSocketImpl
jccollet
parents: 715
diff changeset
   352
                serverPort = ((InetSocketAddress) p.address()).getPort();
31529
31d7d82b39ff 8129444: socksProxyVersion system property ignored for Socket(Proxy)
asmotrak
parents: 29986
diff changeset
   353
                useV4 = useV4(p);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   354
90ce3da70b43 Initial load
duke
parents:
diff changeset
   355
                // Connects to the SOCKS server
90ce3da70b43 Initial load
duke
parents:
diff changeset
   356
                try {
5147
96642e83ad41 6223635: Code hangs at connect call even when Timeout is specified when using a socks proxy
chegar
parents: 3464
diff changeset
   357
                    privilegedConnect(server, serverPort, remainingMillis(deadlineMillis));
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   358
                    // Worked, let's get outta here
90ce3da70b43 Initial load
duke
parents:
diff changeset
   359
                    break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   360
                } catch (IOException e) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   361
                    // Ooops, let's notify the ProxySelector
90ce3da70b43 Initial load
duke
parents:
diff changeset
   362
                    sel.connectFailed(uri,p.address(),e);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   363
                    server = null;
3051
9481bd560a57 6852108: Remove Preferences dependance from SocksSocketImpl
jccollet
parents: 715
diff changeset
   364
                    serverPort = -1;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   365
                    savedExc = e;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   366
                    // Will continue the while loop and try the next proxy
90ce3da70b43 Initial load
duke
parents:
diff changeset
   367
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   368
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   369
90ce3da70b43 Initial load
duke
parents:
diff changeset
   370
            /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   371
             * If server is still null at this point, none of the proxy
90ce3da70b43 Initial load
duke
parents:
diff changeset
   372
             * worked
90ce3da70b43 Initial load
duke
parents:
diff changeset
   373
             */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   374
            if (server == null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   375
                throw new SocketException("Can't connect to SOCKS proxy:"
90ce3da70b43 Initial load
duke
parents:
diff changeset
   376
                                          + savedExc.getMessage());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   377
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   378
        } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   379
            // Connects to the SOCKS server
90ce3da70b43 Initial load
duke
parents:
diff changeset
   380
            try {
5147
96642e83ad41 6223635: Code hangs at connect call even when Timeout is specified when using a socks proxy
chegar
parents: 3464
diff changeset
   381
                privilegedConnect(server, serverPort, remainingMillis(deadlineMillis));
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   382
            } catch (IOException e) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   383
                throw new SocketException(e.getMessage());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   384
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   385
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   386
21278
ef8a3a2a72f2 8022746: List of spelling errors in API doc
malenkov
parents: 19069
diff changeset
   387
        // cmdIn & cmdOut were initialized during the privilegedConnect() call
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   388
        BufferedOutputStream out = new BufferedOutputStream(cmdOut, 512);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   389
        InputStream in = cmdIn;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   390
90ce3da70b43 Initial load
duke
parents:
diff changeset
   391
        if (useV4) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   392
            // SOCKS Protocol version 4 doesn't know how to deal with
90ce3da70b43 Initial load
duke
parents:
diff changeset
   393
            // DOMAIN type of addresses (unresolved addresses here)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   394
            if (epoint.isUnresolved())
90ce3da70b43 Initial load
duke
parents:
diff changeset
   395
                throw new UnknownHostException(epoint.toString());
5147
96642e83ad41 6223635: Code hangs at connect call even when Timeout is specified when using a socks proxy
chegar
parents: 3464
diff changeset
   396
            connectV4(in, out, epoint, deadlineMillis);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   397
            return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   398
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   399
90ce3da70b43 Initial load
duke
parents:
diff changeset
   400
        // This is SOCKS V5
90ce3da70b43 Initial load
duke
parents:
diff changeset
   401
        out.write(PROTO_VERS);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   402
        out.write(2);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   403
        out.write(NO_AUTH);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   404
        out.write(USER_PASSW);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   405
        out.flush();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   406
        byte[] data = new byte[2];
5147
96642e83ad41 6223635: Code hangs at connect call even when Timeout is specified when using a socks proxy
chegar
parents: 3464
diff changeset
   407
        int i = readSocksReply(in, data, deadlineMillis);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   408
        if (i != 2 || ((int)data[0]) != PROTO_VERS) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   409
            // Maybe it's not a V5 sever after all
90ce3da70b43 Initial load
duke
parents:
diff changeset
   410
            // Let's try V4 before we give up
90ce3da70b43 Initial load
duke
parents:
diff changeset
   411
            // SOCKS Protocol version 4 doesn't know how to deal with
90ce3da70b43 Initial load
duke
parents:
diff changeset
   412
            // DOMAIN type of addresses (unresolved addresses here)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   413
            if (epoint.isUnresolved())
90ce3da70b43 Initial load
duke
parents:
diff changeset
   414
                throw new UnknownHostException(epoint.toString());
5147
96642e83ad41 6223635: Code hangs at connect call even when Timeout is specified when using a socks proxy
chegar
parents: 3464
diff changeset
   415
            connectV4(in, out, epoint, deadlineMillis);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   416
            return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   417
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   418
        if (((int)data[1]) == NO_METHODS)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   419
            throw new SocketException("SOCKS : No acceptable methods");
5147
96642e83ad41 6223635: Code hangs at connect call even when Timeout is specified when using a socks proxy
chegar
parents: 3464
diff changeset
   420
        if (!authenticate(data[1], in, out, deadlineMillis)) {
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   421
            throw new SocketException("SOCKS : authentication failed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   422
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   423
        out.write(PROTO_VERS);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   424
        out.write(CONNECT);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   425
        out.write(0);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   426
        /* Test for IPV4/IPV6/Unresolved */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   427
        if (epoint.isUnresolved()) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   428
            out.write(DOMAIN_NAME);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   429
            out.write(epoint.getHostName().length());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   430
            try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   431
                out.write(epoint.getHostName().getBytes("ISO-8859-1"));
90ce3da70b43 Initial load
duke
parents:
diff changeset
   432
            } catch (java.io.UnsupportedEncodingException uee) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   433
                assert false;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   434
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   435
            out.write((epoint.getPort() >> 8) & 0xff);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   436
            out.write((epoint.getPort() >> 0) & 0xff);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   437
        } else if (epoint.getAddress() instanceof Inet6Address) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   438
            out.write(IPV6);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   439
            out.write(epoint.getAddress().getAddress());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   440
            out.write((epoint.getPort() >> 8) & 0xff);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   441
            out.write((epoint.getPort() >> 0) & 0xff);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   442
        } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   443
            out.write(IPV4);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   444
            out.write(epoint.getAddress().getAddress());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   445
            out.write((epoint.getPort() >> 8) & 0xff);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   446
            out.write((epoint.getPort() >> 0) & 0xff);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   447
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   448
        out.flush();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   449
        data = new byte[4];
5147
96642e83ad41 6223635: Code hangs at connect call even when Timeout is specified when using a socks proxy
chegar
parents: 3464
diff changeset
   450
        i = readSocksReply(in, data, deadlineMillis);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   451
        if (i != 4)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   452
            throw new SocketException("Reply from SOCKS server has bad length");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   453
        SocketException ex = null;
3051
9481bd560a57 6852108: Remove Preferences dependance from SocksSocketImpl
jccollet
parents: 715
diff changeset
   454
        int len;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   455
        byte[] addr;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   456
        switch (data[1]) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   457
        case REQUEST_OK:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   458
            // success!
90ce3da70b43 Initial load
duke
parents:
diff changeset
   459
            switch(data[3]) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   460
            case IPV4:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   461
                addr = new byte[4];
5147
96642e83ad41 6223635: Code hangs at connect call even when Timeout is specified when using a socks proxy
chegar
parents: 3464
diff changeset
   462
                i = readSocksReply(in, addr, deadlineMillis);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   463
                if (i != 4)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   464
                    throw new SocketException("Reply from SOCKS server badly formatted");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   465
                data = new byte[2];
5147
96642e83ad41 6223635: Code hangs at connect call even when Timeout is specified when using a socks proxy
chegar
parents: 3464
diff changeset
   466
                i = readSocksReply(in, data, deadlineMillis);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   467
                if (i != 2)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   468
                    throw new SocketException("Reply from SOCKS server badly formatted");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   469
                break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   470
            case DOMAIN_NAME:
22276
7fc4c8b08e49 7100957: SOCKS proxying does not work with IPv6 connections
chegar
parents: 21278
diff changeset
   471
                byte[] lenByte = new byte[1];
7fc4c8b08e49 7100957: SOCKS proxying does not work with IPv6 connections
chegar
parents: 21278
diff changeset
   472
                i = readSocksReply(in, lenByte, deadlineMillis);
7fc4c8b08e49 7100957: SOCKS proxying does not work with IPv6 connections
chegar
parents: 21278
diff changeset
   473
                if (i != 1)
7fc4c8b08e49 7100957: SOCKS proxying does not work with IPv6 connections
chegar
parents: 21278
diff changeset
   474
                    throw new SocketException("Reply from SOCKS server badly formatted");
7fc4c8b08e49 7100957: SOCKS proxying does not work with IPv6 connections
chegar
parents: 21278
diff changeset
   475
                len = lenByte[0] & 0xFF;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   476
                byte[] host = new byte[len];
5147
96642e83ad41 6223635: Code hangs at connect call even when Timeout is specified when using a socks proxy
chegar
parents: 3464
diff changeset
   477
                i = readSocksReply(in, host, deadlineMillis);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   478
                if (i != len)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   479
                    throw new SocketException("Reply from SOCKS server badly formatted");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   480
                data = new byte[2];
5147
96642e83ad41 6223635: Code hangs at connect call even when Timeout is specified when using a socks proxy
chegar
parents: 3464
diff changeset
   481
                i = readSocksReply(in, data, deadlineMillis);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   482
                if (i != 2)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   483
                    throw new SocketException("Reply from SOCKS server badly formatted");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   484
                break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   485
            case IPV6:
22276
7fc4c8b08e49 7100957: SOCKS proxying does not work with IPv6 connections
chegar
parents: 21278
diff changeset
   486
                len = 16;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   487
                addr = new byte[len];
5147
96642e83ad41 6223635: Code hangs at connect call even when Timeout is specified when using a socks proxy
chegar
parents: 3464
diff changeset
   488
                i = readSocksReply(in, addr, deadlineMillis);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   489
                if (i != len)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   490
                    throw new SocketException("Reply from SOCKS server badly formatted");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   491
                data = new byte[2];
5147
96642e83ad41 6223635: Code hangs at connect call even when Timeout is specified when using a socks proxy
chegar
parents: 3464
diff changeset
   492
                i = readSocksReply(in, data, deadlineMillis);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   493
                if (i != 2)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   494
                    throw new SocketException("Reply from SOCKS server badly formatted");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   495
                break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   496
            default:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   497
                ex = new SocketException("Reply from SOCKS server contains wrong code");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   498
                break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   499
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   500
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   501
        case GENERAL_FAILURE:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   502
            ex = new SocketException("SOCKS server general failure");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   503
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   504
        case NOT_ALLOWED:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   505
            ex = new SocketException("SOCKS: Connection not allowed by ruleset");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   506
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   507
        case NET_UNREACHABLE:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   508
            ex = new SocketException("SOCKS: Network unreachable");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   509
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   510
        case HOST_UNREACHABLE:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   511
            ex = new SocketException("SOCKS: Host unreachable");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   512
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   513
        case CONN_REFUSED:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   514
            ex = new SocketException("SOCKS: Connection refused");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   515
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   516
        case TTL_EXPIRED:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   517
            ex =  new SocketException("SOCKS: TTL expired");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   518
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   519
        case CMD_NOT_SUPPORTED:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   520
            ex = new SocketException("SOCKS: Command not supported");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   521
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   522
        case ADDR_TYPE_NOT_SUP:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   523
            ex = new SocketException("SOCKS: address type not supported");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   524
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   525
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   526
        if (ex != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   527
            in.close();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   528
            out.close();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   529
            throw ex;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   530
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   531
        external_address = epoint;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   532
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   533
54155
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
   534
    @Override
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
   535
    protected void listen(int backlog) {
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
   536
        throw new InternalError("should not get here");
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
   537
    }
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   538
54155
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
   539
    @Override
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
   540
    protected void accept(SocketImpl s) {
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
   541
        throw new InternalError("should not get here");
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
   542
    }
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   543
90ce3da70b43 Initial load
duke
parents:
diff changeset
   544
    /**
19069
1d9cb0d080e3 8021833: javadoc cleanup in java.net
juh
parents: 18212
diff changeset
   545
     * Returns the value of this socket's {@code address} field.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   546
     *
19069
1d9cb0d080e3 8021833: javadoc cleanup in java.net
juh
parents: 18212
diff changeset
   547
     * @return  the value of this socket's {@code address} field.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   548
     * @see     java.net.SocketImpl#address
90ce3da70b43 Initial load
duke
parents:
diff changeset
   549
     */
3051
9481bd560a57 6852108: Remove Preferences dependance from SocksSocketImpl
jccollet
parents: 715
diff changeset
   550
    @Override
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   551
    protected InetAddress getInetAddress() {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   552
        if (external_address != null)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   553
            return external_address.getAddress();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   554
        else
54155
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
   555
            return delegate.getInetAddress();
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   556
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   557
90ce3da70b43 Initial load
duke
parents:
diff changeset
   558
    /**
19069
1d9cb0d080e3 8021833: javadoc cleanup in java.net
juh
parents: 18212
diff changeset
   559
     * Returns the value of this socket's {@code port} field.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   560
     *
19069
1d9cb0d080e3 8021833: javadoc cleanup in java.net
juh
parents: 18212
diff changeset
   561
     * @return  the value of this socket's {@code port} field.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   562
     * @see     java.net.SocketImpl#port
90ce3da70b43 Initial load
duke
parents:
diff changeset
   563
     */
3051
9481bd560a57 6852108: Remove Preferences dependance from SocksSocketImpl
jccollet
parents: 715
diff changeset
   564
    @Override
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   565
    protected int getPort() {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   566
        if (external_address != null)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   567
            return external_address.getPort();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   568
        else
54155
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
   569
            return delegate.getPort();
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   570
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   571
3051
9481bd560a57 6852108: Remove Preferences dependance from SocksSocketImpl
jccollet
parents: 715
diff changeset
   572
    @Override
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   573
    protected void close() throws IOException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   574
        if (cmdsock != null)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   575
            cmdsock.close();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   576
        cmdsock = null;
54155
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
   577
        delegate.close();
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   578
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   579
3450
2f08a8bb9b83 6801071: Remote sites can compromise user privacy and possibly hijack web sessions
chegar
parents: 715
diff changeset
   580
    private String getUserName() {
53473
9366628d727b 8216986: Remove unused code from SocksSocketImpl
michaelm
parents: 52499
diff changeset
   581
        return StaticProperty.userName();
3450
2f08a8bb9b83 6801071: Remote sites can compromise user privacy and possibly hijack web sessions
chegar
parents: 715
diff changeset
   582
    }
54155
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
   583
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
   584
    @Override
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
   585
    void reset() {
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
   586
        throw new InternalError("should not get here");
b5a73f22b2bd 8220493: Prepare Socket/ServerSocket for alternative platform SocketImpl
alanb
parents: 53473
diff changeset
   587
    }
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   588
}