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