Annotation of 42BSD/usr.bin/uucp/uusend.c, revision 1.1.1.1

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

unix.superglobalmegacorp.com

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