view usr/src/cmd/refer/addbib.c @ 14050:0c8d9998d589

3747 txg commit callbacks don't work Reviewed by: Matthew Ahrens <mahrens@delphix.com> Reviewed by: Eric Schrock <eric.schrock@delphix.com> Approved by: Christopher Siden <christopher.siden@delphix.com>
author Will Andrews <willa@spectralogic.com>
date Tue, 11 Jun 2013 09:13:51 -0800
parents 6c26331bc6b8
children
line wrap: on
line source

/*
 * Copyright 2005 Sun Microsystems, Inc.  All rights reserved.
 * Use is subject to license terms.
 */

/*	Copyright (c) 1984, 1986, 1987, 1988, 1989 AT&T	*/
/*	  All Rights Reserved  	*/

/*
 * Copyright (c) 1980 Regents of the University of California.
 * All rights reserved. The Berkeley software License Agreement
 * specifies the terms and conditions for redistribution.
 */

#pragma ident	"%Z%%M%	%I%	%E% SMI"


#include <locale.h>
#include <stdio.h>
#include <ctype.h>
#include <signal.h>
#define	MAXENT 50

struct skeleton {
	char prompt[20];	/* prompt user for entry */
	char keylet[5]; 	/* key letter for database */
} bibskel[MAXENT] = {
	"   Author:",	"%A",
	"    Title:",	"%T",
	"  Journal:",	"%J",
	"   Volume:", 	"%V",
	"    Pages:",	"%P",
	"Publisher:",	"%I",
	"     City:", 	"%C",
	"     Date:", 	"%D",
	"    Other:",	"%O",
	" Keywords:",	"%K",	};

int entries = 10;	/* total number of entries in bibskel */
int abstract = 1;	/* asking for abstracts is the default */

static void addbib(FILE *, char *);
void bibedit(FILE *, char *, char *);
static void instruct(void);
static void rd_skel(char *);
static void trim(char []);

static void
usage(void)			/* print proper usage and exit */
{
	puts(gettext("Usage:  addbib [-p promptfile] [-a] database\n\
\t-p: the promptfile defines alternate fields\n\
\t-a: don't include prompting for the abstract\n"));
	exit(1);
}

int
main(int argc, char *argv[])	/* addbib: bibliography entry program */
{
	FILE *fp, *fopen();
	int i;

	(void) setlocale(LC_ALL, "");

#if !defined(TEXT_DOMAIN)
#define	TEXT_DOMAIN "SYS_TEST"
#endif
	(void) textdomain(TEXT_DOMAIN);

	if (argc == 1) {
		puts(gettext(
		    "You must specify a bibliography file (database)."));
		usage();
	}
	for (i = 1; argv[i][0] == '-'; i++) {
		if (argv[i][1] == 'p') {
			if (i >= argc - 2) {
				puts(gettext("Not enough arguments for "
				    "-p option."));
				usage();
			}
			rd_skel(argv[++i]);
		} else if (argv[i][1] == 'a') {
			if (i >= argc - 1) {
				puts(gettext(
				    "No bibliofile specified after -a."));
				usage();
			}
			abstract = 0;
		} else {	/* neither -p nor -a */
			printf(gettext(
			    "Invalid command line flag: %s\n"), argv[i]);
			usage();
		}
	}
	if (i < argc - 1) {
		puts(gettext("Too many arguments with no options."));
		usage();
	}
	if ((fp = fopen(argv[i], "a")) == NULL) {
		perror(argv[i]);
		exit(1);
	}
	addbib(fp, argv[i]);	/* loop for input */
	return (0);
}

