2007-03-21 05:09:52 +01:00
|
|
|
/**
|
|
|
|
* $Revision:$
|
|
|
|
* $Date:$
|
|
|
|
*
|
|
|
|
* Copyright 2003-2007 Jive Software.
|
|
|
|
*
|
|
|
|
* All rights reserved. Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
package org.jivesoftware.smackx.filetransfer;
|
|
|
|
|
|
|
|
import org.jivesoftware.smack.util.Cache;
|
|
|
|
import org.jivesoftware.smack.XMPPException;
|
|
|
|
import org.jivesoftware.smack.PacketCollector;
|
|
|
|
import org.jivesoftware.smack.SmackConfiguration;
|
|
|
|
import org.jivesoftware.smack.XMPPConnection;
|
|
|
|
import org.jivesoftware.smack.filter.PacketIDFilter;
|
|
|
|
import org.jivesoftware.smack.packet.IQ;
|
|
|
|
import org.jivesoftware.smackx.ServiceDiscoveryManager;
|
|
|
|
import org.jivesoftware.smackx.packet.DiscoverItems;
|
|
|
|
import org.jivesoftware.smackx.packet.Bytestream;
|
|
|
|
import org.jivesoftware.smackx.packet.DiscoverInfo;
|
|
|
|
|
|
|
|
import java.net.ServerSocket;
|
|
|
|
import java.net.Socket;
|
|
|
|
import java.net.SocketException;
|
|
|
|
import java.net.SocketTimeoutException;
|
|
|
|
import java.util.*;
|
|
|
|
import java.io.*;
|
|
|
|
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
public class Socks5TransferNegotiatorManager implements FileTransferNegotiatorManager {
|
|
|
|
|
|
|
|
private static final long BLACKLIST_LIFETIME = 60 * 1000 * 120;
|
|
|
|
// locks the proxies during their initialization process
|
|
|
|
private final Object proxyLock = new Object();
|
|
|
|
|
|
|
|
private static ProxyProcess proxyProcess;
|
|
|
|
|
|
|
|
// locks on the proxy process during its initiatilization process
|
|
|
|
private final Object processLock = new Object();
|
|
|
|
|
|
|
|
private final Cache<String, Integer> addressBlacklist
|
|
|
|
= new Cache<String, Integer>(100, BLACKLIST_LIFETIME);
|
|
|
|
|
|
|
|
private XMPPConnection connection;
|
|
|
|
|
|
|
|
private List<String> proxies;
|
|
|
|
|
|
|
|
private List<Bytestream.StreamHost> streamHosts;
|
|
|
|
|
|
|
|
public Socks5TransferNegotiatorManager(XMPPConnection connection) {
|
|
|
|
this.connection = connection;
|
|
|
|
}
|
|
|
|
|
|
|
|
public StreamNegotiator createNegotiator() {
|
|
|
|
return new Socks5TransferNegotiator(this, connection);
|
|
|
|
}
|
|
|
|
|
|
|
|
public void incrementConnectionFailures(String address) {
|
|
|
|
Integer count = addressBlacklist.get(address);
|
|
|
|
if (count == null) {
|
|
|
|
count = 1;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
count += 1;
|
|
|
|
}
|
|
|
|
addressBlacklist.put(address, count);
|
|
|
|
}
|
|
|
|
|
|
|
|
public int getConnectionFailures(String address) {
|
|
|
|
Integer count = addressBlacklist.get(address);
|
|
|
|
return count != null ? count : 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
public ProxyProcess addTransfer() throws IOException {
|
|
|
|
synchronized (processLock) {
|
|
|
|
if (proxyProcess == null) {
|
|
|
|
proxyProcess = new ProxyProcess(new ServerSocket(7777));
|
|
|
|
proxyProcess.start();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
proxyProcess.addTransfer();
|
|
|
|
return proxyProcess;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void removeTransfer() {
|
|
|
|
if (proxyProcess == null) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
proxyProcess.removeTransfer();
|
|
|
|
}
|
|
|
|
|
|
|
|
public Collection<Bytestream.StreamHost> getStreamHosts() {
|
|
|
|
synchronized (proxyLock) {
|
|
|
|
if (proxies == null) {
|
|
|
|
initProxies();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return Collections.unmodifiableCollection(streamHosts);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Checks the service discovery item returned from a server component to verify if it is
|
|
|
|
* a File Transfer proxy or not.
|
|
|
|
*
|
|
|
|
* @param manager the service discovery manager which will be used to query the component
|
|
|
|
* @param item the discovered item on the server relating
|
|
|
|
* @return returns the JID of the proxy if it is a proxy or null if the item is not a proxy.
|
|
|
|
*/
|
|
|
|
private String checkIsProxy(ServiceDiscoveryManager manager, DiscoverItems.Item item) {
|
|
|
|
DiscoverInfo info;
|
|
|
|
try {
|
|
|
|
info = manager.discoverInfo(item.getEntityID());
|
|
|
|
}
|
|
|
|
catch (XMPPException e) {
|
|
|
|
return null;
|
|
|
|
}
|
2009-06-04 13:58:25 +02:00
|
|
|
Iterator<DiscoverInfo.Identity> itx = info.getIdentities();
|
2007-03-21 05:09:52 +01:00
|
|
|
while (itx.hasNext()) {
|
2009-06-04 13:58:25 +02:00
|
|
|
DiscoverInfo.Identity identity = itx.next();
|
2007-03-21 05:09:52 +01:00
|
|
|
if ("proxy".equalsIgnoreCase(identity.getCategory())
|
|
|
|
&& "bytestreams".equalsIgnoreCase(
|
|
|
|
identity.getType())) {
|
|
|
|
return info.getFrom();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
private void initProxies() {
|
|
|
|
proxies = new ArrayList<String>();
|
|
|
|
ServiceDiscoveryManager manager = ServiceDiscoveryManager
|
|
|
|
.getInstanceFor(connection);
|
|
|
|
try {
|
|
|
|
DiscoverItems discoItems = manager.discoverItems(connection.getServiceName());
|
2009-06-04 13:58:25 +02:00
|
|
|
Iterator<DiscoverItems.Item> it = discoItems.getItems();
|
2007-03-21 05:09:52 +01:00
|
|
|
while (it.hasNext()) {
|
2009-06-04 13:58:25 +02:00
|
|
|
DiscoverItems.Item item = it.next();
|
2007-03-21 05:09:52 +01:00
|
|
|
String proxy = checkIsProxy(manager, item);
|
|
|
|
if (proxy != null) {
|
|
|
|
proxies.add(proxy);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
catch (XMPPException e) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (proxies.size() > 0) {
|
|
|
|
initStreamHosts();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Loads streamhost address and ports from the proxies on the local server.
|
|
|
|
*/
|
|
|
|
private void initStreamHosts() {
|
|
|
|
List<Bytestream.StreamHost> streamHosts = new ArrayList<Bytestream.StreamHost>();
|
2009-06-04 13:58:25 +02:00
|
|
|
Iterator<String> it = proxies.iterator();
|
2007-03-21 05:09:52 +01:00
|
|
|
IQ query;
|
|
|
|
PacketCollector collector;
|
|
|
|
Bytestream response;
|
|
|
|
while (it.hasNext()) {
|
2009-06-04 13:58:25 +02:00
|
|
|
String jid = it.next();
|
2007-03-21 05:09:52 +01:00
|
|
|
query = new IQ() {
|
|
|
|
public String getChildElementXML() {
|
|
|
|
return "<query xmlns=\"http://jabber.org/protocol/bytestreams\"/>";
|
|
|
|
}
|
|
|
|
};
|
|
|
|
query.setType(IQ.Type.GET);
|
|
|
|
query.setTo(jid);
|
|
|
|
|
|
|
|
collector = connection.createPacketCollector(new PacketIDFilter(
|
|
|
|
query.getPacketID()));
|
|
|
|
connection.sendPacket(query);
|
|
|
|
|
|
|
|
response = (Bytestream) collector.nextResult(SmackConfiguration
|
|
|
|
.getPacketReplyTimeout());
|
|
|
|
if (response != null) {
|
|
|
|
streamHosts.addAll(response.getStreamHosts());
|
|
|
|
}
|
|
|
|
collector.cancel();
|
|
|
|
}
|
|
|
|
this.streamHosts = streamHosts;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void cleanup() {
|
|
|
|
synchronized (processLock) {
|
|
|
|
if (proxyProcess != null) {
|
|
|
|
proxyProcess.stop();
|
|
|
|
proxyProcess = null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class ProxyProcess implements Runnable {
|
|
|
|
|
|
|
|
private final ServerSocket listeningSocket;
|
|
|
|
|
|
|
|
private final Map<String, Socket> connectionMap = new HashMap<String, Socket>();
|
|
|
|
|
|
|
|
private boolean done = false;
|
|
|
|
|
|
|
|
private Thread thread;
|
|
|
|
private int transfers;
|
|
|
|
|
|
|
|
public void run() {
|
|
|
|
try {
|
|
|
|
try {
|
|
|
|
listeningSocket.setSoTimeout(10000);
|
|
|
|
}
|
|
|
|
catch (SocketException e) {
|
|
|
|
// There was a TCP error, lets print the stack trace
|
|
|
|
e.printStackTrace();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
while (!done) {
|
|
|
|
Socket conn = null;
|
|
|
|
synchronized (ProxyProcess.this) {
|
|
|
|
while (transfers <= 0 && !done) {
|
|
|
|
transfers = -1;
|
|
|
|
try {
|
|
|
|
ProxyProcess.this.wait();
|
|
|
|
}
|
|
|
|
catch (InterruptedException e) {
|
|
|
|
/* Do nothing */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (done) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
synchronized (listeningSocket) {
|
|
|
|
conn = listeningSocket.accept();
|
|
|
|
}
|
|
|
|
if (conn == null) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
String digest = establishSocks5UploadConnection(conn);
|
|
|
|
synchronized (connectionMap) {
|
|
|
|
connectionMap.put(digest, conn);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
catch (SocketTimeoutException e) {
|
|
|
|
/* Do Nothing */
|
|
|
|
}
|
|
|
|
catch (IOException e) {
|
|
|
|
/* Do Nothing */
|
|
|
|
}
|
|
|
|
catch (XMPPException e) {
|
|
|
|
e.printStackTrace();
|
|
|
|
if (conn != null) {
|
|
|
|
try {
|
|
|
|
conn.close();
|
|
|
|
}
|
|
|
|
catch (IOException e1) {
|
|
|
|
/* Do Nothing */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
finally {
|
|
|
|
try {
|
|
|
|
listeningSocket.close();
|
|
|
|
}
|
|
|
|
catch (IOException e) {
|
|
|
|
/* Do Nothing */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Negotiates the Socks 5 bytestream when the local computer is acting as
|
|
|
|
* the proxy.
|
|
|
|
*
|
|
|
|
* @param connection the socket connection with the peer.
|
|
|
|
* @return the SHA-1 digest that is used to uniquely identify the file
|
|
|
|
* transfer.
|
|
|
|
* @throws XMPPException
|
|
|
|
* @throws IOException
|
|
|
|
*/
|
|
|
|
private String establishSocks5UploadConnection(Socket connection) throws XMPPException, IOException {
|
|
|
|
OutputStream out = new DataOutputStream(connection.getOutputStream());
|
|
|
|
InputStream in = new DataInputStream(connection.getInputStream());
|
|
|
|
|
|
|
|
// first byte is version should be 5
|
|
|
|
int b = in.read();
|
|
|
|
if (b != 5) {
|
|
|
|
throw new XMPPException("Only SOCKS5 supported");
|
|
|
|
}
|
|
|
|
|
|
|
|
// second byte number of authentication methods supported
|
|
|
|
b = in.read();
|
|
|
|
int[] auth = new int[b];
|
|
|
|
for (int i = 0; i < b; i++) {
|
|
|
|
auth[i] = in.read();
|
|
|
|
}
|
|
|
|
|
|
|
|
int authMethod = -1;
|
|
|
|
for (int anAuth : auth) {
|
|
|
|
authMethod = (anAuth == 0 ? 0 : -1); // only auth method
|
|
|
|
// 0, no
|
|
|
|
// authentication,
|
|
|
|
// supported
|
|
|
|
if (authMethod == 0) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (authMethod != 0) {
|
|
|
|
throw new XMPPException("Authentication method not supported");
|
|
|
|
}
|
|
|
|
byte[] cmd = new byte[2];
|
|
|
|
cmd[0] = (byte) 0x05;
|
|
|
|
cmd[1] = (byte) 0x00;
|
|
|
|
out.write(cmd);
|
|
|
|
|
|
|
|
String responseDigest = Socks5TransferNegotiator.createIncomingSocks5Message(in);
|
|
|
|
cmd = Socks5TransferNegotiator.createOutgoingSocks5Message(0, responseDigest);
|
|
|
|
|
|
|
|
if (!connection.isConnected()) {
|
|
|
|
throw new XMPPException("Socket closed by remote user");
|
|
|
|
}
|
|
|
|
out.write(cmd);
|
|
|
|
return responseDigest;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public void start() {
|
|
|
|
thread.start();
|
|
|
|
}
|
|
|
|
|
|
|
|
public void stop() {
|
|
|
|
done = true;
|
|
|
|
synchronized (this) {
|
|
|
|
this.notify();
|
|
|
|
}
|
|
|
|
synchronized (listeningSocket) {
|
|
|
|
listeningSocket.notify();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public int getPort() {
|
|
|
|
return listeningSocket.getLocalPort();
|
|
|
|
}
|
|
|
|
|
|
|
|
ProxyProcess(ServerSocket listeningSocket) {
|
|
|
|
thread = new Thread(this, "File Transfer Connection Listener");
|
|
|
|
this.listeningSocket = listeningSocket;
|
|
|
|
}
|
|
|
|
|
|
|
|
public Socket getSocket(String digest) {
|
|
|
|
synchronized (connectionMap) {
|
|
|
|
return connectionMap.get(digest);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public void addTransfer() {
|
|
|
|
synchronized (this) {
|
|
|
|
if (transfers == -1) {
|
|
|
|
transfers = 1;
|
|
|
|
this.notify();
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
transfers++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public void removeTransfer() {
|
|
|
|
synchronized (this) {
|
|
|
|
transfers--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|