Files
sudo/find_path.c
Todd C. Miller f12f25204d moved cwd stuff
1994-05-24 20:17:43 +00:00

329 lines
7.8 KiB
C

/*
* CU sudo version 1.3.1 (based on Root Group sudo version 1.1)
*
* This software comes with no waranty whatsoever, use at your own risk.
*
* Please send bugs, changes, problems to sudo-bugs.cs.colorado.edu
*
*/
/*
* sudo version 1.1 allows users to execute commands as root
* Copyright (C) 1991 The Root Group, Inc.
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 1, or (at your option)
* any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
*
*******************************************************************
*
* This module contains the find_path() command that returns
* a pointer to a static area with the absolute path of the
* command or NULL if the command is not found in the path
*
* I also added the strdup() function in here after I found most
* systems don't have it...
*
* Jeff Nieusma Thu Mar 21 23:11:23 MST 1991
*/
/*
* Most of this code has been rewritten to fix bugs and bears little
* resemblence to the original.
*
* Todd C. Miller (millert@colorado.edu) Sat Sep 4 12:22:04 MDT 1993
*/
#ifndef lint
static char rcsid[] = "$Id$";
#endif /* lint */
#include "config.h"
#include <stdio.h>
#ifdef STDC_HEADERS
#include <stdlib.h>
#endif /* STDC_HEADERS */
#ifdef HAVE_UNISTD_H
#include <unistd.h>
#endif /* HAVE_UNISTD_H */
#ifdef HAVE_STRING_H
#include <string.h>
#endif /* HAVE_STRING_H */
#ifdef HAVE_STRINGS_H
#include <strings.h>
#endif /* HAVE_STRINGS_H */
#ifdef HAVE_MALLOC_H
#include <malloc.h>
#endif /* HAVE_MALLOC_H */
#include <errno.h>
#include <sys/types.h>
#include <sys/param.h>
#include <sys/stat.h>
#include "sudo.h"
#ifndef STDC_HEADERS
extern char *malloc();
extern char *getenv();
extern char *strcpy();
extern int fprintf();
extern int readlink();
extern int stat();
extern int lstat();
#ifdef HAVE_STRDUP
extern char *strdup();
#endif /* HAVE_STRDUP */
#endif /* !STDC_HEADERS */
#ifndef _S_IFMT
#define _S_IFMT S_IFMT
#endif /* _S_IFMT */
#ifndef _S_IFLNK
#define _S_IFLNK S_IFLNK
#endif /* _S_IFLNK */
/*******************************************************************
*
* find_path()
*
* this function finds the full pathname for a command
*/
char *find_path(file)
char *file;
{
register char *n; /* for traversing path */
char *path = NULL; /* contents of PATH env var */
char fn[MAXPATHLEN + 1]; /* filename (path + file) */
struct stat statbuf; /* for stat(2) */
int statfailed; /* stat(2) return value */
int checkdot = 0; /* check current dir? */
char *qualify();
if (strlen(file) > MAXPATHLEN) {
(void) fprintf(stderr, "%s: path too long: %s\n", Argv[0], file);
exit(1);
}
/*
* do we need to search the path?
*/
if (strchr(file, '/'))
return (qualify(file));
/*
* grab PATH out of environment and make a local copy
*/
if ((path = getenv("PATH")) == NULL)
return (NULL);
if ((path = strdup(path)) == NULL) {
perror("find_path: malloc");
exit(1);
}
do {
if ((n = strchr(path, ':')))
*n = '\0';
/*
* search current dir last if it is in PATH This will miss sneaky
* things like using './' or './/'
*/
if (*path == '\0' || (*path == '.' && *(path + 1) == '\0')) {
checkdot = 1;
path = n + 1;
continue;
}
(void) sprintf(fn, "%s/%s", path, file);
/*
* stat the file to make sure it exists and is executable
*/
statfailed = stat(fn, &statbuf);
if (!statfailed && (statbuf.st_mode & 0000111))
return (qualify(fn));
else if (statfailed && errno != ENOENT && errno != ENOTDIR &&
errno != EINVAL && errno != EPERM) {
fprintf(stderr, "sudo: Can't stat %s: ", fn);
perror("");
}
path = n + 1;
} while (n);
/*
* check current dir if dot was in the PATH
*/
if (checkdot) {
(void) sprintf(fn, "./%s", file);
/*
* stat the file to make sure it exists and is executable
*/
statfailed = stat(fn, &statbuf);
if (!statfailed && (statbuf.st_mode & 0000111))
return (qualify(fn));
else if (statfailed && errno != ENOENT && errno != ENOTDIR &&
errno != EINVAL && errno != EPERM) {
fprintf(stderr, "sudo: Can't stat %s: ", fn);
perror("");
return (NULL);
}
}
return (NULL);
}
/******************************************************************
*
* qualify()
*
* this function takes a path and makes it fully qualified and resolves
* all symbolic links, returning the fully qualfied path.
*/
char *qualify(n)
char *n; /* name to make fully qualified */
{
char *beg = NULL; /* begining of a path component */
char *end; /* end of a path component */
static char full[MAXPATHLEN + 1]; /* the fully qualified name */
char name[MAXPATHLEN + 1]; /* local copy of n */
struct stat statbuf; /* for lstat() */
char *tmp; /* temporary pointer */
/*
* is it a bogus path?
*/
if (stat(n, &statbuf)) {
if (errno != ENOENT && errno != EPERM) {
fprintf(stderr, "sudo: Can't stat %s: ", n);
perror("");
}
return (NULL);
}
/*
* if n is relative, fill full with working dir
*/
if (*n != '/')
(void) strcpy(full, cwd);
else
full[0] = '\0';
(void) strcpy(name, n); /* working copy... */
do { /* while (end) */
if (beg)
beg = end + 1; /* skip past the NULL */
else
beg = name; /* just starting out... */
/*
* find and terminate end of path component
*/
if ((end = strchr(beg, '/')))
*end = '\0';
if (beg == end)
continue;
else if (!strcmp(beg, ".")); /* ignore "." */
else if (!strcmp(beg, "..")) {
if ((tmp = strrchr(full, '/')))
*tmp = '\0';
} else {
strcat(full, "/");
strcat(full, beg); /* copy in new component */
}
/*
* if we used ../.. to go past the root dir just continue
*/
if (!full[0])
continue;
/*
* check for symbolic links
*/
if (lstat(full, &statbuf)) {
fprintf(stderr, "sudo: Can't lstat %s: ", full);
perror("");
return (NULL);
}
if ((statbuf.st_mode & _S_IFMT) == _S_IFLNK) {
int linklen; /* length of link contents */
char newname[MAXPATHLEN + 1]; /* temp storage to build new
* name */
linklen = readlink(full, newname, sizeof(newname));
newname[linklen] = '\0';
/* check to make sure we don't go past MAXPATHLEN */
++end;
if (end != (char *) 1) {
if (linklen + strlen(end) >= MAXPATHLEN) {
(void )fprintf(stderr, "%s: path too long: %s/%s\n",
Argv[0], newname, end);
exit(1);
}
strcat(newname, "/");
strcat(newname, end); /* copy what's left of end */
}
if (newname[0] == '/') /* reset full if necesary */
full[0] = '\0';
else if ((tmp = strrchr(full, '/')))
*tmp = '\0'; /* remove component from full */
strcpy(name, newname); /* reset name with new path */
beg = NULL; /* since we have a new name */
}
} while (end);
/*
* if we resolved to "/" full[0] will be NULL
*/
if (!full[0])
strcpy(full, "/");
return ((char *) full);
}
#ifndef HAVE_STRDUP
/******************************************************************
*
* strdup()
*
* this function returns a pointer a string copied into
* a malloc()ed buffer
*/
char *strdup(s1)
char *s1;
{
char *s;
if ((s = (char *) malloc(strlen(s1) + 1)) == NULL)
return (NULL);
(void) strcpy(s, s1);
return (s);
}
#endif /* !HAVE_STRDUP */