/* $OpenBSD: edit.c,v 1.30 2007/01/25 11:09:21 xsa Exp $ */ /* * Copyright (c) 2006, 2007 Xavier Santolaria * * 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. * * 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 "log.h" #include "remote.h" #define E_COMMIT 0x01 #define E_EDIT 0x02 #define E_UNEDIT 0x04 #define E_ALL (E_EDIT|E_COMMIT|E_UNEDIT) #define BASE_ADD 0x01 #define BASE_GET 0x02 #define BASE_REMOVE 0x04 static void cvs_edit_local(struct cvs_file *); static void cvs_editors_local(struct cvs_file *); static void cvs_unedit_local(struct cvs_file *); static RCSNUM *cvs_base_handle(struct cvs_file *, int); static int edit_aflags = 0; struct cvs_cmd cvs_cmd_edit = { CVS_OP_EDIT, 0, "edit", { }, "Get ready to edit a watched file", "[-lR] [-a action] [file ...]", "a:lR", NULL, cvs_edit }; struct cvs_cmd cvs_cmd_editors = { CVS_OP_EDITORS, 0, "editors", { }, "See who is editing a watched file", "[-lR] [file ...]", "lR", NULL, cvs_editors }; struct cvs_cmd cvs_cmd_unedit = { CVS_OP_UNEDIT, 0, "unedit", { }, "Undo an edit command", "[-lR] [file ...]", "lR", NULL, cvs_unedit }; int cvs_edit(int argc, char **argv) { int ch; int flags; struct cvs_recursion cr; flags = CR_RECURSE_DIRS; while ((ch = getopt(argc, argv, cvs_cmd_edit.cmd_opts)) != -1) { switch (ch) { case 'a': if (strcmp(optarg, "edit") == 0) edit_aflags |= E_EDIT; else if (strcmp(optarg, "unedit") == 0) edit_aflags |= E_UNEDIT; else if (strcmp(optarg, "commit") == 0) edit_aflags |= E_COMMIT; else if (strcmp(optarg, "all") == 0) edit_aflags |= E_ALL; else if (strcmp(optarg, "none") == 0) edit_aflags &= ~E_ALL; else fatal("%s", cvs_cmd_edit.cmd_synopsis); break; case 'l': flags &= ~CR_RECURSE_DIRS; break; case 'R': break; default: fatal("%s", cvs_cmd_edit.cmd_synopsis); } } argc -= optind; argv += optind; if (argc == 0) fatal("%s", cvs_cmd_edit.cmd_synopsis); if (edit_aflags == 0) edit_aflags |= E_ALL; cr.enterdir = NULL; cr.leavedir = NULL; if (current_cvsroot->cr_method != CVS_METHOD_LOCAL) { cvs_client_connect_to_server(); cr.fileproc = cvs_client_sendfile; if (!(flags & CR_RECURSE_DIRS)) cvs_client_send_request("Argument -l"); } else { cr.fileproc = cvs_edit_local; } cr.flags = flags; cvs_file_run(argc, argv, &cr); if (current_cvsroot->cr_method != CVS_METHOD_LOCAL) { cvs_client_send_files(argv, argc); cvs_client_senddir("."); cvs_client_send_request("edit"); cvs_client_get_responses(); } return (0); } int cvs_editors(int argc, char **argv) { int ch; int flags; struct cvs_recursion cr; flags = CR_RECURSE_DIRS; while ((ch = getopt(argc, argv, cvs_cmd_editors.cmd_opts)) != -1) { switch (ch) { case 'l': flags &= ~CR_RECURSE_DIRS; break; case 'R': break; default: fatal("%s", cvs_cmd_editors.cmd_synopsis); } } argc -= optind; argv += optind; if (argc == 0) fatal("%s", cvs_cmd_editors.cmd_synopsis); cr.enterdir = NULL; cr.leavedir = NULL; if (current_cvsroot->cr_method != CVS_METHOD_LOCAL) { cvs_client_connect_to_server(); cr.fileproc = cvs_client_sendfile; if (!(flags & CR_RECURSE_DIRS)) cvs_client_send_request("Argument -l"); } else { cr.fileproc = cvs_editors_local; } cr.flags = flags; cvs_file_run(argc, argv, &cr); if (current_cvsroot->cr_method != CVS_METHOD_LOCAL) { cvs_client_send_files(argv, argc); cvs_client_senddir("."); cvs_client_send_request("editors"); cvs_client_get_responses(); } return (0); } int cvs_unedit(int argc, char **argv) { int ch; int flags; struct cvs_recursion cr; flags = CR_RECURSE_DIRS; while ((ch = getopt(argc, argv, cvs_cmd_unedit.cmd_opts)) != -1) { switch (ch) { case 'l': flags &= ~CR_RECURSE_DIRS; break; case 'R': break; default: fatal("%s", cvs_cmd_unedit.cmd_synopsis); } } argc -= optind; argv += optind; if (argc == 0) fatal("%s", cvs_cmd_unedit.cmd_synopsis); cr.enterdir = NULL; cr.leavedir = NULL; if (current_cvsroot->cr_method != CVS_METHOD_LOCAL) { cvs_client_connect_to_server(); cr.fileproc = cvs_client_sendfile; if (!(flags & CR_RECURSE_DIRS)) cvs_client_send_request("Argument -l"); } else { cr.fileproc = cvs_unedit_local; } cr.flags = flags; cvs_file_run(argc, argv, &cr); if (current_cvsroot->cr_method != CVS_METHOD_LOCAL) { cvs_client_send_files(argv, argc); cvs_client_senddir("."); cvs_client_send_request("unedit"); cvs_client_get_responses(); } return (0); } static void cvs_edit_local(struct cvs_file *cf) { FILE *fp; struct tm *t; time_t now; char *bfpath, timebuf[64], thishost[MAXHOSTNAMELEN], wdir[MAXPATHLEN]; if (cvs_noexec == 1) return; cvs_log(LP_TRACE, "cvs_edit_local(%s)", cf->file_path); cvs_file_classify(cf, NULL, 0); if ((fp = fopen(CVS_PATH_NOTIFY, "a")) == NULL) fatal("cvs_edit_local: fopen: `%s': %s", CVS_PATH_NOTIFY, strerror(errno)); (void)time(&now); if ((t = gmtime(&now)) == NULL) fatal("gmtime failed"); asctime_r(t, timebuf); if (timebuf[strlen(timebuf) - 1] == '\n') timebuf[strlen(timebuf) - 1] = '\0'; if (gethostname(thishost, sizeof(thishost)) == -1) fatal("gethostname failed"); if (getcwd(wdir, sizeof(wdir)) == NULL) fatal("getcwd failed"); (void)fprintf(fp, "E%s\t%s GMT\t%s\t%s\t", cf->file_name, timebuf, thishost, wdir); if (edit_aflags & E_EDIT) (void)fprintf(fp, "E"); if (edit_aflags & E_UNEDIT) (void)fprintf(fp, "U"); if (edit_aflags & E_COMMIT) (void)fprintf(fp, "C"); (void)fprintf(fp, "\n"); (void)fclose(fp); if (fchmod(cf->fd, 0644) == -1) fatal("cvs_edit_local: fchmod %s", strerror(errno)); bfpath = xmalloc(MAXPATHLEN); if (cvs_path_cat(CVS_PATH_BASEDIR, cf->file_name, bfpath, MAXPATHLEN) >= MAXPATHLEN) fatal("cvs_edit_local: truncation"); if (mkdir(CVS_PATH_BASEDIR, 0755) == -1 && errno != EEXIST) fatal("cvs_edit_local: `%s': %s", CVS_PATH_BASEDIR, strerror(errno)); if (cvs_file_copy(cf->file_path, bfpath) == -1) fatal("cvs_edit_local: cvs_file_copy failed"); xfree(bfpath); (void)cvs_base_handle(cf, BASE_ADD); } static void cvs_editors_local(struct cvs_file *cf) { } static void cvs_unedit_local(struct cvs_file *cf) { FILE *fp; struct stat st; struct tm *t; time_t now; char *bfpath, timebuf[64], thishost[MAXHOSTNAMELEN], wdir[MAXPATHLEN]; RCSNUM *ba_rev; if (cvs_noexec == 1) return; cvs_log(LP_TRACE, "cvs_unedit_local(%s)", cf->file_path); cvs_file_classify(cf, NULL, 0); bfpath = xmalloc(MAXPATHLEN); if (cvs_path_cat(CVS_PATH_BASEDIR, cf->file_name, bfpath, MAXPATHLEN) >= MAXPATHLEN) fatal("cvs_unedit_local: truncation"); if (stat(bfpath, &st) == -1) { xfree(bfpath); return; } if (cvs_file_cmp(cf->file_path, bfpath) != 0) { cvs_printf("%s has been modified; revert changes? ", cf->file_name); if (cvs_yesno() == -1) { xfree(bfpath); return; } } cvs_rename(bfpath, cf->file_path); xfree(bfpath); if ((fp = fopen(CVS_PATH_NOTIFY, "a")) == NULL) fatal("cvs_unedit_local: fopen: `%s': %s", CVS_PATH_NOTIFY, strerror(errno)); (void)time(&now); if ((t = gmtime(&now)) == NULL) fatal("gmtime failed"); asctime_r(t, timebuf); if (timebuf[strlen(timebuf) - 1] == '\n') timebuf[strlen(timebuf) - 1] = '\0'; if (gethostname(thishost, sizeof(thishost)) == -1) fatal("gethostname failed"); if (getcwd(wdir, sizeof(wdir)) == NULL) fatal("getcwd failed"); (void)fprintf(fp, "U%s\t%s GMT\t%s\t%s\t\n", cf->file_name, timebuf, thishost, wdir); (void)fclose(fp); if ((ba_rev = cvs_base_handle(cf, BASE_GET)) == NULL) { cvs_log(LP_ERR, "%s not mentioned in %s", cf->file_name, CVS_PATH_BASEREV); return; } if (cf->file_ent != NULL) { CVSENTRIES *entlist; struct cvs_ent *ent; char *entry, rbuf[16]; entlist = cvs_ent_open(cf->file_wd); if ((ent = cvs_ent_get(entlist, cf->file_name)) == NULL) fatal("cvs_unedit_local: cvs_ent_get failed"); (void)rcsnum_tostr(ba_rev, rbuf, sizeof(rbuf)); memset(timebuf, 0, sizeof(timebuf)); ctime_r(&cf->file_ent->ce_mtime, timebuf); if (timebuf[strlen(timebuf) - 1] == '\n') timebuf[strlen(timebuf) - 1] = '\0'; (void)xasprintf(&entry, "/%s/%s/%s/%s/%s", cf->file_name, rbuf, timebuf, (cf->file_ent->ce_tag) ? cf->file_ent->ce_tag : "", (cf->file_ent->ce_opts) ? cf->file_ent->ce_opts : ""); cvs_ent_add(entlist, entry); cvs_ent_free(ent); cvs_ent_close(entlist, ENT_SYNC); xfree(entry); } rcsnum_free(ba_rev); (void)cvs_base_handle(cf, BASE_REMOVE); if (fchmod(cf->fd, 0644) == -1) fatal("cvs_unedit_local: fchmod %s", strerror(errno)); } static RCSNUM * cvs_base_handle(struct cvs_file *cf, int flags) { FILE *fp, *tfp; RCSNUM *ba_rev; size_t len; int i; char *dp, *sp; char buf[MAXPATHLEN], *fields[2], rbuf[16]; cvs_log(LP_TRACE, "cvs_base_handle(%s)", cf->file_path); tfp = NULL; ba_rev = NULL; if (((fp = fopen(CVS_PATH_BASEREV, "r")) == NULL) && errno != ENOENT) { cvs_log(LP_ERRNO, "%s", CVS_PATH_BASEREV); goto out; } if (flags & (BASE_ADD|BASE_REMOVE)) { if ((tfp = fopen(CVS_PATH_BASEREVTMP, "w")) == NULL) { cvs_log(LP_ERRNO, "%s", CVS_PATH_BASEREVTMP); goto out; } } if (fp != NULL) { while(fgets(buf, sizeof(buf), fp)) { len = strlen(buf); if (len > 0 && buf[len - 1] == '\n') buf[len - 1] = '\0'; if (buf[0] != 'B') continue; sp = buf + 1; i = 0; do { if ((dp = strchr(sp, '/')) != NULL) *(dp++) = '\0'; fields[i++] = sp; sp = dp; } while (dp != NULL && i < 2); if (cvs_file_cmpname(fields[0], cf->file_path) == 0) { if (flags & BASE_GET) { ba_rev = rcsnum_parse(fields[1]); if (ba_rev == NULL) fatal("cvs_base_handle: " "rcsnum_parse"); goto got_rev; } } else { if (flags & (BASE_ADD|BASE_REMOVE)) (void)fprintf(tfp, "%s\n", buf); } } } got_rev: if (flags & (BASE_ADD)) { (void)rcsnum_tostr(cf->file_ent->ce_rev, rbuf, sizeof(rbuf)); (void)fprintf(tfp, "B%s/%s/\n", cf->file_path, rbuf); } out: if (fp != NULL) (void)fclose(fp); if (tfp != NULL) { (void)fclose(tfp); (void)cvs_rename(CVS_PATH_BASEREVTMP, CVS_PATH_BASEREV); } return (ba_rev); }