xref: /netbsd-src/games/quiz/quiz.c (revision 952d3d39d2b7734fba58bb7c5b1d3af135d87736)
1*952d3d39Srillig /*	$NetBSD: quiz.c,v 1.29 2023/01/22 17:19:11 rillig Exp $	*/
2ee0f9d10Scgd 
35bfb98b1Scgd /*-
4a432dc58Scgd  * Copyright (c) 1991, 1993
5a432dc58Scgd  *	The Regents of the University of California.  All rights reserved.
65bfb98b1Scgd  *
75bfb98b1Scgd  * This code is derived from software contributed to Berkeley by
8a432dc58Scgd  * Jim R. Oldroyd at The Instruction Set and Keith Gabryelski at
9a432dc58Scgd  * Commodore Business Machines.
105bfb98b1Scgd  *
115bfb98b1Scgd  * Redistribution and use in source and binary forms, with or without
125bfb98b1Scgd  * modification, are permitted provided that the following conditions
135bfb98b1Scgd  * are met:
145bfb98b1Scgd  * 1. Redistributions of source code must retain the above copyright
155bfb98b1Scgd  *    notice, this list of conditions and the following disclaimer.
165bfb98b1Scgd  * 2. Redistributions in binary form must reproduce the above copyright
175bfb98b1Scgd  *    notice, this list of conditions and the following disclaimer in the
185bfb98b1Scgd  *    documentation and/or other materials provided with the distribution.
19e5aeb4eaSagc  * 3. Neither the name of the University nor the names of its contributors
205bfb98b1Scgd  *    may be used to endorse or promote products derived from this software
215bfb98b1Scgd  *    without specific prior written permission.
225bfb98b1Scgd  *
235bfb98b1Scgd  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
245bfb98b1Scgd  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
255bfb98b1Scgd  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
265bfb98b1Scgd  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
275bfb98b1Scgd  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
285bfb98b1Scgd  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
295bfb98b1Scgd  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
305bfb98b1Scgd  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
315bfb98b1Scgd  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
325bfb98b1Scgd  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
335bfb98b1Scgd  * SUCH DAMAGE.
345bfb98b1Scgd  */
355bfb98b1Scgd 
36d8297446Slukem #include <sys/cdefs.h>
375bfb98b1Scgd #ifndef lint
382fe2731dSlukem __COPYRIGHT("@(#) Copyright (c) 1991, 1993\
392fe2731dSlukem  The Regents of the University of California.  All rights reserved.");
405bfb98b1Scgd #endif /* not lint */
415bfb98b1Scgd 
425bfb98b1Scgd #ifndef lint
43ee0f9d10Scgd #if 0
442ea7b8f1Stls static char sccsid[] = "@(#)quiz.c	8.3 (Berkeley) 5/4/95";
45ee0f9d10Scgd #else
46*952d3d39Srillig __RCSID("$NetBSD: quiz.c,v 1.29 2023/01/22 17:19:11 rillig Exp $");
47ee0f9d10Scgd #endif
485bfb98b1Scgd #endif /* not lint */
495bfb98b1Scgd 
505bfb98b1Scgd #include <sys/types.h>
512ea7b8f1Stls 
522ea7b8f1Stls #include <ctype.h>
535bfb98b1Scgd #include <errno.h>
545bfb98b1Scgd #include <stdio.h>
555bfb98b1Scgd #include <stdlib.h>
565bfb98b1Scgd #include <string.h>
57c541862aSpk #include <err.h>
582ea7b8f1Stls #include <time.h>
592ea7b8f1Stls #include <unistd.h>
605bfb98b1Scgd #include "quiz.h"
615bfb98b1Scgd #include "pathnames.h"
625bfb98b1Scgd 
635bfb98b1Scgd static QE qlist;
645bfb98b1Scgd static int catone, cattwo, tflag;
65cf97b0fdSdholland static unsigned qsize;
665bfb98b1Scgd 
675305281bSdholland static char *appdstr(char *, const char *, size_t);
685305281bSdholland static void downcase(char *);
695305281bSdholland static void get_cats(char *, char *);
705305281bSdholland static void get_file(const char *);
715305281bSdholland static const char *next_cat(const char *);
725305281bSdholland static void quiz(void);
73cf97b0fdSdholland static void score(unsigned, unsigned, unsigned);
745305281bSdholland static void show_index(void);
755305281bSdholland static void usage(void) __dead;
765bfb98b1Scgd 
775bfb98b1Scgd int
main(int argc,char * argv[])7804ed5236Sdholland main(int argc, char *argv[])
795bfb98b1Scgd {
80d8297446Slukem 	int ch;
81092d3130Sjsm 	const char *indexfile;
825bfb98b1Scgd 
835367f340Sjsm 	/* Revoke setgid privileges */
84f9eca697Smycroft 	setgid(getgid());
855367f340Sjsm 
865bfb98b1Scgd 	indexfile = _PATH_QUIZIDX;
87d8297446Slukem 	while ((ch = getopt(argc, argv, "i:t")) != -1)
885bfb98b1Scgd 		switch(ch) {
895bfb98b1Scgd 		case 'i':
905bfb98b1Scgd 			indexfile = optarg;
915bfb98b1Scgd 			break;
925bfb98b1Scgd 		case 't':
935bfb98b1Scgd 			tflag = 1;
945bfb98b1Scgd 			break;
955bfb98b1Scgd 		case '?':
965bfb98b1Scgd 		default:
975bfb98b1Scgd 			usage();
985bfb98b1Scgd 		}
995bfb98b1Scgd 	argc -= optind;
1005bfb98b1Scgd 	argv += optind;
1015bfb98b1Scgd 
1025bfb98b1Scgd 	switch(argc) {
1035bfb98b1Scgd 	case 0:
1045bfb98b1Scgd 		get_file(indexfile);
1055bfb98b1Scgd 		show_index();
1065bfb98b1Scgd 		break;
1075bfb98b1Scgd 	case 2:
1085bfb98b1Scgd 		get_file(indexfile);
1095bfb98b1Scgd 		get_cats(argv[0], argv[1]);
1105bfb98b1Scgd 		quiz();
1115bfb98b1Scgd 		break;
1125bfb98b1Scgd 	default:
1135bfb98b1Scgd 		usage();
1145bfb98b1Scgd 	}
1155bfb98b1Scgd 	exit(0);
1165bfb98b1Scgd }
1175bfb98b1Scgd 
1185305281bSdholland static void
get_file(const char * file)11904ed5236Sdholland get_file(const char *file)
1205bfb98b1Scgd {
121d8297446Slukem 	FILE *fp;
122d8297446Slukem 	QE *qp;
1235bfb98b1Scgd 	size_t len;
1245bfb98b1Scgd 	char *lp;
1255bfb98b1Scgd 
1265bfb98b1Scgd 	if ((fp = fopen(file, "r")) == NULL)
127c541862aSpk 		err(1, "%s", file);
1285bfb98b1Scgd 
1295bfb98b1Scgd 	/*
1305bfb98b1Scgd 	 * XXX
1315bfb98b1Scgd 	 * Should really free up space from any earlier read list
1325bfb98b1Scgd 	 * but there are no reverse pointers to do so with.
1335bfb98b1Scgd 	 */
1345bfb98b1Scgd 	qp = &qlist;
1355bfb98b1Scgd 	qsize = 0;
13602254e0cScgd 	while ((lp = fgetln(fp, &len)) != NULL) {
137d8297446Slukem 		if (lp[len - 1] == '\n')
1382037c75eScgd 			lp[--len] = '\0';
139a432dc58Scgd 		if (qp->q_text && qp->q_text[strlen(qp->q_text) - 1] == '\\')
140a432dc58Scgd 			qp->q_text = appdstr(qp->q_text, lp, len);
141a432dc58Scgd 		else {
1425bfb98b1Scgd 			if ((qp->q_next = malloc(sizeof(QE))) == NULL)
143d8297446Slukem 				errx(1, "malloc");
1445bfb98b1Scgd 			qp = qp->q_next;
145d8297446Slukem 			if ((qp->q_text = malloc(len + 1)) == NULL)
146d8297446Slukem 				errx(1, "malloc");
147d8297446Slukem 			strncpy(qp->q_text, lp, len);
148d8297446Slukem 			qp->q_text[len] = '\0';
1495bfb98b1Scgd 			qp->q_asked = qp->q_answered = FALSE;
1505bfb98b1Scgd 			qp->q_next = NULL;
1515bfb98b1Scgd 			++qsize;
1525bfb98b1Scgd 		}
1535bfb98b1Scgd 	}
1545bfb98b1Scgd 	(void)fclose(fp);
1555bfb98b1Scgd }
1565bfb98b1Scgd 
1575305281bSdholland static void
show_index(void)15804ed5236Sdholland show_index(void)
1595bfb98b1Scgd {
160d8297446Slukem 	QE *qp;
161092d3130Sjsm 	const char *p, *s;
1625bfb98b1Scgd 	FILE *pf;
163b2d99aa2Sjsm 	const char *pager;
1645bfb98b1Scgd 
165b2d99aa2Sjsm 	if (!isatty(1))
166b2d99aa2Sjsm 		pager = "cat";
167b2d99aa2Sjsm 	else {
168b2d99aa2Sjsm 		if (!(pager = getenv("PAGER")) || (*pager == 0))
169b2d99aa2Sjsm 			pager = _PATH_PAGER;
170b2d99aa2Sjsm 	}
171b2d99aa2Sjsm 	if ((pf = popen(pager, "w")) == NULL)
172b2d99aa2Sjsm 		err(1, "%s", pager);
1735bfb98b1Scgd 	(void)fprintf(pf, "Subjects:\n\n");
1745bfb98b1Scgd 	for (qp = qlist.q_next; qp; qp = qp->q_next) {
1755bfb98b1Scgd 		for (s = next_cat(qp->q_text); s; s = next_cat(s)) {
1765bfb98b1Scgd 			if (!rxp_compile(s))
177c541862aSpk 				errx(1, "%s", rxperr);
178d8297446Slukem 			if ((p = rxp_expand()) != NULL)
1795bfb98b1Scgd 				(void)fprintf(pf, "%s ", p);
1805bfb98b1Scgd 		}
1815bfb98b1Scgd 		(void)fprintf(pf, "\n");
1825bfb98b1Scgd 	}
1835bfb98b1Scgd 	(void)fprintf(pf, "\n%s\n%s\n%s\n",
1845bfb98b1Scgd "For example, \"quiz victim killer\" prints a victim's name and you reply",
1855bfb98b1Scgd "with the killer, and \"quiz killer victim\" works the other way around.",
1865bfb98b1Scgd "Type an empty line to get the correct answer.");
1875bfb98b1Scgd 	(void)pclose(pf);
1885bfb98b1Scgd }
1895bfb98b1Scgd 
1905305281bSdholland static void
get_cats(char * cat1,char * cat2)19104ed5236Sdholland get_cats(char *cat1, char *cat2)
1925bfb98b1Scgd {
193d8297446Slukem 	QE *qp;
1945bfb98b1Scgd 	int i;
195092d3130Sjsm 	const char *s;
1965bfb98b1Scgd 
1975bfb98b1Scgd 	downcase(cat1);
1985bfb98b1Scgd 	downcase(cat2);
1995bfb98b1Scgd 	for (qp = qlist.q_next; qp; qp = qp->q_next) {
2005bfb98b1Scgd 		s = next_cat(qp->q_text);
2015bfb98b1Scgd 		catone = cattwo = i = 0;
2025bfb98b1Scgd 		while (s) {
2035bfb98b1Scgd 			if (!rxp_compile(s))
204c541862aSpk 				errx(1, "%s", rxperr);
2055bfb98b1Scgd 			i++;
2065bfb98b1Scgd 			if (rxp_match(cat1))
2075bfb98b1Scgd 				catone = i;
2085bfb98b1Scgd 			if (rxp_match(cat2))
2095bfb98b1Scgd 				cattwo = i;
2105bfb98b1Scgd 			s = next_cat(s);
2115bfb98b1Scgd 		}
2125bfb98b1Scgd 		if (catone && cattwo && catone != cattwo) {
2135bfb98b1Scgd 			if (!rxp_compile(qp->q_text))
214c541862aSpk 				errx(1, "%s", rxperr);
2155bfb98b1Scgd 			get_file(rxp_expand());
2165bfb98b1Scgd 			return;
2175bfb98b1Scgd 		}
2185bfb98b1Scgd 	}
219c541862aSpk 	errx(1, "invalid categories");
2205bfb98b1Scgd }
2215bfb98b1Scgd 
2225305281bSdholland static void
quiz(void)22304ed5236Sdholland quiz(void)
2245bfb98b1Scgd {
225d8297446Slukem 	QE *qp;
226d8297446Slukem 	int i;
227a432dc58Scgd 	size_t len;
228cf97b0fdSdholland 	unsigned guesses, rights, wrongs;
229920b84d8Sdholland 	unsigned next, j;
230092d3130Sjsm 	char *answer, *t, question[LINE_SZ];
231092d3130Sjsm 	const char *s;
2325bfb98b1Scgd 
2335bfb98b1Scgd 	srandom(time(NULL));
2345bfb98b1Scgd 	guesses = rights = wrongs = 0;
2355bfb98b1Scgd 	for (;;) {
2365bfb98b1Scgd 		if (qsize == 0)
2375bfb98b1Scgd 			break;
2385bfb98b1Scgd 		next = random() % qsize;
2395bfb98b1Scgd 		qp = qlist.q_next;
240920b84d8Sdholland 		for (j = 0; j < next; j++)
2415bfb98b1Scgd 			qp = qp->q_next;
2425bfb98b1Scgd 		while (qp && qp->q_answered)
2435bfb98b1Scgd 			qp = qp->q_next;
2445bfb98b1Scgd 		if (!qp) {
2455bfb98b1Scgd 			qsize = next;
2465bfb98b1Scgd 			continue;
2475bfb98b1Scgd 		}
2485bfb98b1Scgd 		if (tflag && random() % 100 > 20) {
2495bfb98b1Scgd 			/* repeat questions in tutorial mode */
2505bfb98b1Scgd 			while (qp && (!qp->q_asked || qp->q_answered))
2515bfb98b1Scgd 				qp = qp->q_next;
2525bfb98b1Scgd 			if (!qp)
2535bfb98b1Scgd 				continue;
2545bfb98b1Scgd 		}
2555bfb98b1Scgd 		s = qp->q_text;
2565bfb98b1Scgd 		for (i = 0; i < catone - 1; i++)
2575bfb98b1Scgd 			s = next_cat(s);
2585bfb98b1Scgd 		if (!rxp_compile(s))
259c541862aSpk 			errx(1, "%s", rxperr);
2605bfb98b1Scgd 		t = rxp_expand();
2615bfb98b1Scgd 		if (!t || *t == '\0') {
2625bfb98b1Scgd 			qp->q_answered = TRUE;
2635bfb98b1Scgd 			continue;
2645bfb98b1Scgd 		}
2655bfb98b1Scgd 		(void)strcpy(question, t);
2665bfb98b1Scgd 		s = qp->q_text;
2675bfb98b1Scgd 		for (i = 0; i < cattwo - 1; i++)
2685bfb98b1Scgd 			s = next_cat(s);
2695bfb98b1Scgd 		if (!rxp_compile(s))
270c541862aSpk 			errx(1, "%s", rxperr);
2715bfb98b1Scgd 		t = rxp_expand();
2725bfb98b1Scgd 		if (!t || *t == '\0') {
2735bfb98b1Scgd 			qp->q_answered = TRUE;
2745bfb98b1Scgd 			continue;
2755bfb98b1Scgd 		}
2765bfb98b1Scgd 		qp->q_asked = TRUE;
2775bfb98b1Scgd 		(void)printf("%s?\n", question);
2785bfb98b1Scgd 		for (;; ++guesses) {
279d8297446Slukem 			if ((answer = fgetln(stdin, &len)) == NULL ||
280d8297446Slukem 			    answer[len - 1] != '\n') {
2815bfb98b1Scgd 				score(rights, wrongs, guesses);
2825bfb98b1Scgd 				exit(0);
2835bfb98b1Scgd 			}
284a60129f5Scgd 			answer[len - 1] = '\0';
2855bfb98b1Scgd 			downcase(answer);
2865bfb98b1Scgd 			if (rxp_match(answer)) {
2875bfb98b1Scgd 				(void)printf("Right!\n");
2885bfb98b1Scgd 				++rights;
2895bfb98b1Scgd 				qp->q_answered = TRUE;
2905bfb98b1Scgd 				break;
2915bfb98b1Scgd 			}
2925bfb98b1Scgd 			if (*answer == '\0') {
2935bfb98b1Scgd 				(void)printf("%s\n", t);
2945bfb98b1Scgd 				++wrongs;
2955bfb98b1Scgd 				if (!tflag)
2965bfb98b1Scgd 					qp->q_answered = TRUE;
2975bfb98b1Scgd 				break;
2985bfb98b1Scgd 			}
2995bfb98b1Scgd 			(void)printf("What?\n");
3005bfb98b1Scgd 		}
3015bfb98b1Scgd 	}
3025bfb98b1Scgd 	score(rights, wrongs, guesses);
3035bfb98b1Scgd }
3045bfb98b1Scgd 
3055305281bSdholland static const char *
next_cat(const char * s)30604ed5236Sdholland next_cat(const char *s)
3075bfb98b1Scgd {
3083353b6d2Smycroft 	int esc;
3093353b6d2Smycroft 
3103353b6d2Smycroft 	esc = 0;
3115bfb98b1Scgd 	for (;;)
3125bfb98b1Scgd 		switch (*s++) {
3135bfb98b1Scgd 		case '\0':
3145bfb98b1Scgd 			return (NULL);
3155bfb98b1Scgd 		case '\\':
3163353b6d2Smycroft 			esc = 1;
3175bfb98b1Scgd 			break;
3185bfb98b1Scgd 		case ':':
3193353b6d2Smycroft 			if (!esc)
3205bfb98b1Scgd 				return (s);
321fbffadb9Smrg 			/* FALLTHROUGH */
3223353b6d2Smycroft 		default:
3233353b6d2Smycroft 			esc = 0;
3243353b6d2Smycroft 			break;
3255bfb98b1Scgd 		}
3265bfb98b1Scgd 	/* NOTREACHED */
3275bfb98b1Scgd }
3285bfb98b1Scgd 
3295305281bSdholland static char *
appdstr(char * s,const char * tp,size_t len)33004ed5236Sdholland appdstr(char *s, const char *tp, size_t len)
3315bfb98b1Scgd {
332092d3130Sjsm 	char *mp;
333092d3130Sjsm 	const char *sp;
334d8297446Slukem 	int ch;
3355bfb98b1Scgd 	char *m;
3365bfb98b1Scgd 
337a432dc58Scgd 	if ((m = malloc(strlen(s) + len + 1)) == NULL)
338d8297446Slukem 		errx(1, "malloc");
339ba40c8d6Sjsm 	for (mp = m, sp = s; (*mp++ = *sp++) != '\0'; )
340d8297446Slukem 		;
3412037c75eScgd 	--mp;
3425bfb98b1Scgd 	if (*(mp - 1) == '\\')
3435bfb98b1Scgd 		--mp;
3442037c75eScgd 
345d8297446Slukem 	while ((ch = *mp++ = *tp++) && ch != '\n')
346d8297446Slukem 		;
3475bfb98b1Scgd 	*mp = '\0';
3485bfb98b1Scgd 
3495bfb98b1Scgd 	free(s);
3505bfb98b1Scgd 	return (m);
3515bfb98b1Scgd }
3525bfb98b1Scgd 
3535305281bSdholland static void
score(unsigned r,unsigned w,unsigned g)35404ed5236Sdholland score(unsigned r, unsigned w, unsigned g)
3555bfb98b1Scgd {
3565bfb98b1Scgd 	(void)printf("Rights %d, wrongs %d,", r, w);
3575bfb98b1Scgd 	if (g)
3585bfb98b1Scgd 		(void)printf(" extra guesses %d,", g);
3595bfb98b1Scgd 	(void)printf(" score %d%%\n", (r + w + g) ? r * 100 / (r + w + g) : 0);
3605bfb98b1Scgd }
3615bfb98b1Scgd 
3625305281bSdholland static void
downcase(char * p)36304ed5236Sdholland downcase(char *p)
3645bfb98b1Scgd {
365*952d3d39Srillig 	unsigned char ch;
3665bfb98b1Scgd 
367d8297446Slukem 	for (; (ch = *p) != '\0'; ++p)
3685bfb98b1Scgd 		if (isascii(ch) && isupper(ch))
3695bfb98b1Scgd 			*p = tolower(ch);
3705bfb98b1Scgd }
3715bfb98b1Scgd 
3725305281bSdholland static void
usage(void)37304ed5236Sdholland usage(void)
3745bfb98b1Scgd {
3755bfb98b1Scgd 	(void)fprintf(stderr, "quiz [-t] [-i file] category1 category2\n");
3765bfb98b1Scgd 	exit(1);
3775bfb98b1Scgd }
378