Annotation of 43BSDReno/usr.bin/uucp/cntrl.c, revision 1.1

1.1     ! root        1: #ifndef lint
        !             2: static char sccsid[] = "@(#)cntrl.c    5.12    (Berkeley) 5/4/88";
        !             3: #endif
        !             4: 
        !             5: #include "uucp.h"
        !             6: #include <sys/stat.h>
        !             7: #include "uust.h"
        !             8: 
        !             9: extern int errno;
        !            10: extern int turntime;
        !            11: int willturn;
        !            12: int HaveSentHup = 0;
        !            13: 
        !            14: struct Proto {
        !            15:        char P_id;
        !            16:        int (*P_turnon)();
        !            17:        int (*P_rdmsg)();
        !            18:        int (*P_wrmsg)();
        !            19:        int (*P_rddata)();
        !            20:        int (*P_wrdata)();
        !            21:        int (*P_turnoff)();
        !            22: };
        !            23: 
        !            24: extern int gturnon(), gturnoff();
        !            25: extern int grdmsg(), grddata();
        !            26: extern int gwrmsg(), gwrdata();
        !            27: extern int imsg(), omsg(), nullf();
        !            28: #ifdef TCPIP
        !            29: extern int twrmsg(), trdmsg();
        !            30: extern int twrdata(), trddata();
        !            31: #endif TCPIP
        !            32: #ifdef PAD
        !            33: extern int fturnon(), fturnoff();
        !            34: extern int frdmsg(), frddata();
        !            35: extern int fwrmsg(), fwrdata();
        !            36: #endif PAD
        !            37: 
        !            38: struct Proto Ptbl[]={
        !            39: #ifdef TCPIP
        !            40:        't', nullf, trdmsg, twrmsg, trddata, twrdata, nullf,
        !            41: #endif TCPIP
        !            42: #ifdef PAD
        !            43:        'f', fturnon, frdmsg, fwrmsg, frddata, fwrdata, fturnoff,
        !            44: #endif PAD
        !            45:        'g', gturnon, grdmsg, gwrmsg, grddata, gwrdata, gturnoff,
        !            46:        '\0'
        !            47: };
        !            48: 
        !            49: int (*Imsg)() = imsg, (*Omsg)() = omsg;
        !            50: 
        !            51: int (*Rdmsg)()=imsg, (*Rddata)();
        !            52: int (*Wrmsg)()=omsg, (*Wrdata)();
        !            53: int (*Turnon)()=nullf, (*Turnoff)() = nullf;
        !            54: 
        !            55: struct timeb Now, LastTurned, LastCheckedNoLogin;
        !            56: 
        !            57: static char *YES = "Y";
        !            58: static char *NO = "N";
        !            59: 
        !            60: int TransferSucceeded = 1;
        !            61: 
        !            62: /*  failure messages  */
        !            63: #define EM_MAX         6
        !            64: #define EM_LOCACC      "N1"    /* local access to file denied */
        !            65: #define EM_RMTACC      "N2"    /* remote access to file/path denied */
        !            66: #define EM_BADUUCP     "N3"    /* a bad uucp command was generated */
        !            67: #define EM_NOTMP       "N4"    /* remote error - can't create temp */
        !            68: #define EM_RMTCP       "N5"    /* can't copy to remote directory - file in public */
        !            69: #define EM_LOCCP       "N6"    /* can't copy on local system */
        !            70: 
        !            71: char *Em_msg[] = {
        !            72:        "COPY FAILED (reason not given by remote)",
        !            73:        "local access to file denied",
        !            74:        "remote access to path/file denied",
        !            75:        "system error - bad uucp command generated",
        !            76:        "remote system can't create temp file",
        !            77:        "can't copy to file/directory - file left in PUBDIR/user/file",
        !            78:        "can't copy to file/directory on local system  - file left in PUBDIR/user/file"
        !            79: };
        !            80: 
        !            81: 
        !            82: #define XUUCP 'X'      /* execute uucp (string) */
        !            83: #define SLTPTCL 'P'    /* select protocol  (string)  */
        !            84: #define USEPTCL 'U'    /* use protocol (character) */
        !            85: #define RCVFILE 'R'    /* receive file (string) */
        !            86: #define SNDFILE 'S'    /* send file (string) */
        !            87: #define RQSTCMPT 'C'   /* request complete (string - yes | no) */
        !            88: #define HUP     'H'    /* ready to hangup (string - yes | no) */
        !            89: #define RESET  'X'     /* reset line modes */
        !            90: 
        !            91: #define W_TYPE         wrkvec[0]
        !            92: #define W_FILE1                wrkvec[1]
        !            93: #define W_FILE2                wrkvec[2]
        !            94: #define W_USER         wrkvec[3]
        !            95: #define W_OPTNS                wrkvec[4]
        !            96: #define W_DFILE                wrkvec[5]
        !            97: #define W_MODE         wrkvec[6]
        !            98: #define W_NUSER                wrkvec[7]
        !            99: 
        !           100: #define        XFRRATE 35000L
        !           101: #define RMESG(m, s, n) if (rmesg(m, s, n) != 0) {(*Turnoff)(); return FAIL;} else
        !           102: #define RAMESG(s, n) if (rmesg('\0', s, n) != 0) {(*Turnoff)(); return FAIL;} else
        !           103: #define WMESG(m, s) if(wmesg(m, s) != 0) {(*Turnoff)(); return FAIL;} else
        !           104: 
        !           105: char Wfile[MAXFULLNAME] = {'\0'};
        !           106: char Dfile[MAXFULLNAME];
        !           107: 
        !           108: /*
        !           109:  * To avoid a huge backlog of X. files, start uuxqt every so often.
        !           110:  */
        !           111: static int nXfiles = 0;        /* number of X files since last uuxqt start */
        !           112: static char send_or_receive;
        !           113: struct stat stbuf;
        !           114: 
        !           115: /*
        !           116:  *     cntrl  -  this routine will execute the conversation
        !           117:  *     between the two machines after both programs are
        !           118:  *     running.
        !           119:  *
        !           120:  *     return codes
        !           121:  *             SUCCESS - ok
        !           122:  *             FAIL - failed
        !           123:  */
        !           124: 
        !           125: cntrl(role, wkpre)
        !           126: int role;
        !           127: char *wkpre;
        !           128: {
        !           129:        char msg[BUFSIZ], rqstr[BUFSIZ];
        !           130:        register FILE *fp;
        !           131:        int filemode;
        !           132:        char filename[MAXFULLNAME], wrktype, *wrkvec[20];
        !           133:        extern (*Rdmsg)(), (*Wrmsg)();
        !           134:        extern char *index(), *lastpart();
        !           135:        int status = 1;
        !           136:        register int i, narg;
        !           137:        int mailopt, ntfyopt;
        !           138:        int ret;
        !           139:        static int pnum, tmpnum = 0;
        !           140:        extern int ReverseRole;
        !           141: 
        !           142:        pnum = getpid();
        !           143:        Wfile[0] = '\0';
        !           144:        willturn = turntime > 0;
        !           145: remaster:
        !           146: #ifdef USG
        !           147:        time(&LastTurned.time);
        !           148:        LastTurned.millitm = 0;
        !           149: #else !USG
        !           150:        ftime(&LastTurned);
        !           151: #endif !USG
        !           152:        send_or_receive = RESET;
        !           153:        HaveSentHup = 0;
        !           154: top:
        !           155:        for (i = 0; i < sizeof wrkvec / sizeof wrkvec[0]; i++)
        !           156:                wrkvec[i] = 0;
        !           157:        DEBUG(4, "*** TOP ***  -  role=%s\n", role ? "MASTER" : "SLAVE");
        !           158:        setproctitle("%s: %s", Rmtname, role ? "MASTER" : "SLAVE");
        !           159:        setupline(RESET);
        !           160:        if (Now.time > (LastCheckedNoLogin.time+60)) {
        !           161:                LastCheckedNoLogin = Now;
        !           162:                if (access(NOLOGIN, 0) == 0) {
        !           163:                        logent(NOLOGIN, "UUCICO SHUTDOWN");
        !           164:                        if (Debug > 4)
        !           165:                                logent("DEBUGGING", "continuing anyway");
        !           166:                        else {
        !           167:                                WMESG(HUP, YES);
        !           168:                                RMESG(HUP, msg, 1);
        !           169:                                goto process;
        !           170:                        }
        !           171:                }
        !           172:        }
        !           173:        if (role == MASTER) {
        !           174:                /* get work */
        !           175:                if (ReverseRole || (narg = gtwvec(Wfile, Spool, wkpre, wrkvec)) == 0) {
        !           176:                        ReverseRole = 0;
        !           177:                        WMESG(HUP, "");
        !           178:                        RMESG(HUP, msg, 1);
        !           179:                        goto process;
        !           180:                }
        !           181:                wrktype = W_TYPE[0];
        !           182: 
        !           183:                msg[0] = '\0';
        !           184:                for (i = 1; i < narg; i++) {
        !           185:                        strcat(msg, " ");
        !           186:                        strcat(msg, wrkvec[i]);
        !           187:                }
        !           188: 
        !           189:                if (wrktype == XUUCP) {
        !           190:                        sprintf(rqstr, "X %s", msg);
        !           191:                        logent(rqstr, "REQUEST");
        !           192:                        goto sendmsg;
        !           193:                }
        !           194:                mailopt = index(W_OPTNS, 'm') != NULL;
        !           195:                ntfyopt = index(W_OPTNS, 'n') != NULL;
        !           196: 
        !           197:                if (narg < 5 || W_TYPE[1] != '\0') {
        !           198:                        char *bnp;
        !           199:                        bnp = rindex(Wfile, '/');
        !           200:                        sprintf(rqstr, "%s/%s", CORRUPT, bnp ? bnp + 1 : Wfile);
        !           201:                        xmv(Wfile, rqstr);
        !           202:                        syslog(LOG_WARNING, "%s CORRUPTED: %d args", Wfile,
        !           203:                                narg);
        !           204:                        Wfile[0] = '\0';
        !           205:                        goto top;
        !           206:                }
        !           207:                sprintf(User, "%.9s", W_USER);
        !           208:                sprintf(rqstr, "(%s %s %s %s)", W_TYPE, W_FILE1,
        !           209:                  W_FILE2, W_USER);
        !           210:                logent(rqstr, "REQUEST");
        !           211:                if (wrktype == SNDFILE ) {
        !           212:                        strcpy(filename, W_FILE1);
        !           213:                        i = expfile(filename);
        !           214:                        DEBUG(4, "expfile type - %d, ", i);
        !           215:                        if (i != 0 && chkpth(User, "", filename))
        !           216:                                goto e_access;
        !           217:                        strcpy(Dfile, W_DFILE);
        !           218:                        fp = NULL;
        !           219:                        if (index(W_OPTNS, 'c') == NULL) {
        !           220:                                fp = fopen(subfile(Dfile), "r");
        !           221:                                if (fp != NULL)
        !           222:                                        i = 0;
        !           223:                        }
        !           224:                        if (fp == NULL &&
        !           225:                           (fp = fopen(subfile(filename), "r")) == NULL) {
        !           226:                                /*  can not read data file  */
        !           227:                                logent("CAN'T READ DATA", _FAILED);
        !           228:                                TransferSucceeded = 1; /* else will keep sending */
        !           229:                                USRF(USR_LOCACC);
        !           230:                                unlinkdf(Dfile);
        !           231:                                lnotify(User, filename, "can't access");
        !           232:                                goto top;
        !           233:                        }
        !           234:                        /* if file exists but is not generally readable... */
        !           235:                        if (i != 0 && fstat(fileno(fp), &stbuf) == 0
        !           236:                        &&  (stbuf.st_mode & ANYREAD) == 0) {
        !           237:                e_access:;
        !           238:                                /*  access denied  */
        !           239:                                if (fp != NULL) {
        !           240:                                        fclose(fp);
        !           241:                                        fp = NULL;
        !           242:                                }
        !           243:                                TransferSucceeded = 1; /* else will keep sending */
        !           244:                                logent("DENIED", "ACCESS");
        !           245:                                USRF(USR_LOCACC);
        !           246:                                unlinkdf(W_DFILE);
        !           247:                                lnotify(User, filename, "access denied");
        !           248:                                goto top;
        !           249:                        }
        !           250: 
        !           251:                        setupline(SNDFILE);
        !           252:                }
        !           253: 
        !           254:                if (wrktype == RCVFILE) {
        !           255:                        strcpy(filename, W_FILE2);
        !           256:                        expfile(filename);
        !           257:                        if (chkpth(User, "", filename)
        !           258:                         || chkperm(filename, index(W_OPTNS, 'd'))) {
        !           259:                                /*  access denied  */
        !           260:                                logent("DENIED", "ACCESS");
        !           261:                                TransferSucceeded = 1; /* else will keep trying */
        !           262:                                USRF(USR_LOCACC);
        !           263:                                lnotify(User, filename, "access denied");
        !           264:                                goto top;
        !           265:                        }
        !           266:                        sprintf(Dfile, "%s/TM.%05d.%03d", Spool, pnum, tmpnum++);
        !           267:                        if ((fp = fopen(subfile(Dfile), "w")) == NULL) {
        !           268:                                /*  can not create temp  */
        !           269:                                logent("CAN'T CREATE TM", _FAILED);
        !           270:                                USRF(USR_LNOTMP);
        !           271:                                unlinkdf(Dfile);
        !           272:                                goto top;
        !           273:                        }
        !           274:                        setupline(RCVFILE);
        !           275:                }
        !           276: sendmsg:
        !           277:                DEBUG(4, "wrktype - %c\n", wrktype);
        !           278:                WMESG(wrktype, msg);
        !           279:                RMESG(wrktype, msg, 1);
        !           280:                goto process;
        !           281:        }
        !           282: 
        !           283:        /* role is slave */
        !           284:        RAMESG(msg, 1);
        !           285:        if (willturn < 0)
        !           286:                willturn = msg[0] == HUP;
        !           287:                        
        !           288: process:
        !           289:        DEBUG(4, "PROCESS: msg - %s\n", msg);
        !           290:        switch (msg[0]) {
        !           291: 
        !           292:        case RQSTCMPT:
        !           293:                DEBUG(4, "RQSTCMPT:\n", CNULL);
        !           294:                if (msg[1] == 'N') {
        !           295:                        i = atoi(&msg[2]);
        !           296:                        if (i<0 || i>EM_MAX)
        !           297:                                i = 0;
        !           298:                        USRF( 1 << i );
        !           299:                        logent(Em_msg[i], "REQUEST FAILED");
        !           300:                        TransferSucceeded = 1; /* He had his chance */
        !           301:                }
        !           302:                if (msg[1] == 'Y') {
        !           303:                        USRF(USR_COK);
        !           304:                        TransferSucceeded = 1;
        !           305:                }
        !           306:                if (role == MASTER)
        !           307:                        notify(mailopt, W_USER, W_FILE1, Rmtname, &msg[1]);
        !           308: 
        !           309:                if (msg[2] == 'M' && role == MASTER) {
        !           310:                        extern int Nfiles;
        !           311:                        WMESG(HUP, "");
        !           312:                        RMESG(HUP, msg, 1);
        !           313:                        logent(Rmtname, "TURNAROUND");
        !           314: #ifdef USG
        !           315:                                time(&LastTurned.time);
        !           316:                                LastTurned.millitm = 0;
        !           317: #else !USG
        !           318:                                ftime(&LastTurned);
        !           319: #endif !USG
        !           320:                        Nfiles = 0; /* force rescan of queue for work */
        !           321:                        goto process;
        !           322:                }
        !           323:                goto top;
        !           324: 
        !           325:        case HUP:
        !           326:                DEBUG(4, "HUP:\n", CNULL);
        !           327:                HaveSentHup = 1;
        !           328:                if (msg[1] == 'Y') {
        !           329:                        if (role == MASTER)
        !           330:                                WMESG(HUP, YES);
        !           331:                        (*Turnoff)();
        !           332:                        Rdmsg = Imsg;
        !           333:                        Wrmsg = Omsg;
        !           334:                        return SUCCESS;
        !           335:                }
        !           336: 
        !           337:                if (msg[1] == 'N') {
        !           338:                        if (role != MASTER) {
        !           339:                                syslog(LOG_ERR, "Wrong Role - HUP");
        !           340:                                cleanup(FAIL);
        !           341:                        }
        !           342:                        role = SLAVE;
        !           343:                        goto remaster;
        !           344:                }
        !           345: 
        !           346:                /* get work */
        !           347:                if (!iswrk(Wfile, "chk", Spool, wkpre)) {
        !           348:                        WMESG(HUP, YES);
        !           349:                        RMESG(HUP, msg, 1);
        !           350:                        goto process;
        !           351:                }
        !           352: 
        !           353:                WMESG(HUP, NO);
        !           354:                /*
        !           355:                 * want to create an orphan uuxqt,
        !           356:                 * so a double-fork is needed.
        !           357:                 */
        !           358:                if (fork() == 0) {
        !           359:                        xuuxqt();
        !           360:                        _exit(0);
        !           361:                }
        !           362:                wait((int *)0);
        !           363:                role = MASTER;
        !           364:                goto remaster;
        !           365: 
        !           366:        case XUUCP:
        !           367:                if (role == MASTER) {
        !           368:                        goto top;
        !           369:                }
        !           370: 
        !           371:                /*  slave part  */
        !           372:                i = getargs(msg, wrkvec, 20);
        !           373:                strcpy(filename, W_FILE1);
        !           374:                if (index(filename, ';') != NULL || index(W_FILE2, ';') != NULL
        !           375:                    || i < 3) {
        !           376:                        WMESG(XUUCP, NO);
        !           377:                        goto top;
        !           378:                }
        !           379:                expfile(filename);
        !           380:                if (chkpth("", Rmtname, filename)) {
        !           381:                        WMESG(XUUCP, NO);
        !           382:                        logent("XUUCP DENIED", filename);
        !           383:                                USRF(USR_XUUCP);
        !           384:                        goto top;
        !           385:                }
        !           386:                sprintf(rqstr, "%s %s", filename, W_FILE2);
        !           387:                xuucp(rqstr);
        !           388:                WMESG(XUUCP, YES);
        !           389:                goto top;
        !           390: 
        !           391:        case SNDFILE:
        !           392:                /*  MASTER section of SNDFILE  */
        !           393: 
        !           394:                DEBUG(4, "%s\n", "SNDFILE:");
        !           395:                if (msg[1] == 'N') {
        !           396:                        i = atoi(&msg[2]);
        !           397:                        if (i < 0 || i > EM_MAX)
        !           398:                                i = 0;
        !           399:                        logent(Em_msg[i], "REQUEST FAILED");
        !           400:                        USRF( 1 << i );
        !           401:                        fclose(fp);
        !           402:                        fp = NULL;
        !           403:                        /* dont send him files he can't save */
        !           404:                        if (strcmp(&msg[1], EM_NOTMP) == 0) {
        !           405:                                WMESG(HUP, "");
        !           406:                                RMESG(HUP, msg, 1);
        !           407:                                goto process;
        !           408:                        }
        !           409:                        notify(mailopt, W_USER, W_FILE1, Rmtname, &msg[1]);
        !           410:                        if (role != MASTER) {
        !           411:                                syslog(LOG_ERR, "Wrong Role - SN");
        !           412:                                cleanup(FAIL);
        !           413:                        }
        !           414:                        unlinkdf(W_DFILE);
        !           415:                        goto top;
        !           416:                }
        !           417: 
        !           418:                if (msg[1] == 'Y') {
        !           419:                        /* send file */
        !           420:                        if (role != MASTER) {
        !           421:                                syslog(LOG_ERR, "Wrong Role - SY");
        !           422:                                cleanup(FAIL);
        !           423:                        }
        !           424:                        if (fstat(fileno(fp), &stbuf) < 0) {
        !           425:                                syslog(LOG_ERR, "stat(%s) failed: %m",filename);
        !           426:                                cleanup(FAIL);
        !           427:                        }
        !           428:                        i = 1 + (int)(stbuf.st_size / XFRRATE);
        !           429:                        if (send_or_receive != SNDFILE) {
        !           430:                                send_or_receive = SNDFILE;
        !           431:                                systat(Rmtname, SS_INPROGRESS, "SENDING");
        !           432:                        }
        !           433:                        ret = (*Wrdata)(fp, Ofn);
        !           434:                        fclose(fp);
        !           435:                        fp = NULL;
        !           436:                        if (ret != SUCCESS) {
        !           437:                                (*Turnoff)();
        !           438:                                USRF(USR_CFAIL);
        !           439:                                return FAIL;
        !           440:                        }
        !           441:                        RMESG(RQSTCMPT, msg, i);
        !           442:                        unlinkdf(W_DFILE);
        !           443:                        goto process;
        !           444:                }
        !           445: 
        !           446:                /*  SLAVE section of SNDFILE  */
        !           447:                if (role != SLAVE) {
        !           448:                        syslog(LOG_ERR, "Wrong Role - SLAVE");
        !           449:                        cleanup(FAIL);
        !           450:                }
        !           451: 
        !           452:                /* request to receive file */
        !           453:                /* check permissions */
        !           454:                i = getargs(msg, wrkvec, 20);
        !           455:                if (i < 5) {
        !           456:                        char *bnp;
        !           457:                        bnp = rindex(Wfile, '/');
        !           458:                        sprintf(rqstr, "%s/%s", CORRUPT, bnp ? bnp + 1 : Wfile);
        !           459:                        xmv(Wfile, rqstr);
        !           460:                        syslog(LOG_WARNING, "%s CORRUPTED: %d args", Wfile, i);
        !           461:                        Wfile[0] = '\0';
        !           462:                        goto top;
        !           463:                }
        !           464:                sprintf(rqstr, "(%s %s %s %s)", W_TYPE, W_FILE1, W_FILE2,
        !           465:                        W_USER);
        !           466:                logent(rqstr, "REQUESTED");
        !           467:                DEBUG(4, "msg - %s\n", msg);
        !           468:                strcpy(filename, W_FILE2);
        !           469:                /* Run uuxqt occasionally */
        !           470:                if (filename[0] == XQTPRE) {
        !           471:                        if (++nXfiles > 10) {
        !           472:                                nXfiles = 0;
        !           473:                                /*
        !           474:                                 * want to create an orphan uuxqt,
        !           475:                                 * so a double-fork is needed.
        !           476:                                 */
        !           477:                                if (fork() == 0) {
        !           478:                                        xuuxqt();
        !           479:                                        _exit(0);
        !           480:                                }
        !           481:                                wait((int *)0);
        !           482:                        }
        !           483:                }
        !           484:                /* expand filename, i is set to 0 if this is
        !           485:                 * is a vanilla spool file, so no stat(II)s are needed */
        !           486:                i = expfile(filename);
        !           487:                DEBUG(4, "expfile type - %d\n", i);
        !           488:                if (i != 0) {
        !           489:                        if (chkpth("", Rmtname, filename)
        !           490:                         || chkperm(filename, index(W_OPTNS, 'd'))) {
        !           491:                                WMESG(SNDFILE, EM_RMTACC);
        !           492:                                logent("DENIED", "PERMISSION");
        !           493:                                goto top;
        !           494:                        }
        !           495:                        if (isdir(filename)) {
        !           496:                                strcat(filename, "/");
        !           497:                                strcat(filename, lastpart(W_FILE1));
        !           498:                        }
        !           499:                }
        !           500:                sprintf(User, "%.9s", W_USER);
        !           501: 
        !           502:                DEBUG(4, "chkpth ok Rmtname - %s\n", Rmtname);
        !           503:                /* speed things up by OKing file before
        !           504:                 * creating TM file.  If the TM file cannot be created,
        !           505:                 * then the conversation bombs, but that seems reasonable,
        !           506:                 * as there are probably serious problems then.
        !           507:                 */
        !           508:                WMESG(SNDFILE, YES);
        !           509:                sprintf(Dfile, "%s/TM.%05d.%03d", Spool, pnum, tmpnum++);
        !           510:                if((fp = fopen(subfile(Dfile), "w")) == NULL) {
        !           511: /*                     WMESG(SNDFILE, EM_NOTMP);*/
        !           512:                        logent("CAN'T OPEN", "TM FILE");
        !           513:                        unlinkdf(Dfile);
        !           514:                        (*Turnoff)();
        !           515:                        return FAIL;
        !           516:                }
        !           517: 
        !           518:                if (send_or_receive != RCVFILE) {
        !           519:                        send_or_receive = RCVFILE;
        !           520:                        systat(Rmtname, SS_INPROGRESS, "RECEIVING");
        !           521:                }
        !           522:                ret = (*Rddata)(Ifn, fp);
        !           523:                fflush(fp);
        !           524:                if (ferror(fp) || fclose(fp))
        !           525:                        ret = FAIL;
        !           526:                
        !           527:                if (ret != SUCCESS) {
        !           528:                        (void) unlinkdf(Dfile);
        !           529:                        (*Turnoff)();
        !           530:                        return FAIL;
        !           531:                }
        !           532:                /* copy to user directory */
        !           533:                ntfyopt = index(W_OPTNS, 'n') != NULL;
        !           534:                status = xmv(Dfile, filename);
        !           535: 
        !           536:                if (willturn && Now.time > (LastTurned.time+turntime)
        !           537:                        && iswrk(Wfile, "chk", Spool, wkpre)) {
        !           538:                                WMESG(RQSTCMPT, status ? EM_RMTCP : "YM");
        !           539:                                willturn = -1;
        !           540:                } else
        !           541:                        WMESG(RQSTCMPT, status ? EM_RMTCP : YES);
        !           542:                if (i == 0)
        !           543:                        ;       /* vanilla file, nothing to do */
        !           544:                else if (status == 0) {
        !           545:                        if (W_MODE == 0 || sscanf(W_MODE, "%o", &filemode) != 1)
        !           546:                                filemode = BASEMODE;
        !           547:                        chmod(subfile(filename), (filemode|BASEMODE)&0777);
        !           548:                        arrived(ntfyopt, filename, W_NUSER, Rmtname, User);
        !           549:                } else {
        !           550:                        logent(_FAILED, "COPY");
        !           551:                        status = putinpub(filename, Dfile, W_USER);
        !           552:                        DEBUG(4, "->PUBDIR %d\n", status);
        !           553:                        if (status == 0)
        !           554:                                arrived(ntfyopt, filename, W_NUSER, Rmtname, User);
        !           555:                }
        !           556: 
        !           557:                goto top;
        !           558: 
        !           559:        case RCVFILE:
        !           560:                /*  MASTER section of RCVFILE  */
        !           561: 
        !           562:                DEBUG(4, "%s\n", "RCVFILE:");
        !           563:                if (msg[1] == 'N') {
        !           564:                        i = atoi(&msg[2]);
        !           565:                        if (i < 0 || i > EM_MAX)
        !           566:                                i = 0;
        !           567:                        logent(Em_msg[i], "REQUEST FAILED");
        !           568:                        USRF( 1 << i );
        !           569:                        fclose(fp);
        !           570:                        fp = NULL;
        !           571:                        notify(mailopt, W_USER, W_FILE1, Rmtname, &msg[1]);
        !           572:                        if (role != MASTER) {
        !           573:                                syslog(LOG_ERR, "Wrong Role - RN");
        !           574:                                cleanup(FAIL);
        !           575:                        }
        !           576:                        unlinkdf(Dfile);
        !           577:                        goto top;
        !           578:                }
        !           579: 
        !           580:                if (msg[1] == 'Y') {
        !           581:                        /* receive file */
        !           582:                        if (role != MASTER) {
        !           583:                                syslog(LOG_ERR, "Wrong Role - RY");
        !           584:                                cleanup(FAIL);
        !           585:                        }
        !           586:                        if (send_or_receive != RCVFILE) {
        !           587:                                send_or_receive = RCVFILE;
        !           588:                                systat(Rmtname, SS_INPROGRESS, "RECEIVING");
        !           589:                        }
        !           590:                        ret = (*Rddata)(Ifn, fp);
        !           591:                        fflush(fp);
        !           592:                        if (ferror(fp) || fclose(fp))
        !           593:                                ret = FAIL;
        !           594:                        if (ret != SUCCESS) {
        !           595:                                unlinkdf(Dfile);
        !           596:                                (*Turnoff)();
        !           597:                                USRF(USR_CFAIL);
        !           598:                                return FAIL;
        !           599:                        }
        !           600:                        /* copy to user directory */
        !           601:                        if (isdir(filename)) {
        !           602:                                strcat(filename, "/");
        !           603:                                strcat(filename, lastpart(W_FILE1));
        !           604:                        }
        !           605:                        status = xmv(Dfile, filename);
        !           606:                        WMESG(RQSTCMPT, status ? EM_RMTCP : YES);
        !           607:                        notify(mailopt, W_USER, filename, Rmtname,
        !           608:                                status ? EM_LOCCP : YES);
        !           609:                        if (status == 0) {
        !           610:                                sscanf(&msg[2], "%o", &filemode);
        !           611:                                if (filemode <= 0)
        !           612:                                        filemode = BASEMODE;
        !           613:                                chmod(subfile(filename), (filemode|BASEMODE)&0777);
        !           614:                                USRF(USR_COK);
        !           615:                        } else {
        !           616:                                logent(_FAILED, "COPY");
        !           617:                                putinpub(filename, Dfile, W_USER);
        !           618:                                USRF(USR_LOCCP);
        !           619:                        }
        !           620:                        if (msg[strlen(msg)-1] == 'M') {
        !           621:                                extern int Nfiles;
        !           622:                                WMESG(HUP, "");
        !           623:                                RMESG(HUP, msg, 1);
        !           624:                                logent(Rmtname, "TURNAROUND");
        !           625: #ifdef USG
        !           626:                                time(&LastTurned.time);
        !           627:                                LastTurned.millitm = 0;
        !           628: #else !USG
        !           629:                                ftime(&LastTurned);
        !           630: #endif !USG
        !           631:                                Nfiles = 0; /* force rescan of queue for work */
        !           632:                                goto process;
        !           633:                        }
        !           634:                        goto top;
        !           635:                }
        !           636: 
        !           637:                /*  SLAVE section of RCVFILE  */
        !           638:                if (role != SLAVE) {
        !           639:                        syslog(LOG_ERR, "Wrong Role - SLAVE RCV");
        !           640:                        cleanup(FAIL);
        !           641:                }
        !           642: 
        !           643:                /* request to send file */
        !           644:                sprintf(rqstr,"(%s)", msg);
        !           645:                logent(rqstr, "REQUESTED");
        !           646: 
        !           647:                /* check permissions */
        !           648:                i = getargs(msg, wrkvec, 20);
        !           649:                if (i < 4) {
        !           650:                        char *bnp;
        !           651:                        bnp = rindex(Wfile, '/');
        !           652:                        sprintf(rqstr, "%s/%s", CORRUPT, bnp ? bnp + 1 : Wfile);
        !           653:                        xmv(Wfile, rqstr);
        !           654:                        syslog(LOG_WARNING, "%s CORRUPTED: %d args", Wfile, i);
        !           655:                        Wfile[0] = '\0';
        !           656:                        goto top;
        !           657:                }
        !           658:                DEBUG(4, "msg - %s\n", msg);
        !           659:                DEBUG(4, "W_FILE1 - %s\n", W_FILE1);
        !           660:                strcpy(filename, W_FILE1);
        !           661:                expfile(filename);
        !           662:                if (isdir(filename)) {
        !           663:                        strcat(filename, "/");
        !           664:                        strcat(filename, lastpart(W_FILE2));
        !           665:                }
        !           666:                sprintf(User, "%.9s", W_USER);
        !           667:                if (chkpth("", Rmtname, filename) || anyread(filename)) {
        !           668:                        WMESG(RCVFILE, EM_RMTACC);
        !           669:                        logent("DENIED", "PERMISSION");
        !           670:                        goto top;
        !           671:                }
        !           672:                DEBUG(4, "chkpth ok Rmtname - %s\n", Rmtname);
        !           673: 
        !           674:                if ((fp = fopen(subfile(filename), "r")) == NULL) {
        !           675:                        WMESG(RCVFILE, EM_RMTACC);
        !           676:                        logent("CAN'T OPEN", "DENIED");
        !           677:                        goto top;
        !           678:                }
        !           679: 
        !           680:                /*  ok to send file */
        !           681:                if (fstat(fileno(fp), &stbuf) < 0) {
        !           682:                        syslog(LOG_ERR, "stat(%s) failed: %m", filename);
        !           683:                        cleanup(FAIL);
        !           684:                }
        !           685: 
        !           686:                i = 1 + (int)(stbuf.st_size / XFRRATE);
        !           687:                if (willturn && Now.time > (LastTurned.time+turntime)
        !           688:                        && iswrk(Wfile, "chk", Spool, wkpre)) {
        !           689:                                willturn = -1;
        !           690:                }
        !           691:                sprintf(msg, "%s %o%s", YES, (int)stbuf.st_mode & 0777,
        !           692:                        willturn < 0 ? " M" : "");
        !           693:                WMESG(RCVFILE, msg);
        !           694:                if (send_or_receive != SNDFILE) {
        !           695:                        send_or_receive = SNDFILE;
        !           696:                        systat(Rmtname, SS_INPROGRESS, "SENDING");
        !           697:                }
        !           698:                ret = (*Wrdata)(fp, Ofn);
        !           699:                fclose(fp);
        !           700:                if (ret != SUCCESS) {
        !           701:                        (*Turnoff)();
        !           702:                        return FAIL;
        !           703:                }
        !           704:                RMESG(RQSTCMPT, msg, i);
        !           705:                goto process;
        !           706:        }
        !           707:        (*Turnoff)();
        !           708:        return FAIL;
        !           709: }
        !           710: 
        !           711: 
        !           712: /*
        !           713:  *     read message 'c'. try 'n' times
        !           714:  *
        !           715:  *     return code:  SUCCESS  |  FAIL
        !           716:  */
        !           717: rmesg(c, msg, n)
        !           718: register char *msg, c;
        !           719: register int n;
        !           720: {
        !           721:        char str[MAXFULLNAME];
        !           722: 
        !           723:        DEBUG(4, "rmesg - '%c' ", c);
        !           724:        while ((*Rdmsg)(msg, Ifn) != SUCCESS) {
        !           725:                if (--n > 0) {
        !           726:                        sprintf(str, "%d", n);
        !           727:                        logent(str, "PATIENCE");
        !           728:                        continue;
        !           729:                }
        !           730:                DEBUG(4, "got FAIL\n", CNULL);
        !           731:                if (c != '\0')
        !           732:                        sprintf(str, "expected '%c' got FAIL (%d)", c, errno);
        !           733:                else
        !           734:                        sprintf(str, "expected ANY got FAIL (%d)", errno);
        !           735:                logent(str, "BAD READ");
        !           736:                return FAIL;
        !           737:        }
        !           738:        if (c != '\0' && msg[0] != c) {
        !           739:                DEBUG(4, "got %s\n", msg);
        !           740:                sprintf(str, "expected '%c' got %s", c, msg);
        !           741:                logent(str, "BAD READ");
        !           742:                return FAIL;
        !           743:        }
        !           744:        DEBUG(4, "got %s\n", msg);
        !           745:        return SUCCESS;
        !           746: }
        !           747: 
        !           748: 
        !           749: /*
        !           750:  *     write a message (type m)
        !           751:  *
        !           752:  *     return codes: SUCCESS - ok | FAIL - ng
        !           753:  */
        !           754: wmesg(m, s)
        !           755: register char *s, m;
        !           756: {
        !           757:        DEBUG(4, "wmesg '%c' ", m);
        !           758:        DEBUG(4, "%s\n", s);
        !           759:        return (*Wrmsg)(m, s, Ofn);
        !           760: }
        !           761: 
        !           762: /*
        !           763:  *     mail results of command
        !           764:  *
        !           765:  *     return codes:  none
        !           766:  */
        !           767: notify(mailopt, user, file, sys, msgcode)
        !           768: char *user, *file, *sys, *msgcode;
        !           769: {
        !           770:        char str[BUFSIZ];
        !           771:        int i;
        !           772:        char *msg;
        !           773: 
        !           774:        if (!mailopt && *msgcode == 'Y')
        !           775:                return;
        !           776:        if (*msgcode == 'Y')
        !           777:                msg = "copy succeeded";
        !           778:        else {
        !           779:                i = atoi(msgcode + 1);
        !           780:                if (i < 1 || i > EM_MAX)
        !           781:                        i = 0;
        !           782:                msg = Em_msg[i];
        !           783:        }
        !           784:        sprintf(str, "file %s!%s -- %s\n",
        !           785:                sys,file, msg);
        !           786:        mailst(user, str, CNULL);
        !           787:        return;
        !           788: }
        !           789: 
        !           790: /*
        !           791:  *     local notify
        !           792:  *
        !           793:  *     return code - none
        !           794:  */
        !           795: lnotify(user, file, mesg)
        !           796: char *user, *file, *mesg;
        !           797: {
        !           798:        char mbuf[200];
        !           799:        sprintf(mbuf, "file %s!%s -- %s\n", Myname, file, mesg);
        !           800:        mailst(user, mbuf, CNULL);
        !           801:        return;
        !           802: }
        !           803: 
        !           804: char UsingProtocol;
        !           805: 
        !           806: /*
        !           807:  *     converse with the remote machine, agree upon a protocol (if possible)
        !           808:  *     and start the protocol.
        !           809:  *
        !           810:  *     return codes:
        !           811:  *             SUCCESS - successful protocol selection
        !           812:  *             FAIL - can't find common or open failed
        !           813:  */
        !           814: startup(role)
        !           815: int role;
        !           816: {
        !           817:        extern (*Rdmsg)(), (*Wrmsg)();
        !           818:        extern char *blptcl(), fptcl();
        !           819:        char msg[BUFSIZ], str[MAXFULLNAME];
        !           820: 
        !           821:        Rdmsg = Imsg;
        !           822:        Wrmsg = Omsg;
        !           823:        if (role == MASTER) {
        !           824:                RMESG(SLTPTCL, msg, 1);
        !           825:                if ((str[0] = fptcl(&msg[1])) == NULL) {
        !           826:                        /* no protocol match */
        !           827:                        WMESG(USEPTCL, NO);
        !           828:                        return FAIL;
        !           829:                }
        !           830:                str[1] = '\0';
        !           831:                WMESG(USEPTCL, str);
        !           832:                if (stptcl(str) != 0)
        !           833:                        return FAIL;
        !           834:                DEBUG(4, "protocol %s\n", str);
        !           835:                UsingProtocol = str[0];
        !           836:                return SUCCESS;
        !           837:        }
        !           838:        else {
        !           839:                WMESG(SLTPTCL, blptcl(str));
        !           840:                RMESG(USEPTCL, msg, 1);
        !           841:                if (msg[1] == 'N') {
        !           842:                        return FAIL;
        !           843:                }
        !           844: 
        !           845:                if (stptcl(&msg[1]) != 0)
        !           846:                        return FAIL;
        !           847:                DEBUG(4, "Protocol %s\n", msg);
        !           848:                UsingProtocol = msg[1];
        !           849:                return SUCCESS;
        !           850:        }
        !           851: }
        !           852: 
        !           853: /*
        !           854:  *     choose a protocol from the input string (str) and return the it
        !           855:  *
        !           856:  *     return codes:
        !           857:  *             '\0'  -  no acceptable protocol
        !           858:  *             any character  -  the chosen protocol
        !           859:  */
        !           860: char
        !           861: fptcl(str)
        !           862: register char *str;
        !           863: {
        !           864:        register struct Proto *p;
        !           865:        extern char LineType[];
        !           866: 
        !           867:        for (p = Ptbl; p->P_id != '\0'; p++) {
        !           868: #ifdef TCPIP
        !           869:                /* Only use 't' on TCP/IP */
        !           870:                if (p->P_id == 't' && strcmp("TCP", LineType))
        !           871:                        continue;
        !           872: #endif TCPIP
        !           873: #ifdef PAD
        !           874:                /* only use 'f' protocol on PAD */
        !           875:                if (p->P_id == 'f' && strcmp("PAD", LineType))
        !           876:                        continue;
        !           877: #endif PAD
        !           878:                if (index(str, p->P_id) != NULL) {
        !           879:                        return p->P_id;
        !           880:                }
        !           881:        }
        !           882: 
        !           883:        return '\0';
        !           884: }
        !           885: 
        !           886: /*
        !           887:  *     build a string of the letters of the available protocols 
        !           888:  */
        !           889: char *
        !           890: blptcl(str)
        !           891: register char *str;
        !           892: {
        !           893:        register struct Proto *p;
        !           894:        register char *s;
        !           895: 
        !           896:        for (p = Ptbl, s = str; (*s++ = p->P_id) != '\0'; p++)
        !           897:                ;
        !           898:        *s = '\0';
        !           899:        return str;
        !           900: }
        !           901: 
        !           902: /*
        !           903:  *     this routine will set up the six routines
        !           904:  *     (Rdmsg, Wrmsg, Rddata, Wrdata, Turnon, Turnoff) for the
        !           905:  *     desired protocol.
        !           906:  *
        !           907:  *     return codes:
        !           908:  *             SUCCESS - ok
        !           909:  *             FAIL - no find or failed to open
        !           910:  *
        !           911:  */
        !           912: stptcl(c)
        !           913: register char *c;
        !           914: {
        !           915:        register struct Proto *p;
        !           916: 
        !           917:        for (p = Ptbl; p->P_id != '\0'; p++) {
        !           918:                if (*c == p->P_id) {
        !           919:                        /* found protocol - set routines */
        !           920:                        Rdmsg = p->P_rdmsg;
        !           921:                        Wrmsg = p->P_wrmsg;
        !           922:                        Rddata = p->P_rddata;
        !           923:                        Wrdata = p->P_wrdata;
        !           924:                        Turnon = p->P_turnon;
        !           925:                        Turnoff = p->P_turnoff;
        !           926:                        if ((*Turnon)() != SUCCESS)
        !           927:                                return FAIL;
        !           928:                        DEBUG(4, "Proto started %c\n", *c);
        !           929:                        return SUCCESS;
        !           930:                }
        !           931:        }
        !           932:        DEBUG(4, "Proto start-fail %c\n", *c);
        !           933:        return FAIL;
        !           934: }
        !           935: 
        !           936: /*
        !           937:  *     put file in public place. if successful, filename is modified
        !           938:  *
        !           939:  *     return code  SUCCESS | FAIL
        !           940:  */
        !           941: 
        !           942: putinpub(file, tmp, user)
        !           943: register char *file, *tmp, *user;
        !           944: {
        !           945:        char fullname[MAXFULLNAME];
        !           946:        char *lastpart();
        !           947:        int status;
        !           948: 
        !           949:        sprintf(fullname, "%s/%s/", PUBDIR, user);
        !           950:        if (mkdirs(fullname) != 0) {
        !           951:                /* can not make directories */
        !           952:                DEBUG(1, "Cannot mkdirs(%s)\n", fullname);
        !           953:                return FAIL;
        !           954:        }
        !           955:        strcat(fullname, lastpart(file));
        !           956:        status = xmv(tmp, fullname);
        !           957:        if (status == 0) {
        !           958:                strcpy(file, fullname);
        !           959:                chmod(subfile(fullname), BASEMODE);
        !           960:        }
        !           961:        return status;
        !           962: }
        !           963: 
        !           964: /*
        !           965:  *     unlink D. file
        !           966:  *
        !           967:  *     return code - none
        !           968:  */
        !           969: 
        !           970: unlinkdf(file)
        !           971: register char *file;
        !           972: {
        !           973:        if (strlen(file) > 6)
        !           974:                unlink(subfile(file));
        !           975:        return;
        !           976: }
        !           977: 
        !           978: /*
        !           979:  *     notify receiver of arrived file
        !           980:  *
        !           981:  *     return code - none
        !           982:  */
        !           983: arrived(opt, file, nuser, rmtsys, rmtuser)
        !           984: char *file, *nuser, *rmtsys, *rmtuser;
        !           985: {
        !           986:        char mbuf[200];
        !           987: 
        !           988:        if (!opt)
        !           989:                return;
        !           990:        sprintf(mbuf, "%s from %s!%s arrived\n", file, rmtsys, rmtuser);
        !           991:        mailst(nuser, mbuf, CNULL);
        !           992:        return;
        !           993: }
        !           994: 
        !           995: nullf()
        !           996: {
        !           997:        return SUCCESS;
        !           998: }

unix.superglobalmegacorp.com

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