681 lines
17 KiB
C
681 lines
17 KiB
C
|
/* $NetBSD: perform.c,v 1.61 2010/02/20 04:40:03 joerg Exp $ */
|
||
|
|
||
|
#if HAVE_CONFIG_H
|
||
|
#include "config.h"
|
||
|
#endif
|
||
|
#ifndef __minix
|
||
|
#include <nbcompat.h>
|
||
|
#endif
|
||
|
#if HAVE_SYS_CDEFS_H
|
||
|
#include <sys/cdefs.h>
|
||
|
#endif
|
||
|
#if HAVE_SYS_QUEUE_H
|
||
|
#include <sys/queue.h>
|
||
|
#endif
|
||
|
#if HAVE_SYS_WAIT_H
|
||
|
#include <sys/wait.h>
|
||
|
#endif
|
||
|
#ifndef __minix
|
||
|
__RCSID("$NetBSD: perform.c,v 1.61 2010/02/20 04:40:03 joerg Exp $");
|
||
|
#endif
|
||
|
|
||
|
/*-
|
||
|
* Copyright (c) 2008 Joerg Sonnenberger <joerg@NetBSD.org>.
|
||
|
* All rights reserved.
|
||
|
*
|
||
|
* 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.
|
||
|
*
|
||
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||
|
* ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||
|
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
|
||
|
* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
|
||
|
* COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
|
||
|
* INCIDENTAL, SPECIAL, EXEMPLARY OR CONSEQUENTIAL DAMAGES (INCLUDING,
|
||
|
* BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
||
|
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
|
||
|
* AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
||
|
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
|
||
|
* OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
||
|
* SUCH DAMAGE.
|
||
|
*/
|
||
|
|
||
|
/*
|
||
|
* 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
|
||
|
* 23 Aug 1993
|
||
|
*
|
||
|
* This is the main body of the info module.
|
||
|
*
|
||
|
*/
|
||
|
|
||
|
#include "lib.h"
|
||
|
#include "info.h"
|
||
|
|
||
|
#if HAVE_SYS_TYPES_H
|
||
|
#include <sys/types.h>
|
||
|
#endif
|
||
|
#if HAVE_SYS_STAT_H
|
||
|
#include <sys/stat.h>
|
||
|
#endif
|
||
|
|
||
|
#ifndef BOOTSTRAP
|
||
|
#include <archive.h>
|
||
|
#include <archive_entry.h>
|
||
|
#endif
|
||
|
#if HAVE_ERR_H
|
||
|
#include <err.h>
|
||
|
#endif
|
||
|
#if HAVE_ERRNO_H
|
||
|
#include <errno.h>
|
||
|
#endif
|
||
|
#if HAVE_FCNTL_H
|
||
|
#include <fcntl.h>
|
||
|
#endif
|
||
|
#if HAVE_SIGNAL_H
|
||
|
#include <signal.h>
|
||
|
#endif
|
||
|
#if HAVE_DIRENT_H
|
||
|
#include <dirent.h>
|
||
|
#endif
|
||
|
#if HAVE_CTYPE_H
|
||
|
#include <ctype.h>
|
||
|
#endif
|
||
|
#include <stddef.h>
|
||
|
|
||
|
#define LOAD_CONTENTS (1 << 0)
|
||
|
#define LOAD_COMMENT (1 << 1)
|
||
|
#define LOAD_DESC (1 << 2)
|
||
|
#define LOAD_INSTALL (1 << 3)
|
||
|
#define LOAD_DEINSTALL (1 << 4)
|
||
|
#define LOAD_DISPLAY (1 << 5)
|
||
|
#define LOAD_MTREE (1 << 6)
|
||
|
#define LOAD_BUILD_VERSION (1 << 7)
|
||
|
#define LOAD_BUILD_INFO (1 << 8)
|
||
|
#define LOAD_SIZE_PKG (1 << 9)
|
||
|
#define LOAD_SIZE_ALL (1 << 10)
|
||
|
#define LOAD_PRESERVE (1 << 11)
|
||
|
#define LOAD_VIEWS (1 << 12)
|
||
|
#define LOAD_REQUIRED_BY (1 << 13)
|
||
|
#define LOAD_INSTALLED_INFO (1 << 14)
|
||
|
|
||
|
static const struct pkg_meta_desc {
|
||
|
size_t entry_offset;
|
||
|
const char *entry_filename;
|
||
|
int entry_mask;
|
||
|
int required_file;
|
||
|
} pkg_meta_descriptors[] = {
|
||
|
{ offsetof(struct pkg_meta, meta_contents), CONTENTS_FNAME,
|
||
|
LOAD_CONTENTS, 1},
|
||
|
{ offsetof(struct pkg_meta, meta_comment), COMMENT_FNAME,
|
||
|
LOAD_COMMENT, 1 },
|
||
|
{ offsetof(struct pkg_meta, meta_desc), DESC_FNAME,
|
||
|
LOAD_DESC, 1 },
|
||
|
{ offsetof(struct pkg_meta, meta_install), INSTALL_FNAME,
|
||
|
LOAD_INSTALL, 0 },
|
||
|
{ offsetof(struct pkg_meta, meta_deinstall), DEINSTALL_FNAME,
|
||
|
LOAD_DEINSTALL, 0 },
|
||
|
{ offsetof(struct pkg_meta, meta_display), DISPLAY_FNAME,
|
||
|
LOAD_DISPLAY, 0 },
|
||
|
{ offsetof(struct pkg_meta, meta_mtree), MTREE_FNAME,
|
||
|
LOAD_MTREE, 0 },
|
||
|
{ offsetof(struct pkg_meta, meta_build_version), BUILD_VERSION_FNAME,
|
||
|
LOAD_BUILD_VERSION, 0 },
|
||
|
{ offsetof(struct pkg_meta, meta_build_info), BUILD_INFO_FNAME,
|
||
|
LOAD_BUILD_INFO, 0 },
|
||
|
{ offsetof(struct pkg_meta, meta_size_pkg), SIZE_PKG_FNAME,
|
||
|
LOAD_SIZE_PKG, 0 },
|
||
|
{ offsetof(struct pkg_meta, meta_size_all), SIZE_ALL_FNAME,
|
||
|
LOAD_SIZE_ALL, 0 },
|
||
|
{ offsetof(struct pkg_meta, meta_preserve), PRESERVE_FNAME,
|
||
|
LOAD_PRESERVE, 0 },
|
||
|
{ offsetof(struct pkg_meta, meta_views), VIEWS_FNAME,
|
||
|
LOAD_VIEWS, 0 },
|
||
|
{ offsetof(struct pkg_meta, meta_required_by), REQUIRED_BY_FNAME,
|
||
|
LOAD_REQUIRED_BY, 0 },
|
||
|
{ offsetof(struct pkg_meta, meta_installed_info), INSTALLED_INFO_FNAME,
|
||
|
LOAD_INSTALLED_INFO, 0 },
|
||
|
{ 0, NULL, 0, 0 },
|
||
|
};
|
||
|
|
||
|
static int desired_meta_data;
|
||
|
|
||
|
static void
|
||
|
free_pkg_meta(struct pkg_meta *meta)
|
||
|
{
|
||
|
const struct pkg_meta_desc *descr;
|
||
|
|
||
|
for (descr = pkg_meta_descriptors; descr->entry_filename; ++descr)
|
||
|
free(*(char **)((char *)meta + descr->entry_offset));
|
||
|
|
||
|
free(meta);
|
||
|
}
|
||
|
|
||
|
#ifndef BOOTSTRAP
|
||
|
static struct pkg_meta *
|
||
|
read_meta_data_from_archive(struct archive *archive,
|
||
|
struct archive_entry *entry)
|
||
|
{
|
||
|
struct pkg_meta *meta;
|
||
|
const char *fname;
|
||
|
const struct pkg_meta_desc *descr, *last_descr;
|
||
|
char **target;
|
||
|
#ifndef __minix
|
||
|
int64_t size;
|
||
|
#else
|
||
|
ssize_t size;
|
||
|
#endif
|
||
|
int r, found_required;
|
||
|
|
||
|
found_required = 0;
|
||
|
|
||
|
meta = xcalloc(1, sizeof(*meta));
|
||
|
|
||
|
last_descr = 0;
|
||
|
if (entry != NULL) {
|
||
|
r = ARCHIVE_OK;
|
||
|
goto has_entry;
|
||
|
}
|
||
|
|
||
|
while ((r = archive_read_next_header(archive, &entry)) == ARCHIVE_OK) {
|
||
|
has_entry:
|
||
|
fname = archive_entry_pathname(entry);
|
||
|
|
||
|
for (descr = pkg_meta_descriptors; descr->entry_filename;
|
||
|
++descr) {
|
||
|
if (strcmp(descr->entry_filename, fname) == 0)
|
||
|
break;
|
||
|
}
|
||
|
if (descr->entry_filename == NULL)
|
||
|
break;
|
||
|
|
||
|
if (descr->required_file)
|
||
|
++found_required;
|
||
|
|
||
|
target = (char **)((char *)meta + descr->entry_offset);
|
||
|
if (*target)
|
||
|
errx(2, "duplicate entry, package corrupt");
|
||
|
if (descr < last_descr)
|
||
|
warnx("misordered package, continuing");
|
||
|
else
|
||
|
last_descr = descr;
|
||
|
|
||
|
if ((descr->entry_mask & desired_meta_data) == 0) {
|
||
|
if (archive_read_data_skip(archive))
|
||
|
errx(2, "cannot read package meta data");
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
size = archive_entry_size(entry);
|
||
|
/* archive_read_data handles this case using multiple reads
|
||
|
* if (size > SSIZE_MAX - 1)
|
||
|
* errx(2, "package meta data too large to process");
|
||
|
*/
|
||
|
*target = xmalloc(size + 1);
|
||
|
if (archive_read_data(archive, *target, size) != size)
|
||
|
errx(2, "cannot read package meta data");
|
||
|
(*target)[size] = '\0';
|
||
|
}
|
||
|
|
||
|
for (descr = pkg_meta_descriptors; descr->entry_filename; ++descr) {
|
||
|
if (descr->required_file)
|
||
|
--found_required;
|
||
|
}
|
||
|
|
||
|
meta->is_installed = 0;
|
||
|
if (found_required != 0 || (r != ARCHIVE_OK && r != ARCHIVE_EOF)) {
|
||
|
free_pkg_meta(meta);
|
||
|
meta = NULL;
|
||
|
}
|
||
|
|
||
|
return meta;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
static struct pkg_meta *
|
||
|
read_meta_data_from_pkgdb(const char *pkg)
|
||
|
{
|
||
|
struct pkg_meta *meta;
|
||
|
const struct pkg_meta_desc *descr;
|
||
|
char **target;
|
||
|
char *fname;
|
||
|
int fd;
|
||
|
struct stat st;
|
||
|
|
||
|
meta = xcalloc(1, sizeof(*meta));
|
||
|
|
||
|
for (descr = pkg_meta_descriptors; descr->entry_filename; ++descr) {
|
||
|
if ((descr->entry_mask & desired_meta_data) == 0)
|
||
|
continue;
|
||
|
|
||
|
fname = pkgdb_pkg_file(pkg, descr->entry_filename);
|
||
|
fd = open(fname, O_RDONLY, 0);
|
||
|
free(fname);
|
||
|
if (fd == -1) {
|
||
|
if (errno == ENOENT && descr->required_file == 0)
|
||
|
continue;
|
||
|
err(2, "cannot read meta data file %s of package %s",
|
||
|
descr->entry_filename, pkg);
|
||
|
}
|
||
|
target = (char **)((char *)meta + descr->entry_offset);
|
||
|
|
||
|
if (fstat(fd, &st) == -1)
|
||
|
err(2, "cannot stat meta data");
|
||
|
if ((st.st_mode & S_IFMT) != S_IFREG)
|
||
|
errx(1, "meta data is not regular file");
|
||
|
/*
|
||
|
if (st.st_size > SSIZE_MAX - 1)
|
||
|
err(2, "meta data file too large to process");
|
||
|
*target = xmalloc(st.st_size + 1);
|
||
|
if (read(fd, *target, st.st_size) != st.st_size)
|
||
|
err(2, "cannot read meta data");
|
||
|
*/
|
||
|
*target = xmalloc(st.st_size + 1);
|
||
|
if (readfile(fd, *target, st.st_size) < 0)
|
||
|
err(2, "cannot read meta data");
|
||
|
(*target)[st.st_size] = '\0';
|
||
|
close(fd);
|
||
|
}
|
||
|
|
||
|
meta->is_installed = 1;
|
||
|
|
||
|
return meta;
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
build_full_reqby(lpkg_head_t *reqby, struct pkg_meta *meta, int limit)
|
||
|
{
|
||
|
char *iter, *eol, *next;
|
||
|
lpkg_t *lpp;
|
||
|
struct pkg_meta *meta_dep;
|
||
|
|
||
|
if (limit == 65536)
|
||
|
errx(1, "Cycle in the dependency tree, bailing out");
|
||
|
|
||
|
if (meta->is_installed == 0 || meta->meta_required_by == NULL)
|
||
|
return;
|
||
|
|
||
|
for (iter = meta->meta_required_by; *iter != '\0'; iter = next) {
|
||
|
eol = iter + strcspn(iter, "\n");
|
||
|
if (*eol == '\n')
|
||
|
next = eol + 1;
|
||
|
else
|
||
|
next = eol;
|
||
|
if (iter == eol)
|
||
|
continue;
|
||
|
TAILQ_FOREACH(lpp, reqby, lp_link) {
|
||
|
if (strlen(lpp->lp_name) + iter != eol)
|
||
|
continue;
|
||
|
if (memcmp(lpp->lp_name, iter, eol - iter) == 0)
|
||
|
break;
|
||
|
}
|
||
|
if (lpp != NULL)
|
||
|
continue;
|
||
|
*eol = '\0';
|
||
|
lpp = alloc_lpkg(iter);
|
||
|
if (next != eol)
|
||
|
*eol = '\n';
|
||
|
|
||
|
meta_dep = read_meta_data_from_pkgdb(lpp->lp_name);
|
||
|
if (meta_dep == NULL)
|
||
|
continue;
|
||
|
build_full_reqby(reqby, meta_dep, limit + 1);
|
||
|
free_pkg_meta(meta_dep);
|
||
|
|
||
|
TAILQ_INSERT_HEAD(reqby, lpp, lp_link);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static lfile_head_t files;
|
||
|
|
||
|
static int
|
||
|
pkg_do(const char *pkg)
|
||
|
{
|
||
|
struct pkg_meta *meta;
|
||
|
int code = 0;
|
||
|
const char *binpkgfile = NULL;
|
||
|
char *pkgdir;
|
||
|
|
||
|
if (IS_URL(pkg) || (fexists(pkg) && isfile(pkg))) {
|
||
|
#ifdef BOOTSTRAP
|
||
|
errx(2, "Binary packages not supported during bootstrap");
|
||
|
#else
|
||
|
struct archive *archive;
|
||
|
struct archive_entry *entry;
|
||
|
char *archive_name, *pkgname;
|
||
|
|
||
|
archive = open_archive(pkg, &archive_name);
|
||
|
if (archive == NULL) {
|
||
|
warnx("can't find package `%s', skipped", pkg);
|
||
|
return -1;
|
||
|
}
|
||
|
pkgname = NULL;
|
||
|
entry = NULL;
|
||
|
pkg_verify_signature(archive_name, &archive, &entry, &pkgname);
|
||
|
if (archive == NULL)
|
||
|
return -1;
|
||
|
free(pkgname);
|
||
|
|
||
|
meta = read_meta_data_from_archive(archive, entry);
|
||
|
archive_read_finish(archive);
|
||
|
if (!IS_URL(pkg))
|
||
|
binpkgfile = pkg;
|
||
|
#endif
|
||
|
} else {
|
||
|
/*
|
||
|
* It's not an uninstalled package, try and find it among the
|
||
|
* installed
|
||
|
*/
|
||
|
pkgdir = pkgdb_pkg_dir(pkg);
|
||
|
if (!fexists(pkgdir) || !(isdir(pkgdir) || islinktodir(pkgdir))) {
|
||
|
switch (add_installed_pkgs_by_basename(pkg, &pkgs)) {
|
||
|
case 1:
|
||
|
return 0;
|
||
|
case 0:
|
||
|
/* No match */
|
||
|
warnx("can't find package `%s'", pkg);
|
||
|
return 1;
|
||
|
case -1:
|
||
|
errx(EXIT_FAILURE, "Error during search in pkgdb for %s", pkg);
|
||
|
}
|
||
|
}
|
||
|
free(pkgdir);
|
||
|
meta = read_meta_data_from_pkgdb(pkg);
|
||
|
}
|
||
|
|
||
|
if (meta == NULL) {
|
||
|
warnx("invalid package `%s' skipped", pkg);
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* Index is special info type that has to override all others to make
|
||
|
* any sense.
|
||
|
*/
|
||
|
if (Flags & SHOW_INDEX) {
|
||
|
char tmp[MaxPathSize];
|
||
|
|
||
|
(void) snprintf(tmp, sizeof(tmp), "%-19s ", pkg);
|
||
|
show_index(meta->meta_comment, tmp);
|
||
|
} else if (Flags & SHOW_BI_VAR) {
|
||
|
if (strcspn(BuildInfoVariable, "ABCDEFGHIJKLMNOPQRSTUVWXYZ")
|
||
|
== strlen(BuildInfoVariable)) {
|
||
|
if (meta->meta_installed_info)
|
||
|
show_var(meta->meta_installed_info, BuildInfoVariable);
|
||
|
} else {
|
||
|
if (meta->meta_build_info)
|
||
|
show_var(meta->meta_build_info, BuildInfoVariable);
|
||
|
else
|
||
|
warnx("Build information missing");
|
||
|
}
|
||
|
} else {
|
||
|
package_t plist;
|
||
|
|
||
|
/* Read the contents list */
|
||
|
parse_plist(&plist, meta->meta_contents);
|
||
|
|
||
|
/* Start showing the package contents */
|
||
|
if (!Quiet && !(Flags & SHOW_SUMMARY)) {
|
||
|
printf("%sInformation for %s:\n\n", InfoPrefix, pkg);
|
||
|
if (meta->meta_preserve) {
|
||
|
printf("*** PACKAGE MAY NOT BE DELETED ***\n");
|
||
|
}
|
||
|
}
|
||
|
if (Flags & SHOW_SUMMARY) {
|
||
|
show_summary(meta, &plist, binpkgfile);
|
||
|
}
|
||
|
if (Flags & SHOW_COMMENT) {
|
||
|
show_file(meta->meta_comment, "Comment:\n", TRUE);
|
||
|
}
|
||
|
if (Flags & SHOW_DEPENDS) {
|
||
|
show_depends("Requires:\n", &plist);
|
||
|
}
|
||
|
if (Flags & SHOW_BLD_DEPENDS) {
|
||
|
show_bld_depends("Built using:\n", &plist);
|
||
|
}
|
||
|
if ((Flags & SHOW_REQBY) && meta->meta_required_by) {
|
||
|
show_file(meta->meta_required_by, "Required by:\n", TRUE);
|
||
|
}
|
||
|
if ((Flags & SHOW_FULL_REQBY) && meta->is_installed) {
|
||
|
lpkg_head_t reqby;
|
||
|
TAILQ_INIT(&reqby);
|
||
|
build_full_reqby(&reqby, meta, 0);
|
||
|
show_list(&reqby, "Full required by list:\n");
|
||
|
}
|
||
|
if (Flags & SHOW_DESC) {
|
||
|
show_file(meta->meta_desc, "Description:\n", TRUE);
|
||
|
}
|
||
|
if ((Flags & SHOW_DISPLAY) && meta->meta_display) {
|
||
|
show_file(meta->meta_display, "Install notice:\n",
|
||
|
TRUE);
|
||
|
}
|
||
|
if (Flags & SHOW_PLIST) {
|
||
|
show_plist("Packing list:\n", &plist, PLIST_SHOW_ALL);
|
||
|
}
|
||
|
if ((Flags & SHOW_INSTALL) && meta->meta_install) {
|
||
|
show_file(meta->meta_install, "Install script:\n",
|
||
|
TRUE);
|
||
|
}
|
||
|
if ((Flags & SHOW_DEINSTALL) && meta->meta_deinstall) {
|
||
|
show_file(meta->meta_deinstall, "De-Install script:\n",
|
||
|
TRUE);
|
||
|
}
|
||
|
if ((Flags & SHOW_MTREE) && meta->meta_mtree) {
|
||
|
show_file(meta->meta_mtree, "mtree file:\n", TRUE);
|
||
|
}
|
||
|
if (Flags & SHOW_PREFIX) {
|
||
|
show_plist("Prefix(s):\n", &plist, PLIST_CWD);
|
||
|
}
|
||
|
if (Flags & SHOW_FILES) {
|
||
|
show_files("Files:\n", &plist);
|
||
|
}
|
||
|
if ((Flags & SHOW_BUILD_VERSION) && meta->meta_build_version) {
|
||
|
show_file(meta->meta_build_version, "Build version:\n",
|
||
|
TRUE);
|
||
|
}
|
||
|
if (Flags & SHOW_BUILD_INFO) {
|
||
|
if (meta->meta_build_info) {
|
||
|
show_file(meta->meta_build_info, "Build information:\n",
|
||
|
TRUE);
|
||
|
}
|
||
|
if (meta->meta_installed_info) {
|
||
|
show_file(meta->meta_installed_info, "Installed information:\n",
|
||
|
TRUE);
|
||
|
}
|
||
|
}
|
||
|
if ((Flags & SHOW_PKG_SIZE) && meta->meta_size_pkg) {
|
||
|
show_file(meta->meta_size_pkg, "Size of this package in bytes: ",
|
||
|
TRUE);
|
||
|
}
|
||
|
if ((Flags & SHOW_ALL_SIZE) && meta->meta_size_all) {
|
||
|
show_file(meta->meta_size_all, "Size in bytes including required pkgs: ",
|
||
|
TRUE);
|
||
|
}
|
||
|
if (!Quiet && !(Flags & SHOW_SUMMARY)) {
|
||
|
if (meta->meta_preserve) {
|
||
|
printf("*** PACKAGE MAY NOT BE DELETED ***\n\n");
|
||
|
}
|
||
|
puts(InfoPrefix);
|
||
|
}
|
||
|
free_plist(&plist);
|
||
|
}
|
||
|
free_pkg_meta(meta);
|
||
|
return code;
|
||
|
}
|
||
|
|
||
|
struct print_matching_arg {
|
||
|
const char *pattern;
|
||
|
int got_match;
|
||
|
};
|
||
|
|
||
|
static int
|
||
|
print_matching_pkg(const char *pkgname, void *cookie)
|
||
|
{
|
||
|
struct print_matching_arg *arg= cookie;
|
||
|
|
||
|
if (pkg_match(arg->pattern, pkgname)) {
|
||
|
if (!Quiet)
|
||
|
puts(pkgname);
|
||
|
arg->got_match = 1;
|
||
|
}
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* Returns 0 if at least one package matching pkgname.
|
||
|
* Returns 1 otherwise.
|
||
|
*
|
||
|
* If -q was not specified, print all matching packages to stdout.
|
||
|
*/
|
||
|
int
|
||
|
CheckForPkg(const char *pkgname)
|
||
|
{
|
||
|
struct print_matching_arg arg;
|
||
|
|
||
|
arg.pattern = pkgname;
|
||
|
arg.got_match = 0;
|
||
|
|
||
|
if (iterate_pkg_db(print_matching_pkg, &arg) == -1) {
|
||
|
warnx("cannot iterate pkgdb");
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
if (arg.got_match == 0 && !ispkgpattern(pkgname)) {
|
||
|
char *pattern;
|
||
|
|
||
|
pattern = xasprintf("%s-[0-9]*", pkgname);
|
||
|
|
||
|
arg.pattern = pattern;
|
||
|
arg.got_match = 0;
|
||
|
|
||
|
if (iterate_pkg_db(print_matching_pkg, &arg) == -1) {
|
||
|
free(pattern);
|
||
|
warnx("cannot iterate pkgdb");
|
||
|
return 1;
|
||
|
}
|
||
|
free(pattern);
|
||
|
}
|
||
|
|
||
|
if (arg.got_match)
|
||
|
return 0;
|
||
|
else
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* Returns 0 if at least one package matching pkgname.
|
||
|
* Returns 1 otherwise.
|
||
|
*
|
||
|
* If -q was not specified, print best match to stdout.
|
||
|
*/
|
||
|
int
|
||
|
CheckForBestPkg(const char *pkgname)
|
||
|
{
|
||
|
char *pattern, *best_match;
|
||
|
|
||
|
best_match = find_best_matching_installed_pkg(pkgname);
|
||
|
if (best_match == NULL) {
|
||
|
if (ispkgpattern(pkgname))
|
||
|
return 1;
|
||
|
|
||
|
pattern = xasprintf("%s-[0-9]*", pkgname);
|
||
|
best_match = find_best_matching_installed_pkg(pattern);
|
||
|
free(pattern);
|
||
|
}
|
||
|
|
||
|
if (best_match == NULL)
|
||
|
return 1;
|
||
|
if (!Quiet)
|
||
|
puts(best_match);
|
||
|
free(best_match);
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static int
|
||
|
perform_single_pkg(const char *pkg, void *cookie)
|
||
|
{
|
||
|
int *err_cnt = cookie;
|
||
|
|
||
|
if (Which == WHICH_ALL || !is_automatic_installed(pkg))
|
||
|
*err_cnt += pkg_do(pkg);
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int
|
||
|
pkg_perform(lpkg_head_t *pkghead)
|
||
|
{
|
||
|
int err_cnt = 0;
|
||
|
|
||
|
TAILQ_INIT(&files);
|
||
|
|
||
|
desired_meta_data = 0;
|
||
|
if ((Flags & (SHOW_INDEX | SHOW_BI_VAR)) == 0)
|
||
|
desired_meta_data |= LOAD_PRESERVE;
|
||
|
if ((Flags & (SHOW_INDEX | SHOW_BI_VAR)) == 0)
|
||
|
desired_meta_data |= LOAD_CONTENTS;
|
||
|
if (Flags & (SHOW_COMMENT | SHOW_INDEX | SHOW_SUMMARY))
|
||
|
desired_meta_data |= LOAD_COMMENT;
|
||
|
if (Flags & (SHOW_BI_VAR | SHOW_BUILD_INFO | SHOW_SUMMARY))
|
||
|
desired_meta_data |= LOAD_BUILD_INFO | LOAD_INSTALLED_INFO;
|
||
|
if (Flags & (SHOW_SUMMARY | SHOW_PKG_SIZE))
|
||
|
desired_meta_data |= LOAD_SIZE_PKG;
|
||
|
if (Flags & SHOW_ALL_SIZE)
|
||
|
desired_meta_data |= LOAD_SIZE_ALL;
|
||
|
if (Flags & (SHOW_SUMMARY | SHOW_DESC))
|
||
|
desired_meta_data |= LOAD_DESC;
|
||
|
if (Flags & (SHOW_REQBY | SHOW_FULL_REQBY))
|
||
|
desired_meta_data |= LOAD_REQUIRED_BY;
|
||
|
if (Flags & SHOW_DISPLAY)
|
||
|
desired_meta_data |= LOAD_DISPLAY;
|
||
|
if (Flags & SHOW_INSTALL)
|
||
|
desired_meta_data |= LOAD_INSTALL;
|
||
|
if (Flags & SHOW_DEINSTALL)
|
||
|
desired_meta_data |= LOAD_DEINSTALL;
|
||
|
if (Flags & SHOW_MTREE)
|
||
|
desired_meta_data |= LOAD_MTREE;
|
||
|
if (Flags & SHOW_BUILD_VERSION)
|
||
|
desired_meta_data |= LOAD_BUILD_VERSION;
|
||
|
|
||
|
if (Which != WHICH_LIST) {
|
||
|
if (File2Pkg) {
|
||
|
/* Show all files with the package they belong to */
|
||
|
if (pkgdb_dump() == -1)
|
||
|
err_cnt = 1;
|
||
|
} else {
|
||
|
if (iterate_pkg_db(perform_single_pkg, &err_cnt) == -1)
|
||
|
err_cnt = 1;
|
||
|
}
|
||
|
} else {
|
||
|
/* Show info on individual pkg(s) */
|
||
|
lpkg_t *lpp;
|
||
|
|
||
|
while ((lpp = TAILQ_FIRST(pkghead)) != NULL) {
|
||
|
TAILQ_REMOVE(pkghead, lpp, lp_link);
|
||
|
err_cnt += pkg_do(lpp->lp_name);
|
||
|
free_lpkg(lpp);
|
||
|
}
|
||
|
}
|
||
|
return err_cnt;
|
||
|
}
|