123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228 |
- #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) \
- { \
- if (Ver.end() == false) \
- 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;
- }
|