/* * The new sysinstall program. * * This is probably the last program in the `sysinstall' line - the next * generation being essentially a complete rewrite. * * $Id: package.c,v 1.68 1999/05/07 20:31:36 jkh Exp $ * * Copyright (c) 1995 * Jordan Hubbard. 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, * verbatim and that no modifications are made prior to this * point in the file. * 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 JORDAN HUBBARD ``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 JORDAN HUBBARD OR HIS PETS 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, LIFE 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. * */ #include "sysinstall.h" #include #include #include #include #include static Boolean sigpipe_caught = FALSE; static void catch_pipe(int sig) { sigpipe_caught = TRUE; } /* Like package_extract, but assumes current media device */ int package_add(char *name) { if (!mediaVerify()) return DITEM_FAILURE; return package_extract(mediaDevice, name, FALSE); } /* For use by dispatch */ int packageAdd(dialogMenuItem *self) { char *cp; cp = variable_get(VAR_PACKAGE); if (!cp) { msgDebug("packageAdd: No package name passed in package variable\n"); return DITEM_FAILURE; } else return package_add(cp); } Boolean package_exists(char *name) { char fname[FILENAME_MAX]; int status /* = vsystem("pkg_info -e %s", name) */; /* If in "Latest" syntax, ignore; can't tell with these */ if (name[0] == '@') return FALSE; /* XXX KLUDGE ALERT! This makes evil assumptions about how XXX * packages register themselves and should *really be done with * `pkg_info -e ' except that this it's too slow for an * item check routine.. :-( */ snprintf(fname, FILENAME_MAX, "/var/db/pkg/%s", name); status = access(fname, R_OK); if (isDebug()) msgDebug("package check for %s returns %s.\n", name, status ? "failure" : "success"); return !status; } /* Extract a package based on a namespec and a media device */ int package_extract(Device *dev, char *name, Boolean depended) { char tmp[200], path[511]; int ret, last_msg = 0; FILE *fp; /* Check to make sure it's not already there */ if (package_exists(name)) return DITEM_SUCCESS; /* If necessary, initialize the ldconfig hints */ if (!file_readable("/var/run/ld.so.hints")) vsystem("ldconfig /usr/lib /usr/local/lib /usr/X11R6/lib"); if (!dev->init(dev)) { msgConfirm("Unable to initialize media type for package extract."); return DITEM_FAILURE; } /* Be initially optimistic */ ret = DITEM_SUCCESS | DITEM_RESTORE; /* Make a couple of paranoid locations for temp files to live if user specified none */ if (!variable_get(VAR_PKG_TMPDIR)) { /* Set it to a location with as much space as possible */ variable_set2(VAR_PKG_TMPDIR, "/usr/tmp", 0); } Mkdir(variable_get(VAR_PKG_TMPDIR)); vsystem("chmod 1777 %s", variable_get(VAR_PKG_TMPDIR)); if (name[0] == '@') { /* @ at the beginning of the package name means "get latest" */ sprintf(path, "packages/Latest/%s.tgz", ++name); } else if (!index(name, '/')) sprintf(path, "packages/All/%s%s", name, strstr(name, ".tgz") ? "" : ".tgz"); else sprintf(path, "%s%s", name, strstr(name, ".tgz") ? "" : ".tgz"); /* Set hints for pkg_add */ snprintf(tmp, sizeof tmp, "%s/Packages/All/", dev->name); setenv("PKG_ADD_BASE", tmp, 1); if (isDebug()) msgDebug("Set PKG_ADD_BASE to %s\n", tmp); fp = dev->get(dev, path, TRUE); if (fp) { int i = 0, tot, pfd[2]; pid_t pid; signal(SIGPIPE, catch_pipe); msgNotify("Adding %s%s\nfrom %s", path, depended ? " (as a dependency)" : "", dev->name); pipe(pfd); pid = fork(); if (!pid) { dup2(pfd[0], 0); close(pfd[0]); dup2(DebugFD, 1); close(2); close(pfd[1]); i = execl("/usr/sbin/pkg_add", "/usr/sbin/pkg_add", "-", 0); if (isDebug()) msgDebug("pkg_add returns %d status\n", i); } else { char buf[BUFSIZ]; WINDOW *w = savescr(); struct timeval start, stop; close(pfd[0]); tot = 0; (void)gettimeofday(&start, (struct timezone *)0); while (!sigpipe_caught && (i = fread(buf, 1, BUFSIZ, fp)) > 0) { int seconds; tot += i; /* Print statistics about how we're doing */ (void) gettimeofday(&stop, (struct timezone *)0); stop.tv_sec = stop.tv_sec - start.tv_sec; stop.tv_usec = stop.tv_usec - start.tv_usec; if (stop.tv_usec < 0) stop.tv_sec--, stop.tv_usec += 1000000; seconds = stop.tv_sec + (stop.tv_usec / 1000000.0); if (!seconds) seconds = 1; if (seconds != last_msg) { last_msg = seconds; msgInfo("%10d bytes read from package %s @ %4.1f KBytes/second", tot, name, (tot / seconds) / 1024.0); } /* Write it out */ if (sigpipe_caught || write(pfd[1], buf, i) != i) { msgInfo("Write failure to pkg_add! Package may be corrupt."); break; } } close(pfd[1]); fclose(fp); if (sigpipe_caught) msgInfo("pkg_add(1) apparently did not like the %s package.", name); else if (i == -1) msgInfo("I/O error while reading in the %s package.", name); else msgInfo("Package %s read successfully - waiting for pkg_add(1)", name); refresh(); i = waitpid(pid, &tot, 0); if (sigpipe_caught || i < 0 || WEXITSTATUS(tot)) { ret = DITEM_FAILURE | DITEM_RESTORE; if (variable_get(VAR_NO_CONFIRM)) msgNotify("Add of package %s aborted, error code %d -\n" "Please check the debug screen for more info.", name, WEXITSTATUS(tot)); else msgConfirm("Add of package %s aborted, error code %d -\n" "Please check the debug screen for more info.", name, WEXITSTATUS(tot)); } else msgNotify("Package %s was added successfully", name); /* Now catch any stragglers */ while (wait3(&tot, WNOHANG, NULL) > 0); sleep(1); restorescr(w); sigpipe_caught = FALSE; } } else { dialog_clear_norefresh(); if (variable_get(VAR_NO_CONFIRM)) msgNotify("Unable to fetch package %s from selected media.\n" "No package add will be done.", name); else msgConfirm("Unable to fetch package %s from selected media.\n" "No package add will be done.", name); ret = DITEM_FAILURE | DITEM_RESTORE; } return ret; }