2020-05-22 16:18:41 +03:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2013-2020, The PurpleI2P Project
|
|
|
|
*
|
|
|
|
* This file is part of Purple i2pd project and licensed under BSD3
|
|
|
|
*
|
|
|
|
* See full license text in LICENSE file at top of project tree
|
|
|
|
*/
|
|
|
|
|
2015-01-07 19:09:59 +01:00
|
|
|
#include "Destination.h"
|
|
|
|
#include "Identity.h"
|
|
|
|
#include "ClientContext.h"
|
|
|
|
#include "I2PService.h"
|
2017-08-31 12:08:22 -04:00
|
|
|
#include <boost/asio/error.hpp>
|
2015-01-07 19:09:59 +01:00
|
|
|
|
|
|
|
namespace i2p
|
|
|
|
{
|
|
|
|
namespace client
|
|
|
|
{
|
2018-12-06 13:13:20 -05:00
|
|
|
static const i2p::data::SigningKeyType I2P_SERVICE_DEFAULT_KEY_TYPE = i2p::data::SIGNING_KEY_TYPE_EDDSA_SHA512_ED25519;
|
2015-01-07 19:09:59 +01:00
|
|
|
|
2015-02-24 15:40:50 -05:00
|
|
|
I2PService::I2PService (std::shared_ptr<ClientDestination> localDestination):
|
2015-01-07 19:09:59 +01:00
|
|
|
m_LocalDestination (localDestination ? localDestination :
|
2017-08-31 12:08:22 -04:00
|
|
|
i2p::client::context.CreateNewLocalDestination (false, I2P_SERVICE_DEFAULT_KEY_TYPE)),
|
2017-10-04 20:15:29 +03:00
|
|
|
m_ReadyTimer(m_LocalDestination->GetService()),
|
2018-05-28 17:00:47 -04:00
|
|
|
m_ReadyTimerTriggered(false),
|
2017-10-04 20:15:29 +03:00
|
|
|
m_ConnectTimeout(0),
|
2017-08-31 12:08:22 -04:00
|
|
|
isUpdated (true)
|
2015-01-07 19:09:59 +01:00
|
|
|
{
|
2017-08-31 12:08:22 -04:00
|
|
|
m_LocalDestination->Acquire ();
|
2015-01-07 19:09:59 +01:00
|
|
|
}
|
2017-08-31 12:08:22 -04:00
|
|
|
|
2015-01-07 21:15:04 +01:00
|
|
|
I2PService::I2PService (i2p::data::SigningKeyType kt):
|
2017-07-28 15:12:15 -04:00
|
|
|
m_LocalDestination (i2p::client::context.CreateNewLocalDestination (false, kt)),
|
2017-08-31 12:08:22 -04:00
|
|
|
m_ReadyTimer(m_LocalDestination->GetService()),
|
|
|
|
m_ConnectTimeout(0),
|
2017-07-28 15:12:15 -04:00
|
|
|
isUpdated (true)
|
2015-01-07 21:15:04 +01:00
|
|
|
{
|
2017-07-06 16:12:06 -04:00
|
|
|
m_LocalDestination->Acquire ();
|
2015-01-07 21:15:04 +01:00
|
|
|
}
|
2017-08-31 12:08:22 -04:00
|
|
|
|
|
|
|
I2PService::~I2PService ()
|
|
|
|
{
|
|
|
|
ClearHandlers ();
|
|
|
|
if (m_LocalDestination) m_LocalDestination->Release ();
|
2017-07-06 16:12:06 -04:00
|
|
|
}
|
|
|
|
|
2017-08-10 20:29:35 -04:00
|
|
|
void I2PService::ClearHandlers ()
|
|
|
|
{
|
2017-08-31 12:08:22 -04:00
|
|
|
if(m_ConnectTimeout)
|
|
|
|
m_ReadyTimer.cancel();
|
2017-08-10 20:29:35 -04:00
|
|
|
std::unique_lock<std::mutex> l(m_HandlersMutex);
|
|
|
|
for (auto it: m_Handlers)
|
|
|
|
it->Terminate ();
|
|
|
|
m_Handlers.clear();
|
|
|
|
}
|
2017-08-31 12:08:22 -04:00
|
|
|
|
|
|
|
void I2PService::SetConnectTimeout(uint32_t timeout)
|
|
|
|
{
|
|
|
|
m_ConnectTimeout = timeout;
|
|
|
|
}
|
|
|
|
|
|
|
|
void I2PService::AddReadyCallback(ReadyCallback cb)
|
|
|
|
{
|
|
|
|
uint32_t now = i2p::util::GetSecondsSinceEpoch();
|
2018-05-28 17:00:47 -04:00
|
|
|
uint32_t tm = (m_ConnectTimeout) ? now + m_ConnectTimeout : NEVER_TIMES_OUT;
|
|
|
|
|
2017-08-31 12:08:22 -04:00
|
|
|
LogPrint(eLogDebug, "I2PService::AddReadyCallback() ", tm, " ", now);
|
|
|
|
m_ReadyCallbacks.push_back({cb, tm});
|
2018-05-28 17:00:47 -04:00
|
|
|
if (!m_ReadyTimerTriggered) TriggerReadyCheckTimer();
|
2017-08-31 12:08:22 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void I2PService::TriggerReadyCheckTimer()
|
|
|
|
{
|
|
|
|
m_ReadyTimer.expires_from_now(boost::posix_time::seconds (1));
|
2018-12-07 12:25:26 -05:00
|
|
|
m_ReadyTimer.async_wait(std::bind(&I2PService::HandleReadyCheckTimer, shared_from_this (), std::placeholders::_1));
|
2018-05-28 17:00:47 -04:00
|
|
|
m_ReadyTimerTriggered = true;
|
|
|
|
|
2017-08-31 12:08:22 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void I2PService::HandleReadyCheckTimer(const boost::system::error_code &ec)
|
|
|
|
{
|
|
|
|
if(ec || m_LocalDestination->IsReady())
|
|
|
|
{
|
|
|
|
for(auto & itr : m_ReadyCallbacks)
|
|
|
|
itr.first(ec);
|
|
|
|
m_ReadyCallbacks.clear();
|
|
|
|
}
|
|
|
|
else if(!m_LocalDestination->IsReady())
|
|
|
|
{
|
|
|
|
// expire timed out requests
|
|
|
|
uint32_t now = i2p::util::GetSecondsSinceEpoch ();
|
|
|
|
auto itr = m_ReadyCallbacks.begin();
|
|
|
|
while(itr != m_ReadyCallbacks.end())
|
|
|
|
{
|
2020-05-05 02:36:34 +03:00
|
|
|
if(itr->second != NEVER_TIMES_OUT && now >= itr->second)
|
2017-08-31 12:08:22 -04:00
|
|
|
{
|
|
|
|
itr->first(boost::asio::error::timed_out);
|
|
|
|
itr = m_ReadyCallbacks.erase(itr);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
++itr;
|
|
|
|
}
|
|
|
|
}
|
2018-05-28 17:00:47 -04:00
|
|
|
if(!ec && m_ReadyCallbacks.size())
|
2020-05-05 02:36:34 +03:00
|
|
|
TriggerReadyCheckTimer();
|
2018-05-28 17:00:47 -04:00
|
|
|
else
|
2020-05-05 02:36:34 +03:00
|
|
|
m_ReadyTimerTriggered = false;
|
2017-08-31 12:08:22 -04:00
|
|
|
}
|
|
|
|
|
2015-01-07 21:15:04 +01:00
|
|
|
void I2PService::CreateStream (StreamRequestComplete streamRequestComplete, const std::string& dest, int port) {
|
2015-01-07 20:44:24 +01:00
|
|
|
assert(streamRequestComplete);
|
2019-03-28 09:57:34 -04:00
|
|
|
auto address = i2p::client::context.GetAddressBook ().GetAddress (dest);
|
|
|
|
if (address)
|
|
|
|
CreateStream(streamRequestComplete, address, port);
|
2015-01-07 20:44:24 +01:00
|
|
|
else
|
|
|
|
{
|
2016-06-03 00:00:00 +00:00
|
|
|
LogPrint (eLogWarning, "I2PService: Remote destination not found: ", dest);
|
2015-01-07 20:44:24 +01:00
|
|
|
streamRequestComplete (nullptr);
|
|
|
|
}
|
|
|
|
}
|
2015-01-08 01:31:31 +01:00
|
|
|
|
2019-03-28 09:57:34 -04:00
|
|
|
void I2PService::CreateStream(StreamRequestComplete streamRequestComplete, std::shared_ptr<const Address> address, int port)
|
2017-08-31 12:08:22 -04:00
|
|
|
{
|
2019-03-28 09:57:34 -04:00
|
|
|
if(m_ConnectTimeout && !m_LocalDestination->IsReady())
|
2017-08-31 12:08:22 -04:00
|
|
|
{
|
2020-03-01 13:25:50 +03:00
|
|
|
AddReadyCallback([this, streamRequestComplete, address, port] (const boost::system::error_code & ec)
|
|
|
|
{
|
2019-03-28 09:57:34 -04:00
|
|
|
if(ec)
|
|
|
|
{
|
2020-05-05 02:36:34 +03:00
|
|
|
LogPrint(eLogWarning, "I2PService::CreateStream() ", ec.message());
|
2019-03-28 09:57:34 -04:00
|
|
|
streamRequestComplete(nullptr);
|
|
|
|
}
|
|
|
|
else
|
2020-03-01 13:25:50 +03:00
|
|
|
{
|
|
|
|
if (address->IsIdentHash ())
|
2019-03-28 09:57:34 -04:00
|
|
|
this->m_LocalDestination->CreateStream(streamRequestComplete, address->identHash, port);
|
2017-08-31 12:08:22 -04:00
|
|
|
else
|
2020-03-01 13:25:50 +03:00
|
|
|
this->m_LocalDestination->CreateStream (streamRequestComplete, address->blindedPublicKey, port);
|
2019-03-28 09:57:34 -04:00
|
|
|
}
|
|
|
|
});
|
2017-08-31 12:08:22 -04:00
|
|
|
}
|
|
|
|
else
|
2020-03-01 13:25:50 +03:00
|
|
|
{
|
2019-03-28 09:57:34 -04:00
|
|
|
if (address->IsIdentHash ())
|
|
|
|
m_LocalDestination->CreateStream (streamRequestComplete, address->identHash, port);
|
|
|
|
else
|
|
|
|
m_LocalDestination->CreateStream (streamRequestComplete, address->blindedPublicKey, port);
|
|
|
|
}
|
2017-08-31 12:08:22 -04:00
|
|
|
}
|
|
|
|
|
2016-07-28 09:25:05 -04:00
|
|
|
TCPIPPipe::TCPIPPipe(I2PService * owner, std::shared_ptr<boost::asio::ip::tcp::socket> upstream, std::shared_ptr<boost::asio::ip::tcp::socket> downstream) : I2PServiceHandler(owner), m_up(upstream), m_down(downstream)
|
|
|
|
{
|
|
|
|
boost::asio::socket_base::receive_buffer_size option(TCP_IP_PIPE_BUFFER_SIZE);
|
|
|
|
upstream->set_option(option);
|
|
|
|
downstream->set_option(option);
|
|
|
|
}
|
2016-02-26 17:06:11 -05:00
|
|
|
|
|
|
|
TCPIPPipe::~TCPIPPipe()
|
|
|
|
{
|
|
|
|
Terminate();
|
|
|
|
}
|
2017-08-31 12:08:22 -04:00
|
|
|
|
2016-02-26 17:06:11 -05:00
|
|
|
void TCPIPPipe::Start()
|
|
|
|
{
|
|
|
|
AsyncReceiveUpstream();
|
|
|
|
AsyncReceiveDownstream();
|
|
|
|
}
|
|
|
|
|
|
|
|
void TCPIPPipe::Terminate()
|
|
|
|
{
|
|
|
|
if(Kill()) return;
|
2017-10-04 20:15:29 +03:00
|
|
|
if (m_up)
|
|
|
|
{
|
|
|
|
if (m_up->is_open())
|
2016-02-26 17:06:11 -05:00
|
|
|
m_up->close();
|
|
|
|
m_up = nullptr;
|
|
|
|
}
|
2017-10-04 20:15:29 +03:00
|
|
|
if (m_down)
|
|
|
|
{
|
|
|
|
if (m_down->is_open())
|
2016-02-26 17:06:11 -05:00
|
|
|
m_down->close();
|
|
|
|
m_down = nullptr;
|
|
|
|
}
|
2016-11-20 12:13:11 -05:00
|
|
|
Done(shared_from_this());
|
2016-02-26 17:06:11 -05:00
|
|
|
}
|
2017-08-31 12:08:22 -04:00
|
|
|
|
2016-02-26 17:06:11 -05:00
|
|
|
void TCPIPPipe::AsyncReceiveUpstream()
|
|
|
|
{
|
2017-10-04 20:15:29 +03:00
|
|
|
if (m_up)
|
|
|
|
{
|
2016-02-26 17:06:11 -05:00
|
|
|
m_up->async_read_some(boost::asio::buffer(m_upstream_to_down_buf, TCP_IP_PIPE_BUFFER_SIZE),
|
2017-10-04 20:15:29 +03:00
|
|
|
std::bind(&TCPIPPipe::HandleUpstreamReceived, shared_from_this(),
|
2020-03-01 13:25:50 +03:00
|
|
|
std::placeholders::_1, std::placeholders::_2));
|
2016-02-26 17:06:11 -05:00
|
|
|
}
|
2017-10-04 20:15:29 +03:00
|
|
|
else
|
|
|
|
LogPrint(eLogError, "TCPIPPipe: upstream receive: no socket");
|
2016-02-26 17:06:11 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
void TCPIPPipe::AsyncReceiveDownstream()
|
|
|
|
{
|
|
|
|
if (m_down) {
|
|
|
|
m_down->async_read_some(boost::asio::buffer(m_downstream_to_up_buf, TCP_IP_PIPE_BUFFER_SIZE),
|
2017-10-04 20:15:29 +03:00
|
|
|
std::bind(&TCPIPPipe::HandleDownstreamReceived, shared_from_this(),
|
2020-03-01 13:25:50 +03:00
|
|
|
std::placeholders::_1, std::placeholders::_2));
|
2016-02-26 17:06:11 -05:00
|
|
|
}
|
2017-10-04 20:15:29 +03:00
|
|
|
else
|
|
|
|
LogPrint(eLogError, "TCPIPPipe: downstream receive: no socket");
|
2016-02-26 17:06:11 -05:00
|
|
|
}
|
|
|
|
|
2016-11-20 12:13:11 -05:00
|
|
|
void TCPIPPipe::UpstreamWrite(size_t len)
|
2016-02-26 17:06:11 -05:00
|
|
|
{
|
2017-10-04 20:15:29 +03:00
|
|
|
if (m_up)
|
|
|
|
{
|
2016-05-31 00:00:00 +00:00
|
|
|
LogPrint(eLogDebug, "TCPIPPipe: upstream: ", (int) len, " bytes written");
|
2016-11-20 12:13:11 -05:00
|
|
|
boost::asio::async_write(*m_up, boost::asio::buffer(m_upstream_buf, len),
|
2017-10-04 20:15:29 +03:00
|
|
|
boost::asio::transfer_all(),
|
|
|
|
std::bind(&TCPIPPipe::HandleUpstreamWrite,
|
2020-03-01 13:25:50 +03:00
|
|
|
shared_from_this(),
|
|
|
|
std::placeholders::_1));
|
2016-02-26 17:06:11 -05:00
|
|
|
}
|
2017-10-04 20:15:29 +03:00
|
|
|
else
|
|
|
|
LogPrint(eLogError, "TCPIPPipe: upstream write: no socket");
|
2016-02-26 17:06:11 -05:00
|
|
|
}
|
|
|
|
|
2016-11-20 12:13:11 -05:00
|
|
|
void TCPIPPipe::DownstreamWrite(size_t len)
|
2016-02-26 17:06:11 -05:00
|
|
|
{
|
2017-10-04 20:15:29 +03:00
|
|
|
if (m_down)
|
|
|
|
{
|
2016-05-31 00:00:00 +00:00
|
|
|
LogPrint(eLogDebug, "TCPIPPipe: downstream: ", (int) len, " bytes written");
|
2016-11-20 12:13:11 -05:00
|
|
|
boost::asio::async_write(*m_down, boost::asio::buffer(m_downstream_buf, len),
|
2017-10-04 20:15:29 +03:00
|
|
|
boost::asio::transfer_all(),
|
|
|
|
std::bind(&TCPIPPipe::HandleDownstreamWrite,
|
2020-03-01 13:25:50 +03:00
|
|
|
shared_from_this(),
|
|
|
|
std::placeholders::_1));
|
2016-02-26 17:06:11 -05:00
|
|
|
}
|
2017-10-04 20:15:29 +03:00
|
|
|
else
|
|
|
|
LogPrint(eLogError, "TCPIPPipe: downstream write: no socket");
|
2016-02-26 17:06:11 -05:00
|
|
|
}
|
2017-08-31 12:08:22 -04:00
|
|
|
|
|
|
|
|
2016-02-26 17:06:11 -05:00
|
|
|
void TCPIPPipe::HandleDownstreamReceived(const boost::system::error_code & ecode, std::size_t bytes_transfered)
|
|
|
|
{
|
2016-05-31 00:00:00 +00:00
|
|
|
LogPrint(eLogDebug, "TCPIPPipe: downstream: ", (int) bytes_transfered, " bytes received");
|
2017-10-04 20:15:29 +03:00
|
|
|
if (ecode)
|
|
|
|
{
|
2016-05-31 00:00:00 +00:00
|
|
|
LogPrint(eLogError, "TCPIPPipe: downstream read error:" , ecode.message());
|
2016-02-26 17:06:11 -05:00
|
|
|
if (ecode != boost::asio::error::operation_aborted)
|
|
|
|
Terminate();
|
|
|
|
} else {
|
2017-10-04 20:15:29 +03:00
|
|
|
if (bytes_transfered > 0 )
|
2016-02-26 17:06:11 -05:00
|
|
|
memcpy(m_upstream_buf, m_downstream_to_up_buf, bytes_transfered);
|
2016-11-20 12:13:11 -05:00
|
|
|
UpstreamWrite(bytes_transfered);
|
2016-02-26 17:06:11 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void TCPIPPipe::HandleDownstreamWrite(const boost::system::error_code & ecode) {
|
2017-10-04 20:15:29 +03:00
|
|
|
if (ecode)
|
|
|
|
{
|
2016-05-31 00:00:00 +00:00
|
|
|
LogPrint(eLogError, "TCPIPPipe: downstream write error:" , ecode.message());
|
2016-02-26 17:06:11 -05:00
|
|
|
if (ecode != boost::asio::error::operation_aborted)
|
|
|
|
Terminate();
|
|
|
|
}
|
2017-10-04 20:15:29 +03:00
|
|
|
else
|
|
|
|
AsyncReceiveUpstream();
|
2016-02-26 17:06:11 -05:00
|
|
|
}
|
2017-08-31 12:08:22 -04:00
|
|
|
|
2016-02-26 17:06:11 -05:00
|
|
|
void TCPIPPipe::HandleUpstreamWrite(const boost::system::error_code & ecode) {
|
2017-10-04 20:15:29 +03:00
|
|
|
if (ecode)
|
|
|
|
{
|
2016-05-31 00:00:00 +00:00
|
|
|
LogPrint(eLogError, "TCPIPPipe: upstream write error:" , ecode.message());
|
2016-02-26 17:06:11 -05:00
|
|
|
if (ecode != boost::asio::error::operation_aborted)
|
|
|
|
Terminate();
|
|
|
|
}
|
2017-10-04 20:15:29 +03:00
|
|
|
else
|
|
|
|
AsyncReceiveDownstream();
|
2016-02-26 17:06:11 -05:00
|
|
|
}
|
2017-08-31 12:08:22 -04:00
|
|
|
|
2016-02-26 17:06:11 -05:00
|
|
|
void TCPIPPipe::HandleUpstreamReceived(const boost::system::error_code & ecode, std::size_t bytes_transfered)
|
|
|
|
{
|
2016-05-30 21:42:25 -04:00
|
|
|
LogPrint(eLogDebug, "TCPIPPipe: upstream ", (int)bytes_transfered, " bytes received");
|
2017-10-04 20:15:29 +03:00
|
|
|
if (ecode)
|
|
|
|
{
|
2016-05-31 00:00:00 +00:00
|
|
|
LogPrint(eLogError, "TCPIPPipe: upstream read error:" , ecode.message());
|
2016-02-26 17:06:11 -05:00
|
|
|
if (ecode != boost::asio::error::operation_aborted)
|
|
|
|
Terminate();
|
|
|
|
} else {
|
2017-10-04 20:15:29 +03:00
|
|
|
if (bytes_transfered > 0 )
|
2016-11-20 12:13:11 -05:00
|
|
|
memcpy(m_downstream_buf, m_upstream_to_down_buf, bytes_transfered);
|
|
|
|
DownstreamWrite(bytes_transfered);
|
2016-02-26 17:06:11 -05:00
|
|
|
}
|
|
|
|
}
|
2017-08-31 12:08:22 -04:00
|
|
|
|
2015-01-08 01:31:31 +01:00
|
|
|
void TCPIPAcceptor::Start ()
|
|
|
|
{
|
2017-08-02 21:00:04 -04:00
|
|
|
m_Acceptor.reset (new boost::asio::ip::tcp::acceptor (GetService (), m_LocalEndpoint));
|
2019-05-17 11:04:44 +03:00
|
|
|
// update the local end point in case port has been set zero and got updated now
|
2018-01-22 20:47:31 -05:00
|
|
|
m_LocalEndpoint = m_Acceptor->local_endpoint();
|
2020-05-05 02:36:34 +03:00
|
|
|
m_Acceptor->listen ();
|
|
|
|
Accept ();
|
2015-01-08 01:31:31 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void TCPIPAcceptor::Stop ()
|
|
|
|
{
|
2017-08-02 21:00:04 -04:00
|
|
|
if (m_Acceptor)
|
2017-08-31 12:08:22 -04:00
|
|
|
{
|
2017-08-02 21:00:04 -04:00
|
|
|
m_Acceptor->close();
|
|
|
|
m_Acceptor.reset (nullptr);
|
2017-08-31 12:08:22 -04:00
|
|
|
}
|
2015-01-08 01:31:31 +01:00
|
|
|
m_Timer.cancel ();
|
|
|
|
ClearHandlers();
|
|
|
|
}
|
|
|
|
|
|
|
|
void TCPIPAcceptor::Accept ()
|
|
|
|
{
|
2015-04-06 14:41:07 -04:00
|
|
|
auto newSocket = std::make_shared<boost::asio::ip::tcp::socket> (GetService ());
|
2017-08-02 21:00:04 -04:00
|
|
|
m_Acceptor->async_accept (*newSocket, std::bind (&TCPIPAcceptor::HandleAccept, this,
|
2015-01-08 01:31:31 +01:00
|
|
|
std::placeholders::_1, newSocket));
|
|
|
|
}
|
|
|
|
|
2015-04-06 14:41:07 -04:00
|
|
|
void TCPIPAcceptor::HandleAccept (const boost::system::error_code& ecode, std::shared_ptr<boost::asio::ip::tcp::socket> socket)
|
2015-01-08 01:31:31 +01:00
|
|
|
{
|
|
|
|
if (!ecode)
|
|
|
|
{
|
2016-02-11 00:00:00 +00:00
|
|
|
LogPrint(eLogDebug, "I2PService: ", GetName(), " accepted");
|
2015-01-08 01:31:31 +01:00
|
|
|
auto handler = CreateHandler(socket);
|
2017-08-31 12:08:22 -04:00
|
|
|
if (handler)
|
2015-04-06 14:41:07 -04:00
|
|
|
{
|
2015-01-08 01:31:31 +01:00
|
|
|
AddHandler(handler);
|
|
|
|
handler->Handle();
|
2017-08-31 12:08:22 -04:00
|
|
|
}
|
|
|
|
else
|
2015-01-08 01:31:31 +01:00
|
|
|
socket->close();
|
|
|
|
Accept();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (ecode != boost::asio::error::operation_aborted)
|
2016-02-11 00:00:00 +00:00
|
|
|
LogPrint (eLogError, "I2PService: ", GetName(), " closing socket on accept because: ", ecode.message ());
|
2015-01-08 01:31:31 +01:00
|
|
|
}
|
|
|
|
}
|
2015-01-07 19:09:59 +01:00
|
|
|
}
|
|
|
|
}
|