/* -*-  Mode: C++; c-file-style: "gnu"; indent-tabs-mode:nil; -*- */
/*
 * Copyright (c) 2011 University of California, Los Angeles
 *
 * 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 "fw-stats.h"

#include "ns3/ndn-interest-header.h"
#include "ns3/ndn-content-object-header.h"
#include "ns3/ndn-pit.h"
#include "ns3/ndn-pit-entry.h"

#include "ns3/assert.h"
#include "ns3/log.h"
#include "ns3/simulator.h"


#include <boost/foreach.hpp>
#include <boost/lambda/lambda.hpp>
#include <boost/lambda/bind.hpp>
namespace ll = boost::lambda;

NS_LOG_COMPONENT_DEFINE ("NdnSimFwStats");

namespace ns3 {
namespace ndnSIM {

NS_OBJECT_ENSURE_REGISTERED (FwStats);
  
TypeId
FwStats::GetTypeId (void)
{
  static TypeId tid = TypeId ("ns3::ndnSIM::FwStats")
    .SetGroupName ("Ndn")
    .SetParent <BestRoute> ()
    .AddConstructor <FwStats> ()

    .AddTraceSource ("Stats", "Fired every time stats tree is updated",
                     MakeTraceSourceAccessor (&FwStats::m_statsTrace))
    ;
  return tid;
}
    
FwStats::FwStats ()
{
}

void
FwStats::DoDispose ()
{
  BestRoute::DoDispose ();
  m_statsRefreshEvent.Cancel ();
}

void
FwStats::OnInterest (const Ptr<NdnFace> &face,
                     Ptr<NdnInterestHeader> &header,
                     const Ptr<const Packet> &packet)
{
  super::OnInterest (face, header, packet);
  
  m_stats.Rx (header->GetName ().cut (1), face, packet->GetSize ());

  ScheduleRefreshingIfNecessary ();
}

void
FwStats::OnData (const Ptr<NdnFace> &face,
                 Ptr<NdnContentObjectHeader> &header,
                 Ptr<Packet> &payload,
                 const Ptr<const Packet> &packet)
{
  super::OnData (face, header, payload, packet);
  
  m_stats.Rx (header->GetName ().cut (1), face, packet->GetSize ());

  ScheduleRefreshingIfNecessary ();
}


void
FwStats::FailedToCreatePitEntry (const Ptr<NdnFace> &incomingFace,
                                 Ptr<NdnInterestHeader> header,
                                 const Ptr<const Packet> &packet)
{
  super::FailedToCreatePitEntry (incomingFace, header, packet);

  // Kind of cheating... But at least this way we will have some statistics
  m_stats.NewPitEntry (header->GetName ().cut (1));
  m_stats.Incoming (header->GetName ().cut (1), incomingFace);
  m_stats.Timeout (header->GetName ().cut (1));

  ScheduleRefreshingIfNecessary ();
}

void
FwStats::DidCreatePitEntry (const Ptr<NdnFace> &incomingFace,
                            Ptr<NdnInterestHeader> header,
                            const Ptr<const Packet> &packet,
                            Ptr<NdnPitEntry> pitEntry)
{
  super::DidCreatePitEntry (incomingFace, header, packet, pitEntry);
  
  m_stats.NewPitEntry (header->GetName ().cut (1));
  m_stats.Incoming (header->GetName ().cut (1), incomingFace);
  
  ScheduleRefreshingIfNecessary ();
}

void
FwStats::WillSatisfyPendingInterest (const Ptr<NdnFace> &incomingFace,
                                     Ptr<NdnPitEntry> pitEntry)
{
  super::WillSatisfyPendingInterest (incomingFace, pitEntry);
  
  m_stats.Satisfy (pitEntry->GetPrefix ().cut (1));
  
  ScheduleRefreshingIfNecessary ();
}

void
FwStats::DidSendOutInterest (const Ptr<NdnFace> &outgoingFace,
                             Ptr<NdnInterestHeader> header,
                             const Ptr<const Packet> &packet,
                             Ptr<NdnPitEntry> pitEntry)
{
  super::DidSendOutInterest (outgoingFace, header, packet, pitEntry);

  m_stats.Outgoing (header->GetName ().cut (1), outgoingFace);
  m_stats.Tx (header->GetName ().cut (1), outgoingFace, packet->GetSize ());
  
  ScheduleRefreshingIfNecessary ();
}

void
FwStats::DidSendOutData (const Ptr<NdnFace> &face,
                         Ptr<const NdnContentObjectHeader> header,
                         Ptr<const Packet> payload,
                         const Ptr<const Packet> &packet)
{
  super::DidSendOutData (face, header, payload, packet);

  m_stats.Tx (header->GetName ().cut (1), face, packet->GetSize ());
  
  ScheduleRefreshingIfNecessary ();
}


void
FwStats::WillErasePendingInterest (Ptr<NdnPitEntry> pitEntry)
{
  super::WillErasePendingInterest (pitEntry);

  m_stats.Timeout (pitEntry->GetPrefix ().cut (1));
  
  ScheduleRefreshingIfNecessary ();
}

void
FwStats::ScheduleRefreshingIfNecessary ()
{
  if (m_statsRefreshEvent.IsRunning ()) return;
  m_statsRefreshEvent = Simulator::Schedule (Seconds (1.0), &FwStats::RefreshStats, this);
}

void
FwStats::RefreshStats ()
{
  m_stats.Step ();
  m_statsTrace (this, m_stats);
  
  NS_LOG_DEBUG (m_stats["/"]);

  if (!m_stats["/"].IsZero ())
    {
      m_statsRefreshEvent = Simulator::Schedule (Seconds (1.0), &FwStats::RefreshStats, this);
    }
}

void
FwStats::RemoveFace (Ptr<NdnFace> face)
{
  m_stats.RemoveFace (face);

  super::RemoveFace (face);
}


} // namespace ndnSIM
} // namespace ns3
