2014-09-24 12:01:26 -04:00
|
|
|
#ifndef SAM_H__
|
|
|
|
#define SAM_H__
|
|
|
|
|
2014-09-24 14:59:03 -04:00
|
|
|
#include <inttypes.h>
|
|
|
|
#include <string>
|
2014-09-24 16:39:31 -04:00
|
|
|
#include <map>
|
|
|
|
#include <list>
|
2014-09-24 12:01:26 -04:00
|
|
|
#include <thread>
|
2014-10-05 21:59:05 -04:00
|
|
|
#include <mutex>
|
2014-11-22 16:35:58 -05:00
|
|
|
#include <memory>
|
2014-09-24 12:01:26 -04:00
|
|
|
#include <boost/asio.hpp>
|
2014-10-02 12:42:28 -04:00
|
|
|
#include "Identity.h"
|
|
|
|
#include "LeaseSet.h"
|
2014-09-24 14:59:03 -04:00
|
|
|
#include "Streaming.h"
|
2014-10-22 11:46:54 -04:00
|
|
|
#include "Destination.h"
|
2014-09-24 12:01:26 -04:00
|
|
|
|
|
|
|
namespace i2p
|
|
|
|
{
|
2014-10-16 10:28:44 -04:00
|
|
|
namespace client
|
2014-09-24 12:01:26 -04:00
|
|
|
{
|
2015-03-27 09:44:27 -04:00
|
|
|
const size_t SAM_SOCKET_BUFFER_SIZE = 8192;
|
2014-12-26 21:06:24 -05:00
|
|
|
const int SAM_SOCKET_CONNECTION_MAX_IDLE = 3600; // in seconds
|
2017-01-31 11:16:55 -05:00
|
|
|
const int SAM_SESSION_READINESS_CHECK_INTERVAL = 20; // in seconds
|
2014-09-24 14:59:03 -04:00
|
|
|
const char SAM_HANDSHAKE[] = "HELLO VERSION";
|
2014-12-16 15:54:02 -05:00
|
|
|
const char SAM_HANDSHAKE_REPLY[] = "HELLO REPLY RESULT=OK VERSION=%s\n";
|
2018-02-25 08:47:39 -05:00
|
|
|
const char SAM_HANDSHAKE_NOVERSION[] = "HELLO REPLY RESULT=NOVERSION\n";
|
2017-01-31 11:16:55 -05:00
|
|
|
const char SAM_HANDSHAKE_I2P_ERROR[] = "HELLO REPLY RESULT=I2P_ERROR\n";
|
2014-09-25 13:22:25 -04:00
|
|
|
const char SAM_SESSION_CREATE[] = "SESSION CREATE";
|
2014-10-02 21:40:15 -04:00
|
|
|
const char SAM_SESSION_CREATE_REPLY_OK[] = "SESSION STATUS RESULT=OK DESTINATION=%s\n";
|
2014-09-27 08:11:00 -04:00
|
|
|
const char SAM_SESSION_CREATE_DUPLICATED_ID[] = "SESSION STATUS RESULT=DUPLICATED_ID\n";
|
2017-01-31 11:16:55 -05:00
|
|
|
const char SAM_SESSION_CREATE_DUPLICATED_DEST[] = "SESSION STATUS RESULT=DUPLICATED_DEST\n";
|
2018-02-16 06:28:22 -05:00
|
|
|
const char SAM_SESSION_CREATE_INVALID_ID[] = "SESSION STATUS RESULT=INVALID_ID\n";
|
2015-06-24 14:19:10 -05:00
|
|
|
const char SAM_SESSION_STATUS_INVALID_KEY[] = "SESSION STATUS RESULT=INVALID_KEY\n";
|
2017-01-31 11:16:55 -05:00
|
|
|
const char SAM_SESSION_STATUS_I2P_ERROR[] = "SESSION STATUS RESULT=I2P_ERROR MESSAGE=%s\n";
|
2014-09-25 13:58:09 -04:00
|
|
|
const char SAM_STREAM_CONNECT[] = "STREAM CONNECT";
|
2014-09-27 08:11:00 -04:00
|
|
|
const char SAM_STREAM_STATUS_OK[] = "STREAM STATUS RESULT=OK\n";
|
|
|
|
const char SAM_STREAM_STATUS_INVALID_ID[] = "STREAM STATUS RESULT=INVALID_ID\n";
|
|
|
|
const char SAM_STREAM_STATUS_CANT_REACH_PEER[] = "STREAM STATUS RESULT=CANT_REACH_PEER\n";
|
|
|
|
const char SAM_STREAM_STATUS_I2P_ERROR[] = "STREAM STATUS RESULT=I2P_ERROR\n";
|
2017-01-31 11:16:55 -05:00
|
|
|
const char SAM_STREAM_ACCEPT[] = "STREAM ACCEPT";
|
2015-03-26 21:23:59 -04:00
|
|
|
const char SAM_DATAGRAM_SEND[] = "DATAGRAM SEND";
|
2014-09-30 11:08:38 -04:00
|
|
|
const char SAM_DEST_GENERATE[] = "DEST GENERATE";
|
2017-01-31 11:16:55 -05:00
|
|
|
const char SAM_DEST_REPLY[] = "DEST REPLY PUB=%s PRIV=%s\n";
|
2014-09-30 11:08:38 -04:00
|
|
|
const char SAM_DEST_REPLY_I2P_ERROR[] = "DEST REPLY RESULT=I2P_ERROR\n";
|
2014-10-02 16:55:01 -04:00
|
|
|
const char SAM_NAMING_LOOKUP[] = "NAMING LOOKUP";
|
|
|
|
const char SAM_NAMING_REPLY[] = "NAMING REPLY RESULT=OK NAME=ME VALUE=%s\n";
|
2016-01-31 22:37:38 -05:00
|
|
|
const char SAM_DATAGRAM_RECEIVED[] = "DATAGRAM RECEIVED DESTINATION=%s SIZE=%lu\n";
|
2014-10-03 15:08:41 -04:00
|
|
|
const char SAM_NAMING_REPLY_INVALID_KEY[] = "NAMING REPLY RESULT=INVALID_KEY NAME=%s\n";
|
2014-12-16 15:54:02 -05:00
|
|
|
const char SAM_NAMING_REPLY_KEY_NOT_FOUND[] = "NAMING REPLY RESULT=INVALID_KEY_NOT_FOUND NAME=%s\n";
|
2017-01-31 11:16:55 -05:00
|
|
|
const char SAM_PARAM_MIN[] = "MIN";
|
|
|
|
const char SAM_PARAM_MAX[] = "MAX";
|
|
|
|
const char SAM_PARAM_STYLE[] = "STYLE";
|
|
|
|
const char SAM_PARAM_ID[] = "ID";
|
2014-09-29 14:18:06 -04:00
|
|
|
const char SAM_PARAM_SILENT[] = "SILENT";
|
2017-01-31 11:16:55 -05:00
|
|
|
const char SAM_PARAM_DESTINATION[] = "DESTINATION";
|
2014-12-16 16:23:42 -05:00
|
|
|
const char SAM_PARAM_NAME[] = "NAME";
|
2017-01-31 11:16:55 -05:00
|
|
|
const char SAM_PARAM_SIGNATURE_TYPE[] = "SIGNATURE_TYPE";
|
2017-11-14 15:05:07 -05:00
|
|
|
const char SAM_PARAM_CRYPTO_TYPE[] = "CRYPTO_TYPE";
|
2015-03-26 21:23:59 -04:00
|
|
|
const char SAM_PARAM_SIZE[] = "SIZE";
|
2017-01-31 11:16:55 -05:00
|
|
|
const char SAM_VALUE_TRANSIENT[] = "TRANSIENT";
|
2014-10-22 16:42:26 -04:00
|
|
|
const char SAM_VALUE_STREAM[] = "STREAM";
|
|
|
|
const char SAM_VALUE_DATAGRAM[] = "DATAGRAM";
|
2017-01-31 11:16:55 -05:00
|
|
|
const char SAM_VALUE_RAW[] = "RAW";
|
|
|
|
const char SAM_VALUE_TRUE[] = "true";
|
2017-01-29 20:38:18 -05:00
|
|
|
const char SAM_VALUE_FALSE[] = "false";
|
2017-01-31 11:16:55 -05:00
|
|
|
const char SAM_VALUE_HOST[] = "HOST";
|
|
|
|
const char SAM_VALUE_PORT[] = "PORT";
|
2014-09-25 13:22:25 -04:00
|
|
|
|
|
|
|
enum SAMSocketType
|
|
|
|
{
|
|
|
|
eSAMSocketTypeUnknown,
|
|
|
|
eSAMSocketTypeSession,
|
2014-09-26 15:40:57 -04:00
|
|
|
eSAMSocketTypeStream,
|
2014-11-22 21:56:59 -05:00
|
|
|
eSAMSocketTypeAcceptor,
|
|
|
|
eSAMSocketTypeTerminated
|
2014-09-25 13:22:25 -04:00
|
|
|
};
|
2014-09-24 14:59:03 -04:00
|
|
|
|
|
|
|
class SAMBridge;
|
2014-12-29 17:40:03 +01:00
|
|
|
struct SAMSession;
|
2014-11-22 16:35:58 -05:00
|
|
|
class SAMSocket: public std::enable_shared_from_this<SAMSocket>
|
2014-09-24 14:59:03 -04:00
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
2018-01-15 08:19:57 -05:00
|
|
|
typedef boost::asio::ip::tcp::socket Socket_t;
|
|
|
|
SAMSocket (SAMBridge& owner, std::shared_ptr<Socket_t> socket);
|
2017-02-02 03:06:32 +08:00
|
|
|
~SAMSocket ();
|
2014-12-04 11:24:00 -05:00
|
|
|
|
2018-01-15 08:19:57 -05:00
|
|
|
boost::asio::ip::tcp::socket& GetSocket () { return *m_Socket; };
|
2014-09-24 14:59:03 -04:00
|
|
|
void ReceiveHandshake ();
|
2014-12-04 11:24:00 -05:00
|
|
|
void SetSocketType (SAMSocketType socketType) { m_SocketType = socketType; };
|
2015-02-20 22:47:36 -05:00
|
|
|
SAMSocketType GetSocketType () const { return m_SocketType; };
|
2014-09-24 14:59:03 -04:00
|
|
|
|
2018-01-15 08:25:58 -05:00
|
|
|
void Terminate (const char* reason);
|
2017-06-02 23:43:33 +08:00
|
|
|
|
2018-01-15 08:25:58 -05:00
|
|
|
private:
|
2014-09-24 14:59:03 -04:00
|
|
|
|
|
|
|
void HandleHandshakeReceived (const boost::system::error_code& ecode, std::size_t bytes_transferred);
|
|
|
|
void HandleHandshakeReplySent (const boost::system::error_code& ecode, std::size_t bytes_transferred);
|
2014-09-25 13:22:25 -04:00
|
|
|
void HandleMessage (const boost::system::error_code& ecode, std::size_t bytes_transferred);
|
2017-01-31 11:16:55 -05:00
|
|
|
void SendMessageReply (const char * msg, size_t len, bool close);
|
2014-09-25 13:22:25 -04:00
|
|
|
void HandleMessageReplySent (const boost::system::error_code& ecode, std::size_t bytes_transferred, bool close);
|
2014-09-24 14:59:03 -04:00
|
|
|
void Receive ();
|
|
|
|
void HandleReceived (const boost::system::error_code& ecode, std::size_t bytes_transferred);
|
|
|
|
|
2017-01-31 11:16:55 -05:00
|
|
|
void I2PReceive ();
|
2014-09-26 15:40:57 -04:00
|
|
|
void HandleI2PReceive (const boost::system::error_code& ecode, std::size_t bytes_transferred);
|
2014-11-23 11:33:58 -05:00
|
|
|
void HandleI2PAccept (std::shared_ptr<i2p::stream::Stream> stream);
|
2018-01-15 08:25:58 -05:00
|
|
|
void HandleWriteI2PData (const boost::system::error_code& ecode, size_t sz);
|
2015-03-03 15:31:49 -05:00
|
|
|
void HandleI2PDatagramReceive (const i2p::data::IdentityEx& from, uint16_t fromPort, uint16_t toPort, const uint8_t * buf, size_t len);
|
2014-09-24 14:59:03 -04:00
|
|
|
|
2014-09-25 13:22:25 -04:00
|
|
|
void ProcessSessionCreate (char * buf, size_t len);
|
2017-03-29 13:59:48 -04:00
|
|
|
void ProcessStreamConnect (char * buf, size_t len, size_t rem);
|
2014-09-26 15:40:57 -04:00
|
|
|
void ProcessStreamAccept (char * buf, size_t len);
|
2017-03-30 15:43:02 -04:00
|
|
|
void ProcessDestGenerate (char * buf, size_t len);
|
2014-10-02 16:55:01 -04:00
|
|
|
void ProcessNamingLookup (char * buf, size_t len);
|
2017-01-31 11:16:55 -05:00
|
|
|
void SendI2PError(const std::string & msg);
|
|
|
|
size_t ProcessDatagramSend (char * buf, size_t len, const char * data); // from SAM 1.0
|
2015-03-26 21:23:59 -04:00
|
|
|
void ExtractParams (char * buf, std::map<std::string, std::string>& params);
|
2014-09-25 13:22:25 -04:00
|
|
|
|
2015-01-27 11:27:58 -05:00
|
|
|
void Connect (std::shared_ptr<const i2p::data::LeaseSet> remote);
|
2015-04-07 12:02:25 -04:00
|
|
|
void HandleConnectLeaseSetRequestComplete (std::shared_ptr<i2p::data::LeaseSet> leaseSet);
|
2015-11-03 09:15:49 -05:00
|
|
|
void SendNamingLookupReply (std::shared_ptr<const i2p::data::IdentityEx> identity);
|
2015-04-07 12:02:25 -04:00
|
|
|
void HandleNamingLookupLeaseSetRequestComplete (std::shared_ptr<i2p::data::LeaseSet> leaseSet, i2p::data::IdentHash ident);
|
2014-10-16 16:36:12 -04:00
|
|
|
void HandleSessionReadinessCheckTimer (const boost::system::error_code& ecode);
|
|
|
|
void SendSessionCreateReplyOk ();
|
2014-10-02 12:42:28 -04:00
|
|
|
|
2018-01-15 08:25:58 -05:00
|
|
|
void WriteI2PData(size_t sz);
|
|
|
|
void WriteI2PDataImmediate(uint8_t * ptr, size_t sz);
|
2018-01-15 08:19:57 -05:00
|
|
|
|
2018-01-15 08:25:58 -05:00
|
|
|
void HandleWriteI2PDataImmediate(const boost::system::error_code & ec, uint8_t * buff);
|
2014-09-24 14:59:03 -04:00
|
|
|
private:
|
|
|
|
|
|
|
|
SAMBridge& m_Owner;
|
2018-01-15 08:19:57 -05:00
|
|
|
std::shared_ptr<Socket_t> m_Socket;
|
2014-10-02 12:42:28 -04:00
|
|
|
boost::asio::deadline_timer m_Timer;
|
2014-09-24 14:59:03 -04:00
|
|
|
char m_Buffer[SAM_SOCKET_BUFFER_SIZE + 1];
|
2015-03-27 14:02:27 -04:00
|
|
|
size_t m_BufferOffset;
|
2014-09-24 14:59:03 -04:00
|
|
|
uint8_t m_StreamBuffer[SAM_SOCKET_BUFFER_SIZE];
|
2014-09-25 13:22:25 -04:00
|
|
|
SAMSocketType m_SocketType;
|
|
|
|
std::string m_ID; // nickname
|
2014-09-29 14:18:06 -04:00
|
|
|
bool m_IsSilent;
|
2018-01-06 11:48:51 +08:00
|
|
|
bool m_IsAccepting; // for eSAMSocketTypeAcceptor only
|
2014-11-23 11:33:58 -05:00
|
|
|
std::shared_ptr<i2p::stream::Stream> m_Stream;
|
2017-01-31 11:16:55 -05:00
|
|
|
};
|
2014-09-24 14:59:03 -04:00
|
|
|
|
2014-09-24 16:39:31 -04:00
|
|
|
struct SAMSession
|
|
|
|
{
|
2015-02-24 15:40:50 -05:00
|
|
|
std::shared_ptr<ClientDestination> localDestination;
|
2016-04-01 11:36:56 -04:00
|
|
|
std::list<std::shared_ptr<SAMSocket> > m_Sockets;
|
2017-01-31 11:16:55 -05:00
|
|
|
std::shared_ptr<boost::asio::ip::udp::endpoint> UDPEndpoint;
|
2016-04-01 11:36:56 -04:00
|
|
|
std::mutex m_SocketsMutex;
|
|
|
|
|
|
|
|
/** safely add a socket to this session */
|
2018-01-15 08:19:57 -05:00
|
|
|
void AddSocket(std::shared_ptr<SAMSocket> sock) {
|
2016-04-01 11:36:56 -04:00
|
|
|
std::lock_guard<std::mutex> lock(m_SocketsMutex);
|
|
|
|
m_Sockets.push_back(sock);
|
|
|
|
}
|
|
|
|
|
|
|
|
/** safely remove a socket from this session */
|
2018-01-15 08:19:57 -05:00
|
|
|
void DelSocket(SAMSocket * sock) {
|
2016-04-01 11:36:56 -04:00
|
|
|
std::lock_guard<std::mutex> lock(m_SocketsMutex);
|
2018-01-15 08:19:57 -05:00
|
|
|
m_Sockets.remove_if([sock](const std::shared_ptr<SAMSocket> s) -> bool { return s.get() == sock; });
|
2016-04-01 11:36:56 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/** get a list holding a copy of all sam sockets from this session */
|
|
|
|
std::list<std::shared_ptr<SAMSocket> > ListSockets() {
|
|
|
|
std::list<std::shared_ptr<SAMSocket> > l;
|
|
|
|
{
|
|
|
|
std::lock_guard<std::mutex> lock(m_SocketsMutex);
|
2016-08-09 01:53:37 +03:00
|
|
|
for(const auto& sock : m_Sockets ) l.push_back(sock);
|
2016-04-01 11:36:56 -04:00
|
|
|
}
|
|
|
|
return l;
|
|
|
|
}
|
2017-01-31 11:16:55 -05:00
|
|
|
|
|
|
|
SAMSession (std::shared_ptr<ClientDestination> dest);
|
2016-12-22 19:38:17 -05:00
|
|
|
~SAMSession ();
|
2016-12-18 11:49:50 -05:00
|
|
|
|
2014-12-17 15:21:50 -05:00
|
|
|
void CloseStreams ();
|
2014-09-24 16:39:31 -04:00
|
|
|
};
|
|
|
|
|
2014-09-24 12:01:26 -04:00
|
|
|
class SAMBridge
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
2015-11-30 16:44:32 +02:00
|
|
|
SAMBridge (const std::string& address, int port);
|
2014-09-24 12:01:26 -04:00
|
|
|
~SAMBridge ();
|
|
|
|
|
|
|
|
void Start ();
|
|
|
|
void Stop ();
|
2017-01-31 11:16:55 -05:00
|
|
|
|
2014-09-24 14:59:03 -04:00
|
|
|
boost::asio::io_service& GetService () { return m_Service; };
|
2016-04-02 22:16:49 -04:00
|
|
|
std::shared_ptr<SAMSession> CreateSession (const std::string& id, const std::string& destination, // empty string means transient
|
2014-12-16 16:23:42 -05:00
|
|
|
const std::map<std::string, std::string> * params);
|
2014-09-24 16:39:31 -04:00
|
|
|
void CloseSession (const std::string& id);
|
2016-04-02 22:16:49 -04:00
|
|
|
std::shared_ptr<SAMSession> FindSession (const std::string& id) const;
|
2014-09-24 12:01:26 -04:00
|
|
|
|
2017-01-31 11:16:55 -05:00
|
|
|
/** send raw data to remote endpoint from our UDP Socket */
|
|
|
|
void SendTo(const uint8_t * buf, size_t len, std::shared_ptr<boost::asio::ip::udp::endpoint> remote);
|
2017-01-29 20:38:18 -05:00
|
|
|
|
2014-09-24 12:01:26 -04:00
|
|
|
private:
|
|
|
|
|
|
|
|
void Run ();
|
|
|
|
|
|
|
|
void Accept ();
|
2014-11-22 16:35:58 -05:00
|
|
|
void HandleAccept(const boost::system::error_code& ecode, std::shared_ptr<SAMSocket> socket);
|
2014-09-24 12:01:26 -04:00
|
|
|
|
2014-10-22 16:42:26 -04:00
|
|
|
void ReceiveDatagram ();
|
|
|
|
void HandleReceivedDatagram (const boost::system::error_code& ecode, std::size_t bytes_transferred);
|
|
|
|
|
2014-09-24 12:01:26 -04:00
|
|
|
private:
|
|
|
|
|
|
|
|
bool m_IsRunning;
|
2017-01-31 11:16:55 -05:00
|
|
|
std::thread * m_Thread;
|
2014-09-24 12:01:26 -04:00
|
|
|
boost::asio::io_service m_Service;
|
|
|
|
boost::asio::ip::tcp::acceptor m_Acceptor;
|
2014-10-22 16:42:26 -04:00
|
|
|
boost::asio::ip::udp::endpoint m_DatagramEndpoint, m_SenderEndpoint;
|
|
|
|
boost::asio::ip::udp::socket m_DatagramSocket;
|
2015-02-20 22:47:36 -05:00
|
|
|
mutable std::mutex m_SessionsMutex;
|
2016-04-02 22:16:49 -04:00
|
|
|
std::map<std::string, std::shared_ptr<SAMSession> > m_Sessions;
|
2014-10-23 21:14:17 -04:00
|
|
|
uint8_t m_DatagramReceiveBuffer[i2p::datagram::MAX_DATAGRAM_SIZE+1];
|
2015-02-20 22:47:36 -05:00
|
|
|
|
|
|
|
public:
|
|
|
|
|
|
|
|
// for HTTP
|
|
|
|
const decltype(m_Sessions)& GetSessions () const { return m_Sessions; };
|
2017-01-31 11:16:55 -05:00
|
|
|
};
|
2014-09-24 12:01:26 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|