package org.jivesoftware.smack.proxy; import java.io.BufferedReader; import java.io.IOException; import java.io.InputStream; import java.io.StringReader; import java.net.HttpURLConnection; import java.net.InetAddress; import java.net.Socket; import java.net.UnknownHostException; import java.util.regex.Matcher; import java.util.regex.Pattern; import javax.net.SocketFactory; import org.jivesoftware.smack.util.Base64; /** * Http Proxy Socket Factory which returns socket connected to Http Proxy * * @author Atul Aggarwal */ class HTTPProxySocketFactory extends SocketFactory { private final ProxyInfo proxy; private static final Pattern RESPONSE_PATTERN = Pattern .compile("HTTP/\\S+\\s(\\d+)\\s(.*)\\s*"); public HTTPProxySocketFactory(ProxyInfo proxy) { this.proxy = proxy; } @Override public Socket createSocket(InetAddress host, int port) throws IOException { return httpProxifiedSocket(host.getHostAddress(), port); } @Override public Socket createSocket(InetAddress address, int port, InetAddress localAddress, int localPort) throws IOException { return httpProxifiedSocket(address.getHostAddress(), port); } @Override public Socket createSocket(String host, int port) throws IOException, UnknownHostException { return httpProxifiedSocket(host, port); } @Override public Socket createSocket(String host, int port, InetAddress localHost, int localPort) throws IOException, UnknownHostException { return httpProxifiedSocket(host, port); } private Socket httpProxifiedSocket(String host, int port) throws IOException { final String proxyhost = proxy.getProxyAddress(); final int proxyPort = proxy.getProxyPort(); final Socket socket = new Socket(proxyhost, proxyPort); final String hostport = "CONNECT " + host + ":" + port; String proxyLine; final String username = proxy.getProxyUsername(); if (username == null) { proxyLine = ""; } else { final String password = proxy.getProxyPassword(); proxyLine = "\r\nProxy-Authorization: Basic " + new String(Base64.encodeBytes((username + ":" + password) .getBytes("UTF-8"))); } socket.getOutputStream() .write((hostport + " HTTP/1.1\r\nHost: " + hostport + proxyLine + "\r\n\r\n") .getBytes("UTF-8")); final InputStream in = socket.getInputStream(); final StringBuilder got = new StringBuilder(100); int nlchars = 0; while (true) { final char c = (char) in.read(); got.append(c); if (got.length() > 1024) { throw new ProxyException(ProxyInfo.ProxyType.HTTP, "Recieved " + "header of >1024 characters from " + proxyhost + ", cancelling connection"); } if (c == -1) { throw new ProxyException(ProxyInfo.ProxyType.HTTP); } if ((nlchars == 0 || nlchars == 2) && c == '\r') { nlchars++; } else if ((nlchars == 1 || nlchars == 3) && c == '\n') { nlchars++; } else { nlchars = 0; } if (nlchars == 4) { break; } } if (nlchars != 4) { throw new ProxyException(ProxyInfo.ProxyType.HTTP, "Never " + "received blank line from " + proxyhost + ", cancelling connection"); } final String gotstr = got.toString(); final BufferedReader br = new BufferedReader(new StringReader(gotstr)); final String response = br.readLine(); if (response == null) { throw new ProxyException(ProxyInfo.ProxyType.HTTP, "Empty proxy " + "response from " + proxyhost + ", cancelling"); } final Matcher m = RESPONSE_PATTERN.matcher(response); if (!m.matches()) { throw new ProxyException(ProxyInfo.ProxyType.HTTP, "Unexpected " + "proxy response from " + proxyhost + ": " + response); } final int code = Integer.parseInt(m.group(1)); if (code != HttpURLConnection.HTTP_OK) { throw new ProxyException(ProxyInfo.ProxyType.HTTP); } return socket; } }