/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil -*- */
/*
 * Copyright (c) 2011 UCLA
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License version 2 as
 * published by the Free Software Foundation;
 *
 * 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, write to the Free Software
 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
 *
 * Author:  Alexander Afanasyev <alexander.afanasyev@ucla.edu>
 *          Ilya Moiseenko <iliamo@cs.ucla.edu>
 */

#include "ns3/assert.h"
#include "ns3/log.h"
#include "ns3/object.h"
#include "ns3/names.h"
#include "ns3/packet-socket-factory.h"
#include "ns3/config.h"
#include "ns3/simulator.h"
#include "ns3/string.h"
#include "ns3/net-device.h"
#include "ns3/channel.h"
#include "ns3/callback.h"
#include "ns3/node.h"
#include "ns3/core-config.h"
#include "ns3/point-to-point-net-device.h"
#include "ns3/point-to-point-helper.h"
#include "ns3/callback.h"

#include "../model/ndn-net-device-face.hpp"
#include "../model/ndn-l3-protocol.hpp"

#include "ns3/node-list.h"

#include "ns3/data-rate.h"

#include "ndn-face-container.hpp"
#include "ndn-stack-helper.hpp"
#include "utils/dummy-keychain.hpp"

#include <limits>
#include <map>
#include <boost/foreach.hpp>
#include <boost/lexical_cast.hpp>

NS_LOG_COMPONENT_DEFINE("ndn.StackHelper");

