diff options
author | Joris Vink <joris@cvs.openbsd.org> | 2006-05-28 21:11:13 +0000 |
---|---|---|
committer | Joris Vink <joris@cvs.openbsd.org> | 2006-05-28 21:11:13 +0000 |
commit | d83f21246935bfa2da8280ce9cc845c79eb78a01 (patch) | |
tree | fd19554cb1b77de631c594c87fa8f218f18d4c9e /usr.bin/cvs/getlog.c | |
parent | 99661afd38faa75dbd974bd3883d6e6f31ff5fcc (diff) |
add basic 'cvs log' support.
Diffstat (limited to 'usr.bin/cvs/getlog.c')
-rw-r--r-- | usr.bin/cvs/getlog.c | 331 |
1 files changed, 104 insertions, 227 deletions
diff --git a/usr.bin/cvs/getlog.c b/usr.bin/cvs/getlog.c index 8f81cd29338..2c52bd69a0e 100644 --- a/usr.bin/cvs/getlog.c +++ b/usr.bin/cvs/getlog.c @@ -1,51 +1,37 @@ -/* $OpenBSD: getlog.c,v 1.56 2006/04/14 22:33:15 niallo Exp $ */ +/* $OpenBSD: getlog.c,v 1.57 2006/05/28 21:11:12 joris Exp $ */ /* - * Copyright (c) 2004 Jean-Francois Brousseau <jfb@openbsd.org> - * Copyright (c) 2005, 2006 Xavier Santolaria <xsa@openbsd.org> - * All rights reserved. + * Copyright (c) 2006 Joris Vink <joris@openbsd.org> * - * Redistribution and use in source and binary forms, with or without - * modification, are permitted provided that the following conditions - * are met: + * Permission to use, copy, modify, and distribute this software for any + * purpose with or without fee is hereby granted, provided that the above + * copyright notice and this permission notice appear in all copies. * - * 1. Redistributions of source code must retain the above copyright - * notice, this list of conditions and the following disclaimer. - * 2. The name of the author may not be used to endorse or promote products - * derived from this software without specific prior written permission. - * - * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, - * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY - * AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL - * THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, - * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, - * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; - * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, - * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR - * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF - * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES + * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF + * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR + * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES + * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN + * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF + * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. */ #include "includes.h" #include "cvs.h" +#include "diff.h" #include "log.h" #include "proto.h" +#define LOG_REVSEP \ +"----------------------------" -#define CVS_GETLOG_REVSEP "----------------------------" -#define CVS_GETLOG_REVEND \ +#define LOG_REVEND \ "=============================================================================" -#define GLOG_BRANCH (1<<0) -#define GLOG_HEADER (1<<1) -#define GLOG_HEADER_DESCR (1<<2) -#define GLOG_NAME (1<<3) -#define GLOG_NOTAGS (1<<4) +int cvs_getlog(int, char **); +void cvs_log_local(struct cvs_file *); -static int cvs_getlog_init(struct cvs_cmd *, int, char **, int *); -static int cvs_getlog_remote(CVSFILE *, void *); -static int cvs_getlog_local(CVSFILE *, void *); -static int cvs_getlog_pre_exec(struct cvsroot *); +char *logrev = NULL; struct cvs_cmd cvs_cmd_log = { CVS_OP_LOG, CVS_REQ_LOG, "log", @@ -54,243 +40,134 @@ struct cvs_cmd cvs_cmd_log = { "[-bhlNRt] [-d dates] [-r revisions] [-s states] [-w logins]", "bd:hlNRr:s:tw:", NULL, - CF_NOSYMS | CF_IGNORE | CF_SORT | CF_RECURSE, - cvs_getlog_init, - cvs_getlog_pre_exec, - cvs_getlog_remote, - cvs_getlog_local, - NULL, - NULL, - CVS_CMD_SENDDIR | CVS_CMD_ALLOWSPEC | CVS_CMD_SENDARGS2 + cvs_getlog }; - -struct cvs_cmd cvs_cmd_rlog = { - CVS_OP_LOG, CVS_REQ_LOG, "log", - { "lo" }, - "Print out history information for files", - "[-bhlNRt] [-d dates] [-r revisions] [-s states] [-w logins]", - "d:hlRr:", - NULL, - CF_NOSYMS | CF_IGNORE | CF_SORT | CF_RECURSE, - cvs_getlog_init, - cvs_getlog_pre_exec, - cvs_getlog_remote, - cvs_getlog_local, - NULL, - NULL, - CVS_CMD_SENDDIR | CVS_CMD_ALLOWSPEC | CVS_CMD_SENDARGS2 -}; - -static int runflags = 0; - -static int -cvs_getlog_init(struct cvs_cmd *cmd, int argc, char **argv, int *arg) +int +cvs_getlog(int argc, char **argv) { int ch; + int flags; + char *arg = "."; + struct cvs_recursion cr; - while ((ch = getopt(argc, argv, cmd->cmd_opts)) != -1) { + rcsnum_flags |= RCSNUM_NO_MAGIC; + flags = CR_RECURSE_DIRS; + + while ((ch = getopt(argc, argv, cvs_cmd_log.cmd_opts)) != -1) { switch (ch) { - case 'b': - runflags |= GLOG_BRANCH; - break; - case 'd': - break; - case 'h': - runflags |= GLOG_HEADER; - break; case 'l': - cmd->file_flags &= ~CF_RECURSE; - break; - case 'N': - runflags |= GLOG_NOTAGS; - break; - case 'R': - runflags |= GLOG_NAME; + flags &= ~CR_RECURSE_DIRS; break; case 'r': - break; - case 's': - break; - case 't': - runflags |= GLOG_HEADER_DESCR; - break; - case 'w': + logrev = optarg; break; default: - return (CVS_EX_USAGE); + fatal("%s", cvs_cmd_log.cmd_synopsis); } } - *arg = optind; - rcsnum_flags |= RCSNUM_NO_MAGIC; - return (0); -} - -static int -cvs_getlog_pre_exec(struct cvsroot *root) -{ - if (root->cr_method != CVS_METHOD_LOCAL) { - if (runflags & GLOG_BRANCH) - cvs_sendarg(root, "-b", 0); + argc -= optind; + argv += optind; - if (runflags & GLOG_HEADER) - cvs_sendarg(root, "-h", 0); + cr.enterdir = NULL; + cr.leavedir = NULL; + cr.remote = NULL; + cr.local = cvs_log_local; + cr.flags = flags; - if (runflags & GLOG_NOTAGS) - cvs_sendarg(root, "-N", 0); - - if (runflags & GLOG_NAME) - cvs_sendarg(root, "-R", 0); - - if (runflags & GLOG_HEADER_DESCR) - cvs_sendarg(root, "-t", 0); - } - return (0); -} - -/* - * cvs_getlog_remote() - * - */ -static int -cvs_getlog_remote(CVSFILE *cf, void *arg) -{ - char fpath[MAXPATHLEN]; - struct cvsroot *root; - - root = CVS_DIR_ROOT(cf); - - if (cf->cf_type == DT_DIR) { - if (cf->cf_cvstat == CVS_FST_UNKNOWN) - cvs_sendreq(root, CVS_REQ_QUESTIONABLE, cf->cf_name); - else - cvs_senddir(root, cf); - return (0); - } - - cvs_file_getpath(cf, fpath, sizeof(fpath)); - - cvs_sendentry(root, cf); - - switch (cf->cf_cvstat) { - case CVS_FST_UNKNOWN: - cvs_sendreq(root, CVS_REQ_QUESTIONABLE, cf->cf_name); - break; - case CVS_FST_UPTODATE: - cvs_sendreq(root, CVS_REQ_UNCHANGED, cf->cf_name); - break; - case CVS_FST_ADDED: - case CVS_FST_MODIFIED: - cvs_sendreq(root, CVS_REQ_ISMODIFIED, cf->cf_name); - break; - default: - break; - } + if (argc > 0) + cvs_file_run(argc, argv, &cr); + else + cvs_file_run(1, &arg, &cr); return (0); } - - -static int -cvs_getlog_local(CVSFILE *cf, void *arg) +void +cvs_log_local(struct cvs_file *cf) { - int nrev; - char rcspath[MAXPATHLEN], numbuf[64]; - RCSFILE *rf; + u_int nrev; struct rcs_sym *sym; + struct rcs_lock *lkp; struct rcs_delta *rdp; struct rcs_access *acp; + char numb[32], timeb[32]; - nrev = 0; + cvs_file_classify(cf, 0); - if (cf->cf_cvstat == CVS_FST_ADDED) { + if (cf->file_status == FILE_UNKNOWN) { if (verbosity > 0) - cvs_log(LP_WARN, "%s has been added, but not committed", - cf->cf_name); - return (0); - } - - if (cf->cf_cvstat == CVS_FST_UNKNOWN) { + cvs_log(LP_ERR, "nothing known about %s", + cf->file_path); + return; + } else if (cf->file_status == FILE_ADDED) { if (verbosity > 0) - cvs_log(LP_WARN, "nothing known about %s", cf->cf_name); - return (0); + cvs_log(LP_ERR, "%s has been added, but not commited", + cf->file_path); + return; } - if (cf->cf_type == DT_DIR) { - if (verbosity > 1) - cvs_log(LP_NOTICE, "Logging %s", cf->cf_name); - return (0); + printf("\nRCS file: %s", cf->file_rpath); + printf("\nWorking file: %s", cf->file_path); + printf("\nhead:"); + if (cf->file_rcs->rf_head != NULL) + printf(" %s", rcsnum_tostr(cf->file_rcs->rf_head, + numb, sizeof(numb))); + + printf("\nbranch:"); + if (rcs_branch_get(cf->file_rcs) != NULL) { + printf(" %s", rcsnum_tostr(rcs_branch_get(cf->file_rcs), + numb, sizeof(numb))); } - cvs_rcs_getpath(cf, rcspath, sizeof(rcspath)); - - if (runflags & GLOG_NAME) { - cvs_printf("%s\n", rcspath); - return (0); - } + printf("\nlocks: %s", (cf->file_rcs->rf_flags & RCS_SLOCK) + ? "strict" : ""); + TAILQ_FOREACH(lkp, &(cf->file_rcs->rf_locks), rl_list) + printf("\n\t%s: %s", lkp->rl_name, + rcsnum_tostr(lkp->rl_num, numb, sizeof(numb))); - if ((rf = rcs_open(rcspath, RCS_READ|RCS_PARSE_FULLY)) == NULL) - fatal("cvs_getlog_local: rcs_open `%s': %s", rcspath, - rcs_errstr(rcs_errno)); + printf("\naccess list:\n"); + TAILQ_FOREACH(acp, &(cf->file_rcs->rf_access), ra_list) + printf("\t%s\n", acp->ra_name); - cvs_printf("\nRCS file: %s", rcspath); - cvs_printf("\nWorking file: %s", cf->cf_name); - cvs_printf("\nhead:"); - if (rcs_head_get(rf) != NULL) { - cvs_printf(" %s", - rcsnum_tostr(rcs_head_get(rf), numbuf, sizeof(numbuf))); + printf("symbolic names:\n"); + TAILQ_FOREACH(sym, &(cf->file_rcs->rf_symbols), rs_list) { + printf("\t%s: %s\n", sym->rs_name, + rcsnum_tostr(sym->rs_num, numb, sizeof(numb))); } - cvs_printf("\nbranch:"); - if (rcs_branch_get(rf) != NULL) { - cvs_printf(" %s", - rcsnum_tostr(rcs_branch_get(rf), numbuf, sizeof(numbuf))); - } - cvs_printf("\nlocks:%s", (rf->rf_flags & RCS_SLOCK) ? " strict" : ""); - cvs_printf("\naccess list:\n"); - TAILQ_FOREACH(acp, &(rf->rf_access), ra_list) - cvs_printf("\t%s\n", acp->ra_name); + printf("keyword substitution: %s\n", + cf->file_rcs->rf_expand == NULL ? "kv" : cf->file_rcs->rf_expand); - if (!(runflags & GLOG_NOTAGS)) { - cvs_printf("symbolic names:\n"); - TAILQ_FOREACH(sym, &(rf->rf_symbols), rs_list) - cvs_printf("\t%s: %s\n", sym->rs_name, - rcsnum_tostr(sym->rs_num, numbuf, sizeof(numbuf))); - } + printf("total revisions: %u", cf->file_rcs->rf_ndelta); - cvs_printf("keyword substitution: %s\n", - rf->rf_expand == NULL ? "kv" : rf->rf_expand); + if (logrev != NULL) + nrev = 1; + else + nrev = cf->file_rcs->rf_ndelta; - cvs_printf("total revisions: %u", rf->rf_ndelta); + printf(";\tselected revisions: %u", nrev); + printf("\n"); + printf("description:\n%s", cf->file_rcs->rf_desc); - if (!(runflags & GLOG_HEADER) && !(runflags & GLOG_HEADER_DESCR)) - cvs_printf(";\tselected revisions: %u", nrev); + TAILQ_FOREACH(rdp, &(cf->file_rcs->rf_delta), rd_list) { + rcsnum_tostr(rdp->rd_num, numb, sizeof(numb)); - cvs_printf("\n"); + if (logrev != NULL && + strcmp(logrev, numb)) + continue; - if (!(runflags & GLOG_HEADER) || (runflags & GLOG_HEADER_DESCR)) - cvs_printf("description:\n%s", rf->rf_desc); + printf("%s\n", LOG_REVSEP); - if (!(runflags & GLOG_HEADER) && !(runflags & GLOG_HEADER_DESCR)) { - TAILQ_FOREACH(rdp, &(rf->rf_delta), rd_list) { - rcsnum_tostr(rdp->rd_num, numbuf, sizeof(numbuf)); - cvs_printf(CVS_GETLOG_REVSEP "\nrevision %s\n", numbuf); - cvs_printf("date: %d/%02d/%02d %02d:%02d:%02d;" - " author: %s; state: %s;\n", - rdp->rd_date.tm_year + 1900, - rdp->rd_date.tm_mon + 1, - rdp->rd_date.tm_mday, rdp->rd_date.tm_hour, - rdp->rd_date.tm_min, rdp->rd_date.tm_sec, - rdp->rd_author, rdp->rd_state); - cvs_printf("%s", rdp->rd_log); - } - } + printf("revision %s", numb); - cvs_printf(CVS_GETLOG_REVEND "\n"); - - rcs_close(rf); + strftime(timeb, sizeof(timeb), "%Y/%m/%d %H:%M:%S", + &rdp->rd_date); + printf("\ndate: %s; author: %s; state: %s;\n", timeb, + rdp->rd_author, rdp->rd_state); + printf("%s", rdp->rd_log); + } - return (0); + printf("%s\n", LOG_REVEND); } |