static void
addbib(FILE *fp, char *argv)	/* add entries to a bibliographic database */
{
	char line[BUFSIZ];
	int i = 0, firstln, repeat = 0, escape = 0;

	printf(gettext("Instructions? "));
	fgets(line, BUFSIZ, stdin);
	if (line[0] == 'y' || line[0] == 'Y')
		instruct();
	while (1) {
		putchar('\n');
		putc('\n', fp);
		for (i = 0; i < entries; i++) {
			printf("%s\t", gettext(bibskel[i].prompt));
			if (fgets(line, BUFSIZ, stdin) == NULL) {
				clearerr(stdin);
				break;
			}
			if (line[0] == '-' && line[1] == '\n') {
				i -= 2;
				if (i < -1) {
					printf(gettext("Too far back.\n"));
					i++;
				}
				continue;
			} else if (line[strlen(line)-2] == '\\') {
				if (line[0] != '\\') {
					line[strlen(line)-2] = '\n';
					line[strlen(line)-1] = NULL;
					trim(line);
					fprintf(fp, "%s %s",
					    bibskel[i].keylet, line);
				}
				printf("> ");
				again:
				fgets(line, BUFSIZ, stdin);
				if (line[strlen(line)-2] == '\\') {
					line[strlen(line)-2] = '\n';
					line[strlen(line)-1] = NULL;
					trim(line);
					fputs(line, fp);
					printf("> ");
					goto again;
				}
				trim(line);
				fputs(line, fp);
			} else if (line[0] != '\n') {
				trim(line);
				fprintf(fp, "%s %s", bibskel[i].keylet, line);
			}
		}
		if (abstract) {
			puts(gettext(" Abstract: (ctrl-d to end)"));
			firstln = 1;
			while (fgets(line, BUFSIZ, stdin)) {
				if (firstln && line[0] != '%') {
					fprintf(fp, "%%X ");
					firstln = 0;
				}
				fputs(line, fp);
			}
			clearerr(stdin);
		}
		fflush(fp);	/* write to file at end of each cycle */
		if (ferror(fp)) {
			perror(argv);
			exit(1);
		}
		editloop:
		printf(gettext("\nContinue? "));
			fgets(line, BUFSIZ, stdin);
		if (line[0] == 'e' || line[0] == 'v') {
			bibedit(fp, line, argv);
			goto editloop;
		}
		if (line[0] == 'q' || line[0] == 'n')
			return;
	}
}

static void
trim(char line[])		/* trim line of trailing white space */
{
	int n;

	n = strlen(line);
	while (--n >= 0) {
		if (!isspace(line[n]))
			break;
	}
	line[++n] = '\n';
	line[++n] = NULL;
}

void
bibedit(FILE *fp, char *cmd, char *arg)	/* edit database with edit, ex, or vi */
{
	int i = 0, status;

	fclose(fp);
	while (!isspace(cmd[i]))
		i++;
	cmd[i] = NULL;
	if (fork() == 0) {
		if (cmd[0] == 'v' && cmd[1] == 'i')
			execlp(cmd, cmd, "+$", arg, NULL);
		else /* either ed, ex, or edit */
			execlp(cmd, cmd, arg, NULL);
	}
	signal(SIGINT, SIG_IGN);
	signal(SIGQUIT, SIG_IGN);
	wait(&status);
	signal(SIGINT, SIG_DFL);
	signal(SIGQUIT, SIG_DFL);
	if ((fp = fopen(arg, "a")) == NULL) {
		perror(arg);
		exit(1);
	}
}

static void
instruct(void)		/* give user elementary directions */
{
	putchar('\n');
	puts(gettext(
	    "Addbib will prompt you for various bibliographic fields.\n"
"If you don't need a particular field, just hit RETURN,\n"
"\tand that field will not appear in the output file.\n"
"If you want to return to previous fields in the skeleton,\n"
"\ta single minus sign will go back a field at a time.\n"
"\t(This is the best way to input multiple authors.)\n"
"If you have to continue a field or add an unusual field,\n"
"\ta trailing backslash will allow a temporary escape.\n"
"Finally, (without -a) you will be prompted for an abstract\n"
"Type in as many lines as you need, and end with a ctrl-d.\n"
"To quit, type `q' or `n' when asked if you want to continue.\n"
"To edit the database, type `edit', `vi', or `ex' instead."));

}

static void
rd_skel(char *arg)		/* redo bibskel from user-supplied file */
{
	FILE *pfp, *fopen();
	char str[BUFSIZ];
	int entry, i, j;

	if ((pfp = fopen(arg, "r")) == NULL) {
		fprintf(stderr, gettext("Promptfile "));
		perror(arg);
		exit(1);
	}
	for (entry = 0; fgets(str, BUFSIZ, pfp); entry++) {
		for (i = 0; str[i] != '\t' && str[i] != '\n'; i++)
			bibskel[entry].prompt[i] = str[i];
		bibskel[entry].prompt[i] = NULL;
		if (str[i] == '\n') {
			fprintf(stderr, gettext(
			    "No tabs between promptfile fields.\n"));
			fprintf(stderr, gettext(
			    "Format: prompt-string <TAB> %%key\n"));
			exit(1);
		}
		for (i++, j = 0; str[i] != '\n'; i++, j++)
			bibskel[entry].keylet[j] = str[i];
		bibskel[entry].keylet[j] = NULL;

		if (entry >= MAXENT) {
			fprintf(stderr, gettext(
			    "Too many entries in promptfile.\n"));
			exit(1);
		}
	}
	entries = entry;
	fclose(pfp);
}