/* -*-  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 ("ndn.fw.Stats");

namespace ns3 {
namespace ndn {

using namespace ndnSIM;

namespace fw {

NS_OBJECT_ENSURE_REGISTERED (FwStats);
  
TypeId
FwStats::GetTypeId (void)
{
  static TypeId tid = TypeId ("ns3::ndn::fw::Stats")
    .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<Face> &face,
                     Ptr<InterestHeader> &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<Face> &face,
                 Ptr<ContentObjectHeader> &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<Face> &incomingFace,
                                 Ptr<InterestHeader> 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<Face> &incomingFace,
                            Ptr<InterestHeader> header,
                            const Ptr<const Packet> &packet,
                            Ptr<pit::Entry> 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<Face> &incomingFace,
                                     Ptr<pit::Entry> pitEntry)
{
  super::WillSatisfyPendingInterest (incomingFace, pitEntry);
  
  m_stats.Satisfy (pitEntry->GetPrefix ().cut (1));
  
  ScheduleRefreshingIfNecessary ();
}

void
FwStats::DidSendOutInterest (const Ptr<Face> &outgoingFace,
                             Ptr<InterestHeader> header,
                             const Ptr<const Packet> &packet,
                             Ptr<pit::Entry> 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<Face> &face,
                         Ptr<const ContentObjectHeader> 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::WillEraseTimedOutPendingInterest (Ptr<pit::Entry> pitEntry)
{
  super::WillEraseTimedOutPendingInterest (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<Face> face)
{
  m_stats.RemoveFace (face);

  super::RemoveFace (face);
}


} // namespace fw
} // namespace ndn
} // namespace ns3
