762c7db2e8
- Rework the wrapper support to check libpkg version as well as pkg_install version. - Add libfetch to _prebuild_libs. - There are no new features introduced. Notes: the API is not stable, so basically, do not use libpkg in your projects for now. Also there's no manpage for libpkg yet, because the API will change drastically. I repeat, do not use libpkg for now.
182 lines
4.0 KiB
C
182 lines
4.0 KiB
C
/*
|
|
*
|
|
* FreeBSD install - a package for the installation and maintainance
|
|
* of non-core utilities.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
*
|
|
* Jordan K. Hubbard
|
|
* 18 July 1993
|
|
*
|
|
* This is the delete module.
|
|
*
|
|
*/
|
|
|
|
#include <sys/cdefs.h>
|
|
__FBSDID("$FreeBSD$");
|
|
|
|
#include <sys/types.h>
|
|
#include <sys/stat.h>
|
|
#include <getopt.h>
|
|
#include <err.h>
|
|
|
|
#include <pkg.h>
|
|
#include "delete.h"
|
|
|
|
char *Prefix = NULL;
|
|
Boolean CleanDirs = FALSE;
|
|
Boolean Interactive = FALSE;
|
|
Boolean NoDeInstall = FALSE;
|
|
Boolean Recursive = FALSE;
|
|
match_t MatchType = MATCH_GLOB;
|
|
|
|
static void usage(void);
|
|
|
|
static char opts[] = "adDfGhinp:rvxX";
|
|
static struct option longopts[] = {
|
|
{ "all", no_argument, NULL, 'a' },
|
|
{ "clean-dirs", no_argument, NULL, 'd' },
|
|
{ "dry-run", no_argument, NULL, 'n' },
|
|
{ "extended", no_argument, NULL, 'X' },
|
|
{ "force", no_argument, NULL, 'f' },
|
|
{ "help", no_argument, NULL, 'h' },
|
|
{ "interactive",no_argument, NULL, 'i' },
|
|
{ "prefix", required_argument, NULL, 'p' },
|
|
{ "recursive", no_argument, NULL, 'r' },
|
|
{ "regex", no_argument, NULL, 'x' },
|
|
{ "no-glob", no_argument, NULL, 'G' },
|
|
{ "no-script", no_argument, NULL, 'D' },
|
|
{ "no-scripts", no_argument, NULL, 'D' },
|
|
{ "verbose", no_argument, NULL, 'v' },
|
|
{ NULL, 0, NULL, 0 },
|
|
};
|
|
|
|
int
|
|
main(int argc, char **argv)
|
|
{
|
|
int ch, error;
|
|
char **pkgs, **start;
|
|
char *pkgs_split;
|
|
const char *tmp;
|
|
struct stat stat_s;
|
|
|
|
pkg_wrap(PKG_INSTALL_VERSION, argv);
|
|
|
|
pkgs = start = argv;
|
|
while ((ch = getopt_long(argc, argv, opts, longopts, NULL)) != -1)
|
|
switch(ch) {
|
|
case 'v':
|
|
Verbose++;
|
|
break;
|
|
|
|
case 'f':
|
|
Force = TRUE;
|
|
break;
|
|
|
|
case 'p':
|
|
Prefix = optarg;
|
|
break;
|
|
|
|
case 'D':
|
|
NoDeInstall = TRUE;
|
|
break;
|
|
|
|
case 'd':
|
|
CleanDirs = TRUE;
|
|
break;
|
|
|
|
case 'n':
|
|
Fake = TRUE;
|
|
Verbose = TRUE;
|
|
break;
|
|
|
|
case 'a':
|
|
MatchType = MATCH_ALL;
|
|
break;
|
|
|
|
case 'G':
|
|
MatchType = MATCH_EXACT;
|
|
break;
|
|
|
|
case 'x':
|
|
MatchType = MATCH_REGEX;
|
|
break;
|
|
|
|
case 'X':
|
|
MatchType = MATCH_EREGEX;
|
|
break;
|
|
|
|
case 'i':
|
|
Interactive = TRUE;
|
|
break;
|
|
|
|
case 'r':
|
|
Recursive = TRUE;
|
|
break;
|
|
|
|
case 'h':
|
|
default:
|
|
usage();
|
|
break;
|
|
}
|
|
|
|
argc -= optind;
|
|
argv += optind;
|
|
|
|
/* Get all the remaining package names, if any */
|
|
while (*argv) {
|
|
/* Don't try to apply heuristics if arguments are regexs */
|
|
if (MatchType != MATCH_REGEX)
|
|
while ((pkgs_split = strrchr(*argv, (int)'/')) != NULL) {
|
|
*pkgs_split++ = '\0';
|
|
/*
|
|
* If character after the '/' is alphanumeric, then we've found the
|
|
* package name. Otherwise we've come across a trailing '/' and
|
|
* need to continue our quest.
|
|
*/
|
|
if (isalnum(*pkgs_split) || ((MatchType == MATCH_GLOB) && \
|
|
strpbrk(pkgs_split, "*?[]") != NULL)) {
|
|
*argv = pkgs_split;
|
|
break;
|
|
}
|
|
}
|
|
*pkgs++ = *argv++;
|
|
}
|
|
|
|
/* If no packages, yelp */
|
|
if (pkgs == start && MatchType != MATCH_ALL)
|
|
warnx("missing package name(s)"), usage();
|
|
*pkgs = NULL;
|
|
tmp = LOG_DIR;
|
|
(void) stat(tmp, &stat_s);
|
|
if (!Fake && getuid() && geteuid() != stat_s.st_uid) {
|
|
if (!Force)
|
|
errx(1, "you do not own %s, use -f to force", tmp);
|
|
else
|
|
warnx("you do not own %s (proceeding anyways)", tmp);
|
|
}
|
|
if ((error = pkg_perform(start)) != 0) {
|
|
if (Verbose)
|
|
warnx("%d package deletion(s) failed", error);
|
|
return error;
|
|
}
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
static void
|
|
usage(void)
|
|
{
|
|
fprintf(stderr, "%s\n%s\n",
|
|
"usage: pkg_delete [-dDfGinrvxX] [-p prefix] pkg-name ...",
|
|
" pkg_delete -a [flags]");
|
|
exit(1);
|
|
}
|