/*
* Created on 1 Nov 2006
* Created by Paul Gardner
* Copyright (C) 2006 Aelitis, All Rights Reserved.
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
*
* AELITIS, SAS au capital de 63.529,40 euros
* 8 Allee Lenotre, La Grille Royale, 78600 Le Mesnil le Roi, France.
*
*/
package com.aelitis.azureus.core.networkmanager.admin.impl;
import java.io.IOException;
import java.net.InetAddress;
import java.net.InetSocketAddress;
import java.nio.ByteBuffer;
import java.nio.channels.SocketChannel;
import java.util.StringTokenizer;
import org.gudy.azureus2.core3.util.AESemaphore;
import org.gudy.azureus2.core3.util.Debug;
import org.gudy.azureus2.core3.util.SystemTime;
import com.aelitis.azureus.core.networkmanager.VirtualChannelSelector;
import com.aelitis.azureus.core.networkmanager.admin.NetworkAdminException;
import com.aelitis.azureus.core.networkmanager.admin.NetworkAdminHTTPProxy;
import com.aelitis.azureus.core.networkmanager.impl.tcp.TCPConnectionManager;
import com.aelitis.azureus.core.networkmanager.impl.tcp.ProtocolEndpointTCP;
import com.aelitis.azureus.core.networkmanager.impl.tcp.TCPNetworkManager;
import com.aelitis.azureus.core.networkmanager.impl.tcp.TCPTransportHelperFilterFactory;
import com.aelitis.azureus.core.networkmanager.impl.tcp.TCPTransportImpl;
import com.aelitis.azureus.core.versioncheck.VersionCheckClient;
public class
NetworkAdminHTTPProxyImpl
implements NetworkAdminHTTPProxy
{
private static final String NL = "\015\012";
private final String TARGET_HOST = VersionCheckClient.HTTP_SERVER_ADDRESS_V4;
private final int TARGET_PORT = VersionCheckClient.HTTP_SERVER_PORT;
private String http_host;
private String http_port;
private String https_host;
private String https_port;
private String user;
private String password;
private String[] non_proxy_hosts;
protected
NetworkAdminHTTPProxyImpl()
{
http_host = System.getProperty("http.proxyHost", "" ).trim();
http_port = System.getProperty("http.proxyPort", "" ).trim();
https_host = System.getProperty("https.proxyHost", "" ).trim();
https_port = System.getProperty("https.proxyPort", "" ).trim();
user = System.getProperty("http.proxyUser", "" ).trim();
password = System.getProperty("http.proxyPassword", "" ).trim();
String nph = System.getProperty("http.nonProxyHosts", "" ).trim();
StringTokenizer tok = new StringTokenizer( nph, "|" );
non_proxy_hosts = new String[tok.countTokens()];
int pos = 0;
while( tok.hasMoreTokens()){
non_proxy_hosts[pos++] = tok.nextToken();
}
}
public String
getName()
{
return( http_host + ":" + http_port );
}
protected boolean
isConfigured()
{
return( http_host.length() > 0 || https_host.length() > 0 );
}
public String
getHTTPHost()
{
return( http_host );
}
public String
getHTTPPort()
{
return( http_port );
}
public String
getHTTPSHost()
{
return( https_host );
}
public String
getHTTPSPort()
{
return( https_port );
}
public String
getUser()
{
return( user );
}
public String[]
getNonProxyHosts()
{
return( non_proxy_hosts );
}
public Details
getDetails()
throws NetworkAdminException
{
final int RES_CONNECT_FAILED = 0;
final int RES_PROXY_FAILED = 1;
final int RES_OK = 3;
final AESemaphore sem = new AESemaphore( "NetworkAdminSocksProxy:test" );
final int[] result = { RES_CONNECT_FAILED };
final NetworkAdminException[] error = { null };
final ProxyDetails[] details = {null};
try{
InetSocketAddress socks_address =
new InetSocketAddress( InetAddress.getByName( http_host ), Integer.parseInt(http_port));
final InetSocketAddress target_address = new InetSocketAddress( TARGET_HOST, TARGET_PORT );
TCPConnectionManager.ConnectListener connect_listener =
new TCPConnectionManager.ConnectListener()
{
public void
connectAttemptStarted()
{
}
public void
connectSuccess(
SocketChannel channel )
{
final TCPTransportImpl transport =
new TCPTransportImpl(
new ProtocolEndpointTCP( target_address ), false, false, null );
transport.setFilter( TCPTransportHelperFilterFactory.createTransparentFilter( channel ));
final long start_time = SystemTime.getCurrentTime();
try{
String get_str = VersionCheckClient.getSingleton().getHTTPGetString( true, false );
ByteBuffer request = ByteBuffer.wrap( get_str.getBytes());
while( request.hasRemaining()){
if ( transport.write( new ByteBuffer[]{ request }, 0, 1 ) < 1 ) {
if( SystemTime.getCurrentTime() - start_time > 30*1000 ) {
String error = "proxy handshake message send timed out after 30sec";
Debug.out( error );
throw new IOException( error );
}
try{
Thread.sleep( 50 );
}catch( Throwable t ){
t.printStackTrace();
}
}
}
TCPNetworkManager.getSingleton().getReadSelector().register(
transport.getSocketChannel(),
new VirtualChannelSelector.VirtualSelectorListener()
{
private byte[] reply_buffer = new byte[8192];
private ByteBuffer reply = ByteBuffer.wrap( reply_buffer );
public boolean
selectSuccess(
VirtualChannelSelector selector,
SocketChannel sc,
Object attachment )
{
try{
if( SystemTime.getCurrentTime() - start_time > 30*1000 ){
throw( new Exception( "Timeout" ));
}
long len = transport.read( new ByteBuffer[]{ reply }, 0, 1 );
if ( len <= 0 ){
return( false );
}
String str = new String( reply_buffer, 0, reply.position());
if ( str.indexOf( NL + NL ) != -1 ){
System.out.println( str );
String server_name = "unknown";
String auth = "none";
String response = "unknown";
StringTokenizer tok = new StringTokenizer( str, "\n" );
int line_num = 0;
while( tok.hasMoreTokens()){
String token = tok.nextToken().trim();
if ( token.length() == 0 ){
continue;
}
line_num++;
if ( line_num == 1 ){
int pos = token.indexOf(' ');
if ( pos != -1 ){
response = token.substring( pos + 1 ).trim();
}
}else{
int pos = token.indexOf(':');
if ( pos != -1 ){
String lhs = token.substring( 0, pos ).trim().toLowerCase();
String rhs = token.substring( pos+1 ).trim();
if ( lhs.equals( "server" )){
if ( !response.startsWith( "200" )){
server_name = rhs;
}
}else if ( lhs.equals( "via" )){
server_name = rhs;
int p = server_name.indexOf(' ');
if ( p != -1 ){
server_name = server_name.substring( p+1 ).trim();
}
}else if ( lhs.equals( "proxy-authenticate" )){
auth = rhs;
}
}
}
}
details[0] =
new ProxyDetails(
server_name,
response,
auth );
transport.close( "Done" );
result[0] = RES_OK;
sem.release();
}else{
TCPNetworkManager.getSingleton().getReadSelector().resumeSelects( transport.getSocketChannel() );
}
return( true );
}
catch( Throwable t ) {
return false;
}
}
public void
selectFailure(
VirtualChannelSelector selector,
SocketChannel sc,
Object attachment,
Throwable msg )
{
result[0] = RES_PROXY_FAILED;
error[0] = new NetworkAdminException( "Proxy error", msg );
transport.close( "Proxy error" );
sem.release();
}
},
null );
}catch( Throwable t ) {
result[0] = RES_PROXY_FAILED;
error[0] = new NetworkAdminException( "Proxy connect failed", t );
sem.release(); }
}
public void
connectFailure(
Throwable failure_msg )
{
result[0] = RES_CONNECT_FAILED;
error[0] = new NetworkAdminException( "Connect failed", failure_msg );
sem.release();
}
};
TCPNetworkManager.getSingleton().getConnectDisconnectManager().requestNewConnection(
socks_address, connect_listener );
}catch( Throwable e ){
result[0] = RES_CONNECT_FAILED;
error[0] = new NetworkAdminException( "Connect failed", e );
sem.release();
}
if ( !sem.reserve(10000)){
result[0] = RES_CONNECT_FAILED;
error[0] = new NetworkAdminException( "Connect timeout" );
}
if ( result[0] == RES_OK ){
return( details[0] );
}
throw( error[0] );
}
protected class
ProxyDetails
implements Details
{
private String name;
private String response;
private String auth_type;
protected
ProxyDetails(
String _name,
String _response,
String _auth_type )
{
name = _name;
response = _response;
auth_type = _auth_type;
}
public String
getServerName()
{
return( name );
}
public String
getResponse()
{
return( response );
}
public String
getAuthenticationType()
{
return( auth_type );
}
}
}
|