/* $NetBSD: file.c,v 1.71 2003/12/20 10:09:20 grant Exp $ */ #include #ifndef lint #if 0 static const char *rcsid = "from FreeBSD Id: file.c,v 1.29 1997/10/08 07:47:54 charnier Exp"; #else __RCSID("$NetBSD: file.c,v 1.71 2003/12/20 10:09:20 grant 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 * 18 July 1993 * * Miscellaneous file access utilities. * */ #include "lib.h" #include #include #include #include #include #include #include #include /* * Quick check to see if a file (or dir ...) exists */ Boolean fexists(const char *fname) { struct stat dummy; if (!lstat(fname, &dummy)) return TRUE; return FALSE; } /* * Quick check to see if something is a directory */ Boolean isdir(const char *fname) { struct stat sb; if (lstat(fname, &sb) != FAIL && S_ISDIR(sb.st_mode)) return TRUE; else return FALSE; } /* * Check if something is a link to a directory */ Boolean islinktodir(const char *fname) { struct stat sb; if (lstat(fname, &sb) != FAIL && S_ISLNK(sb.st_mode)) { if (stat(fname, &sb) != FAIL && S_ISDIR(sb.st_mode)) return TRUE; /* link to dir! */ else return FALSE; /* link to non-dir */ } else return FALSE; /* non-link */ } /* * Check to see if file is a dir, and is empty */ Boolean isemptydir(const char *fname) { if (isdir(fname) || islinktodir(fname)) { DIR *dirp; struct dirent *dp; dirp = opendir(fname); if (!dirp) return FALSE; /* no perms, leave it alone */ for (dp = readdir(dirp); dp != NULL; dp = readdir(dirp)) { if (strcmp(dp->d_name, ".") && strcmp(dp->d_name, "..")) { closedir(dirp); return FALSE; } } (void) closedir(dirp); return TRUE; } return FALSE; } /* * Check if something is a regular file */ Boolean isfile(const char *fname) { struct stat sb; if (stat(fname, &sb) != FAIL && S_ISREG(sb.st_mode)) return TRUE; return FALSE; } /* * Check to see if file is a file and is empty. If nonexistent or not * a file, say "it's empty", otherwise return TRUE if zero sized. */ Boolean isemptyfile(const char *fname) { struct stat sb; if (stat(fname, &sb) != FAIL && S_ISREG(sb.st_mode)) { if (sb.st_size != 0) return FALSE; } return TRUE; } /* This struct defines the leading part of a valid URL name */ typedef struct url_t { char *u_s; /* the leading part of the URL */ int u_len; /* its length */ } url_t; /* A table of valid leading strings for URLs */ static const url_t urls[] = { {"ftp://", 6}, {"http://", 7}, {NULL} }; /* * Returns length of leading part of any URL from urls table, or -1 */ int URLlength(const char *fname) { const url_t *up; int i; if (fname != (char *) NULL) { for (i = 0; isspace((unsigned char) *fname); i++) { fname++; } for (up = urls; up->u_s; up++) { if (strncmp(fname, up->u_s, up->u_len) == 0) { return i + up->u_len; /* ... + sizeof(up->u_s); - HF */ } } } return -1; } /* * Returns the host part of a URL */ const char * fileURLHost(const char *fname, char *where, int max) { const char *ret; int i; assert(where != NULL); assert(max > 0); if ((i = URLlength(fname)) < 0) { /* invalid URL? */ errx(EXIT_FAILURE, "fileURLhost called with a bad URL: `%s'", fname); } fname += i; /* Do we have a place to stick our work? */ ret = where; while (*fname && *fname != '/' && --max) *where++ = *fname++; *where = '\0'; return ret; } /* * Returns the filename part of a URL */ const char * fileURLFilename(const char *fname, char *where, int max) { const char *ret; int i; assert(where != NULL); assert(max > 0); if ((i = URLlength(fname)) < 0) { /* invalid URL? */ errx(EXIT_FAILURE, "fileURLFilename called with a bad URL: `%s'", fname); } fname += i; /* Do we have a place to stick our work? */ ret = where; while (*fname && *fname != '/') ++fname; if (*fname == '/') { while (*fname && --max) *where++ = *fname++; } *where = '\0'; return ret; } /* * Try and fetch a file by URL, returning the directory name for where * it's unpacked, if successful. To be handed to leave_playpen() later. */ char * fileGetURL(const char *spec) { char host[MAXHOSTNAMELEN], file[FILENAME_MAX]; const char *cp; char *rp; char pen[FILENAME_MAX]; int rc; rp = NULL; if (!IS_URL(spec)) { errx(EXIT_FAILURE, "fileGetURL was called with non-URL arg '%s'", spec); } /* Some sanity checks on the URL */ cp = fileURLHost(spec, host, MAXHOSTNAMELEN); if (!*cp) { warnx("URL `%s' has bad host part!", spec); return NULL; } cp = fileURLFilename(spec, file, FILENAME_MAX); if (!*cp) { warnx("URL `%s' has bad filename part!", spec); return NULL; } if (Verbose) printf("Trying to fetch %s.\n", spec); pen[0] = '\0'; rp = make_playpen(pen, sizeof(pen), 0); if (rp == NULL) { printf("Error: Unable to construct a new playpen for FTP!\n"); return NULL; } rp = strdup(pen); rc = unpackURL(spec, pen); if (rc < 0) { leave_playpen(rp); /* Don't leave dir hang around! */ printf("Error on unpackURL('%s', '%s')\n", spec, pen); return NULL; } return rp; } static char * resolvepattern1(const char *name) { static char tmp[FILENAME_MAX]; char *cp; if (IS_URL(name)) { /* some package depends on a wildcard pkg */ int rc; rc = expandURL(tmp, name); if (rc < 0) { return NULL; } if (Verbose) printf("'%s' expanded to '%s'\n", name, tmp); return tmp; /* return expanded URL w/ corrent pkg */ } else if (ispkgpattern(name)) { cp = findbestmatchingname( dirname_of(name), basename_of(name)); if (cp) { snprintf(tmp, sizeof(tmp), "%s/%s", dirname_of(name), cp); free(cp); return tmp; } } else { if (isfile(name)) { strlcpy(tmp, name, sizeof(tmp)); return tmp; } } return NULL; } static char * resolvepattern(const char *name) { char tmp[FILENAME_MAX]; char *cp; const char *suf; cp = resolvepattern1(name); if (cp != NULL) return cp; if (ispkgpattern(name)) return NULL; suf = suffix_of(name); if (!strcmp(suf, "tbz") || !strcmp(suf, "tgz")) return NULL; /* add suffix and try */ snprintf(tmp, sizeof(tmp), "%s.tbz", name); cp = resolvepattern1(tmp); if (cp != NULL) return cp; snprintf(tmp, sizeof(tmp), "%s.tgz", name); cp = resolvepattern1(tmp); if (cp != NULL) return cp; /* add version number wildcard and try */ snprintf(tmp, sizeof(tmp), "%s-[0-9]*.t[bg]z", name); return resolvepattern1(tmp); } /* * Look for filename/pattern "fname" in * Returns a full path/URL where the pkg can be found */ char * fileFindByPath(const char *fname) { char tmp[FILENAME_MAX]; struct path *path; /* * 1. if fname is an absolute pathname or a URL, * just use it. */ if (IS_FULLPATH(fname) || IS_URL(fname)) return resolvepattern(fname); /* * 2. otherwise, use PKG_PATH. */ TAILQ_FOREACH(path, &PkgPath, pl_entry) { char *cp; const char *cp2 = path->pl_path; if (Verbose) printf("trying PKG_PATH %s\n", cp2); if (IS_FULLPATH(cp2) || IS_URL(cp2)) { snprintf(tmp, sizeof(tmp), "%s/%s", cp2, fname); } else { char cwdtmp[MAXPATHLEN]; if (getcwd(cwdtmp, sizeof(cwdtmp)) == NULL) errx(EXIT_FAILURE, "getcwd"); snprintf(tmp, sizeof(tmp), "%s/%s/%s", cwdtmp, cp2, fname); } cp = resolvepattern(tmp); if (cp) return cp; } #if 0 /* * 3. finally, search current directory. */ snprintf(tmp, sizeof(tmp), "./%s", fname); return resolvepattern(tmp); #else return NULL; #endif } /* * Expect "fname" to point at a file, and read it into * the buffer returned. */ char * fileGetContents(char *fname) { char *contents; struct stat sb; int fd; if (stat(fname, &sb) == FAIL) { cleanup(0); errx(2, "can't stat '%s'", fname); } contents = (char *) malloc((size_t) (sb.st_size) + 1); fd = open(fname, O_RDONLY, 0); if (fd == FAIL) { cleanup(0); errx(2, "unable to open '%s' for reading", fname); } if (read(fd, contents, (size_t) sb.st_size) != (size_t) sb.st_size) { cleanup(0); errx(2, "short read on '%s' - did not get %lld bytes", fname, (long long) sb.st_size); } close(fd); contents[(size_t) sb.st_size] = '\0'; return contents; } /* * Takes a filename and package name, returning (in "try") the canonical * "preserve" name for it. */ Boolean make_preserve_name(char *try, size_t max, char *name, char *file) { int len, i; if ((len = strlen(file)) == 0) return FALSE; else i = len - 1; strncpy(try, file, max); if (try[i] == '/') /* Catch trailing slash early and save checking in the loop */ --i; for (; i; i--) { if (try[i] == '/') { try[i + 1] = '.'; strncpy(&try[i + 2], &file[i + 1], max - i - 2); break; } } if (!i) { try[0] = '.'; strncpy(try + 1, file, max - 1); } /* I should probably be called rude names for these inline assignments */ strncat(try, ".", max -= strlen(try)); strncat(try, name, max -= strlen(name)); strncat(try, ".", max--); strncat(try, "backup", max -= 6); return TRUE; } /* * Write the contents of "str" to a file */ void write_file(char *name, char *str) { size_t len; FILE *fp; if ((fp = fopen(name, "w")) == (FILE *) NULL) { cleanup(0); errx(2, "cannot fopen '%s' for writing", name); } len = strlen(str); if (fwrite(str, 1, len, fp) != len) { cleanup(0); errx(2, "short fwrite on '%s', tried to write %ld bytes", name, (long) len); } if (fclose(fp)) { cleanup(0); errx(2, "failure to fclose '%s'", name); } } void copy_file(char *dir, char *fname, char *to) { char fpath[FILENAME_MAX]; (void) snprintf(fpath, sizeof(fpath), "%s%s%s", (fname[0] != '/') ? dir : "", (fname[0] != '/') ? "/" : "", fname); if (fexec("cp", "-r", fpath, to, NULL)) { cleanup(0); errx(2, "could not perform 'cp -r %s %s'", fpath, to); } } void move_file(char *dir, char *fname, char *to) { char fpath[FILENAME_MAX]; (void) snprintf(fpath, sizeof(fpath), "%s%s%s", (fname[0] != '/') ? dir : "", (fname[0] != '/') ? "/" : "", fname); if (fexec("mv", fpath, to, NULL)) { cleanup(0); errx(2, "could not perform 'mv %s %s'", fpath, to); } } void remove_files(const char *path, const char *pattern) { char fpath[FILENAME_MAX]; glob_t globbed; int i; (void) snprintf(fpath, sizeof(fpath), "%s/%s", path, pattern); if ((i=glob(fpath, GLOB_NOSORT, NULL, &globbed)) != 0) { switch(i) { case GLOB_NOMATCH: warn("no files matching ``%s'' found", fpath); break; case GLOB_ABORTED: warn("globbing aborted"); break; case GLOB_NOSPACE: warn("out-of-memory during globbing"); break; default: warn("unknown error during globbing"); break; } return; } /* deleting globbed files */ for (i=0; i