#ifndef DATAGRAM_H__
#define DATAGRAM_H__

#include <inttypes.h>
#include <memory>
#include <functional>
#include <map>
#include "Base.h"
#include "Identity.h"
#include "LeaseSet.h"
#include "I2NPProtocol.h"
#include "Garlic.h"

namespace i2p
{
namespace client
{
	class ClientDestination;
}
namespace datagram
{

  // seconds interval for cleanup timer
  const int DATAGRAM_SESSION_CLEANUP_INTERVAL = 3;
  // milliseconds for max session idle time (10 minutes)
  const uint64_t DATAGRAM_SESSION_MAX_IDLE = 3600 * 1000;

  
  class DatagramSession
  {
  public:
    DatagramSession(i2p::client::ClientDestination * localDestination,
                    const i2p::data::IdentHash & remoteIdent);

    /** send an i2np message to remote endpoint for this session */
    void SendMsg(std::shared_ptr<I2NPMessage> msg);
    /** get the last time in milliseconds for when we used this datagram session */
    uint64_t LastActivity() const { return m_LastUse; }
  private:

    /** get next usable routing path, try reusing outbound tunnels  */
    std::shared_ptr<i2p::garlic::GarlicRoutingPath> GetNextRoutingPath();
    /** 
     *  mark current routing path as invalid and clear it
     *  if the outbound tunnel we were using was okay don't use the IBGW in the routing path's lease next time
     */
    void ResetRoutingPath();

    /** get next usable lease, does not fetch or update if expired or have no lease set */
    std::shared_ptr<const i2p::data::Lease> GetNextLease();
    
    void HandleSend(std::shared_ptr<I2NPMessage> msg);
    void HandleGotLeaseSet(std::shared_ptr<const i2p::data::LeaseSet> remoteIdent,
                           std::shared_ptr<I2NPMessage> msg);
    void UpdateLeaseSet(std::shared_ptr<I2NPMessage> msg=nullptr);
    
  private:
    i2p::client::ClientDestination * m_LocalDestination;
    i2p::data::IdentHash m_RemoteIdentity;
    std::shared_ptr<i2p::garlic::GarlicRoutingSession> m_RoutingSession;
    // Ident hash of IBGW that are invalid
    std::vector<i2p::data::IdentHash> m_InvalidIBGW;
    std::shared_ptr<const i2p::data::LeaseSet> m_RemoteLeaseSet;
    uint64_t m_LastUse;
  };
  
	const size_t MAX_DATAGRAM_SIZE = 32768;  
	class DatagramDestination
	{
		typedef std::function<void (const i2p::data::IdentityEx& from, uint16_t fromPort, uint16_t toPort, const uint8_t * buf, size_t len)> Receiver;

		public:

			DatagramDestination (std::shared_ptr<i2p::client::ClientDestination> owner);
			~DatagramDestination ();				

			void SendDatagramTo (const uint8_t * payload, size_t len, const i2p::data::IdentHash& ident, uint16_t fromPort = 0, uint16_t toPort = 0);
			void HandleDataMessagePayload (uint16_t fromPort, uint16_t toPort, const uint8_t * buf, size_t len);

			void SetReceiver (const Receiver& receiver) { m_Receiver = receiver; };
			void ResetReceiver () { m_Receiver = nullptr; };

			void SetReceiver (const Receiver& receiver, uint16_t port) { m_ReceiversByPorts[port] = receiver; };
			void ResetReceiver (uint16_t port) { m_ReceiversByPorts.erase (port); };
    
		private:
      // clean up after next tick
      void ScheduleCleanup();
    
      // clean up stale sessions and expire tags
      void HandleCleanUp(const boost::system::error_code & ecode);
      
      std::shared_ptr<DatagramSession> ObtainSession(const i2p::data::IdentHash & ident);
			
			std::shared_ptr<I2NPMessage> CreateDataMessage (const uint8_t * payload, size_t len, uint16_t fromPort, uint16_t toPort);

			void HandleDatagram (uint16_t fromPort, uint16_t toPort, const uint8_t * buf, size_t len);

		private:
			i2p::client::ClientDestination * m_Owner;
      boost::asio::deadline_timer m_CleanupTimer;
			Receiver m_Receiver; // default
      std::mutex m_SessionsMutex;
      std::map<i2p::data::IdentHash, std::shared_ptr<DatagramSession> > m_Sessions;
			std::map<uint16_t, Receiver> m_ReceiversByPorts;

			i2p::data::GzipInflator m_Inflator;
			i2p::data::GzipDeflator m_Deflator;
	};		
}
}

#endif