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