/*
 * synergy -- mouse and keyboard sharing utility
 * Copyright (C) 2002 Chris Schoeneman
 * 
 * This package is free software; you can redistribute it and/or
 * modify it under the terms of the GNU General Public License
 * found in the file COPYING that should have accompanied this file.
 * 
 * This package 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.
 */

#include "CNetworkAddress.h"
#include "XSocket.h"
#include "CArch.h"
#include "XArch.h"
#include <cstdlib>

//
// CNetworkAddress
//

// name re-resolution adapted from a patch by Brent Priddy.

CNetworkAddress::CNetworkAddress() :
	m_address(NULL),
	m_hostname(),
	m_port(0)
{
	// note -- make no calls to CNetwork socket interface here;
	// we're often called prior to CNetwork::init().
}

CNetworkAddress::CNetworkAddress(int port) :
	m_address(NULL),
	m_hostname(),
	m_port(port)
{
	checkPort();
	m_address = ARCH->newAnyAddr(IArchNetwork::kINET);
	ARCH->setAddrPort(m_address, m_port);
}

CNetworkAddress::CNetworkAddress(const CNetworkAddress& addr) :
	m_address(addr.m_address != NULL ? ARCH->copyAddr(addr.m_address) : NULL),
	m_hostname(addr.m_hostname),
	m_port(addr.m_port)
{
	// do nothing
}

CNetworkAddress::CNetworkAddress(const CString& hostname, int port) :
	m_address(NULL),
	m_hostname(hostname),
	m_port(port)
{
	// check for port suffix
	CString::size_type i = m_hostname.rfind(':');
	if (i != CString::npos && i + 1 < m_hostname.size()) {
		// found a colon.  see if it looks like an IPv6 address.
		bool colonNotation = false;
		bool dotNotation   = false;
		bool doubleColon   = false;
		for (CString::size_type j = 0; j < i; ++j) {
			if (m_hostname[j] == ':') {
				colonNotation = true;
				dotNotation   = false;
				if (m_hostname[j + 1] == ':') {
					doubleColon = true;
				}
			}
			else if (m_hostname[j] == '.' && colonNotation) {
				dotNotation = true;
			}
		}

		// port suffix is ambiguous with IPv6 notation if there's
		// a double colon and the end of the address is not in dot
		// notation.  in that case we assume it's not a port suffix.
		// the user can replace the double colon with zeros to
		// disambiguate.
		if ((!doubleColon || dotNotation) || !colonNotation) {
			// parse port from hostname
			char* end;
			const char* chostname = m_hostname.c_str();
			long suffixPort = strtol(chostname + i + 1, &end, 10);
			if (end == chostname + i + 1 || *end != '\0') {
				throw XSocketAddress(XSocketAddress::kBadPort,
											m_hostname, m_port);
			}

			// trim port from hostname
			m_hostname.erase(i);

			// save port
			m_port = static_cast<int>(suffixPort);
		}
	}

	// check port number
	checkPort();
}

CNetworkAddress::~CNetworkAddress()
{
	if (m_address != NULL) {
		ARCH->closeAddr(m_address);
	}
}

CNetworkAddress&
CNetworkAddress::operator=(const CNetworkAddress& addr)
{
	CArchNetAddress newAddr = NULL;
	if (addr.m_address != NULL) {
		newAddr = ARCH->copyAddr(addr.m_address);
	}
	if (m_address != NULL) {
		ARCH->closeAddr(m_address);
	}
	m_address  = newAddr;
	m_hostname = addr.m_hostname;
	m_port     = addr.m_port;
	return *this;
}

void
CNetworkAddress::resolve()
{
	// discard previous address
	if (m_address != NULL) {
		ARCH->closeAddr(m_address);
		m_address = NULL;
	}

	try {
		// if hostname is empty then use wildcard address otherwise look
		// up the name.
		if (m_hostname.empty()) {
			m_address = ARCH->newAnyAddr(IArchNetwork::kINET);
		}
		else {
			m_address = ARCH->nameToAddr(m_hostname);
		}
	}
	catch (XArchNetworkNameUnknown&) {
		throw XSocketAddress(XSocketAddress::kNotFound, m_hostname, m_port);
	}
	catch (XArchNetworkNameNoAddress&) {
		throw XSocketAddress(XSocketAddress::kNoAddress, m_hostname, m_port);
	}
	catch (XArchNetworkNameUnsupported&) {
		throw XSocketAddress(XSocketAddress::kUnsupported, m_hostname, m_port);
	}
	catch (XArchNetworkName&) {
		throw XSocketAddress(XSocketAddress::kUnknown, m_hostname, m_port);
	}

	// set port in address
	ARCH->setAddrPort(m_address, m_port);
}

bool
CNetworkAddress::operator==(const CNetworkAddress& addr) const
{
	return ARCH->isEqualAddr(m_address, addr.m_address);
}

bool
CNetworkAddress::operator!=(const CNetworkAddress& addr) const
{
	return !operator==(addr);
}

bool
CNetworkAddress::isValid() const
{
	return (m_address != NULL);
}

const CArchNetAddress&
CNetworkAddress::getAddress() const
{
	return m_address;
}

int
CNetworkAddress::getPort() const
{
	return m_port;
}

CString
CNetworkAddress::getHostname() const
{
	return m_hostname;
}

void
CNetworkAddress::checkPort()
{
	// check port number
	if (m_port <= 0 || m_port > 65535) {
		throw XSocketAddress(XSocketAddress::kBadPort, m_hostname, m_port);
	}
}