blob: d41465ea764824559e0153cb1701cd2a15c58342 [file] [log] [blame]
/*
* dpkg - main program for package management
* remove.c - functionality for removing packages
*
* Copyright © 1995 Ian Jackson <ian@chiark.greenend.org.uk>
*
* This is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This 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, see <http://www.gnu.org/licenses/>.
*/
#include <config.h>
#include <compat.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <errno.h>
#include <ctype.h>
#include <string.h>
#include <fcntl.h>
#include <dirent.h>
#include <unistd.h>
#include <stdlib.h>
#include <stdio.h>
#include <dpkg/i18n.h>
#include <dpkg/dpkg.h>
#include <dpkg/dpkg-db.h>
#include <dpkg/dir.h>
#include <dpkg/options.h>
#include <dpkg/triglib.h>
#include "infodb.h"
#include "filesdb.h"
#include "main.h"
/*
* pkgdepcheck may be a virtual pkg.
*/
static void checkforremoval(struct pkginfo *pkgtoremove,
struct pkginfo *pkgdepcheck,
int *rokp, struct varbuf *raemsgs) {
struct deppossi *possi;
struct pkginfo *depender;
int before, ok;
for (possi = pkgdepcheck->installed.depended; possi; possi = possi->rev_next) {
if (possi->up->type != dep_depends && possi->up->type != dep_predepends) continue;
depender= possi->up->up;
debug(dbg_depcon,"checking depending package `%s'",depender->name);
if (!(depender->status == stat_installed ||
depender->status == stat_triggerspending ||
depender->status == stat_triggersawaited))
continue;
if (ignore_depends(depender)) {
debug(dbg_depcon, "ignoring depending package '%s'", depender->name);
continue;
}
if (dependtry > 1) { if (findbreakcycle(pkgtoremove)) sincenothing= 0; }
before= raemsgs->used;
ok= dependencies_ok(depender,pkgtoremove,raemsgs);
if (ok == 0 && depender->clientdata->istobe == itb_remove) ok= 1;
if (ok == 1)
/* Don't burble about reasons for deferral. */
varbuf_trunc(raemsgs, before);
if (ok < *rokp) *rokp= ok;
}
}
void deferred_remove(struct pkginfo *pkg) {
struct varbuf raemsgs = VARBUF_INIT;
int rok;
struct dependency *dep;
debug(dbg_general,"deferred_remove package %s",pkg->name);
if (pkg->status == stat_notinstalled) {
warning(_("ignoring request to remove %.250s which isn't installed."),
pkg->name);
pkg->clientdata->istobe= itb_normal;
return;
} else if (!f_pending &&
pkg->status == stat_configfiles &&
cipaction->arg_int != act_purge) {
warning(_("ignoring request to remove %.250s, only the config\n"
" files of which are on the system. Use --purge to remove them too."),
pkg->name);
pkg->clientdata->istobe= itb_normal;
return;
}
if (pkg->installed.essential && pkg->status != stat_configfiles)
forcibleerr(fc_removeessential, _("This is an essential package -"
" it should not be removed."));
if (!f_pending)
pkg->want = (cipaction->arg_int == act_purge) ? want_purge : want_deinstall;
if (!f_noact) modstatdb_note(pkg);
debug(dbg_general,"checking dependencies for remove `%s'",pkg->name);
rok= 2;
checkforremoval(pkg,pkg,&rok,&raemsgs);
for (dep= pkg->installed.depends; dep; dep= dep->next) {
if (dep->type != dep_provides) continue;
debug(dbg_depcon,"checking virtual package `%s'",dep->list->ed->name);
checkforremoval(pkg,dep->list->ed,&rok,&raemsgs);
}
if (rok == 1) {
varbuf_destroy(&raemsgs);
pkg->clientdata->istobe= itb_remove;
add_to_queue(pkg);
return;
} else if (rok == 0) {
sincenothing= 0;
varbuf_end_str(&raemsgs);
fprintf(stderr,
_("dpkg: dependency problems prevent removal of %s:\n%s"),
pkg->name, raemsgs.buf);
ohshit(_("dependency problems - not removing"));
} else if (raemsgs.used) {
varbuf_end_str(&raemsgs);
fprintf(stderr,
_("dpkg: %s: dependency problems, but removing anyway as you requested:\n%s"),
pkg->name, raemsgs.buf);
}
varbuf_destroy(&raemsgs);
sincenothing= 0;
if (pkg->eflag & eflag_reinstreq)
forcibleerr(fc_removereinstreq,
_("Package is in a very bad inconsistent state - you should\n"
" reinstall it before attempting a removal."));
ensure_allinstfiles_available();
filesdbinit();
if (f_noact) {
printf(_("Would remove or purge %s ...\n"),pkg->name);
pkg->status= stat_notinstalled;
pkg->clientdata->istobe= itb_normal;
return;
}
oldconffsetflags(pkg->installed.conffiles);
printf(_("Removing %s ...\n"),pkg->name);
log_action("remove", pkg);
trig_activate_packageprocessing(pkg);
if (pkg->status >= stat_halfconfigured) {
static enum pkgstatus oldpkgstatus;
oldpkgstatus= pkg->status;
pkg->status= stat_halfconfigured;
modstatdb_note(pkg);
push_cleanup(cu_prermremove, ~ehflag_normaltidy, NULL, 0, 2,
(void *)pkg, (void *)&oldpkgstatus);
maintainer_script_installed(pkg, PRERMFILE, "pre-removal",
"remove", NULL);
/* Will turn into ‘half-installed’ soon ... */
pkg->status = stat_unpacked;
}
removal_bulk(pkg);
}
static void push_leftover(struct fileinlist **leftoverp,
struct filenamenode *namenode) {
struct fileinlist *newentry;
newentry= nfmalloc(sizeof(struct fileinlist));
newentry->next= *leftoverp;
newentry->namenode= namenode;
*leftoverp= newentry;
}
static void
removal_bulk_remove_file(const char *filename, const char *filetype)
{
/* We need the postrm and list files for --purge. */
if (strcmp(filetype, LISTFILE) == 0 ||
strcmp(filetype, POSTRMFILE) == 0)
return;
debug(dbg_stupidlyverbose, "removal_bulk info not postrm or list");
if (unlink(filename))
ohshite(_("unable to delete control info file `%.250s'"), filename);
debug(dbg_scripts, "removal_bulk info unlinked %s", filename);
}
static void
removal_bulk_remove_files(struct pkginfo *pkg)
{
int before;
struct reversefilelistiter rlistit;
struct fileinlist *leftover;
struct filenamenode *namenode;
static struct varbuf fnvb;
struct stat stab;
pkg->status= stat_halfinstalled;
modstatdb_note(pkg);
push_checkpoint(~ehflag_bombout, ehflag_normaltidy);
reversefilelist_init(&rlistit,pkg->clientdata->files);
leftover = NULL;
while ((namenode= reversefilelist_next(&rlistit))) {
struct filenamenode *usenode;
debug(dbg_eachfile, "removal_bulk `%s' flags=%o",
namenode->name, namenode->flags);
if (namenode->flags & fnnf_old_conff) {
push_leftover(&leftover,namenode);
continue;
}
usenode = namenodetouse(namenode, pkg);
trig_file_activate(usenode, pkg);
varbuf_reset(&fnvb);
varbuf_add_str(&fnvb, instdir);
varbuf_add_str(&fnvb, usenode->name);
before= fnvb.used;
varbuf_add_str(&fnvb, DPKGTEMPEXT);
varbuf_end_str(&fnvb);
debug(dbg_eachfiledetail, "removal_bulk cleaning temp `%s'", fnvb.buf);
ensure_pathname_nonexisting(fnvb.buf);
varbuf_trunc(&fnvb, before);
varbuf_add_str(&fnvb, DPKGNEWEXT);
varbuf_end_str(&fnvb);
debug(dbg_eachfiledetail, "removal_bulk cleaning new `%s'", fnvb.buf);
ensure_pathname_nonexisting(fnvb.buf);
varbuf_trunc(&fnvb, before);
varbuf_end_str(&fnvb);
if (!stat(fnvb.buf,&stab) && S_ISDIR(stab.st_mode)) {
debug(dbg_eachfiledetail, "removal_bulk is a directory");
/* Only delete a directory or a link to one if we're the only
* package which uses it. Other files should only be listed
* in this package (but we don't check). */
if (dir_has_conffiles(namenode, pkg)) {
push_leftover(&leftover,namenode);
continue;
}
if (dir_is_used_by_pkg(namenode, pkg, leftover)) {
push_leftover(&leftover, namenode);
continue;
}
if (dir_is_used_by_others(namenode, pkg))
continue;
}
debug(dbg_eachfiledetail, "removal_bulk removing `%s'", fnvb.buf);
if (!rmdir(fnvb.buf) || errno == ENOENT || errno == ELOOP) continue;
if (errno == ENOTEMPTY || errno == EEXIST) {
debug(dbg_eachfiledetail, "removal_bulk `%s' was not empty, will try again later",
fnvb.buf);
push_leftover(&leftover,namenode);
continue;
} else if (errno == EBUSY || errno == EPERM) {
warning(_("while removing %.250s, unable to remove directory '%.250s': "
"%s - directory may be a mount point?"),
pkg->name, namenode->name, strerror(errno));
push_leftover(&leftover,namenode);
continue;
} else if (errno == EINVAL && strcmp(usenode->name, "/.") == 0) {
debug(dbg_eachfiledetail, "removal_bulk '%s' root directory, cannot remove",
fnvb.buf);
push_leftover(&leftover, namenode);
continue;
}
if (errno != ENOTDIR) ohshite(_("cannot remove `%.250s'"),fnvb.buf);
debug(dbg_eachfiledetail, "removal_bulk unlinking `%s'", fnvb.buf);
if (secure_unlink(fnvb.buf))
ohshite(_("unable to securely remove '%.250s'"), fnvb.buf);
}
write_filelist_except(pkg,leftover,0);
maintainer_script_installed(pkg, POSTRMFILE, "post-removal",
"remove", NULL);
trig_parse_ci(pkgadminfile(pkg, TRIGGERSCIFILE),
trig_cicb_interest_delete, NULL, pkg);
trig_file_interests_save();
debug(dbg_general, "removal_bulk cleaning info directory");
pkg_infodb_foreach(pkg, removal_bulk_remove_file);
dir_sync_path(pkgadmindir());
pkg->status= stat_configfiles;
pkg->installed.essential = false;
modstatdb_note(pkg);
push_checkpoint(~ehflag_bombout, ehflag_normaltidy);
}
static void removal_bulk_remove_leftover_dirs(struct pkginfo *pkg) {
struct reversefilelistiter rlistit;
struct fileinlist *leftover;
struct filenamenode *namenode;
static struct varbuf fnvb;
struct stat stab;
/* We may have modified this previously. */
ensure_packagefiles_available(pkg);
modstatdb_note(pkg);
push_checkpoint(~ehflag_bombout, ehflag_normaltidy);
reversefilelist_init(&rlistit,pkg->clientdata->files);
leftover = NULL;
while ((namenode= reversefilelist_next(&rlistit))) {
struct filenamenode *usenode;
debug(dbg_eachfile, "removal_bulk `%s' flags=%o",
namenode->name, namenode->flags);
if (namenode->flags & fnnf_old_conff) {
/* This can only happen if removal_bulk_remove_configfiles() got
* interrupted half way. */
debug(dbg_eachfiledetail, "removal_bulk expecting only left over dirs, "
"ignoring conffile '%s'", namenode->name);
continue;
}
usenode = namenodetouse(namenode, pkg);
trig_file_activate(usenode, pkg);
varbuf_reset(&fnvb);
varbuf_add_str(&fnvb, instdir);
varbuf_add_str(&fnvb, usenode->name);
varbuf_end_str(&fnvb);
if (!stat(fnvb.buf,&stab) && S_ISDIR(stab.st_mode)) {
debug(dbg_eachfiledetail, "removal_bulk is a directory");
/* Only delete a directory or a link to one if we're the only
* package which uses it. Other files should only be listed
* in this package (but we don't check). */
if (dir_is_used_by_pkg(namenode, pkg, leftover)) {
push_leftover(&leftover, namenode);
continue;
}
if (dir_is_used_by_others(namenode, pkg))
continue;
}
debug(dbg_eachfiledetail, "removal_bulk removing `%s'", fnvb.buf);
if (!rmdir(fnvb.buf) || errno == ENOENT || errno == ELOOP) continue;
if (errno == ENOTEMPTY || errno == EEXIST) {
warning(_("while removing %.250s, directory '%.250s' not empty so not removed."),
pkg->name, namenode->name);
push_leftover(&leftover,namenode);
continue;
} else if (errno == EBUSY || errno == EPERM) {
warning(_("while removing %.250s, unable to remove directory '%.250s': "
"%s - directory may be a mount point?"),
pkg->name, namenode->name, strerror(errno));
push_leftover(&leftover,namenode);
continue;
} else if (errno == EINVAL && strcmp(usenode->name, "/.") == 0) {
debug(dbg_eachfiledetail, "removal_bulk '%s' root directory, cannot remove",
fnvb.buf);
push_leftover(&leftover, namenode);
continue;
}
if (errno != ENOTDIR) ohshite(_("cannot remove `%.250s'"),fnvb.buf);
if (lstat(fnvb.buf, &stab) == 0 && S_ISLNK(stab.st_mode)) {
debug(dbg_eachfiledetail, "removal_bulk is a symlink to a directory");
if (unlink(fnvb.buf))
ohshite(_("cannot remove '%.250s'"), fnvb.buf);
continue;
}
push_leftover(&leftover,namenode);
continue;
}
write_filelist_except(pkg,leftover,0);
modstatdb_note(pkg);
push_checkpoint(~ehflag_bombout, ehflag_normaltidy);
}
static void removal_bulk_remove_configfiles(struct pkginfo *pkg) {
static const char *const removeconffexts[] = { REMOVECONFFEXTS, NULL };
int r, removevbbase;
int conffnameused, conffbasenamelen;
char *conffbasename;
struct conffile *conff, **lconffp;
struct fileinlist *searchfile;
DIR *dsd;
struct dirent *de;
char *p;
const char *const *ext;
printf(_("Purging configuration files for %s ...\n"),pkg->name);
log_action("purge", pkg);
trig_activate_packageprocessing(pkg);
/* We may have modified this above. */
ensure_packagefiles_available(pkg);
/* We're about to remove the configuration, so remove the note
* about which version it was ... */
blankversion(&pkg->configversion);
modstatdb_note(pkg);
/* Remove from our list any conffiles that aren't ours any more or
* are involved in diversions, except if we are the package doing the
* diverting. */
for (lconffp = &pkg->installed.conffiles; (conff = *lconffp) != NULL; ) {
for (searchfile= pkg->clientdata->files;
searchfile && strcmp(searchfile->namenode->name,conff->name);
searchfile= searchfile->next);
if (!searchfile) {
debug(dbg_conff,"removal_bulk conffile not ours any more `%s'",conff->name);
*lconffp= conff->next;
} else if (searchfile->namenode->divert &&
(searchfile->namenode->divert->camefrom ||
(searchfile->namenode->divert->useinstead &&
searchfile->namenode->divert->pkg != pkg))) {
debug(dbg_conff, "removal_bulk conffile '%s' ignored due to diversion",
conff->name);
*lconffp= conff->next;
} else {
debug(dbg_conffdetail,"removal_bulk set to new conffile `%s'",conff->name);
conff->hash = NEWCONFFILEFLAG;
lconffp= &conff->next;
}
}
modstatdb_note(pkg);
for (conff= pkg->installed.conffiles; conff; conff= conff->next) {
static struct varbuf fnvb, removevb;
if (conff->obsolete) {
debug(dbg_conffdetail, "removal_bulk conffile obsolete %s",
conff->name);
}
varbuf_reset(&fnvb);
r= conffderef(pkg, &fnvb, conff->name);
debug(dbg_conffdetail, "removal_bulk conffile `%s' (= `%s')",
conff->name, r == -1 ? "<r==-1>" : fnvb.buf);
if (r == -1) continue;
conffnameused = fnvb.used;
if (unlink(fnvb.buf) && errno != ENOENT && errno != ENOTDIR)
ohshite(_("cannot remove old config file `%.250s' (= `%.250s')"),
conff->name, fnvb.buf);
p= strrchr(fnvb.buf,'/'); if (!p) continue;
*p = '\0';
varbuf_reset(&removevb);
varbuf_add_str(&removevb, fnvb.buf);
varbuf_add_char(&removevb, '/');
removevbbase= removevb.used;
varbuf_end_str(&removevb);
dsd= opendir(removevb.buf);
if (!dsd) {
int e=errno;
debug(dbg_conffdetail, "removal_bulk conffile no dsd %s %s",
fnvb.buf, strerror(e)); errno= e;
if (errno == ENOENT || errno == ENOTDIR) continue;
ohshite(_("cannot read config file dir `%.250s' (from `%.250s')"),
fnvb.buf, conff->name);
}
debug(dbg_conffdetail, "removal_bulk conffile cleaning dsd %s", fnvb.buf);
push_cleanup(cu_closedir, ~0, NULL, 0, 1, (void *)dsd);
*p= '/';
conffbasenamelen= strlen(++p);
conffbasename= fnvb.buf+conffnameused-conffbasenamelen;
while ((de = readdir(dsd)) != NULL) {
debug(dbg_stupidlyverbose, "removal_bulk conffile dsd entry=`%s'"
" conffbasename=`%s' conffnameused=%d conffbasenamelen=%d",
de->d_name, conffbasename, conffnameused, conffbasenamelen);
if (!strncmp(de->d_name,conffbasename,conffbasenamelen)) {
debug(dbg_stupidlyverbose, "removal_bulk conffile dsd entry starts right");
for (ext= removeconffexts; *ext; ext++)
if (!strcmp(*ext,de->d_name+conffbasenamelen)) goto yes_remove;
p= de->d_name+conffbasenamelen;
if (*p++ == '~') {
while (*p && cisdigit(*p)) p++;
if (*p == '~' && !*++p) goto yes_remove;
}
}
debug(dbg_stupidlyverbose, "removal_bulk conffile dsd entry starts wrong");
if (de->d_name[0] == '#' &&
!strncmp(de->d_name+1,conffbasename,conffbasenamelen) &&
!strcmp(de->d_name+1+conffbasenamelen,"#"))
goto yes_remove;
debug(dbg_stupidlyverbose, "removal_bulk conffile dsd entry not it");
continue;
yes_remove:
varbuf_trunc(&removevb, removevbbase);
varbuf_add_str(&removevb, de->d_name);
varbuf_end_str(&removevb);
debug(dbg_conffdetail, "removal_bulk conffile dsd entry removing `%s'",
removevb.buf);
if (unlink(removevb.buf) && errno != ENOENT && errno != ENOTDIR)
ohshite(_("cannot remove old backup config file `%.250s' (of `%.250s')"),
removevb.buf, conff->name);
}
pop_cleanup(ehflag_normaltidy); /* closedir */
}
/* Remove the conffiles from the file list file. */
write_filelist_except(pkg, pkg->clientdata->files, fnnf_old_conff);
pkg->installed.conffiles = NULL;
modstatdb_note(pkg);
maintainer_script_installed(pkg, POSTRMFILE, "post-removal",
"purge", NULL);
}
/*
* This is used both by deferred_remove() in this file, and at the end of
* process_archive() in archives.c if it needs to finish removing a
* conflicting package.
*/
void removal_bulk(struct pkginfo *pkg) {
bool foundpostrm;
debug(dbg_general,"removal_bulk package %s",pkg->name);
if (pkg->status == stat_halfinstalled || pkg->status == stat_unpacked) {
removal_bulk_remove_files(pkg);
}
foundpostrm = pkg_infodb_has_file(pkg, POSTRMFILE);
debug(dbg_general, "removal_bulk purging? foundpostrm=%d",foundpostrm);
if (!foundpostrm && !pkg->installed.conffiles) {
/* If there are no config files and no postrm script then we
* go straight into ‘purge’. */
debug(dbg_general, "removal_bulk no postrm, no conffiles, purging");
pkg->want= want_purge;
blankversion(&pkg->configversion);
} else if (pkg->want == want_purge) {
removal_bulk_remove_configfiles(pkg);
}
/* I.e., either of the two branches above. */
if (pkg->want == want_purge) {
const char *filename;
/* Retry empty directories, and warn on any leftovers that aren't. */
removal_bulk_remove_leftover_dirs(pkg);
filename = pkgadminfile(pkg, LISTFILE);
debug(dbg_general, "removal_bulk purge done, removing list `%s'",
filename);
if (unlink(filename) && errno != ENOENT)
ohshite(_("cannot remove old files list"));
filename = pkgadminfile(pkg, POSTRMFILE);
debug(dbg_general, "removal_bulk purge done, removing postrm `%s'",
filename);
if (unlink(filename) && errno != ENOENT)
ohshite(_("can't remove old postrm script"));
pkg->status= stat_notinstalled;
pkg->want = want_unknown;
/* This will mess up reverse links, but if we follow them
* we won't go back because pkg->status is stat_notinstalled. */
pkgbin_blank(&pkg->installed);
}
pkg->eflag = eflag_ok;
modstatdb_note(pkg);
debug(dbg_general, "removal done");
}