#include <err.h>
#include <getopt.h>
#include <stdio.h>
#include <unistd.h>
#include <pkg.h>
#include "pkgcli.h"
void
usage_autoremove(void)
{
fprintf(stderr, "Usage: pkg autoremove [-Dynq]\n\n");
fprintf(stderr, "For more information see 'pkg help autoremove'.\n");
}
int
exec_autoremove(int argc, char **argv)
{
struct pkgdb *db = NULL;
struct pkg_jobs *jobs = NULL;
int retcode = EXIT_SUCCESS;
int ch;
int nbactions = 0;
pkg_flags f = PKG_FLAG_FORCE;
bool rc = false;
int lock_type = PKGDB_LOCK_ADVISORY;
struct option longopts[] = {
{ "dry-run", no_argument, NULL, 'n' },
{ "no-scripts", no_argument, NULL, 'D' },
{ "quiet", no_argument, NULL, 'q' },
{ "yes", no_argument, NULL, 'y' },
{ NULL, 0, NULL, 0 },
};
while ((ch = getopt_long(argc, argv, "+Dnqy", longopts, NULL)) != -1) {
switch (ch) {
case 'n':
f |= PKG_FLAG_DRY_RUN;
dry_run = true;
lock_type = PKGDB_LOCK_READONLY;
break;
case 'D':
f |= PKG_FLAG_NOSCRIPT;
break;
case 'q':
quiet = true;
break;
case 'y':
yes = true;
break;
default:
usage_autoremove();
return (EXIT_FAILURE);
}
}
argc -= optind;
if (argc != 0) {
usage_autoremove();
return (EXIT_FAILURE);
}
if (dry_run)
retcode = pkgdb_access(PKGDB_MODE_READ, PKGDB_DB_LOCAL);
else
retcode = pkgdb_access(PKGDB_MODE_READ|PKGDB_MODE_WRITE,
PKGDB_DB_LOCAL);
if (retcode == EPKG_ENOACCESS) {
warnx("Insufficient privileges to autoremove packages");
return (EXIT_FAILURE);
} else if (retcode == EPKG_ENODB) {
warnx("No packages installed. Nothing to do!");
return (EXIT_SUCCESS);
} else if (retcode != EPKG_OK) {
warnx("Error accessing the package database");
return (EXIT_FAILURE);
}
if (pkgdb_open(&db, PKGDB_DEFAULT) != EPKG_OK) {
return (EXIT_FAILURE);
}
if (pkgdb_obtain_lock(db, lock_type) != EPKG_OK) {
pkgdb_close(db);
warnx("Cannot get an advisory lock on a database, it is locked by another process");
return (EXIT_FAILURE);
}
if (pkg_jobs_new(&jobs, PKG_JOBS_AUTOREMOVE, db) != EPKG_OK) {
pkgdb_close(db);
return (EXIT_FAILURE);
}
pkg_jobs_set_flags(jobs, f);
if ((retcode = pkg_jobs_solve(jobs)) != EPKG_OK) {
retcode = EXIT_FAILURE;
goto cleanup;
}
if ((nbactions = pkg_jobs_count(jobs)) == 0) {
if (!quiet)
printf("Nothing to do.\n");
goto cleanup;
}
if (!quiet || dry_run) {
print_jobs_summary(jobs,
"Deinstallation has been requested for the following %d packages:\n\n", nbactions);
if (!dry_run)
rc = query_yesno(false,
"\nProceed with deinstalling packages? ");
}
if ((yes || rc ) && !dry_run && ((retcode = pkg_jobs_apply(jobs)) != EPKG_OK)) {
goto cleanup;
}
if (!yes && !rc)
retcode = EXIT_FAILURE;
pkgdb_compact(db);
cleanup:
pkg_jobs_free(jobs);
pkgdb_release_lock(db, lock_type);
pkgdb_close(db);
return (retcode);
}