Annotation of 43BSD/usr.bin/uucp/uusend.c, revision 1.1

1.1     ! root        1: #ifndef lint
        !             2: static char sccsid[] = "@(#)uusend.c   5.2 (Berkeley) 1/22/85";
        !             3: #endif
        !             4: 
        !             5: /*
        !             6:  * uusend: primitive operation to allow uucp like copy of binary files
        !             7:  * but handle indirection over systems.
        !             8:  *
        !             9:  * usage: uusend [-r] [-m ooo] localfile sysname1!sysname2!...!destfile
        !            10:  *        uusend [-r] [-m ooo]     -     sysname1!sysname2!...!destfile
        !            11:  *
        !            12:  * Author: Mark Horton, May 1980.
        !            13:  *
        !            14:  * "-r" switch added.  Has same effect as "-r" in uux. 11/82  CCW
        !            15:  *
        !            16:  * Error recovery (a la uucp) added & ifdefs for ruusend (as in rmail).
        !            17:  * Checks for illegal access to /usr/lib/uucp.
        !            18:  *                             February 1983  Christopher Woodbury
        !            19:  * Fixed mode set[ug]id loophole. 4/8/83  CCW
        !            20:  *
        !            21:  * Add '-f' to make uusend syntax more similar to UUCP.  "destname"
        !            22:  * can now be a directory.     June 1983  CCW
        !            23:  */
        !            24: 
        !            25: #include <stdio.h>
        !            26: #include <pwd.h>
        !            27: #include <sys/types.h>
        !            28: #include <sys/stat.h>
        !            29: 
        !            30: /*
        !            31:  * define RECOVER to permit requests like 'uusend file sys1!sys2!~uucp'
        !            32:  * (abbreviation for 'uusend file sys1!sys2!~uucp/file').
        !            33:  * define DEBUG to keep log of uusend uusage.
        !            34:  * define RUUSEND if neighboring sites permit 'ruusend',
        !            35:  * which they certainly should to avoid security holes
        !            36:  */
        !            37: #define        RECOVER
        !            38: /*#define      DEBUG   "/usr/spool/uucp/uusend.log"/**/
        !            39: 
        !            40: FILE   *in, *out;
        !            41: FILE   *dout;
        !            42: 
        !            43: extern FILE    *popen();
        !            44: extern char    *index(), *strcpy(), *strcat(), *ctime();
        !            45: 
        !            46: #ifdef RUUSEND
        !            47: int    rsend;
        !            48: #endif  RUUSEND
        !            49: int    mode = -1;      /* mode to chmod new file to */
        !            50: char   *nextsys;       /* next system in the chain */
        !            51: char   dnbuf[200];     /* buffer for result of ~user/file */
        !            52: char   cmdbuf[256];    /* buffer to build uux command in */
        !            53: char   *rflg = "";     /* default value of rflg  ccw -- 1 Nov '82 */
        !            54: 
        !            55: struct passwd *user;   /* entry  in /etc/passwd for ~user */
        !            56: struct passwd *getpwnam();
        !            57: struct stat    stbuf;
        !            58: 
        !            59: char   *excl;          /* location of first ! in destname */
        !            60: char   *sl;            /* location of first / in destname */
        !            61: char   *sourcename;    /* argv[1] */
        !            62: char   *destname;      /* argv[2] */
        !            63: char   *UULIB = "/usr/lib/uucp";         /* UUCP lib directory */
        !            64: 
        !            65: #ifdef RECOVER
        !            66: char   *UUPUB = "/usr/spool/uucppublic/";  /* public UUCP directory */
        !            67: char   *filename;      /* file name from end of destname */
        !            68: char   *getfname();    /* routine to get filename from destname */
        !            69: int    fflg;
        !            70: char   f[100];         /* name of default output file */
        !            71: #else  !RECOVER
        !            72: char   *f      = "";   /* so we waste a little space */
        !            73: #endif !RECOVER
        !            74: 
        !            75: main(argc, argv)
        !            76: int    argc;
        !            77: char   **argv;
        !            78: {
        !            79:        register int c;
        !            80:        long count;
        !            81:        extern char **environ;
        !            82: 
        !            83: #ifdef DEBUG
        !            84:        long t;
        !            85:        umask(022);
        !            86:        dout = fopen(DEBUG, "a");
        !            87:        if (dout == NULL) {
        !            88:                printf("Cannot append to %s\n", DEBUG);
        !            89:                exit(1);
        !            90:        }
        !            91:        freopen(DEBUG, "a", stdout);
        !            92:        fprintf(dout, "\nuusend run: ");
        !            93:        for (c=0; c<argc; c++)
        !            94:                fprintf(dout, "%s ", argv[c]);
        !            95:        time(&t);
        !            96:        fprintf(dout, "%s", ctime(&t));
        !            97: #endif DEBUG
        !            98: 
        !            99: #ifdef RUUSEND
        !           100:        if(argv[0][0] == 'r')
        !           101:                rsend++;
        !           102: #endif RUUSEND
        !           103:        while (argc > 1 && argv[1][0] == '-' && argv[1][1]) {
        !           104:                switch(argv[1][1]) {
        !           105:                case 'm':
        !           106:                        sscanf(argv[2], "%o", &mode);
        !           107:                        mode &= 0777;  /* fix set[ug]id loophole */
        !           108:                        argc--; argv++;
        !           109:                        break;
        !           110:                case 'r':               /* -r flag for uux */
        !           111:                        rflg = "-r ";
        !           112:                        break;
        !           113: #ifdef RECOVER
        !           114:                case 'f':
        !           115:                        fflg++;
        !           116:                        strcpy(f, argv[1]);
        !           117:                        break;
        !           118: #endif RECOVER
        !           119:                default:
        !           120:                        fprintf(stderr, "Bad flag: %s\n", argv[1]);
        !           121:                        break;
        !           122:                }
        !           123:                argc--; argv++;
        !           124:        }
        !           125: 
        !           126:        if (argc != 3) {
        !           127:                fprintf(stderr, "Usage: uusend [-m ooo] [-r] -/file sys!sys!..!rfile\n");
        !           128:                exit(1);
        !           129:        }
        !           130: 
        !           131:        sourcename = argv[1];
        !           132:        destname = argv[2];
        !           133: 
        !           134:        if (sourcename[0] == '-')
        !           135:                in = stdin;
        !           136:        else {
        !           137: #ifdef RUUSEND
        !           138:                if (rsend) {
        !           139:                        fprintf(stderr, "illegal input\n");
        !           140:                        exit(2);
        !           141:                }
        !           142: #endif RUUSEND
        !           143:                in = fopen(sourcename, "r");
        !           144:                if (in == NULL) {
        !           145:                        perror(argv[1]);
        !           146:                        exit(2);
        !           147:                }
        !           148:                if (!fflg || f[2] == '\0') {
        !           149:                        strcpy(f, "-f");
        !           150:                        strcat(f, getfname(sourcename));
        !           151:                        fflg++;
        !           152:                }
        !           153:        }
        !           154: 
        !           155:        excl = index(destname, '!');
        !           156:        if (excl) {
        !           157:                /*
        !           158:                 * destname is on a remote system.
        !           159:                 */
        !           160:                nextsys = destname;
        !           161:                *excl++ = 0;
        !           162:                destname = excl;
        !           163:                if (mode < 0) {
        !           164:                        fstat(fileno(in), &stbuf);
        !           165:                        mode = stbuf.st_mode & 0777;
        !           166:                }
        !           167: #ifdef RUUSEND
        !           168:                sprintf(cmdbuf,"uux -gn -z %s- \"%s!ruusend %s -m %o - (%s)\"",
        !           169: #else !RUUSEND
        !           170:                sprintf(cmdbuf, "uux -gn -z %s- \"%s!uusend %s -m %o - (%s)\"",
        !           171: #endif !RUUSEND
        !           172:                        rflg, nextsys, f, mode, destname);
        !           173: #ifdef DEBUG
        !           174:                fprintf(dout, "remote: nextsys='%s', destname='%s', cmd='%s'\n", nextsys, destname, cmdbuf);
        !           175: #endif DEBUG
        !           176:                out = popen(cmdbuf, "w");
        !           177:        } else {
        !           178:                /*
        !           179:                 * destname is local.
        !           180:                 */
        !           181:                if (destname[0] == '~') {
        !           182: #ifdef DEBUG
        !           183:                        fprintf(dout, "before ~: '%s'\n", destname);
        !           184: fflush(dout);
        !           185: #endif DEBUG
        !           186:                        sl = index(destname, '/');
        !           187: #ifdef RECOVER
        !           188:                        if (sl == NULL && !fflg) {
        !           189:                                fprintf(stderr, "Illegal ~user\n");
        !           190:                                exit(3);
        !           191:                        }
        !           192:                        for (sl = destname; *sl != '\0'; sl++)
        !           193:                                ;       /* boy, is this a hack! */
        !           194: #else !RECOVER
        !           195:                        if (sl == NULL) {
        !           196:                                fprintf(stderr, "Illegal ~user\n");
        !           197:                                exit(3);
        !           198:                        }
        !           199:                        *sl++ = 0;
        !           200: #endif !RECOVER
        !           201:                        user = getpwnam(destname+1);
        !           202:                        if (user == NULL) {
        !           203:                                fprintf(stderr, "No such user as %s\n",
        !           204:                                        destname);
        !           205: #ifdef RECOVER
        !           206:                                if ((filename =getfname(sl)) == NULL &&
        !           207:                                     !fflg)
        !           208:                                        exit(4);
        !           209:                                strcpy(dnbuf, UUPUB);
        !           210:                                if (fflg)
        !           211:                                        strcat(dnbuf, &f[2]);
        !           212:                                else
        !           213:                                        strcat(dnbuf, filename);
        !           214:                        }
        !           215:                        else {
        !           216:                                strcpy(dnbuf, user->pw_dir);
        !           217:                                strcat(dnbuf, "/");
        !           218:                                strcat(dnbuf, sl);
        !           219:                        }
        !           220: #else !RECOVER
        !           221:                                exit(4);
        !           222:                        }
        !           223:                        strcpy(dnbuf, user->pw_dir);
        !           224:                        strcat(dnbuf, "/");
        !           225:                        strcat(dnbuf, sl);
        !           226: #endif !RECOVER
        !           227:                        destname = dnbuf;
        !           228:                }
        !           229: #ifdef RECOVER
        !           230:                else
        !           231:                        destname = strcpy(dnbuf, destname);
        !           232: #endif !RECOVER
        !           233:                if(strncmp(UULIB, destname, strlen(UULIB)) == 0) {
        !           234:                        fprintf(stderr, "illegal file: %s", destname);
        !           235:                        exit(4);
        !           236:                }
        !           237: #ifdef RECOVER
        !           238:                if (stat(destname, &stbuf) == 0 &&
        !           239:                    (stbuf.st_mode & S_IFMT) == S_IFDIR &&
        !           240:                     fflg) {
        !           241:                        strcat(destname, "/");
        !           242:                        strcat(destname, &f[2]);
        !           243:                }
        !           244: #endif RECOVER
        !           245:                out = fopen(destname, "w");
        !           246: #ifdef DEBUG
        !           247:                fprintf(dout, "local, file='%s'\n", destname);
        !           248: #endif DEBUG
        !           249:                if (out == NULL) {
        !           250:                        perror(destname);
        !           251: #ifdef RECOVER
        !           252:                        if (strncmp(destname,UUPUB,strlen(UUPUB)) == 0)
        !           253:                                exit(5);        /* forget it! */
        !           254:                        filename = getfname(destname);
        !           255:                        if (destname == dnbuf) /* cmdbuf is scratch */
        !           256:                                filename = strcpy(cmdbuf, filename);
        !           257:                        destname = strcpy(dnbuf, UUPUB);
        !           258:                        if (user != NULL) {
        !           259:                                strcat(destname, user->pw_name);
        !           260:                                if (stat(destname, &stbuf) == -1) {
        !           261:                                        mkdir(destname, 0777);
        !           262:                                }
        !           263:                                strcat(destname, "/");
        !           264:                        }
        !           265:                        if (fflg)
        !           266:                                strcat(destname, &f[2]);
        !           267:                        else
        !           268:                                strcat(destname, filename);
        !           269:                        if ((out = fopen(destname, "w")) == NULL)
        !           270:                                exit(5); /* all for naught! */
        !           271: #else !RECOVER
        !           272:                        exit(5);
        !           273: #endif !RECOVER
        !           274:                }
        !           275:                if (mode > 0)
        !           276:                        chmod(destname, mode);  /* don't bother to check it */
        !           277:        }
        !           278: 
        !           279:        /*
        !           280:         * Now, in any case, copy from in to out.
        !           281:         */
        !           282: 
        !           283:        count = 0;
        !           284:        while ((c=getc(in)) != EOF) {
        !           285:                putc(c, out);
        !           286:                count++;
        !           287:        }
        !           288: #ifdef DEBUG
        !           289:        fprintf(dout, "count %ld bytes\n", count);
        !           290:        fclose(dout);
        !           291: #endif DEBUG
        !           292: 
        !           293:        fclose(in);
        !           294:        fclose(out);    /* really should pclose in that case */
        !           295:        exit(0);
        !           296: }
        !           297: 
        !           298: /*
        !           299:  * Return the ptr in sp at which the character c appears;
        !           300:  * NULL if not found.  Included so I don't have to fight the
        !           301:  * index/strchr battle.
        !           302:  */
        !           303: 
        !           304: #define        NULL    0
        !           305: 
        !           306: char *
        !           307: index(sp, c)
        !           308: register char *sp, c;
        !           309: {
        !           310:        do {
        !           311:                if (*sp == c)
        !           312:                        return(sp);
        !           313:        } while (*sp++);
        !           314:        return(NULL);
        !           315: }
        !           316: 
        !           317: #ifdef RECOVER
        !           318: char *
        !           319: getfname(p)
        !           320: register char *p;
        !           321: {
        !           322:        register char *s;
        !           323:        s = p;
        !           324:        while (*p != '\0')
        !           325:                p++;
        !           326:        if (p == s)
        !           327:                return (NULL);
        !           328:        for (;p != s; p--)
        !           329:                if (*p == '/') {
        !           330:                        p++;
        !           331:                        break;
        !           332:                }
        !           333:        return (p);
        !           334: }
        !           335: 
        !           336: #ifndef BSD4_2
        !           337: makedir(dirname, mode)
        !           338: char *dirname;
        !           339: int mode;
        !           340: {
        !           341:        register int pid;
        !           342:        int retcode, status;
        !           343:        switch ((pid = fork())) {
        !           344:            case -1:            /* error */
        !           345:                return (-1);
        !           346:            case 0:             /* child */
        !           347:                umask(0);
        !           348:                execl("/bin/mkdir", "mkdir", dirname, (char *)0);
        !           349:                exit(1);
        !           350:                /* NOTREACHED */
        !           351:            default:            /* parent */
        !           352:                while ((retcode=wait(&status)) != pid && retcode != -1)
        !           353:                        ;
        !           354:                if (retcode == -1)
        !           355:                        return  -1;
        !           356:                else {
        !           357:                        chmod(dirname, mode);
        !           358:                        return status;
        !           359:                }
        !           360:        }
        !           361:        /* NOTREACHED */
        !           362: }
        !           363: #endif !BSD4_2
        !           364: #endif RECOVER

unix.superglobalmegacorp.com

This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.