]> diplodocus.org Git - nmh/blob - sbr/readconfig.c
Update manpages to use .TP for tagged paragraphs (part I).
[nmh] / sbr / readconfig.c
1
2 /*
3 * readconfig.c -- base routine to read nmh configuration files
4 * -- such as nmh profile, context file, or mhn.defaults.
5 *
6 * This code is Copyright (c) 2002, by the authors of nmh. See the
7 * COPYRIGHT file in the root directory of the nmh distribution for
8 * complete copyright information.
9 */
10
11 #include <h/mh.h>
12 #include <h/utils.h>
13
14 struct procstr {
15 char *procname;
16 char **procnaddr;
17 };
18
19 static struct procstr procs[] = {
20 { "context", &context },
21 { "mh-sequences", &mh_seq },
22 { "buildmimeproc", &buildmimeproc },
23 { "fileproc", &fileproc },
24 { "formatproc", &formatproc },
25 { "incproc", &incproc },
26 { "installproc", &installproc },
27 { "lproc", &lproc },
28 { "mailproc", &mailproc },
29 { "mhlproc", &mhlproc },
30 { "moreproc", &moreproc },
31 { "mshproc", &mshproc },
32 { "packproc", &packproc },
33 { "postproc", &postproc },
34 { "rmmproc", &rmmproc },
35 { "sendproc", &sendproc },
36 { "showmimeproc", &showmimeproc },
37 { "showproc", &showproc },
38 { "vmhproc", &vmhproc },
39 { "whatnowproc", &whatnowproc },
40 { "whomproc", &whomproc },
41 { NULL, NULL }
42 };
43
44 static struct node **opp = NULL;
45
46
47 void
48 readconfig (struct node **npp, FILE *ib, char *file, int ctx)
49 {
50 register int state;
51 register char *cp;
52 char name[NAMESZ], field[BUFSIZ];
53 register struct node *np;
54 register struct procstr *ps;
55 m_getfld_state_t gstate = 0;
56
57 if (npp == NULL && (npp = opp) == NULL) {
58 admonish (NULL, "bug: readconfig called but pump not primed");
59 return;
60 }
61
62 for (;;) {
63 int fieldsz = sizeof field;
64 switch (state = m_getfld (&gstate, name, field, &fieldsz, ib)) {
65 case FLD:
66 case FLDPLUS:
67 np = (struct node *) mh_xmalloc (sizeof(*np));
68 *npp = np;
69 *(npp = &np->n_next) = NULL;
70 np->n_name = getcpy (name);
71 if (state == FLDPLUS) {
72 cp = getcpy (field);
73 while (state == FLDPLUS) {
74 fieldsz = sizeof field;
75 state = m_getfld (&gstate, name, field, &fieldsz, ib);
76 cp = add (field, cp);
77 }
78 np->n_field = trimcpy (cp);
79 free (cp);
80 } else {
81 np->n_field = trimcpy (field);
82 }
83 np->n_context = ctx;
84
85 /*
86 * Now scan the list of `procs' and link in the
87 * field value to the global variable.
88 */
89 for (ps = procs; ps->procname; ps++)
90 if (strcmp (np->n_name, ps->procname) == 0) {
91 *ps->procnaddr = np->n_field;
92 break;
93 }
94 continue;
95
96 case BODY:
97 adios (NULL, "no blank lines are permitted in %s", file);
98
99 case FILEEOF:
100 break;
101
102 default:
103 adios (NULL, "%s is poorly formatted", file);
104 }
105 break;
106 }
107 m_getfld_state_destroy (&gstate);
108
109 /*
110 * Special handling for the pager processes: lproc and moreproc.
111 *
112 * If they are not set by the profile, use the callers $PAGER if
113 * available, otherwise set them to DEFAULT_PAGER.
114 */
115 if (lproc == NULL) {
116 lproc = getenv("PAGER");
117 if (lproc == NULL || lproc[0] == '\0')
118 lproc = DEFAULT_PAGER;
119 }
120 if (moreproc == NULL) {
121 moreproc = getenv("PAGER");
122 if (moreproc == NULL || moreproc[0] == '\0')
123 moreproc = DEFAULT_PAGER;
124 }
125
126 if (opp == NULL) {
127 /* Check for duplicated non-null profile entries. Except
128 allow multiple profile entries named "#", because that's
129 what the mh-profile man page suggests using for comments.
130
131 Only do this check on the very first call from
132 context_read(), when opp is NULL. That way, entries in
133 mhn.defaults can be overridden without triggering
134 warnings.
135
136 Note that that mhn.defaults, $MHN, $MHBUILD, $MHSHOW, and
137 $MHSTORE all put their entries into just one list, m_defs,
138 the same list that the profile uses. */
139
140 struct node *np;
141 for (np = m_defs; np; np = np->n_next) {
142 /* Yes, this is O(N^2). The profile should be small enough so
143 that's not a performance problem. */
144 if (strlen (np->n_name) > 0 && strcmp ("#", np->n_name)) {
145 struct node *np2;
146 for (np2 = np->n_next; np2; np2 = np2->n_next) {
147 if (! mh_strcasecmp (np->n_name, np2->n_name)) {
148 admonish (NULL, "multiple \"%s\" profile components "
149 "in %s, ignoring \"%s\"",
150 np->n_name, defpath, np2->n_field);
151 }
152 }
153 }
154 }
155 }
156
157 opp = npp;
158 }