/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
/*
 * Copyright (c) 2013-2025 Regents of the University of California.
 *
 * This file is part of ndn-cxx library (NDN C++ library with eXperimental eXtensions).
 *
 * ndn-cxx library is free software: you can redistribute it and/or modify it under the
 * terms of the GNU Lesser General Public License as published by the Free Software
 * Foundation, either version 3 of the License, or (at your option) any later version.
 *
 * ndn-cxx library 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 Lesser General Public License for more details.
 *
 * You should have received copies of the GNU General Public License and GNU Lesser
 * General Public License along with ndn-cxx, e.g., in COPYING.md file.  If not, see
 * <http://www.gnu.org/licenses/>.
 *
 * See AUTHORS.md for complete list of ndn-cxx authors and contributors.
 */

#include "ndn-cxx/util/dummy-client-face.hpp"
#include "ndn-cxx/mgmt/nfd/control-command.hpp"
#include "ndn-cxx/mgmt/nfd/controller.hpp"

#include "tests/test-common.hpp"
#include "tests/unit/io-key-chain-fixture.hpp"

namespace ndn::tests {

BOOST_AUTO_TEST_SUITE(Util)
BOOST_FIXTURE_TEST_SUITE(TestDummyClientFace, IoKeyChainFixture)

BOOST_AUTO_TEST_CASE(ProcessEventsOverride)
{
  bool isOverrideInvoked = false;
  auto override = [&] (time::milliseconds timeout) {
    isOverrideInvoked = true;
    BOOST_CHECK_EQUAL(timeout, 200_ms);
  };

  DummyClientFace face(m_io, {false, false, override});
  face.processEvents(200_ms);
  BOOST_CHECK(isOverrideInvoked);
}

BOOST_AUTO_TEST_CASE(RegistrationReply)
{
  DummyClientFace::Options opts;
  opts.enableRegistrationReply = true;
  opts.registrationReplyFaceId = 3001;
  DummyClientFace face(m_io, m_keyChain, opts);

  ndn::nfd::Controller controller(face, m_keyChain);
  ndn::nfd::ControlParameters params;
  bool didRibRegisterSucceed = false;
  controller.start<ndn::nfd::RibRegisterCommand>(
    ndn::nfd::ControlParameters()
      .setName("/Q")
      .setOrigin(ndn::nfd::ROUTE_ORIGIN_NLSR)
      .setCost(2400)
      .setFlags(0),
    [&] (const ndn::nfd::ControlParameters& p) {
      BOOST_CHECK_EQUAL(p.getName(), "/Q");
      BOOST_CHECK_EQUAL(p.getFaceId(), 3001);
      BOOST_CHECK_EQUAL(p.getOrigin(), ndn::nfd::ROUTE_ORIGIN_NLSR);
      BOOST_CHECK_EQUAL(p.getCost(), 2400);
      BOOST_CHECK_EQUAL(p.getFlags(), 0);
      didRibRegisterSucceed = true;
    },
    [] (const ndn::nfd::ControlResponse& r) {
      BOOST_TEST_FAIL("RibRegisterCommand failed " << r);
    });
  advanceClocks(1_ms, 2);
  BOOST_CHECK(didRibRegisterSucceed);

  bool didRibUnregisterSucceed = false;
  controller.start<ndn::nfd::RibUnregisterCommand>(
    ndn::nfd::ControlParameters()
      .setName("/Q")
      .setOrigin(ndn::nfd::ROUTE_ORIGIN_NLSR),
    [&] (const ndn::nfd::ControlParameters& p) {
      BOOST_CHECK_EQUAL(p.getName(), "/Q");
      BOOST_CHECK_EQUAL(p.getFaceId(), 3001);
      BOOST_CHECK_EQUAL(p.getOrigin(), ndn::nfd::ROUTE_ORIGIN_NLSR);
      didRibUnregisterSucceed = true;
    },
    [] (const ndn::nfd::ControlResponse& r) {
      BOOST_TEST_FAIL("RibUnregisterCommand failed " << r);
    });
  advanceClocks(1_ms, 2);
  BOOST_CHECK(didRibUnregisterSucceed);
}

BOOST_AUTO_TEST_CASE(BroadcastLink)
{
  DummyClientFace face1(m_io, m_keyChain, {true, true});
  DummyClientFace face2(m_io, m_keyChain, {true, true});
  face1.linkTo(face2);

  int nFace1Interest = 0;
  int nFace2Interest = 0;
  face1.setInterestFilter("/face1",
                          [&] (const InterestFilter&, const Interest& interest) {
                            BOOST_CHECK_EQUAL(interest.getName(), "/face1/data");
                            nFace1Interest++;
                            face1.put(makeNack(interest, lp::NackReason::NO_ROUTE));
                          }, nullptr, nullptr);
  face2.setInterestFilter("/face2",
                          [&] (const InterestFilter&, const Interest& interest) {
                            BOOST_CHECK_EQUAL(interest.getName(), "/face2/data");
                            nFace2Interest++;
                            face2.put(*makeData("/face2/data"));
                            return;
                          }, nullptr, nullptr);

  advanceClocks(25_ms, 4);

  int nFace1Data = 0;
  int nFace2Nack = 0;
  face1.expressInterest(*makeInterest("/face2/data"),
                        [&] (const Interest& i, const Data& d) {
                          BOOST_CHECK_EQUAL(d.getName(), "/face2/data");
                          nFace1Data++;
                        }, nullptr, nullptr);
  face2.expressInterest(*makeInterest("/face1/data"),
                        [&] (const Interest& i, const Data& d) {
                          BOOST_CHECK(false);
                        },
                        [&] (const Interest& i, const lp::Nack& n) {
                          BOOST_CHECK_EQUAL(n.getInterest().getName(), "/face1/data");
                          nFace2Nack++;
                        }, nullptr);

  advanceClocks(10_ms, 100);

  BOOST_CHECK_EQUAL(nFace1Data, 1);
  BOOST_CHECK_EQUAL(nFace2Nack, 1);
  BOOST_CHECK_EQUAL(nFace1Interest, 1);
  BOOST_CHECK_EQUAL(nFace2Interest, 1);
}

BOOST_AUTO_TEST_CASE(BroadcastLinkDestroy)
{
  DummyClientFace face1(m_io, m_keyChain, {true, true});
  DummyClientFace face2(m_io, m_keyChain, {true, true});

  face1.linkTo(face2);
  face2.unlink();
  BOOST_CHECK(face1.m_bcastLink == nullptr);

  DummyClientFace face3(m_io, m_keyChain, {true, true});
  face1.linkTo(face2);
  face3.linkTo(face1);
  face2.unlink();
  BOOST_CHECK(face1.m_bcastLink != nullptr);
}

BOOST_AUTO_TEST_CASE(AlreadyLinkException)
{
  DummyClientFace face1(m_io, m_keyChain, DummyClientFace::Options{true, true});
  DummyClientFace face2(m_io, m_keyChain, DummyClientFace::Options{true, true});
  DummyClientFace face3(m_io, m_keyChain, DummyClientFace::Options{true, true});
  DummyClientFace face4(m_io, m_keyChain, DummyClientFace::Options{true, true});

  face1.linkTo(face2);
  face3.linkTo(face4);

  BOOST_CHECK_THROW(face2.linkTo(face3), DummyClientFace::AlreadyLinkedError);
}

BOOST_AUTO_TEST_SUITE_END() // TestDummyClientFace
BOOST_AUTO_TEST_SUITE_END() // Util

} // namespace ndn::tests
