430 lines
10 KiB
C
430 lines
10 KiB
C
/* $NetBSD: show.c,v 1.34 2006/04/24 13:36:23 dillo Exp $ */
|
|
|
|
#if HAVE_CONFIG_H
|
|
#include "config.h"
|
|
#endif
|
|
#include <nbcompat.h>
|
|
#if HAVE_SYS_CDEFS_H
|
|
#include <sys/cdefs.h>
|
|
#endif
|
|
#ifndef lint
|
|
#if 0
|
|
static const char *rcsid = "from FreeBSD Id: show.c,v 1.11 1997/10/08 07:47:38 charnier Exp";
|
|
#else
|
|
__RCSID("$NetBSD: show.c,v 1.34 2006/04/24 13:36:23 dillo Exp $");
|
|
#endif
|
|
#endif
|
|
|
|
/*
|
|
* 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
|
|
*
|
|
* Various display routines for the info module.
|
|
*
|
|
*/
|
|
/*
|
|
* Copyright (c) 1999 Hubert Feyrer. 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.
|
|
* 3. All advertising materials mentioning features or use of this software
|
|
* must display the following acknowledgement:
|
|
* This product includes software developed by Hubert Feyrer for
|
|
* the NetBSD Project.
|
|
* 4. The name of the author may not be used to endorse or promote products
|
|
* derived from this software without specific prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``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 AUTHOR 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.
|
|
*/
|
|
|
|
#if HAVE_ERR_H
|
|
#include <err.h>
|
|
#endif
|
|
|
|
#include "lib.h"
|
|
#include "info.h"
|
|
|
|
/* Structure to define entries for the "show table" */
|
|
typedef struct show_t {
|
|
pl_ent_t sh_type; /* type of entry */
|
|
char *sh_quiet; /* message when quiet */
|
|
char *sh_verbose; /* message when verbose */
|
|
} show_t;
|
|
|
|
/*
|
|
* The entries in this table must be ordered the same as
|
|
* pl_ent_t constants
|
|
*/
|
|
static const show_t showv[] = {
|
|
{PLIST_FILE, "%s", "\tFile: %s"},
|
|
{PLIST_CWD, "@cwd %s", "\tCWD to: %s"},
|
|
{PLIST_CMD, "@exec %s", "\tEXEC '%s'"},
|
|
{PLIST_CHMOD, "@chmod %s", "\tCHMOD to %s"},
|
|
{PLIST_CHOWN, "@chown %s", "\tCHOWN to %s"},
|
|
{PLIST_CHGRP, "@chgrp %s", "\tCHGRP to %s"},
|
|
{PLIST_COMMENT, "@comment %s", "\tComment: %s"},
|
|
{PLIST_IGNORE, "@ignore", "Ignore next file:"},
|
|
{PLIST_NAME, "@name %s", "\tPackage name: %s"},
|
|
{PLIST_UNEXEC, "@unexec %s", "\tUNEXEC '%s'"},
|
|
{PLIST_SRC, "@src: %s", "\tSRC to: %s"},
|
|
{PLIST_DISPLAY, "@display %s", "\tInstall message file: %s"},
|
|
{PLIST_PKGDEP, "@pkgdep %s", "\tPackage depends on: %s"},
|
|
{PLIST_MTREE, "@mtree %s", "\tPackage mtree file: %s"},
|
|
{PLIST_DIR_RM, "@dirrm %s", "\tDeinstall directory remove: %s"},
|
|
{PLIST_IGNORE_INST, "@ignore_inst ??? doesn't belong here",
|
|
"\tIgnore next file installation directive (doesn't belong)"},
|
|
{PLIST_OPTION, "@option %s", "\tPackage has option: %s"},
|
|
{PLIST_PKGCFL, "@pkgcfl %s", "\tPackage conflicts with: %s"},
|
|
{PLIST_BLDDEP, "@blddep %s", "\tPackage depends exactly on: %s"},
|
|
{-1, NULL, NULL}
|
|
};
|
|
|
|
static int print_file_as_var(const char *, const char *);
|
|
|
|
void
|
|
show_file(char *pkg, char *title, char *fname, Boolean separator)
|
|
{
|
|
FILE *fp;
|
|
char line[1024];
|
|
int n;
|
|
|
|
if (!Quiet) {
|
|
printf("%s%s", InfoPrefix, title);
|
|
}
|
|
if ((fp = fopen(fname, "r")) == (FILE *) NULL) {
|
|
printf("ERROR: show_file: package \"%s\": can't open '%s' for reading\n", pkg, fname);
|
|
} else {
|
|
int append_nl = 0;
|
|
while ((n = fread(line, 1, sizeof(line), fp)) != 0) {
|
|
fwrite(line, 1, n, stdout);
|
|
append_nl = (line[n - 1] != '\n');
|
|
}
|
|
(void) fclose(fp);
|
|
if (append_nl)
|
|
printf("\n");
|
|
}
|
|
if (!Quiet || separator) {
|
|
printf("\n"); /* just in case */
|
|
}
|
|
}
|
|
|
|
void
|
|
show_var(const char *fname, const char *variable)
|
|
{
|
|
char *value;
|
|
|
|
if ((value=var_get(fname, variable)) != NULL) {
|
|
(void) printf("%s\n", value);
|
|
free(value);
|
|
}
|
|
}
|
|
|
|
void
|
|
show_index(char *pkg, char *title, char *fname)
|
|
{
|
|
FILE *fp;
|
|
char *line;
|
|
size_t linelen;
|
|
size_t maxline = termwidth;
|
|
|
|
if (!Quiet) {
|
|
printf("%s%s", InfoPrefix, title);
|
|
maxline -= MAX(MAXNAMESIZE, strlen(title));
|
|
}
|
|
if ((fp = fopen(fname, "r")) == (FILE *) NULL) {
|
|
warnx("show_index: package \"%s\": can't open '%s' for reading", pkg, fname);
|
|
return;
|
|
}
|
|
if ((line = fgetln(fp, &linelen))) {
|
|
line[linelen - 1] = '\0'; /* tromp newline & terminate string */
|
|
if (termwidth && (linelen > maxline)) {
|
|
/* XXX -1 if term does NOT have xn (or xenl) quirk */
|
|
line[maxline] = '\0';
|
|
}
|
|
(void) printf("%s\n", line);
|
|
}
|
|
(void) fclose(fp);
|
|
}
|
|
|
|
/*
|
|
* Show a packing list item type. If type is PLIST_SHOW_ALL, show all
|
|
*/
|
|
void
|
|
show_plist(char *title, package_t *plist, pl_ent_t type)
|
|
{
|
|
plist_t *p;
|
|
Boolean ign;
|
|
|
|
if (!Quiet) {
|
|
printf("%s%s", InfoPrefix, title);
|
|
}
|
|
for (ign = FALSE, p = plist->head; p; p = p->next) {
|
|
if (p->type == type || type == PLIST_SHOW_ALL) {
|
|
switch (p->type) {
|
|
case PLIST_FILE:
|
|
printf(Quiet ? showv[p->type].sh_quiet : showv[p->type].sh_verbose, p->name);
|
|
if (ign) {
|
|
if (!Quiet) {
|
|
printf(" (ignored)");
|
|
}
|
|
ign = FALSE;
|
|
}
|
|
break;
|
|
case PLIST_CHMOD:
|
|
case PLIST_CHOWN:
|
|
case PLIST_CHGRP:
|
|
printf(Quiet ? showv[p->type].sh_quiet : showv[p->type].sh_verbose,
|
|
p->name ? p->name : "(clear default)");
|
|
break;
|
|
case PLIST_IGNORE:
|
|
printf(Quiet ? showv[p->type].sh_quiet : showv[p->type].sh_verbose);
|
|
ign = TRUE;
|
|
break;
|
|
case PLIST_IGNORE_INST:
|
|
printf(Quiet ? showv[p->type].sh_quiet : showv[p->type].sh_verbose, p->name);
|
|
ign = TRUE;
|
|
break;
|
|
case PLIST_CWD:
|
|
case PLIST_CMD:
|
|
case PLIST_SRC:
|
|
case PLIST_UNEXEC:
|
|
case PLIST_COMMENT:
|
|
case PLIST_NAME:
|
|
case PLIST_DISPLAY:
|
|
case PLIST_PKGDEP:
|
|
case PLIST_MTREE:
|
|
case PLIST_DIR_RM:
|
|
case PLIST_OPTION:
|
|
case PLIST_PKGCFL:
|
|
case PLIST_BLDDEP:
|
|
printf(Quiet ? showv[p->type].sh_quiet : showv[p->type].sh_verbose, p->name);
|
|
break;
|
|
default:
|
|
warnx("unknown command type %d (%s)", p->type, p->name);
|
|
}
|
|
(void) fputc('\n', stdout);
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Show all files in the packing list (except ignored ones)
|
|
*/
|
|
void
|
|
show_files(char *title, package_t *plist)
|
|
{
|
|
plist_t *p;
|
|
Boolean ign;
|
|
char *dir = ".";
|
|
|
|
if (!Quiet) {
|
|
printf("%s%s", InfoPrefix, title);
|
|
}
|
|
for (ign = FALSE, p = plist->head; p; p = p->next) {
|
|
switch (p->type) {
|
|
case PLIST_FILE:
|
|
if (!ign) {
|
|
printf("%s%s%s\n", dir,
|
|
(strcmp(dir, "/") == 0) ? "" : "/", p->name);
|
|
}
|
|
ign = FALSE;
|
|
break;
|
|
case PLIST_CWD:
|
|
dir = p->name;
|
|
break;
|
|
case PLIST_IGNORE:
|
|
ign = TRUE;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Show dependencies (packages this pkg requires)
|
|
*/
|
|
void
|
|
show_depends(char *title, package_t *plist)
|
|
{
|
|
plist_t *p;
|
|
int nodepends;
|
|
|
|
nodepends = 1;
|
|
for (p = plist->head; p && nodepends; p = p->next) {
|
|
switch (p->type) {
|
|
case PLIST_PKGDEP:
|
|
nodepends = 0;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
if (nodepends)
|
|
return;
|
|
|
|
if (!Quiet) {
|
|
printf("%s%s", InfoPrefix, title);
|
|
}
|
|
for (p = plist->head; p; p = p->next) {
|
|
switch (p->type) {
|
|
case PLIST_PKGDEP:
|
|
printf("%s\n", p->name);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
printf("\n");
|
|
}
|
|
|
|
/*
|
|
* Show exact dependencies (packages this pkg was built with)
|
|
*/
|
|
void
|
|
show_bld_depends(char *title, package_t *plist)
|
|
{
|
|
plist_t *p;
|
|
int nodepends;
|
|
|
|
nodepends = 1;
|
|
for (p = plist->head; p && nodepends; p = p->next) {
|
|
switch (p->type) {
|
|
case PLIST_BLDDEP:
|
|
nodepends = 0;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
if (nodepends)
|
|
return;
|
|
|
|
if (!Quiet) {
|
|
printf("%s%s", InfoPrefix, title);
|
|
}
|
|
for (p = plist->head; p; p = p->next) {
|
|
switch (p->type) {
|
|
case PLIST_BLDDEP:
|
|
printf("%s\n", p->name);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
printf("\n");
|
|
}
|
|
|
|
|
|
/*
|
|
* Show entry for pkg_summary.txt file.
|
|
*/
|
|
void
|
|
show_summary(package_t *plist, const char *binpkgfile)
|
|
{
|
|
static const char *bi_vars[] = {
|
|
"PKGPATH",
|
|
"CATEGORIES",
|
|
"PROVIDES",
|
|
"REQUIRES",
|
|
"PKG_OPTIONS",
|
|
"OPSYS",
|
|
"OS_VERSION",
|
|
"MACHINE_ARCH",
|
|
"LICENSE",
|
|
"HOMEPAGE",
|
|
"PKGTOOLS_VERSION",
|
|
"BUILD_DATE",
|
|
NULL
|
|
};
|
|
|
|
plist_t *p;
|
|
struct stat st;
|
|
|
|
for (p = plist->head; p; p = p->next) {
|
|
switch (p->type) {
|
|
case PLIST_NAME:
|
|
printf("PKGNAME=%s\n", p->name);
|
|
break;
|
|
case PLIST_PKGDEP:
|
|
printf("DEPENDS=%s\n", p->name);
|
|
break;
|
|
case PLIST_PKGCFL:
|
|
printf("CONFLICTS=%s\n", p->name);
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
print_file_as_var("COMMENT", COMMENT_FNAME);
|
|
print_file_as_var("SIZE_PKG", SIZE_PKG_FNAME);
|
|
|
|
var_copy_list(BUILD_INFO_FNAME, bi_vars);
|
|
|
|
if (binpkgfile != NULL && stat(binpkgfile, &st) == 0) {
|
|
printf("FILE_SIZE=%" PRIu64 "\n", (uint64_t)st.st_size);
|
|
/* XXX: DIGETS */
|
|
}
|
|
|
|
print_file_as_var("DESCRIPTION", DESC_FNAME);
|
|
putc('\n', stdout);
|
|
}
|
|
|
|
/*
|
|
* Print the contents of file fname as value of variable var to stdout.
|
|
*/
|
|
static int
|
|
print_file_as_var(const char *var, const char *fname)
|
|
{
|
|
FILE *fp;
|
|
char *line;
|
|
size_t len;
|
|
|
|
fp = fopen(fname, "r");
|
|
if (!fp) {
|
|
warn("unable to open %s file", fname);
|
|
return -1;
|
|
}
|
|
|
|
while ((line = fgetln(fp, &len)) != (char *) NULL) {
|
|
if (line[len - 1] == '\n')
|
|
--len;
|
|
printf("%s=%.*s\n", var, (int)len, line);
|
|
}
|
|
|
|
fclose(fp);
|
|
|
|
return 0;
|
|
}
|