sources-request@panda.UUCP (11/15/85)
Mod.sources: Volume 3, Issue 44 Submitted by: philabs!mcnc!ikonas!!roger [ moderators note: This program ONLY works properly on System III/V where chown(3) allows you to give away ownership of a file. It might be useful to superusers on V7/BSD type systems, perhaps? ] Wednesday, 13 November 1985, 5:32 p.m. Enclosed find the manual page and C source for a program to change owner and group id's for one or more files. If a named file is a directory, the entire tree beginning with (and including) that directory is descended and changed. Roger L. Cordes, Jr. William G. Daniel & Associates ...!mcnc!ikonas!dedalus!roger #!/bin/sh # This is a shar archive: extract with sh, not csh. # This archive contains: # give.1 # give.c echo x - give.1 cat > give.1 << '!Funky!Stuff!' .TH GIVE sd .SH NAME give \- change owner and group information for one or more files .SH SYNOPSIS .B give .I file... .B to .I user [\c .I group\c ] .SH DESCRIPTION .B Give will change ownership and group information for the files specified. If any of the files is a directory, .B give will descend and change the entire directory tree, including the named directory. .sp AS might be expected, .B give will only work for the current owner of the named files or root. .SH DIAGNOSTICS .B Give will exit with a descriptive error message if any of the following occur: .br .in +0.5i access to /etc/passwd (and /etc/group, if applicable) is denied .br the named user (group) does not occur in the file .br files specified cannot be accessed or modified .in -0.5i .SH FILES /etc/passwd \- user list .br /etc/group \- group list .br !Funky!Stuff! echo x - give.c cat > give.c << '!Funky!Stuff!' # /**** **** Wednesday, 13 November 1985, 1:29 p.m. **** give: change owner and group for one or more files as indicated: **** usage: **** give file... to user [group] **** - may be used succesfully only by current owner or root **** - if file is a directory, descend and change the entire tree **** beginning with (including) the named directory **** compile: **** cc -O -o give give.c ****/ main(argc,argv) int argc; char *argv[]; { int i, to, uid = -1, gid = -1; /* * find end of filename list in argv[] and verify command-line syntax: * must say at least: "give SOMETHING to SOMEBODY" */ for ( i=0; i<argc; i++ ) if ( !strcmp(argv[i],"to") ) { to = i; break; } if ( (argc<4) || (to<2) || (to==(argc-1)) || (argc<4) ) exit(printf("usage: %s File... to User [Group]\n",argv[0])); /* * get target integer user id */ if ( (uid=getuser(argv[to+1])) < 0 ) exit(printf("unknown user: `%s'...\n",argv[to+1])); /* * get target integer group id (if applicable) */ if ( (to<(argc-2)) && ((gid=getgroup(argv[to+2]))<0) ) exit(printf("unknown group: `%s'...\n",argv[to+2])); give(&argv[1],(to-1),uid,gid,0); } #include <stdio.h> /* * getuser: get integer user id: returns -1 if unknown */ #define is_digit(C) ( ((C)>='0') && ((C)<='9') ) char line[256], *c = (char *)0, *strchr(); getuser(name) char *name; { FILE *file; int uid = -1, i, len, strlen(), intname = 1; len = strlen(name); for ( i=0; i<len; i++ ) if ( !is_digit(name[i]) ) intname = 0; if ( !(file=fopen("/etc/passwd","r")) ) exit(printf("cannot read user name list...\n")); while ( fgets(line,128,file) ) if ( intname || !strncmp(line,name,len) ) { c = line; for ( i=0; i<2; i++ ) if ( !(c=strchr((c+1),':')) ) if ( intname ) break; else goto out; if ( !intname || !strncmp((c+1),name,len) ) { sscanf((c+1),"%d",&uid); break; } } out: fclose(file); return(uid); } /* * getgroup: get integer group id: returns -1 on error */ getgroup(name) char *name; { int i, gid = -1, intname = 1, len; FILE *file; len = strlen(name); for ( i=0; i<len; i++ ) if ( !is_digit(name[i]) ) intname = 0; if ( !(file=fopen("/etc/group","r")) ) return(-1); while ( fgets(line,128,file) ) if ( intname || !strncmp(line,name,len) ) { if ( !(c=strchr(line,':')) ) goto out; if ( !(c=strchr((c+1),':')) ) goto out; if ( intname && strncmp((c+1),name,len) ) continue; sscanf((c+1),"%d",&gid); } out: fclose(file); return(gid); } /* * give: recursively descend a directory structure changing * owner and group ids. */ #include <sys/types.h> #include <sys/stat.h> #define indent() for ( j=0; j<pass; j++ ) printf(" ") #define is_dir() ((st.st_mode&S_IFDIR)==S_IFDIR) give(names,n,uid,gid,pass) char **names; int n, uid, gid, pass; { int i, j, grp, nn = 0, stat(); struct stat st; char **next; for ( i=0; i<n; i++ ) { if ( stat(names[i],&st) < 0 ) { indent(); printf("`%s': cannot stat...\n",names[i]); continue; } if ( gid < 0 ) grp = st.st_gid; else grp = gid; if ( is_dir() ) { indent(); printf("%s:\n",names[i]); if ( chdir(names[i]) < 0 ) { indent(); printf("cannot cd to `%s'...\n", names[i]); continue; } if ( !(nn=getflist(&next)) ) { indent(); printf("cannot ls `%s'...\n",names[i]); } else give(next,nn,uid,gid,(pass+1)); if ( chdir("..") < 0 ) { indent(); printf("cannot cd to `..'...\n"); exit(3); } } if ( chown(names[i],uid,grp) < 0 ) { indent(); printf("`%s': cannot change...\n",names[i]); } } } /* * getflist: get names of files in current working directory */ #include <sys/dir.h> char ln[4096], *strcat(); getflist(names) char ***names; { int n = 0; struct direct dir; FILE *file; if ( !(file=fopen(".","r")) ) return(0); ln[0] = 0; while ( fread(&dir,sizeof(struct direct),1,file) ) if ( !strcmp(dir.d_name,".") || !strcmp(dir.d_name,"..") ) continue; else if ( dir.d_ino ) { strcat(ln,dir.d_name); strcat(ln," "); } fclose(file); n = get_words(ln,names); return(n); } /* * get_words: split a character string into individual words */ #define is_white(C) ( ((C)==' ') || ((C)=='\t') ) int get_words(line,wordlist) char *line; char ***wordlist; { int i, j, k, nwords = 0, len = 0, longest = 0; char *c, **words, *malloc(), was_white = 0, got_a_word = 0; /* * count words, find longest word */ for ( c=line; *c; c++ ) if ( is_white(*c) ) { if ( got_a_word && !was_white ) { if ( len > longest ) longest = len; len = 0; nwords++; } was_white = 1; } else { was_white = 0; got_a_word = 1; len++; } if ( got_a_word && !was_white ) { if ( len > longest ) longest = len; len = 0; nwords++; } if ( !nwords ) return(0); /* * allocate the array of individual words */ if ( !(words=(char **)malloc(nwords*sizeof(char *))) ) { printf("cannot create the word list...\n"); return(0); } longest++; /* plus '\0' */ for ( i=0; i<nwords; i++ ) if ( (words[i]=malloc(longest*sizeof(char))) == NULL ) { printf("cannot allocate an individual word...\n"); for ( --i; i>=0; i-- ) free(words[i]); free((char *)words); return(0); } /* * load the array of individual words */ got_a_word = was_white = j = k = 0; for ( i=0; line[i]!='\0'; i++ ) if ( !is_white(line[i]) ) { got_a_word = 1; words[j][k] = line[i]; was_white = 0; k++; } else if ( got_a_word && !was_white ) { words[j][k] = '\0'; was_white = 1; k = 0; j++; } if ( got_a_word && !was_white ) words[j][k] = '\0'; *wordlist = words; return(nwords); } !Funky!Stuff!