[BACK]Return to admin.c CVS log [TXT][DIR] Up to [local] / src / usr.bin / cvs

Annotation of src/usr.bin/cvs/admin.c, Revision 1.5

1.5     ! joris       1: /*     $OpenBSD: admin.c,v 1.4 2005/03/08 00:20:39 joris Exp $ */
1.1       joris       2: /*
                      3:  * Copyright (c) 2004 Jean-Francois Brousseau <jfb@openbsd.org>
                      4:  * Copyright (c) 2005 Joris Vink <joris@openbsd.org>
                      5:  * All rights reserved.
                      6:  *
                      7:  * Redistribution and use in source and binary forms, with or without
                      8:  * modification, are permitted provided that the following conditions
                      9:  * are met:
                     10:  *
                     11:  * 1. Redistributions of source code must retain the above copyright
                     12:  *    notice, this list of conditions and the following disclaimer.
                     13:  * 2. The name of the author may not be used to endorse or promote products
                     14:  *    derived from this software without specific prior written permission.
                     15:  *
                     16:  * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES,
                     17:  * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
                     18:  * AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL
                     19:  * THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
                     20:  * EXEMPLARY, OR CONSEQUENTIAL  DAMAGES (INCLUDING, BUT NOT LIMITED TO,
                     21:  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
                     22:  * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
                     23:  * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
                     24:  * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
                     25:  * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
                     26:  */
                     27:
                     28: #include <sys/types.h>
                     29: #include <sys/stat.h>
                     30:
                     31: #include <errno.h>
                     32: #include <stdio.h>
                     33: #include <fcntl.h>
                     34: #include <stdlib.h>
                     35: #include <unistd.h>
                     36: #include <string.h>
                     37: #include <sysexits.h>
                     38:
                     39: #include "cvs.h"
                     40: #include "log.h"
                     41: #include "proto.h"
                     42:
                     43:
                     44: int cvs_admin_file (CVSFILE *, void *);
                     45:
                     46: #define LOCK_SET       0x01
                     47: #define LOCK_REMOVE    0x02
                     48:
                     49: #define FLAG_BRANCH            0x01
                     50: #define FLAG_DELUSER           0x02
                     51: #define FLAG_INTERACTIVE       0x04
                     52: #define FLAG_QUIET             0x08
                     53:
                     54: /*
                     55:  * cvs_admin()
                     56:  *
                     57:  * Handler for the `cvs admin' command.
                     58:  * Returns 0 on success, or one of the known exit codes on error.
                     59:  */
                     60: int
                     61: cvs_admin(int argc, char **argv)
                     62: {
                     63:        int i, ch, flags;
1.3       joris      64:        int runflags, kflag, lockrev, strictlock;
1.5     ! joris      65:        char *q, *Ntag, *ntag, *comment, *replace_msg;
1.1       joris      66:        char *alist, *subst, *lockrev_arg, *unlockrev_arg;
1.5     ! joris      67:        char *state, *userfile, *branch_arg, *elist, *range;
1.1       joris      68:        struct cvsroot *root;
                     69:        RCSNUM *rcs;
                     70:
                     71:        runflags = strictlock = lockrev = 0;
1.5     ! joris      72:        Ntag = ntag = comment = replace_msg = NULL;
        !            73:        state = alist = subst = elist = lockrev_arg = NULL;
        !            74:        range = userfile = branch_arg = unlockrev_arg = NULL;
1.1       joris      75:        flags = CF_SORT|CF_IGNORE|CF_RECURSE;
                     76:
                     77:        /* option-o-rama ! */
                     78:        while ((ch = getopt(argc, argv, "a:A:b::c:e::Ik:l::Lm:n:N:o:qs:t:u::U"))
                     79:            != -1) {
                     80:                switch (ch) {
                     81:                case 'a':
                     82:                        alist = optarg;
                     83:                        break;
                     84:                case 'A':
                     85:                        userfile = optarg;
                     86:                        break;
                     87:                case 'b':
                     88:                        runflags |= FLAG_BRANCH;
                     89:                        if (optarg)
                     90:                                branch_arg = optarg;
                     91:                        break;
                     92:                case 'c':
                     93:                        comment = optarg;
                     94:                        break;
                     95:                case 'e':
                     96:                        runflags |= FLAG_DELUSER;
                     97:                        if (optarg)
                     98:                                elist = optarg;
                     99:                        break;
                    100:                case 'I':
                    101:                        runflags |= FLAG_INTERACTIVE;
                    102:                        break;
                    103:                case 'k':
                    104:                        subst = optarg;
1.3       joris     105:                        kflag = rcs_kflag_get(subst);
                    106:                        if (RCS_KWEXP_INVAL(kflag)) {
                    107:                                cvs_log(LP_ERR,
                    108:                                    "invalid RCS keyword expansion mode");
                    109:                                rcs_kflag_usage();
                    110:                                return (EX_USAGE);
                    111:                        }
1.1       joris     112:                        break;
                    113:                case 'l':
                    114:                        lockrev |= LOCK_SET;
                    115:                        if (optarg)
                    116:                                lockrev_arg = optarg;
                    117:                        break;
                    118:                case 'L':
                    119:                        strictlock |= LOCK_SET;
                    120:                        break;
                    121:                case 'm':
                    122:                        replace_msg = optarg;
                    123:                        break;
                    124:                case 'n':
1.5     ! joris     125:                        ntag = optarg;
1.1       joris     126:                        break;
                    127:                case 'N':
1.5     ! joris     128:                        Ntag = optarg;
1.1       joris     129:                        break;
                    130:                case 'o':
1.5     ! joris     131:                        range = optarg;
1.1       joris     132:                        break;
                    133:                case 'q':
                    134:                        runflags |= FLAG_QUIET;
                    135:                        break;
                    136:                case 's':
1.5     ! joris     137:                        state = optarg;
1.1       joris     138:                        break;
                    139:                case 't':
                    140:                        break;
                    141:                case 'u':
                    142:                        lockrev |= LOCK_REMOVE;
                    143:                        if (optarg)
                    144:                                unlockrev_arg = optarg;
                    145:                        break;
                    146:                case 'U':
                    147:                        strictlock |= LOCK_REMOVE;
                    148:                        break;
                    149:                default:
                    150:                        return (EX_USAGE);
                    151:                }
                    152:        }
                    153:
                    154:        argc -= optind;
                    155:        argv += optind;
                    156:
                    157:        /* do some sanity checking on the arguments */
                    158:        if ((strictlock & LOCK_SET) && (strictlock & LOCK_REMOVE)) {
                    159:                cvs_log(LP_ERR, "-L and -U are incompatible");
                    160:                return (EX_PROTOCOL);
                    161:        }
                    162:
                    163:        if (lockrev_arg != NULL) {
                    164:                if ((rcs = rcsnum_parse(lockrev_arg)) == NULL) {
                    165:                        cvs_log(LP_ERR, "%s is not a numeric branch",
                    166:                            lockrev_arg);
                    167:                        return (EX_USAGE);
                    168:                }
                    169:                rcsnum_free(rcs);
                    170:        }
                    171:
                    172:        if (unlockrev_arg != NULL) {
                    173:                if ((rcs = rcsnum_parse(unlockrev_arg)) == NULL) {
                    174:                        cvs_log(LP_ERR, "%s is not a numeric branch",
                    175:                            unlockrev_arg);
                    176:                        return (EX_PROTOCOL);
                    177:                }
                    178:                rcsnum_free(rcs);
                    179:        }
                    180:
                    181:        if (replace_msg != NULL) {
                    182:                if ((q = strchr(replace_msg, ':')) == NULL) {
                    183:                        cvs_log(LP_ERR, "invalid option for -m");
                    184:                        return (EX_USAGE);
                    185:                }
                    186:                *q = '\0';
                    187:                if ((rcs = rcsnum_parse(replace_msg)) == NULL) {
                    188:                        cvs_log(LP_ERR, "%s is not a numeric revision",
                    189:                            replace_msg);
                    190:                        return (EX_PROTOCOL);
                    191:                }
1.2       joris     192:                rcsnum_free(rcs);
1.1       joris     193:                *q = ':';
                    194:        }
                    195:
                    196:        if (argc == 0) {
                    197:                cvs_files = cvs_file_get(".", flags);
                    198:        } else {
                    199:                cvs_files = cvs_file_getspec(argv, argc, 0);
                    200:        }
                    201:        if (cvs_files == NULL)
                    202:                return (EX_DATAERR);
                    203:
                    204:        root = CVS_DIR_ROOT(cvs_files);
                    205:        if (root == NULL) {
                    206:                cvs_log(LP_ERR,
                    207:                    "No CVSROOT specified!  Please use the `-d' option");
                    208:                cvs_log(LP_ERR,
                    209:                    "or set the CVSROOT environment variable.");
                    210:                return (EX_USAGE);
                    211:        }
                    212:
                    213:        if (root->cr_method != CVS_METHOD_LOCAL) {
                    214:                if (cvs_connect(root) < 0)
                    215:                        return (EX_PROTOCOL);
                    216:
                    217:                if ((alist != NULL) && ((cvs_sendarg(root, "-a", 0) < 0) ||
                    218:                    (cvs_sendarg(root, alist, 0) < 0)))
                    219:                        return (EX_PROTOCOL);
                    220:
                    221:                if ((userfile != NULL) && ((cvs_sendarg(root, "-A", 0) < 0) ||
                    222:                    (cvs_sendarg(root, userfile, 0) < 0)))
                    223:                        return (EX_PROTOCOL);
                    224:
                    225:                if (runflags & FLAG_BRANCH) {
                    226:                        if (cvs_sendarg(root, "-b", 0) < 0)
                    227:                                return (EX_PROTOCOL);
                    228:                        if ((branch_arg != NULL) &&
                    229:                            (cvs_sendarg(root, branch_arg, 0) < 0))
                    230:                                return (EX_PROTOCOL);
                    231:                }
                    232:
1.4       joris     233:                if ((comment != NULL) && ((cvs_sendarg(root, "-c", 0) < 0) ||
                    234:                    (cvs_sendarg(root, comment, 0) < 0)))
1.1       joris     235:                        return (EX_PROTOCOL);
                    236:
                    237:                if (runflags & FLAG_DELUSER)  {
                    238:                        if (cvs_sendarg(root, "-e", 0) < 0)
                    239:                                return (EX_PROTOCOL);
                    240:                        if ((elist != NULL) &&
                    241:                            (cvs_sendarg(root, elist, 0) < 0))
                    242:                                return (EX_PROTOCOL);
                    243:                }
                    244:
                    245:                if (runflags & FLAG_INTERACTIVE) {
                    246:                        if (cvs_sendarg(root, "-I", 0) < 0)
                    247:                                return (EX_PROTOCOL);
                    248:                }
                    249:
                    250:                if ((subst != NULL) && ((cvs_sendarg(root, "-k", 0) < 0) ||
                    251:                    (cvs_sendarg(root, subst, 0) < 0)))
                    252:                        return (EX_PROTOCOL);
                    253:
                    254:                if (lockrev & LOCK_SET) {
                    255:                        if (cvs_sendarg(root, "-l", 0) < 0)
                    256:                                return (EX_PROTOCOL);
                    257:                        if ((lockrev_arg != NULL) &&
                    258:                            (cvs_sendarg(root, lockrev_arg, 0) < 0))
                    259:                                return (0);
                    260:                }
                    261:
                    262:                if ((strictlock & LOCK_SET) &&
                    263:                    (cvs_sendarg(root, "-L", 0) < 0))
                    264:                        return (EX_PROTOCOL);
                    265:
                    266:                if ((replace_msg != NULL) && ((cvs_sendarg(root, "-m", 0) < 0)
                    267:                    || (cvs_sendarg(root, replace_msg, 0) < 0)))
1.5     ! joris     268:                        return (EX_PROTOCOL);
        !           269:
        !           270:                if ((ntag != NULL) && ((cvs_sendarg(root, "-n", 0) < 0) ||
        !           271:                    (cvs_sendarg(root, ntag, 0) < 0)))
        !           272:                        return (EX_PROTOCOL);
        !           273:
        !           274:                if ((Ntag != NULL) && ((cvs_sendarg(root, "-N", 0) < 0) ||
        !           275:                    (cvs_sendarg(root, Ntag, 0) < 0)))
        !           276:                        return (EX_PROTOCOL);
        !           277:
        !           278:                if ((range != NULL) && ((cvs_sendarg(root, "-o", 0) < 0) ||
        !           279:                    (cvs_sendarg(root, range, 0) < 0)))
        !           280:                        return (EX_PROTOCOL);
        !           281:
        !           282:                if ((state != NULL) && ((cvs_sendarg(root, "-s", 0) < 0) ||
        !           283:                    (cvs_sendarg(root, state, 0) < 0)))
1.1       joris     284:                        return (EX_PROTOCOL);
                    285:
                    286:                if (lockrev & LOCK_REMOVE) {
                    287:                        if (cvs_sendarg(root, "-u", 0) < 0)
                    288:                                return (EX_PROTOCOL);
                    289:                        if ((unlockrev_arg != NULL) &&
                    290:                            (cvs_sendarg(root, unlockrev_arg, 0) < 0))
                    291:                                return (EX_PROTOCOL);
                    292:                }
                    293:
                    294:                if ((strictlock & LOCK_REMOVE) &&
                    295:                    (cvs_sendarg(root, "-U", 0) < 0))
                    296:                        return (EX_PROTOCOL);
                    297:        }
                    298:
                    299:        cvs_file_examine(cvs_files, cvs_admin_file, NULL);
                    300:
                    301:        if (root->cr_method != CVS_METHOD_LOCAL) {
                    302:                if (cvs_senddir(root, cvs_files) < 0)
                    303:                        return (EX_PROTOCOL);
                    304:                for (i = 0; i < argc; i++)
                    305:                        if (cvs_sendarg(root, argv[i], 0) < 0)
                    306:                                return (EX_PROTOCOL);
                    307:                if (cvs_sendreq(root, CVS_REQ_ADMIN, NULL) < 0)
                    308:                        return (EX_PROTOCOL);
                    309:        }
                    310:
                    311:        return (0);
                    312: }
                    313:
                    314:
                    315: /*
                    316:  * cvs_admin_file()
                    317:  *
                    318:  * Perform admin commands on each file.
                    319:  */
                    320: int
                    321: cvs_admin_file(CVSFILE *cfp, void *arg)
                    322: {
                    323:        int ret;
                    324:        char *repo, fpath[MAXPATHLEN], rcspath[MAXPATHLEN];
                    325:        RCSFILE *rf;
                    326:        struct cvs_ent *entp;
                    327:        struct cvsroot *root;
                    328:
                    329:        ret = 0;
                    330:        rf = NULL;
                    331:        root = CVS_DIR_ROOT(cfp);
                    332:        repo = CVS_DIR_REPO(cfp);
                    333:
                    334:        if (cfp->cf_type == DT_DIR) {
                    335:                if (root->cr_method != CVS_METHOD_LOCAL) {
                    336:                        if (cfp->cf_cvstat == CVS_FST_UNKNOWN)
                    337:                                ret = cvs_sendreq(root, CVS_REQ_QUESTIONABLE,
                    338:                                    CVS_FILE_NAME(cfp));
                    339:                        else
                    340:                                ret = cvs_senddir(root, cfp);
                    341:                }
                    342:
                    343:                return (ret);
                    344:        }
                    345:
                    346:        cvs_file_getpath(cfp, fpath, sizeof(fpath));
                    347:        entp = cvs_ent_getent(fpath);
                    348:
                    349:        if (root->cr_method != CVS_METHOD_LOCAL) {
                    350:                if ((entp != NULL) && (cvs_sendentry(root, entp) < 0)) {
                    351:                        cvs_ent_free(entp);
                    352:                        return (-1);
                    353:                }
                    354:
                    355:                switch (cfp->cf_cvstat) {
                    356:                case CVS_FST_UNKNOWN:
                    357:                        ret = cvs_sendreq(root, CVS_REQ_QUESTIONABLE,
                    358:                            CVS_FILE_NAME(cfp));
                    359:                        break;
                    360:                case CVS_FST_UPTODATE:
                    361:                        ret = cvs_sendreq(root, CVS_REQ_UNCHANGED,
                    362:                            CVS_FILE_NAME(cfp));
                    363:                        break;
                    364:                case CVS_FST_MODIFIED:
                    365:                        ret = cvs_sendreq(root, CVS_REQ_MODIFIED,
                    366:                            CVS_FILE_NAME(cfp));
                    367:                        if (ret == 0)
                    368:                                ret = cvs_sendfile(root, fpath);
                    369:                default:
                    370:                        break;
                    371:                }
                    372:        } else {
                    373:                if (cfp->cf_cvstat == CVS_FST_UNKNOWN) {
                    374:                        cvs_log(LP_WARN, "I know nothing about %s", fpath);
                    375:                        return (0);
                    376:                }
                    377:
                    378:                snprintf(rcspath, sizeof(rcspath), "%s/%s/%s%s",
                    379:                    root->cr_dir, repo, CVS_FILE_NAME(cfp), RCS_FILE_EXT);
                    380:
                    381:                rf = rcs_open(rcspath, RCS_READ);
                    382:                if (rf == NULL) {
                    383:                        cvs_ent_free(entp);
                    384:                        return (-1);
                    385:                }
                    386:
                    387:                rcs_close(rf);
                    388:        }
                    389:
                    390:        if (entp != NULL)
                    391:                cvs_ent_free(entp);
                    392:        return (ret);
                    393: }