mirror of
https://github.com/deskflow/deskflow.git
synced 2026-07-01 21:02:39 +08:00
211 lines
5.8 KiB
C++
211 lines
5.8 KiB
C++
/*
|
|
* Deskflow -- mouse and keyboard sharing utility
|
|
* SPDX-FileCopyrightText: (C) 2012 - 2016 Symless Ltd.
|
|
* SPDX-FileCopyrightText: (C) 2002 Chris Schoeneman
|
|
* SPDX-License-Identifier: GPL-2.0-only WITH LicenseRef-OpenSSL-Exception
|
|
*/
|
|
|
|
#include "net/NetworkAddress.h"
|
|
|
|
#include "arch/Arch.h"
|
|
#include "arch/XArch.h"
|
|
#include "net/XSocket.h"
|
|
|
|
#include <algorithm>
|
|
#include <cstdlib>
|
|
|
|
//
|
|
// NetworkAddress
|
|
//
|
|
|
|
// name re-resolution adapted from a patch by Brent Priddy.
|
|
|
|
NetworkAddress::NetworkAddress(int port) : m_port(port)
|
|
{
|
|
checkPort();
|
|
m_address = ARCH->newAnyAddr(IArchNetwork::AddressFamily::INet);
|
|
ARCH->setAddrPort(m_address, m_port);
|
|
}
|
|
|
|
NetworkAddress::NetworkAddress(const NetworkAddress &addr) : m_hostname(addr.m_hostname), m_port(addr.m_port)
|
|
{
|
|
*this = addr;
|
|
}
|
|
|
|
NetworkAddress::NetworkAddress(const std::string &hostname, int port) : m_hostname(hostname), m_port(port)
|
|
{
|
|
// detect internet protocol version with colon count
|
|
auto isColon = [](char c) { return c == ':'; };
|
|
|
|
if (auto colonCount = std::count_if(m_hostname.begin(), m_hostname.end(), isColon); colonCount == 1) {
|
|
// ipv4 with port part
|
|
auto hostIt = m_hostname.find(':');
|
|
try {
|
|
m_port = std::stoi(m_hostname.substr(hostIt + 1));
|
|
} catch (...) {
|
|
throw XSocketAddress(XSocketAddress::SocketError::BadPort, m_hostname, m_port);
|
|
}
|
|
|
|
auto endHostnameIt = static_cast<int>(hostIt);
|
|
m_hostname = m_hostname.substr(0, endHostnameIt > 0 ? endHostnameIt : 0);
|
|
} else if (colonCount > 1) {
|
|
// ipv6 part
|
|
if (m_hostname[0] == '[') {
|
|
// ipv6 with port part
|
|
std::string portDelimeter = "]:";
|
|
auto hostIt = m_hostname.find(portDelimeter);
|
|
|
|
// bad syntax of ipv6 with port
|
|
if (hostIt == std::string::npos) {
|
|
throw XSocketAddress(XSocketAddress::SocketError::Unknown, m_hostname, m_port);
|
|
}
|
|
|
|
auto portSuffix = m_hostname.substr(hostIt + portDelimeter.size());
|
|
// port is implied but omitted
|
|
if (portSuffix.empty()) {
|
|
throw XSocketAddress(XSocketAddress::SocketError::BadPort, m_hostname, m_port);
|
|
}
|
|
try {
|
|
m_port = std::stoi(portSuffix);
|
|
} catch (...) {
|
|
// port is not a number
|
|
throw XSocketAddress(XSocketAddress::SocketError::BadPort, m_hostname, m_port);
|
|
}
|
|
|
|
auto endHostnameIt = static_cast<int>(hostIt) - 1;
|
|
m_hostname = m_hostname.substr(1, endHostnameIt > 0 ? endHostnameIt : 0);
|
|
}
|
|
|
|
// ensure that ipv6 link-local adress ended with scope id
|
|
if (m_hostname.rfind("fe80:", 0) == 0 && m_hostname.find('%') == std::string::npos) {
|
|
throw XSocketAddress(XSocketAddress::SocketError::Unknown, m_hostname, m_port);
|
|
}
|
|
}
|
|
|
|
// check port number
|
|
checkPort();
|
|
}
|
|
|
|
NetworkAddress::~NetworkAddress()
|
|
{
|
|
if (m_address != nullptr) {
|
|
ARCH->closeAddr(m_address);
|
|
m_address = nullptr;
|
|
}
|
|
}
|
|
|
|
NetworkAddress &NetworkAddress::operator=(const NetworkAddress &addr)
|
|
{
|
|
if (m_address != nullptr) {
|
|
ARCH->closeAddr(m_address);
|
|
m_address = nullptr;
|
|
}
|
|
|
|
ArchNetAddress newAddr = nullptr;
|
|
if (addr.m_address != nullptr) {
|
|
newAddr = ARCH->copyAddr(addr.m_address);
|
|
}
|
|
m_address = newAddr;
|
|
|
|
m_hostname = addr.m_hostname;
|
|
m_port = addr.m_port;
|
|
return *this;
|
|
}
|
|
|
|
size_t NetworkAddress::resolve(size_t index)
|
|
{
|
|
size_t resolvedAddressesCount = 0;
|
|
// discard previous address
|
|
if (m_address != nullptr) {
|
|
ARCH->closeAddr(m_address);
|
|
m_address = nullptr;
|
|
}
|
|
|
|
try {
|
|
// if hostname is empty then use wildcard address otherwise look
|
|
// up the name.
|
|
if (m_hostname.empty()) {
|
|
m_address = ARCH->newAnyAddr(IArchNetwork::AddressFamily::INet);
|
|
resolvedAddressesCount = 1;
|
|
} else {
|
|
// Logic for temporary filtring only ipv4 addresses
|
|
std::vector<ArchNetAddress> ipv4OnlyAddresses;
|
|
{
|
|
auto addresses = ARCH->nameToAddr(m_hostname);
|
|
for (auto address : addresses) {
|
|
if (ARCH->getAddrFamily(address) == IArchNetwork::AddressFamily::INet) {
|
|
ipv4OnlyAddresses.emplace_back(address);
|
|
} else {
|
|
ARCH->closeAddr(address);
|
|
}
|
|
}
|
|
}
|
|
|
|
resolvedAddressesCount = ipv4OnlyAddresses.size();
|
|
assert(resolvedAddressesCount > 0);
|
|
if (index < resolvedAddressesCount - 1) {
|
|
m_address = ipv4OnlyAddresses[index];
|
|
} else {
|
|
m_address = ipv4OnlyAddresses[resolvedAddressesCount - 1];
|
|
}
|
|
|
|
for (auto address : ipv4OnlyAddresses) {
|
|
if (m_address != address) {
|
|
ARCH->closeAddr(address);
|
|
}
|
|
}
|
|
}
|
|
} catch (XArchNetworkNameUnknown &) {
|
|
throw XSocketAddress(XSocketAddress::SocketError::NotFound, m_hostname, m_port);
|
|
} catch (XArchNetworkNameNoAddress &) {
|
|
throw XSocketAddress(XSocketAddress::SocketError::NoAddress, m_hostname, m_port);
|
|
} catch (XArchNetworkNameUnsupported &) {
|
|
throw XSocketAddress(XSocketAddress::SocketError::Unsupported, m_hostname, m_port);
|
|
} catch (XArchNetworkName &) {
|
|
throw XSocketAddress(XSocketAddress::SocketError::Unknown, m_hostname, m_port);
|
|
}
|
|
|
|
// set port in address
|
|
ARCH->setAddrPort(m_address, m_port);
|
|
|
|
return resolvedAddressesCount;
|
|
}
|
|
|
|
bool NetworkAddress::operator==(const NetworkAddress &addr) const
|
|
{
|
|
return m_address == addr.m_address || ARCH->isEqualAddr(m_address, addr.m_address);
|
|
}
|
|
|
|
bool NetworkAddress::operator!=(const NetworkAddress &addr) const
|
|
{
|
|
return !operator==(addr);
|
|
}
|
|
|
|
bool NetworkAddress::isValid() const
|
|
{
|
|
return (m_address != nullptr);
|
|
}
|
|
|
|
const ArchNetAddress &NetworkAddress::getAddress() const
|
|
{
|
|
return m_address;
|
|
}
|
|
|
|
int NetworkAddress::getPort() const
|
|
{
|
|
return m_port;
|
|
}
|
|
|
|
std::string NetworkAddress::getHostname() const
|
|
{
|
|
return m_hostname;
|
|
}
|
|
|
|
void NetworkAddress::checkPort() const
|
|
{
|
|
// check port number
|
|
if (m_port < 0 || m_port > 65535) {
|
|
throw XSocketAddress(XSocketAddress::SocketError::BadPort, m_hostname, m_port);
|
|
}
|
|
}
|