/**
* $RCSfile: ICEResolver.java,v $
* $Revision: 1.1 $
* $Date: 2007/07/02 17:41:07 $
*
* Copyright 2003-2005 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.jingle.nat;
import java.net.InetAddress;
import java.net.NetworkInterface;
import java.net.SocketException;
import java.net.UnknownHostException;
import java.util.Enumeration;
import java.util.HashMap;
import java.util.Map;
import java.util.Random;
import org.jivesoftware.smack.Connection;
import org.jivesoftware.smack.XMPPException;
import org.jivesoftware.smackx.jingle.JingleSession;
import org.jivesoftware.smackx.jingle.SmackLogger;
import de.javawi.jstun.test.demo.ice.Candidate;
import de.javawi.jstun.test.demo.ice.ICENegociator;
import de.javawi.jstun.util.UtilityException;
/**
* ICE Resolver for Jingle transport method that results in sending data between
* two entities using the Interactive Connectivity Establishment (ICE)
* methodology. (XEP-0176) The goal of this resolver is to make possible to
* establish and manage out-of-band connections between two XMPP entities, even
* if they are behind Network Address Translators (NATs) or firewalls. To use
* this resolver you must have a STUN Server and be in a non STUN blocked
* network. Or use a XMPP server with public IP detection Service.
*
* @author Thiago Camargo
*/
public class ICEResolver extends TransportResolver {
private static final SmackLogger LOGGER = SmackLogger
.getLogger(ICEResolver.class);
Connection connection;
Random random = new Random();
long sid;
String server;
int port;
static Map<String, ICENegociator> negociatorsMap = new HashMap<String, ICENegociator>();
// ICENegociator iceNegociator = null;
public ICEResolver(Connection connection, String server, int port) {
super();
this.connection = connection;
this.server = server;
this.port = port;
setType(Type.ice);
}
@Override
public void cancel() throws XMPPException {
}
@Override
public void initialize() throws XMPPException {
if (!isResolving() && !isResolved()) {
LOGGER.debug("Initialized");
// Negotiation with a STUN server for a set of interfaces is quite
// slow, but the results
// never change over then instance of a JVM. To increase connection
// performance considerably
// we now cache established/initialized negotiators for each STUN
// server, so that subsequent uses
// of the STUN server are much, much faster.
if (negociatorsMap.get(server) == null) {
final ICENegociator iceNegociator = new ICENegociator(server,
port, (short) 1);
negociatorsMap.put(server, iceNegociator);
// gather candidates
iceNegociator.gatherCandidateAddresses();
// priorize candidates
iceNegociator.prioritizeCandidates();
}
}
setInitialized();
}
/**
* Resolve the IP and obtain a valid transport method.
*/
@Override
public synchronized void resolve(JingleSession session)
throws XMPPException {
setResolveInit();
for (final TransportCandidate candidate : getCandidatesList()) {
if (candidate instanceof ICECandidate) {
final ICECandidate iceCandidate = (ICECandidate) candidate;
iceCandidate.removeCandidateEcho();
}
}
clear();
// Create a transport candidate for each ICE negotiator candidate we
// have.
final ICENegociator iceNegociator = negociatorsMap.get(server);
for (final Candidate candidate : iceNegociator.getSortedCandidates()) {
try {
final Candidate.CandidateType type = candidate
.getCandidateType();
ICECandidate.Type iceType = ICECandidate.Type.local;
if (type.equals(Candidate.CandidateType.ServerReflexive)) {
iceType = ICECandidate.Type.srflx;
} else if (type.equals(Candidate.CandidateType.PeerReflexive)) {
iceType = ICECandidate.Type.prflx;
} else if (type.equals(Candidate.CandidateType.Relayed)) {
iceType = ICECandidate.Type.relay;
} else {
iceType = ICECandidate.Type.host;
}
// JBW/GW - 17JUL08: Figure out the zero-based NIC number for
// this candidate.
short nicNum = 0;
try {
final Enumeration<NetworkInterface> nics = NetworkInterface
.getNetworkInterfaces();
short i = 0;
final NetworkInterface nic = NetworkInterface
.getByInetAddress(candidate.getAddress()
.getInetAddress());
while (nics.hasMoreElements()) {
final NetworkInterface checkNIC = nics.nextElement();
if (checkNIC.equals(nic)) {
nicNum = i;
break;
}
i++;
}
} catch (final SocketException e1) {
// TODO Auto-generated catch block
e1.printStackTrace();
}
final TransportCandidate transportCandidate = new ICECandidate(
candidate.getAddress().getInetAddress()
.getHostAddress(), 1, nicNum,
String.valueOf(Math.abs(random.nextLong())),
candidate.getPort(), "1", candidate.getPriority(),
iceType);
transportCandidate.setLocalIp(candidate.getBase().getAddress()
.getInetAddress().getHostAddress());
transportCandidate.setPort(getFreePort());
try {
transportCandidate.addCandidateEcho(session);
} catch (final SocketException e) {
e.printStackTrace();
}
addCandidate(transportCandidate);
LOGGER.debug("Candidate addr: "
+ candidate.getAddress().getInetAddress() + "|"
+ candidate.getBase().getAddress().getInetAddress()
+ " Priority:" + candidate.getPriority());
} catch (final UtilityException e) {
e.printStackTrace();
} catch (final UnknownHostException e) {
e.printStackTrace();
}
}
// Get a Relay Candidate from XMPP Server
if (RTPBridge.serviceAvailable(connection)) {
// try {
String localIp;
int network;
// JBW/GW - 17JUL08: ICENegotiator.getPublicCandidate() always
// returned null in JSTUN 1.7.0, and now the API doesn't exist in
// JSTUN 1.7.1
// if (iceNegociator.getPublicCandidate() != null) {
// localIp =
// iceNegociator.getPublicCandidate().getBase().getAddress().getInetAddress().getHostAddress();
// network = iceNegociator.getPublicCandidate().getNetwork();
// }
// else {
{
localIp = BridgedResolver.getLocalHost();
network = 0;
}
sid = Math.abs(random.nextLong());
final RTPBridge rtpBridge = RTPBridge.getRTPBridge(connection,
String.valueOf(sid));
final TransportCandidate localCandidate = new ICECandidate(
rtpBridge.getIp(), 1, network, String.valueOf(Math
.abs(random.nextLong())), rtpBridge.getPortA(),
"1", 0, ICECandidate.Type.relay);
localCandidate.setLocalIp(localIp);
final TransportCandidate remoteCandidate = new ICECandidate(
rtpBridge.getIp(), 1, network, String.valueOf(Math
.abs(random.nextLong())), rtpBridge.getPortB(),
"1", 0, ICECandidate.Type.relay);
remoteCandidate.setLocalIp(localIp);
localCandidate.setSymmetric(remoteCandidate);
remoteCandidate.setSymmetric(localCandidate);
localCandidate.setPassword(rtpBridge.getPass());
remoteCandidate.setPassword(rtpBridge.getPass());
localCandidate.setSessionId(rtpBridge.getSid());
remoteCandidate.setSessionId(rtpBridge.getSid());
localCandidate.setConnection(connection);
remoteCandidate.setConnection(connection);
addCandidate(localCandidate);
// }
// catch (UtilityException e) {
// e.printStackTrace();
// }
// catch (UnknownHostException e) {
// e.printStackTrace();
// }
// Get Public Candidate From XMPP Server
// JBW/GW - 17JUL08 - ICENegotiator.getPublicCandidate() always
// returned null in JSTUN 1.7.0, and now it doesn't exist in JSTUN
// 1.7.1
// if (iceNegociator.getPublicCandidate() == null) {
if (true) {
final String publicIp = RTPBridge.getPublicIP(connection);
if (publicIp != null && !publicIp.equals("")) {
Enumeration ifaces = null;
try {
ifaces = NetworkInterface.getNetworkInterfaces();
} catch (final SocketException e) {
e.printStackTrace();
}
// If detect this address in local machine, don't use it.
boolean found = false;
while (ifaces.hasMoreElements() && !false) {
final NetworkInterface iface = (NetworkInterface) ifaces
.nextElement();
final Enumeration iaddresses = iface.getInetAddresses();
while (iaddresses.hasMoreElements()) {
final InetAddress iaddress = (InetAddress) iaddresses
.nextElement();
if (iaddress.getHostAddress().indexOf(publicIp) > -1) {
found = true;
break;
}
}
}
if (!found) {
try {
final TransportCandidate publicCandidate = new ICECandidate(
publicIp, 1, 0, String.valueOf(Math
.abs(random.nextLong())),
getFreePort(), "1", 0,
ICECandidate.Type.srflx);
publicCandidate.setLocalIp(InetAddress
.getLocalHost().getHostAddress());
try {
publicCandidate.addCandidateEcho(session);
} catch (final SocketException e) {
e.printStackTrace();
}
addCandidate(publicCandidate);
} catch (final UnknownHostException e) {
e.printStackTrace();
}
}
}
}
}
setResolveEnd();
}
}