189 lines
3.6 KiB
C
189 lines
3.6 KiB
C
|
/* $NetBSD: edit.c,v 1.1 2007/02/18 22:13:42 rmind Exp $ */
|
||
|
/* $OpenBSD: edit.c,v 1.14 2006/05/25 03:20:32 ray Exp $ */
|
||
|
|
||
|
/*
|
||
|
* Written by Raymond Lai <ray@cyth.net>.
|
||
|
* Public domain.
|
||
|
*/
|
||
|
|
||
|
#include <sys/types.h>
|
||
|
#include <sys/wait.h>
|
||
|
|
||
|
#include <ctype.h>
|
||
|
#include <err.h>
|
||
|
#include <stdio.h>
|
||
|
#include <stdlib.h>
|
||
|
#include <string.h>
|
||
|
#include <unistd.h>
|
||
|
|
||
|
#include "common.h"
|
||
|
#include "extern.h"
|
||
|
|
||
|
static void edit(const char *);
|
||
|
|
||
|
/*
|
||
|
* Takes the name of a file and opens it with an editor.
|
||
|
*/
|
||
|
static void
|
||
|
edit(const char *filename)
|
||
|
{
|
||
|
int status;
|
||
|
pid_t pid;
|
||
|
const char *editor;
|
||
|
|
||
|
editor = getenv("VISUAL");
|
||
|
if (editor == NULL)
|
||
|
editor = getenv("EDITOR");
|
||
|
if (editor == NULL)
|
||
|
editor = "vi";
|
||
|
|
||
|
/* Start editor on temporary file. */
|
||
|
switch (pid = fork()) {
|
||
|
case 0:
|
||
|
/* child */
|
||
|
execlp(editor, editor, filename, (void *)NULL);
|
||
|
warn("could not execute editor: %s", editor);
|
||
|
cleanup(filename);
|
||
|
case -1:
|
||
|
warn("could not fork");
|
||
|
cleanup(filename);
|
||
|
}
|
||
|
|
||
|
/* parent */
|
||
|
/* Wait for editor to exit. */
|
||
|
if (waitpid(pid, &status, 0) == -1) {
|
||
|
warn("waitpid");
|
||
|
cleanup(filename);
|
||
|
}
|
||
|
|
||
|
/* Check that editor terminated normally. */
|
||
|
if (!WIFEXITED(status)) {
|
||
|
warn("%s terminated abnormally", editor);
|
||
|
cleanup(filename);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* Parse edit command. Returns 0 on success, -1 on error.
|
||
|
*/
|
||
|
int
|
||
|
eparse(const char *cmd, const char *left, const char *right)
|
||
|
{
|
||
|
FILE *file;
|
||
|
size_t nread, nwritten;
|
||
|
int fd;
|
||
|
char *filename;
|
||
|
char buf[BUFSIZ], *text;
|
||
|
|
||
|
/* Skip whitespace. */
|
||
|
while (isspace((int)(*cmd)))
|
||
|
++cmd;
|
||
|
|
||
|
text = NULL;
|
||
|
switch (*cmd) {
|
||
|
case '\0':
|
||
|
/* Edit empty file. */
|
||
|
break;
|
||
|
|
||
|
case 'b':
|
||
|
/* Both strings. */
|
||
|
if (left == NULL)
|
||
|
goto RIGHT;
|
||
|
if (right == NULL)
|
||
|
goto LEFT;
|
||
|
|
||
|
/* Neither column is blank, so print both. */
|
||
|
if (asprintf(&text, "%s\n%s\n", left, right) == -1)
|
||
|
err(2, "could not allocate memory");
|
||
|
break;
|
||
|
|
||
|
case 'l':
|
||
|
LEFT:
|
||
|
/* Skip if there is no left column. */
|
||
|
if (left == NULL)
|
||
|
break;
|
||
|
|
||
|
if (asprintf(&text, "%s\n", left) == -1)
|
||
|
err(2, "could not allocate memory");
|
||
|
|
||
|
break;
|
||
|
|
||
|
case 'r':
|
||
|
RIGHT:
|
||
|
/* Skip if there is no right column. */
|
||
|
if (right == NULL)
|
||
|
break;
|
||
|
|
||
|
if (asprintf(&text, "%s\n", right) == -1)
|
||
|
err(2, "could not allocate memory");
|
||
|
|
||
|
break;
|
||
|
|
||
|
default:
|
||
|
return (-1);
|
||
|
}
|
||
|
|
||
|
/* Create temp file. */
|
||
|
if (asprintf(&filename, "%s/sdiff.XXXXXXXXXX", tmpdir) == -1)
|
||
|
err(2, "asprintf");
|
||
|
if ((fd = mkstemp(filename)) == -1)
|
||
|
err(2, "mkstemp");
|
||
|
if (text != NULL) {
|
||
|
size_t len;
|
||
|
|
||
|
len = strlen(text);
|
||
|
if ((nwritten = write(fd, text, len)) == -1 ||
|
||
|
nwritten != len) {
|
||
|
warn("error writing to temp file");
|
||
|
cleanup(filename);
|
||
|
}
|
||
|
}
|
||
|
close(fd);
|
||
|
|
||
|
/* text is no longer used. */
|
||
|
free(text);
|
||
|
|
||
|
/* Edit temp file. */
|
||
|
edit(filename);
|
||
|
|
||
|
/* Open temporary file. */
|
||
|
if (!(file = fopen(filename, "r"))) {
|
||
|
warn("could not open edited file: %s", filename);
|
||
|
cleanup(filename);
|
||
|
}
|
||
|
|
||
|
/* Copy temporary file contents to output file. */
|
||
|
for (nread = sizeof(buf); nread == sizeof(buf);) {
|
||
|
nread = fread(buf, sizeof(*buf), sizeof(buf), file);
|
||
|
/* Test for error or end of file. */
|
||
|
if (nread != sizeof(buf) &&
|
||
|
(ferror(file) || !feof(file))) {
|
||
|
warnx("error reading edited file: %s", filename);
|
||
|
cleanup(filename);
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* If we have nothing to read, break out of loop
|
||
|
* instead of writing nothing.
|
||
|
*/
|
||
|
if (!nread)
|
||
|
break;
|
||
|
|
||
|
/* Write data we just read. */
|
||
|
nwritten = fwrite(buf, sizeof(*buf), nread, outfile);
|
||
|
if (nwritten != nread) {
|
||
|
warnx("error writing to output file");
|
||
|
cleanup(filename);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* We've reached the end of the temporary file, so remove it. */
|
||
|
if (unlink(filename))
|
||
|
warn("could not delete: %s", filename);
|
||
|
fclose(file);
|
||
|
|
||
|
free(filename);
|
||
|
|
||
|
return (0);
|
||
|
}
|