#include <apt-pkg/pkgcache.h>
#include <apt-pkg/cacheset.h>
#include <apt-pkg/debsystem.h>
#include <apt-pkg/fileutl.h>
#include <apt-pkg/statechanges.h>
#include <apt-pkg/prettyprinters.h>

#include <algorithm>
#include <memory>

namespace APT
{

class StateChanges::Private
{
public:
   APT::VersionVector hold;
   APT::VersionVector unhold;
   APT::VersionVector install;
   APT::VersionVector deinstall;
   APT::VersionVector purge;
   APT::VersionVector error;
};

#define APT_GETTERSETTER(Name, Container) \
void StateChanges::Name(pkgCache::VerIterator const &Ver) \
{ \
   Container.push_back(Ver); \
}\
APT::VersionVector& StateChanges::Name() \
{ \
   return Container; \
}
APT_GETTERSETTER(Hold, d->hold)
APT_GETTERSETTER(Unhold, d->unhold)
APT_GETTERSETTER(Install, d->install)
APT_GETTERSETTER(Remove, d->deinstall)
APT_GETTERSETTER(Purge, d->purge)
#undef APT_GETTERSETTER
APT::VersionVector& StateChanges::Error()
{
   return d->error;
}

void StateChanges::clear()
{
   d->hold.clear();
   d->unhold.clear();
   d->install.clear();
   d->deinstall.clear();
   d->purge.clear();
   d->error.clear();
}

bool StateChanges::empty() const
{
   return d->hold.empty() &&
      d->unhold.empty() &&
      d->install.empty() &&
      d->deinstall.empty() &&
      d->purge.empty() &&
      d->error.empty();
}

bool StateChanges::Save(bool const DiscardOutput)
{
   bool const Debug = _config->FindB("Debug::pkgDpkgPm", false);
   d->error.clear();
   if (d->hold.empty() && d->unhold.empty() && d->install.empty() && d->deinstall.empty() && d->purge.empty())
      return true;

   std::vector<std::string> Args = debSystem::GetDpkgBaseCommand();
   // ensure dpkg knows about the package so that it keeps the status we set
   if (d->hold.empty() == false || d->install.empty() == false)
   {
      APT::VersionVector makeDpkgAvailable;
      auto const notInstalled = [](pkgCache::VerIterator const &V) { return V.ParentPkg()->CurrentVer == 0; };
      std::copy_if(d->hold.begin(), d->hold.end(), std::back_inserter(makeDpkgAvailable), notInstalled);
      std::copy_if(d->install.begin(), d->install.end(), std::back_inserter(makeDpkgAvailable), notInstalled);

      if (makeDpkgAvailable.empty() == false)
      {
	 auto const BaseArgs = Args.size();
	 Args.push_back("--merge-avail");
	 // FIXME: supported only since 1.17.7 in dpkg
	 Args.push_back("-");
	 int dummyAvail = -1;
	 if (Debug)
	 {
	    for (auto const &V: makeDpkgAvailable)
	    {
	       std::clog << "echo 'Dummy record for " << V.ParentPkg().FullName(false) << "' | ";
	       std::copy(Args.begin(), Args.end(), std::ostream_iterator<std::string>(std::clog, " "));
	       std::clog << std::endl;
	    }
	 }
	 else
	 {
	    pid_t const dpkgMergeAvail = debSystem::ExecDpkg(Args, &dummyAvail, nullptr, true);

	    FILE* dpkg = fdopen(dummyAvail, "w");
	    for (auto const &V: makeDpkgAvailable)
	       fprintf(dpkg, "Package: %s\nVersion: 0~\nArchitecture: %s\nMaintainer: Dummy Example <dummy@example.org>\n"
		     "Description: dummy package record\n A record is needed to put a package on hold, so here it is.\n\n", V.ParentPkg().Name(), V.Arch());
	    fclose(dpkg);

	    ExecWait(dpkgMergeAvail, "dpkg --merge-avail", true);
	 }
	 Args.erase(Args.begin() + BaseArgs, Args.end());
      }
   }
   bool const dpkgMultiArch = _system->MultiArchSupported();

   Args.push_back("--set-selections");
   if (Debug)
   {
      std::string state;
      auto const dpkgName = [&](pkgCache::VerIterator const &V) {
	 pkgCache::PkgIterator P = V.ParentPkg();
	 if (strcmp(V.Arch(), "none") == 0)
	    ioprintf(std::clog, "echo '%s %s' | ", P.Name(), state.c_str());
	 else if (dpkgMultiArch == false)
	    ioprintf(std::clog, "echo '%s %s' | ", P.FullName(true).c_str(), state.c_str());
	 else
	    ioprintf(std::clog, "echo '%s:%s %s' | ", P.Name(), V.Arch(), state.c_str());
	 std::copy(Args.begin(), Args.end(), std::ostream_iterator<std::string>(std::clog, " "));
	 std::clog << std::endl;
      };
      for (auto const &V: d->unhold)
      {
	 if (V.ParentPkg()->CurrentVer != 0)
	    state = "install";
	 else
	    state = "deinstall";
	 dpkgName(V);
      }
      if (d->purge.empty() == false)
      {
	 state = "purge";
	 std::for_each(d->purge.begin(), d->purge.end(), dpkgName);
      }
      if (d->deinstall.empty() == false)
      {
	 state = "deinstall";
	 std::for_each(d->deinstall.begin(), d->deinstall.end(), dpkgName);
      }
      if (d->hold.empty() == false)
      {
	 state = "hold";
	 std::for_each(d->hold.begin(), d->hold.end(), dpkgName);
      }
      if (d->install.empty() == false)
      {
	 state = "install";
	 std::for_each(d->install.begin(), d->install.end(), dpkgName);
      }
   }
   else
   {
      int selections = -1;
      pid_t const dpkgSelections = debSystem::ExecDpkg(Args, &selections, nullptr, DiscardOutput);

      FILE* dpkg = fdopen(selections, "w");
      std::string state;
      auto const dpkgName = [&](pkgCache::VerIterator const &V) {
	 pkgCache::PkgIterator P = V.ParentPkg();
	 if (strcmp(V.Arch(), "none") == 0)
	    fprintf(dpkg, "%s %s\n", P.Name(), state.c_str());
	 else if (dpkgMultiArch == false)
	    fprintf(dpkg, "%s %s\n", P.FullName(true).c_str(), state.c_str());
	 else
	    fprintf(dpkg, "%s:%s %s\n", P.Name(), V.Arch(), state.c_str());
      };
      for (auto const &V: d->unhold)
      {
	 if (V.ParentPkg()->CurrentVer != 0)
	    state = "install";
	 else
	    state = "deinstall";
	 dpkgName(V);
      }
      if (d->purge.empty() == false)
      {
	 state = "purge";
	 std::for_each(d->purge.begin(), d->purge.end(), dpkgName);
      }
      if (d->deinstall.empty() == false)
      {
	 state = "deinstall";
	 std::for_each(d->deinstall.begin(), d->deinstall.end(), dpkgName);
      }
      if (d->hold.empty() == false)
      {
	 state = "hold";
	 std::for_each(d->hold.begin(), d->hold.end(), dpkgName);
      }
      if (d->install.empty() == false)
      {
	 state = "install";
	 std::for_each(d->install.begin(), d->install.end(), dpkgName);
      }
      fclose(dpkg);

      if (ExecWait(dpkgSelections, "dpkg --set-selections") == false)
      {
	 std::move(d->purge.begin(), d->purge.end(), std::back_inserter(d->error));
	 d->purge.clear();
	 std::move(d->deinstall.begin(), d->deinstall.end(), std::back_inserter(d->error));
	 d->deinstall.clear();
	 std::move(d->hold.begin(), d->hold.end(), std::back_inserter(d->error));
	 d->hold.clear();
	 std::move(d->unhold.begin(), d->unhold.end(), std::back_inserter(d->error));
	 d->unhold.clear();
	 std::move(d->install.begin(), d->install.end(), std::back_inserter(d->error));
	 d->install.clear();
      }
   }
   return d->error.empty();
}

StateChanges::StateChanges() : d(new StateChanges::Private()) {}
StateChanges::StateChanges(StateChanges&&) = default;
StateChanges& StateChanges::operator=(StateChanges&&) = default;
StateChanges::~StateChanges() = default;

}