mirror of
https://github.com/badaix/snapcast.git
synced 2025-04-28 17:57:05 +02:00
264 lines
9.6 KiB
C++
264 lines
9.6 KiB
C++
/***
|
|
This file is part of snapcast
|
|
Copyright (C) 2014-2022 Johannes Pohl
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program 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.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
***/
|
|
|
|
// prototype/interface header file
|
|
#include "client_connection.hpp"
|
|
|
|
// local headers
|
|
#include "common/aixlog.hpp"
|
|
#include "common/snap_exception.hpp"
|
|
#include "common/str_compat.hpp"
|
|
#include "message/hello.hpp"
|
|
|
|
// standard headers
|
|
#include <iostream>
|
|
#include <mutex>
|
|
|
|
|
|
using namespace std;
|
|
|
|
static constexpr auto LOG_TAG = "Connection";
|
|
|
|
ClientConnection::ClientConnection(boost::asio::io_context& io_context, const ClientSettings::Server& server)
|
|
: io_context_(io_context), strand_(boost::asio::make_strand(io_context_.get_executor())), resolver_(strand_), socket_(strand_), reqId_(1), server_(server)
|
|
{
|
|
base_msg_size_ = base_message_.getSize();
|
|
buffer_.resize(base_msg_size_);
|
|
}
|
|
|
|
|
|
ClientConnection::~ClientConnection()
|
|
{
|
|
disconnect();
|
|
}
|
|
|
|
|
|
std::string ClientConnection::getMacAddress()
|
|
{
|
|
std::string mac =
|
|
#ifndef WINDOWS
|
|
::getMacAddress(socket_.native_handle());
|
|
#else
|
|
::getMacAddress(socket_.local_endpoint().address().to_string());
|
|
#endif
|
|
if (mac.empty())
|
|
mac = "00:00:00:00:00:00";
|
|
LOG(INFO, LOG_TAG) << "My MAC: \"" << mac << "\", socket: " << socket_.native_handle() << "\n";
|
|
return mac;
|
|
}
|
|
|
|
|
|
void ClientConnection::connect(const ResultHandler& handler)
|
|
{
|
|
tcp::resolver::query query(server_.host, cpt::to_string(server_.port), boost::asio::ip::resolver_query_base::numeric_service);
|
|
boost::system::error_code ec;
|
|
LOG(INFO, LOG_TAG) << "Resolving host IP for: " << server_.host << "\n";
|
|
auto iterator = resolver_.resolve(query, ec);
|
|
if (ec)
|
|
{
|
|
LOG(ERROR, LOG_TAG) << "Failed to resolve host '" << server_.host << "', error: " << ec.message() << "\n";
|
|
handler(ec);
|
|
return;
|
|
}
|
|
|
|
LOG(INFO, LOG_TAG) << "Connecting\n";
|
|
socket_.connect(*iterator, ec);
|
|
if (ec)
|
|
{
|
|
LOG(ERROR, LOG_TAG) << "Failed to connect to host '" << server_.host << "', error: " << ec.message() << "\n";
|
|
handler(ec);
|
|
return;
|
|
}
|
|
LOG(NOTICE, LOG_TAG) << "Connected to " << socket_.remote_endpoint().address().to_string() << endl;
|
|
handler(ec);
|
|
|
|
#if 0
|
|
resolver_.async_resolve(query, host_, cpt::to_string(port_), [this, handler](const boost::system::error_code& ec, tcp::resolver::results_type results) {
|
|
if (ec)
|
|
{
|
|
LOG(ERROR, LOG_TAG) << "Failed to resolve host '" << host_ << "', error: " << ec.message() << "\n";
|
|
handler(ec);
|
|
return;
|
|
}
|
|
|
|
resolver_.cancel();
|
|
socket_.async_connect(*results, [this, handler](const boost::system::error_code& ec) {
|
|
if (ec)
|
|
{
|
|
LOG(ERROR, LOG_TAG) << "Failed to connect to host '" << host_ << "', error: " << ec.message() << "\n";
|
|
handler(ec);
|
|
return;
|
|
}
|
|
|
|
LOG(NOTICE, LOG_TAG) << "Connected to " << socket_.remote_endpoint().address().to_string() << endl;
|
|
handler(ec);
|
|
getNextMessage();
|
|
});
|
|
});
|
|
#endif
|
|
}
|
|
|
|
|
|
void ClientConnection::disconnect()
|
|
{
|
|
LOG(DEBUG, LOG_TAG) << "Disconnecting\n";
|
|
if (!socket_.is_open())
|
|
{
|
|
LOG(DEBUG, LOG_TAG) << "Not connected\n";
|
|
return;
|
|
}
|
|
boost::system::error_code ec;
|
|
socket_.shutdown(boost::asio::ip::tcp::socket::shutdown_both, ec);
|
|
if (ec)
|
|
LOG(ERROR, LOG_TAG) << "Error in socket shutdown: " << ec.message() << endl;
|
|
socket_.close(ec);
|
|
if (ec)
|
|
LOG(ERROR, LOG_TAG) << "Error in socket close: " << ec.message() << endl;
|
|
boost::asio::post(strand_, [this]() { pendingRequests_.clear(); });
|
|
LOG(DEBUG, LOG_TAG) << "Disconnected\n";
|
|
}
|
|
|
|
|
|
void ClientConnection::sendNext()
|
|
{
|
|
auto& message = messages_.front();
|
|
static boost::asio::streambuf streambuf;
|
|
std::ostream stream(&streambuf);
|
|
tv t;
|
|
message.msg->sent = t;
|
|
message.msg->serialize(stream);
|
|
auto handler = message.handler;
|
|
|
|
boost::asio::async_write(socket_, streambuf, [this, handler](boost::system::error_code ec, std::size_t length) {
|
|
if (ec)
|
|
LOG(ERROR, LOG_TAG) << "Failed to send message, error: " << ec.message() << "\n";
|
|
else
|
|
LOG(TRACE, LOG_TAG) << "Wrote " << length << " bytes to socket\n";
|
|
|
|
messages_.pop_front();
|
|
if (handler)
|
|
handler(ec);
|
|
|
|
if (!messages_.empty())
|
|
sendNext();
|
|
});
|
|
}
|
|
|
|
|
|
void ClientConnection::send(const msg::message_ptr& message, const ResultHandler& handler)
|
|
{
|
|
boost::asio::post(strand_, [this, message, handler]() {
|
|
messages_.emplace_back(message, handler);
|
|
if (messages_.size() > 1)
|
|
{
|
|
LOG(DEBUG, LOG_TAG) << "outstanding async_write\n";
|
|
return;
|
|
}
|
|
sendNext();
|
|
});
|
|
}
|
|
|
|
|
|
void ClientConnection::sendRequest(const msg::message_ptr& message, const chronos::usec& timeout, const MessageHandler<msg::BaseMessage>& handler)
|
|
{
|
|
boost::asio::post(strand_, [this, message, timeout, handler]() {
|
|
pendingRequests_.erase(
|
|
std::remove_if(pendingRequests_.begin(), pendingRequests_.end(), [](std::weak_ptr<PendingRequest> request) { return request.expired(); }),
|
|
pendingRequests_.end());
|
|
unique_ptr<msg::BaseMessage> response(nullptr);
|
|
if (++reqId_ >= 10000)
|
|
reqId_ = 1;
|
|
message->id = reqId_;
|
|
auto request = make_shared<PendingRequest>(strand_, reqId_, handler);
|
|
pendingRequests_.push_back(request);
|
|
request->startTimer(timeout);
|
|
send(message, [handler](const boost::system::error_code& ec) {
|
|
if (ec)
|
|
handler(ec, nullptr);
|
|
});
|
|
});
|
|
}
|
|
|
|
|
|
void ClientConnection::getNextMessage(const MessageHandler<msg::BaseMessage>& handler)
|
|
{
|
|
boost::asio::async_read(socket_, boost::asio::buffer(buffer_, base_msg_size_), [this, handler](boost::system::error_code ec, std::size_t length) mutable {
|
|
if (ec)
|
|
{
|
|
LOG(ERROR, LOG_TAG) << "Error reading message header of length " << length << ": " << ec.message() << "\n";
|
|
if (handler)
|
|
handler(ec, nullptr);
|
|
return;
|
|
}
|
|
|
|
base_message_.deserialize(buffer_.data());
|
|
tv t;
|
|
base_message_.received = t;
|
|
// LOG(TRACE, LOG_TAG) << "getNextMessage: " << base_message_.type << ", size: " << base_message_.size << ", id: " <<
|
|
// base_message_.id << ", refers: " << base_message_.refersTo << "\n";
|
|
if (base_message_.type > message_type::kLast)
|
|
{
|
|
LOG(ERROR, LOG_TAG) << "unknown message type received: " << base_message_.type << ", size: " << base_message_.size << "\n";
|
|
if (handler)
|
|
handler(boost::asio::error::invalid_argument, nullptr);
|
|
return;
|
|
}
|
|
else if (base_message_.size > msg::max_size)
|
|
{
|
|
LOG(ERROR, LOG_TAG) << "received message of type " << base_message_.type << " to large: " << base_message_.size << "\n";
|
|
if (handler)
|
|
handler(boost::asio::error::invalid_argument, nullptr);
|
|
return;
|
|
}
|
|
|
|
if (base_message_.size > buffer_.size())
|
|
buffer_.resize(base_message_.size);
|
|
|
|
boost::asio::async_read(socket_, boost::asio::buffer(buffer_, base_message_.size),
|
|
[this, handler](boost::system::error_code ec, std::size_t length) mutable {
|
|
if (ec)
|
|
{
|
|
LOG(ERROR, LOG_TAG) << "Error reading message body of length " << length << ": " << ec.message() << "\n";
|
|
if (handler)
|
|
handler(ec, nullptr);
|
|
return;
|
|
}
|
|
|
|
auto response = msg::factory::createMessage(base_message_, buffer_.data());
|
|
if (!response)
|
|
LOG(WARNING, LOG_TAG) << "Failed to deserialize message of type: " << base_message_.type << "\n";
|
|
for (auto iter = pendingRequests_.begin(); iter != pendingRequests_.end(); ++iter)
|
|
{
|
|
auto request = *iter;
|
|
if (auto req = request.lock())
|
|
{
|
|
if (req->id() == base_message_.refersTo)
|
|
{
|
|
req->setValue(std::move(response));
|
|
pendingRequests_.erase(iter);
|
|
getNextMessage(handler);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (handler)
|
|
handler(ec, std::move(response));
|
|
});
|
|
});
|
|
}
|