/* $NetBSD: show.c,v 1.34 2006/04/24 13:36:23 dillo Exp $ */ #if HAVE_CONFIG_H #include "config.h" #endif #include #if HAVE_SYS_CDEFS_H #include #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 #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; }