File: [local] / src / usr.bin / cvs / release.c (download)
Revision 1.18, Mon Jul 25 12:13:08 2005 UTC (18 years, 10 months ago) by xsa
Branch: MAIN
Changes since 1.17: +5 -5 lines
KNF;
|
/* $OpenBSD: release.c,v 1.18 2005/07/25 12:13:08 xsa Exp $ */
/*
* Copyright (c) 2005 Xavier Santolaria <xsa@openbsd.org>
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
*
* 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.
*/
#include <sys/types.h>
#include <sys/stat.h>
#include <errno.h>
#include <fcntl.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include "cvs.h"
#include "log.h"
#include "proto.h"
#define UPDCMD_FLAGS "-n -q -d"
extern char *__progname;
static int cvs_release_init(struct cvs_cmd *, int, char **, int *);
static int cvs_release_pre_exec(struct cvsroot *);
static int cvs_release_yesno(void);
static int cvs_release_dir(CVSFILE *, void *);
struct cvs_cmd cvs_cmd_release = {
CVS_OP_RELEASE, CVS_REQ_RELEASE, "release",
{ },
"Release",
"[-d]",
"d",
NULL,
0,
cvs_release_init,
cvs_release_pre_exec,
cvs_release_dir,
cvs_release_dir,
NULL,
NULL,
CVS_CMD_SENDDIR | CVS_CMD_SENDARGS2 | CVS_CMD_ALLOWSPEC
};
static int dflag; /* -d option */
static int
cvs_release_init(struct cvs_cmd *cmd, int argc, char **argv, int *arg)
{
int ch;
while ((ch = getopt(argc, argv, cmd->cmd_opts)) != -1) {
switch (ch) {
case 'd':
dflag = 1;
break;
default:
return (CVS_EX_USAGE);
}
}
argc -= optind;
argv += optind;
*arg = optind;
if (argc == 0)
return (CVS_EX_USAGE);
return (0);
}
static int
cvs_release_pre_exec(struct cvsroot *root)
{
if (root->cr_method != CVS_METHOD_LOCAL) {
if (dflag && cvs_sendarg(root, "-d", 0) < 0)
return (CVS_EX_PROTO);
}
return (0);
}
/*
* cvs_release_yesno()
*
* Read from standart input for `y' or `Y' character.
* Returns 0 on success, or -1 on failure.
*/
static int
cvs_release_yesno(void)
{
int c, ret;
ret = 0;
fflush (stderr);
fflush (stdout);
if ((c = getchar()) != 'y' && c != 'Y')
ret = -1;
else
while (c != EOF && c != '\n')
c = getchar();
return (ret);
}
/*
* cvs_release_dir()
*
* Release specified directorie(s).
* Returns 0 on success, or -1 on failure.
*/
static int
cvs_release_dir(CVSFILE *cf, void *arg)
{
FILE *fp;
int j, l;
size_t len;
char *wdir, cwd[MAXPATHLEN];
char cdpath[MAXPATHLEN], dpath[MAXPATHLEN];
char buf[256], updcmd[1024];
struct stat st;
struct cvsroot *root;
j = 0; /* number of altered files in the working copy */
root = CVS_DIR_ROOT(cf);
cvs_file_getpath(cf, dpath, sizeof(dpath));
len = cvs_path_cat(dpath, CVS_PATH_CVSDIR, cdpath, sizeof(cdpath));
if (len >= sizeof(cdpath))
return (CVS_EX_DATA);
if (cf->cf_type == DT_DIR) {
if (!strcmp(cf->cf_name, "."))
return (0);
else {
/* test if dir has CVS/ directory */
if (stat(cdpath, &st) == -1) {
if (verbosity > 0)
cvs_log(LP_ERR,
"no repository directory: %s",
dpath);
return (0);
}
}
if (root->cr_method != CVS_METHOD_LOCAL) {
/* XXX kept for compat reason of `cvs update' output */
/* save current working directory for further use */
if ((wdir = getcwd(cwd, sizeof(cwd))) == NULL)
cvs_log(LP_ERRNO, "cannot get current dir");
/* change dir before running the `cvs update' command */
if (cvs_chdir(dpath) == -1)
return (CVS_EX_FILE);
/* construct `cvs update' command */
l = snprintf(updcmd, sizeof(updcmd), "%s %s %s update",
__progname, UPDCMD_FLAGS, root->cr_str);
if (l == -1 || l >= (int)sizeof(updcmd))
return (CVS_EX_DATA);
/* XXX we should try to avoid a new connection ... */
if ((fp = popen(updcmd, "r")) == NULL) {
cvs_log(LP_ERR, "cannot run command `%s'",
updcmd);
return (CVS_EX_DATA);
}
while (fgets(buf, sizeof(buf), fp) != NULL) {
if (strchr("ACMPRU", buf[0]))
j++;
(void)fputs(buf, stdout);
}
if (pclose(fp) != 0) {
cvs_log(LP_ERR, "unable to release `%s'",
dpath);
return (CVS_EX_DATA);
}
printf("You have [%d] altered file%s in this "
"repository.\n", j, j > 1 ? "s" : "");
printf("Are you sure you want to release "
"%sdirectory `%s': ",
dflag ? "(and delete) " : "", dpath);
if (cvs_release_yesno() == -1) { /* No */
(void)fprintf(stderr,
"** `%s' aborted by user choice.\n",
cvs_command);
return (-1);
}
/* change back to original working dir */
if (cvs_chdir(wdir) == -1)
return (CVS_EX_FILE);
if (dflag == 1) {
if (!cvs_noexec && cvs_remove_dir(dpath) != 0) {
cvs_log(LP_ERRNO,
"deletion of directory `%s' failed",
dpath);
return (CVS_EX_FILE);
}
}
}
} else {
if (verbosity > 0)
cvs_log(LP_ERR, "no such directory: %s", dpath);
return (CVS_EX_DATA);
}
return (0);
}