namespace ns3 {
namespace ndn {

StackHelper::StackHelper()
  : m_limitsEnabled(false)
  , m_needSetDefaultRoutes(false)
{
  m_ndnFactory.SetTypeId("ns3::ndn::L3Protocol");
  m_strategyFactory.SetTypeId("ns3::ndn::fw::Flooding");
  m_contentStoreFactory.SetTypeId("ns3::ndn::cs::Lru");
  m_fibFactory.SetTypeId("ns3::ndn::fib::Default");
  m_pitFactory.SetTypeId("ns3::ndn::pit::Persistent");

  m_netDeviceCallbacks.push_back(
    std::make_pair(PointToPointNetDevice::GetTypeId(),
                   MakeCallback(&StackHelper::PointToPointNetDeviceCallback, this)));
  // default callback will be fired if non of others callbacks fit or did the job
}

StackHelper::~StackHelper()
{
}

KeyChain&
StackHelper::getKeyChain()
{
  static ::ndn::DummyKeyChain keyChain;
  return keyChain;
}

void
StackHelper::SetStackAttributes(const std::string& attr1, const std::string& value1,
                                const std::string& attr2, const std::string& value2,
                                const std::string& attr3, const std::string& value3,
                                const std::string& attr4, const std::string& value4)
{
  if (attr1 != "")
    m_ndnFactory.Set(attr1, StringValue(value1));
  if (attr2 != "")
    m_ndnFactory.Set(attr2, StringValue(value2));
  if (attr3 != "")
    m_ndnFactory.Set(attr3, StringValue(value3));
  if (attr4 != "")
    m_ndnFactory.Set(attr4, StringValue(value4));
}

void
StackHelper::SetForwardingStrategy(const std::string& strategy, const std::string& attr1,
                                   const std::string& value1, const std::string& attr2,
                                   const std::string& value2, const std::string& attr3,
                                   const std::string& value3, const std::string& attr4,
                                   const std::string& value4)
{
  m_strategyFactory.SetTypeId(strategy);
  if (attr1 != "")
    m_strategyFactory.Set(attr1, StringValue(value1));
  if (attr2 != "")
    m_strategyFactory.Set(attr2, StringValue(value2));
  if (attr3 != "")
    m_strategyFactory.Set(attr3, StringValue(value3));
  if (attr4 != "")
    m_strategyFactory.Set(attr4, StringValue(value4));
}

void
StackHelper::SetContentStore(const std::string& contentStore, const std::string& attr1,
                             const std::string& value1, const std::string& attr2,
                             const std::string& value2, const std::string& attr3,
                             const std::string& value3, const std::string& attr4,
                             const std::string& value4)
{
  m_contentStoreFactory.SetTypeId(contentStore);
  if (attr1 != "")
    m_contentStoreFactory.Set(attr1, StringValue(value1));
  if (attr2 != "")
    m_contentStoreFactory.Set(attr2, StringValue(value2));
  if (attr3 != "")
    m_contentStoreFactory.Set(attr3, StringValue(value3));
  if (attr4 != "")
    m_contentStoreFactory.Set(attr4, StringValue(value4));
}

void
StackHelper::SetPit(const std::string& pitClass, const std::string& attr1,
                    const std::string& value1, const std::string& attr2, const std::string& value2,
                    const std::string& attr3, const std::string& value3, const std::string& attr4,
                    const std::string& value4)
{
  m_pitFactory.SetTypeId(pitClass);
  if (attr1 != "")
    m_pitFactory.Set(attr1, StringValue(value1));
  if (attr2 != "")
    m_pitFactory.Set(attr2, StringValue(value2));
  if (attr3 != "")
    m_pitFactory.Set(attr3, StringValue(value3));
  if (attr4 != "")
    m_pitFactory.Set(attr4, StringValue(value4));
}

void
StackHelper::SetFib(const std::string& fibClass, const std::string& attr1,
                    const std::string& value1, const std::string& attr2, const std::string& value2,
                    const std::string& attr3, const std::string& value3, const std::string& attr4,
                    const std::string& value4)
{
  m_fibFactory.SetTypeId(fibClass);
  if (attr1 != "")
    m_fibFactory.Set(attr1, StringValue(value1));
  if (attr2 != "")
    m_fibFactory.Set(attr2, StringValue(value2));
  if (attr3 != "")
    m_fibFactory.Set(attr3, StringValue(value3));
  if (attr4 != "")
    m_fibFactory.Set(attr4, StringValue(value4));
}

void
StackHelper::SetDefaultRoutes(bool needSet)
{
  NS_LOG_FUNCTION(this << needSet);
  m_needSetDefaultRoutes = needSet;
}

void
StackHelper::EnableLimits(bool enable /* = true*/, Time avgRtt /*=Seconds(0.1)*/,
                          uint32_t avgData /*=1100*/, uint32_t avgInterest /*=40*/)
{
  NS_LOG_INFO("EnableLimits: " << enable);
  m_limitsEnabled = enable;
  m_avgRtt = avgRtt;
  m_avgDataSize = avgData;
  m_avgInterestSize = avgInterest;
}

Ptr<FaceContainer>
StackHelper::Install(const NodeContainer& c) const
{
  Ptr<FaceContainer> faces = Create<FaceContainer>();
  for (NodeContainer::Iterator i = c.Begin(); i != c.End(); ++i) {
    faces->AddAll(Install(*i));
  }
  return faces;
}

Ptr<FaceContainer>
StackHelper::InstallAll() const
{
  return Install(NodeContainer::GetGlobal());
}

Ptr<FaceContainer>
StackHelper::Install(Ptr<Node> node) const
{
  // NS_ASSERT_MSG (m_forwarding, "SetForwardingHelper() should be set prior calling Install()
  // method");
  Ptr<FaceContainer> faces = Create<FaceContainer>();

  if (node->GetObject<L3Protocol>() != 0) {
    NS_FATAL_ERROR("StackHelper::Install (): Installing "
                   "a NdnStack to a node with an existing Ndn object");
    return 0;
  }

  // Create L3Protocol
  Ptr<L3Protocol> ndn = m_ndnFactory.Create<L3Protocol>();

  // Create and aggregate FIB
  // Ptr<Fib> fib = m_fibFactory.Create<Fib> ();
  // ndn->AggregateObject (fib);

  // Create and aggregate PIT
  // ndn->AggregateObject (m_pitFactory.Create<Pit> ());

  // Create and aggregate forwarding strategy
  // ndn->AggregateObject (m_strategyFactory.Create<ForwardingStrategy> ());

  // Create and aggregate content store
  // ndn->AggregateObject (m_contentStoreFactory.Create<ContentStore> ());

  // Aggregate L3Protocol on node
  node->AggregateObject(ndn);

  for (uint32_t index = 0; index < node->GetNDevices(); index++) {
    Ptr<NetDevice> device = node->GetDevice(index);
    // This check does not make sense: LoopbackNetDevice is installed only if IP stack is installed,
    // Normally, ndnSIM works without IP stack, so no reason to check
    // if (DynamicCast<LoopbackNetDevice> (device) != 0)
    //   continue; // don't create face for a LoopbackNetDevice

    shared_ptr<NetDeviceFace> face;

    for (std::list<std::pair<TypeId, NetDeviceFaceCreateCallback>>::const_iterator item =
           m_netDeviceCallbacks.begin();
         item != m_netDeviceCallbacks.end(); item++) {
      if (device->GetInstanceTypeId() == item->first
          || device->GetInstanceTypeId().IsChildOf(item->first)) {
        face = item->second(node, ndn, device);
        if (face != 0)
          break;
      }
    }
    if (face == 0) {
      face = DefaultNetDeviceCallback(node, ndn, device);
    }

    if (m_needSetDefaultRoutes) {
      // default route with lowest priority possible
      AddRoute(node, "/", StaticCast<Face>(face), std::numeric_limits<int32_t>::max());
    }

    face->SetUp();
    faces->Add(face);
  }

  return faces;
}

void
StackHelper::AddNetDeviceFaceCreateCallback(TypeId netDeviceType,
                                            StackHelper::NetDeviceFaceCreateCallback callback)
{
  m_netDeviceCallbacks.push_back(std::make_pair(netDeviceType, callback));
}

void
StackHelper::UpdateNetDeviceFaceCreateCallback(TypeId netDeviceType,
                                               NetDeviceFaceCreateCallback callback)
{
  for (NetDeviceCallbackList::iterator i = m_netDeviceCallbacks.begin();
       i != m_netDeviceCallbacks.end(); i++) {
    if (i->first == netDeviceType) {
      i->second = callback;
      return;
    }
  }
}

void
StackHelper::RemoveNetDeviceFaceCreateCallback(TypeId netDeviceType,
                                               NetDeviceFaceCreateCallback callback)
{
  for (NetDeviceCallbackList::iterator i = m_netDeviceCallbacks.begin();
       i != m_netDeviceCallbacks.end(); i++) {
    if (i->first == netDeviceType) {
      m_netDeviceCallbacks.erase(i);
      return;
    }
  }
}

shared_ptr<NetDeviceFace>
StackHelper::DefaultNetDeviceCallback(Ptr<Node> node, Ptr<L3Protocol> ndn,
                                      Ptr<NetDevice> netDevice) const
{
  NS_LOG_DEBUG("Creating default NetDeviceFace on node " << node->GetId());

  shared_ptr<NetDeviceFace> face = CreateObject<NetDeviceFace>(node, netDevice);

  ndn->AddFace(face);
  NS_LOG_LOGIC("Node " << node->GetId() << ": added NetDeviceFace as face #" << *face);

  return face;
}

shared_ptr<NetDeviceFace>
StackHelper::PointToPointNetDeviceCallback(Ptr<Node> node, Ptr<L3Protocol> ndn,
                                           Ptr<NetDevice> device) const
{
  NS_LOG_DEBUG("Creating point-to-point NetDeviceFace on node " << node->GetId());

  shared_ptr<NetDeviceFace> face = CreateObject<NetDeviceFace>(node, device);

  ndn->AddFace(face);
  NS_LOG_LOGIC("Node " << node->GetId() << ": added NetDeviceFace as face #" << *face);

  if (m_limitsEnabled) {
    // Ptr<Limits> limits = face->GetObject<Limits> ();
    /*if (limits == 0)
    {
        NS_FATAL_ERROR ("Limits are enabled, but the selected forwarding strategy does not support
    limits. Please revise your scenario");
        exit (1);
    }*/

    NS_LOG_INFO("Limits are enabled");
    Ptr<PointToPointNetDevice> p2p = DynamicCast<PointToPointNetDevice>(device);
    if (p2p != 0) {
      // Setup bucket filtering
      // Assume that we know average data packet size, and this size is equal default size
      // Set maximum buckets (averaging over 1 second)

      DataRateValue dataRate;
      device->GetAttribute("DataRate", dataRate);
      TimeValue linkDelay;
      device->GetChannel()->GetAttribute("Delay", linkDelay);

      NS_LOG_INFO("DataRate for this link is " << dataRate.Get());

      double maxInterestPackets =
        1.0 * dataRate.Get().GetBitRate() / 8.0 / (m_avgDataSize + m_avgInterestSize);
      // NS_LOG_INFO ("Max packets per second: " << maxInterestPackets);
      // NS_LOG_INFO ("Max burst: " << m_avgRtt.ToDouble (Time::S) * maxInterestPackets);
      //         NS_LOG_INFO ("MaxLimit: " << (int)(m_avgRtt.ToDouble (Time::S) *
      //         maxInterestPackets));
      /*
      // Set max to BDP
      limits->SetLimits (maxInterestPackets, m_avgRtt.ToDouble (Time::S));
      limits->SetLinkDelay (linkDelay.Get ().ToDouble (Time::S)); */
    }
  }

  return face;
}

Ptr<FaceContainer>
StackHelper::Install(const std::string& nodeName) const
{
  Ptr<Node> node = Names::Find<Node>(nodeName);
  return Install(node);
}

void
StackHelper::AddRoute(Ptr<Node> node, const std::string& prefix, shared_ptr<Face> face, int32_t metric)
{
  NS_LOG_LOGIC("[" << node->GetId() << "]$ route add " << prefix << " via " << *face << " metric "
                   << metric);

  // Ptr<Fib>  fib  = node->GetObject<Fib> ();

  // NameValue prefixValue;
  // prefixValue.DeserializeFromString (prefix, MakeNameChecker ());
  ::ndn::Name name(prefix);
  // fib->Add (name, face, metric);
}

void
StackHelper::AddRoute(Ptr<Node> node, const std::string& prefix, uint32_t faceId, int32_t metric)
{
  Ptr<L3Protocol> ndn = node->GetObject<L3Protocol>();
  NS_ASSERT_MSG(ndn != 0, "Ndn stack should be installed on the node");

  shared_ptr<Face> face = ndn->GetFace(faceId);
  NS_ASSERT_MSG(face != 0, "Face with ID [" << faceId << "] does not exist on node ["
                                            << node->GetId() << "]");

  AddRoute(node, prefix, face, metric);
}

void
StackHelper::AddRoute(const std::string& nodeName, const std::string& prefix, uint32_t faceId,
                      int32_t metric)
{
  Ptr<Node> node = Names::Find<Node>(nodeName);
  NS_ASSERT_MSG(node != 0, "Node [" << nodeName << "] does not exist");

  Ptr<L3Protocol> ndn = node->GetObject<L3Protocol>();
  NS_ASSERT_MSG(ndn != 0, "Ndn stack should be installed on the node");

  shared_ptr<Face> face = ndn->GetFace(faceId);
  NS_ASSERT_MSG(face != 0, "Face with ID [" << faceId << "] does not exist on node [" << nodeName
                                            << "]");

  AddRoute(node, prefix, face, metric);
}

void
StackHelper::AddRoute(Ptr<Node> node, const std::string& prefix, Ptr<Node> otherNode,
                      int32_t metric)
{
  for (uint32_t deviceId = 0; deviceId < node->GetNDevices(); deviceId++) {
    Ptr<PointToPointNetDevice> netDevice =
      DynamicCast<PointToPointNetDevice>(node->GetDevice(deviceId));
    if (netDevice == 0)
      continue;

    Ptr<Channel> channel = netDevice->GetChannel();
    if (channel == 0)
      continue;

    if (channel->GetDevice(0)->GetNode() == otherNode
        || channel->GetDevice(1)->GetNode() == otherNode) {
      Ptr<L3Protocol> ndn = node->GetObject<L3Protocol>();
      NS_ASSERT_MSG(ndn != 0, "Ndn stack should be installed on the node");

      shared_ptr<Face> face = ndn->GetFaceByNetDevice(netDevice);
      NS_ASSERT_MSG(face != 0, "There is no face associated with the p2p link");

      AddRoute(node, prefix, face, metric);

      return;
    }
  }

  NS_FATAL_ERROR("Cannot add route: Node# " << node->GetId() << " and Node# " << otherNode->GetId()
                                            << " are not connected");
}

void
StackHelper::AddRoute(const std::string& nodeName, const std::string& prefix,
                      const std::string& otherNodeName, int32_t metric)
{
  Ptr<Node> node = Names::Find<Node>(nodeName);
  NS_ASSERT_MSG(node != 0, "Node [" << nodeName << "] does not exist");

  Ptr<Node> otherNode = Names::Find<Node>(otherNodeName);
  NS_ASSERT_MSG(otherNode != 0, "Node [" << otherNodeName << "] does not exist");

  AddRoute(node, prefix, otherNode, metric);
}

} // namespace ndn
} // namespace ns3
