/*
 * Copyright (c) 2011 Tobias Markmann
 * Licensed under the simplified BSD license.
 * See Documentation/Licenses/BSD-simplified.txt for more information.
 */

#include "DefaultRemoteJingleTransportCandidateSelector.h"

#include <boost/smart_ptr/make_shared.hpp>
#include <boost/bind.hpp>

#include <Swiften/Base/Log.h>
#include <Swiften/Base/boost_bsignals.h>
#include <Swiften/Base/foreach.h>
#include <Swiften/Elements/JingleS5BTransportPayload.h>
#include <Swiften/Network/ConnectionFactory.h>
#include <Swiften/FileTransfer/SOCKS5BytestreamRegistry.h>

namespace Swift {

DefaultRemoteJingleTransportCandidateSelector::DefaultRemoteJingleTransportCandidateSelector(ConnectionFactory* connectionFactory, TimerFactory* timerFactory) : connectionFactory(connectionFactory), timerFactory(timerFactory) {
}

DefaultRemoteJingleTransportCandidateSelector::~DefaultRemoteJingleTransportCandidateSelector() {
}

void DefaultRemoteJingleTransportCandidateSelector::addRemoteTransportCandidates(JingleTransportPayload::ref transportPayload) {
	JingleS5BTransportPayload::ref s5bPayload;
	transportSID = transportPayload->getSessionID();
	if ((s5bPayload = boost::dynamic_pointer_cast<JingleS5BTransportPayload>(transportPayload))) {
		foreach(JingleS5BTransportPayload::Candidate c,  s5bPayload->getCandidates()) {
			candidates.push(c);
		}
	}
}

void DefaultRemoteJingleTransportCandidateSelector::selectCandidate() {
	tryNextCandidate(true);
}

void DefaultRemoteJingleTransportCandidateSelector::tryNextCandidate(bool error) {
	if (error) {
		if (s5bSession) {
			SWIFT_LOG(debug) << "failed to connect" << std::endl;
		}
		if (candidates.empty()) {
			// failed to connect to any of the candidates
			// issue an error
			SWIFT_LOG(debug) << "out of candidates )=" << std::endl;
			JingleS5BTransportPayload::ref failed = boost::make_shared<JingleS5BTransportPayload>();
			failed->setCandidateError(true);
			failed->setSessionID(transportSID);
			onRemoteTransportCandidateSelectFinished(failed);
		} else {
			lastCandidate = candidates.top();
			// only try direct or assisted for now
			if (lastCandidate.type == JingleS5BTransportPayload::Candidate::DirectType ||
				lastCandidate.type == JingleS5BTransportPayload::Candidate::AssistedType || lastCandidate.type == JingleS5BTransportPayload::Candidate::ProxyType ) {
				// create connection
				connection = connectionFactory->createConnection();
				s5bSession = boost::make_shared<SOCKS5BytestreamClientSession>(connection, lastCandidate.hostPort, SOCKS5BytestreamRegistry::getHostname(transportSID, requester, target), timerFactory);

				// bind onReady to this method
				s5bSession->onSessionReady.connect(boost::bind(&DefaultRemoteJingleTransportCandidateSelector::tryNextCandidate, this, _1));

				std::string candidateType;
				if (lastCandidate.type == JingleS5BTransportPayload::Candidate::DirectType) {
					candidateType = "direct";
				} else if (lastCandidate.type == JingleS5BTransportPayload::Candidate::AssistedType) {
					candidateType = "assisted";
				} else if (lastCandidate.type == JingleS5BTransportPayload::Candidate::ProxyType) {
					candidateType = "proxy";
				}

				// initiate connect
				SWIFT_LOG(debug) << "try to connect to candidate of type " << candidateType << " : " << lastCandidate.hostPort.toString() << std::endl;
				s5bSession->start();

				// that's it. we're gonna be called back
				candidates.pop();
			} else {
				s5bSession.reset();
				candidates.pop();
				tryNextCandidate(true);
			}
		}
	} else {
		// we have a working connection, hooray
		JingleS5BTransportPayload::ref success = boost::make_shared<JingleS5BTransportPayload>();
		success->setCandidateUsed(lastCandidate.cid);
		success->setSessionID(transportSID);
		onRemoteTransportCandidateSelectFinished(success);
	}
}

void DefaultRemoteJingleTransportCandidateSelector::setMinimumPriority(int priority) {
	SWIFT_LOG(debug) << "priority: " << priority << std::endl;
}

void DefaultRemoteJingleTransportCandidateSelector::setRequesterTargtet(const JID& requester, const JID& target) {
	this->requester = requester;
	this->target = target;
}

SOCKS5BytestreamClientSession::ref DefaultRemoteJingleTransportCandidateSelector::getS5BSession() {
	return s5bSession;
}

bool DefaultRemoteJingleTransportCandidateSelector::isActualCandidate(JingleTransportPayload::ref /* transportPayload */) {
	return false;
}

int DefaultRemoteJingleTransportCandidateSelector::getPriority(JingleTransportPayload::ref /* transportPayload */) {
	return 0;
}

JingleTransport::ref DefaultRemoteJingleTransportCandidateSelector::selectTransport(JingleTransportPayload::ref /* transportPayload */) {
	return JingleTransport::ref();
}

}