/* See LICENSE file for copyright and license details. */
#include "common.h"
#define TEMPFILEPATTERN "tmpXXXXXX"
const char *argv0 = "editasroot";
static const char *unlink_this = NULL;
static int ttyfd = -1;
static int tcattrs_fetched = 0;
static struct termios tcattrs;
static void
usage(void)
{
fprintf(stderr, "usage: %s file\n", argv0);
exit(1);
}
static void
cleanup(void)
{
if (unlink_this) {
unlink(unlink_this);
unlink_this = NULL;
}
if (ttyfd >= 0) {
if (tcattrs_fetched)
tcsetattr(ttyfd, TCSAFLUSH, &tcattrs);
close(ttyfd);
ttyfd = -1;
}
}
static void
graceful_exit(int signo)
{
cleanup();
if (signal(signo, SIG_DFL) != SIG_ERR)
raise(signo);
exit(1);
}
static const char *
get_editor(void)
{
const char *editor = NULL;
int isbg;
isbg = getpgrp() != tcgetpgrp(ttyfd);
if (isbg)
editor = getenv("VISUAL");
if (!editor || !*editor) {
editor = getenv("EDITOR");
if (!editor || !*editor) {
fprintf(stderr, "%s: warning: EDITOR not set, defaulting to 'vi'\n", argv0);
editor = "vi";
}
}
return editor;
}
static char *
get_tmpfile_pattern(void)
{
const char *tmpdir = NULL;
char *path, *p;
tmpdir = getenv("XDG_RUNTIME_DIR");
if (!tmpdir || !*tmpdir)
tmpdir = _PATH_TMP;
path = malloc(strlen(tmpdir) + sizeof("/"TEMPFILEPATTERN));
if (!path) {
fprintf(stderr, "%s: malloc: %s\n", argv0, strerror(ENOMEM));
exit(1);
}
p = stpcpy(path, tmpdir);
p -= p[-1] == '/';
stpcpy(p, "/"TEMPFILEPATTERN);
return path;
}
static pid_t
run_child(const char *file, int fd, int close_this)
{
char fdstr[3 * sizeof(fd) + 2];
pid_t pid;
switch (pid = fork()) {
case -1:
fprintf(stderr, "%s: fork: %s\n", argv0, strerror(errno));
exit(1);
case 0:
close(STDIN_FILENO);
close(STDOUT_FILENO);
close(ttyfd);
close(close_this);
prctl(PR_SET_PDEATHSIG, SIGKILL);
sprintf(fdstr, "%i", fd);
execlp("asroot", "asroot", LIBEXECDIR"/copier", file, fdstr, NULL);
fprintf(stderr, "%s: execlp asroot: %s\n", argv0, strerror(errno));
exit(1);
default:
return pid;
}
}
static void
run_editor(const char *editor, const char *file, int close_this)
{
pid_t pid;
int status;
switch (pid = fork()) {
case -1:
fprintf(stderr, "%s: fork: %s\n", argv0, strerror(errno));
exit(1);
case 0:
close(close_this);
execlp(editor, editor, "--", file, NULL);
fprintf(stderr, "%s: execlp %s: %s\n", argv0, editor, strerror(errno));
exit(1);
default:
break;
}
if (waitpid(pid, &status, 0) != pid) {
fprintf(stderr, "%s: waitpid %s 0: %s\n", argv0, editor, strerror(errno));
exit(1);
}
if (status)
exit(WIFEXITED(status) ? WEXITSTATUS(status) : 1);
}
int
main(int argc, char *argv[])
{
const char *editor;
char *path;
int fd, fds[2], status, i, ok;
pid_t pid;
if (!argc)
usage();
argv0 = *argv++;
if (--argc != 1)
usage();
if (atexit(cleanup)) {
fprintf(stderr, "%s: atexit: %s\n", argv0, strerror(errno));
exit(1);
}
/* Get terminal, needed for attributes and to chekh if VISUAL or EDITOR should be used */
ttyfd = open(_PATH_TTY, O_RDONLY);
if (ttyfd < 0) {
fprintf(stderr, "%s: open %s O_RDONLY: %s\n", argv0, _PATH_TTY, strerror(errno));
exit(1);
}
/* Get terminal attributes, needed in case of interrupt during asroot prompt (didn't find a better alternative) */
memset(&tcattrs, 0, sizeof(tcattrs));
if (tcgetattr(ttyfd, &tcattrs)) {
fprintf(stderr, "%s: tcgetattr %s: %s\n", argv0, _PATH_TTY, strerror(errno));
exit(1);
}
tcattrs_fetched = 1;
/* Check that VISUAL or EDITOR is set before prompting for password,
* that way, the user sees why vi(1) is opened before it is being,
* opened, and can also abort and set EDITOR */
editor = get_editor();
/* Start copier, with a bidirectional channel to it for copying the file */
if (socketpair(PF_LOCAL, SOCK_SEQPACKET, 0, fds)) {
fprintf(stderr, "%s: socketpair PF_LOCAL SOCK_SEQPACKET 0: %s\n", argv0, strerror(errno));
exit(1);
}
pid = run_child(argv[0], fds[1], fds[0]);
close(fds[1]);
/* Make sure (enough) tmpfile is unlinked if killed */
for (i = 1; i < NSIG; i++)
if (i != SIGCHLD)
signal(i, graceful_exit);
/* Create tmpfile from file to edit */
path = get_tmpfile_pattern();
fd = mkstemp(path);
if (fd < 0) {
fprintf(stderr, "%s: mkstemp %s: %s\n", argv0, path, strerror(errno));
exit(1);
}
unlink_this = path;
copy_file(fd, path, fds[0], "<socket to child>", &ok);
if (close(fd)) {
fprintf(stderr, "%s: write %s: %s\n", argv0, path, strerror(errno));
exit(1);
}
if (shutdown(fds[0], SHUT_RD)) {
fprintf(stderr, "%s: shutdown <socket to child> SHUT_RD: %s\n", argv0, strerror(errno));
exit(1);
}
if (!ok) {
fprintf(stderr, "%s: child exited before sending file termination message\n", argv0);
exit(1);
}
/* Don't need to restore terminal attributes, if we got here, asroot as not interrupted */
close(ttyfd);
/* Start file editor */
run_editor(editor, path, fds[0]);
/* Rewrite file from tmpfile and unlink tmpfile */
fd = open(path, O_RDONLY);
if (fd < 0) {
fprintf(stderr, "%s: open %s O_RDONLY: %s\n", argv0, path, strerror(errno));
exit(1);
}
copy_file(fds[0], "<socket to child>", fd, path, NULL);
if (close(fd)) {
fprintf(stderr, "%s: read %s: %s\n", argv0, path, strerror(errno));
exit(1);
}
if (close(fds[0])) {
fprintf(stderr, "%s: write <socket to child>: %s\n", argv0, strerror(errno));
exit(1);
}
unlink(unlink_this);
unlink_this = NULL;
free(path);
/* Wait for exit copier to exit */
if (waitpid(pid, &status, 0) != pid) {
fprintf(stderr, "%s: waitpid <child> 0: %s\n", argv0, strerror(errno));
exit(1);
}
return WIFEXITED(status) ? WEXITSTATUS(status) : 1;
}