Line data Source code
1 : /* helpfile.c - GnuPG's helpfile feature
2 : * Copyright (C) 2007 Free Software Foundation, Inc.
3 : *
4 : * This file is part of GnuPG.
5 : *
6 : * This file is free software; you can redistribute it and/or modify
7 : * it under the terms of either
8 : *
9 : * - the GNU Lesser General Public License as published by the Free
10 : * Software Foundation; either version 3 of the License, or (at
11 : * your option) any later version.
12 : *
13 : * or
14 : *
15 : * - the GNU General Public License as published by the Free
16 : * Software Foundation; either version 2 of the License, or (at
17 : * your option) any later version.
18 : *
19 : * or both in parallel, as here.
20 : *
21 : * This file is distributed in the hope that it will be useful,
22 : * but WITHOUT ANY WARRANTY; without even the implied warranty of
23 : * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
24 : * GNU General Public License for more details.
25 : *
26 : * You should have received a copy of the GNU General Public License
27 : * along with this program; if not, see <http://www.gnu.org/licenses/>.
28 : */
29 :
30 : #include <config.h>
31 : #include <stdlib.h>
32 :
33 :
34 : #include "util.h"
35 : #include "i18n.h"
36 : #include "membuf.h"
37 :
38 :
39 : /* Try to find KEY in the file FNAME. */
40 : static char *
41 0 : findkey_fname (const char *key, const char *fname)
42 : {
43 0 : gpg_error_t err = 0;
44 : FILE *fp;
45 0 : int lnr = 0;
46 : int c;
47 : char *p, line[256];
48 0 : int in_item = 0;
49 0 : membuf_t mb = MEMBUF_ZERO;
50 :
51 0 : fp = fopen (fname, "r");
52 0 : if (!fp)
53 : {
54 0 : if (errno != ENOENT)
55 : {
56 0 : err = gpg_error_from_syserror ();
57 0 : log_error (_("can't open '%s': %s\n"), fname, gpg_strerror (err));
58 : }
59 0 : return NULL;
60 : }
61 :
62 0 : while (fgets (line, DIM(line)-1, fp))
63 : {
64 0 : lnr++;
65 :
66 0 : if (!*line || line[strlen(line)-1] != '\n')
67 : {
68 : /* Eat until end of line. */
69 0 : while ( (c=getc (fp)) != EOF && c != '\n')
70 : ;
71 0 : err = gpg_error (*line? GPG_ERR_LINE_TOO_LONG
72 : : GPG_ERR_INCOMPLETE_LINE);
73 0 : log_error (_("file '%s', line %d: %s\n"),
74 : fname, lnr, gpg_strerror (err));
75 : }
76 : else
77 0 : line[strlen(line)-1] = 0; /* Chop the LF. */
78 :
79 : again:
80 0 : if (!in_item)
81 : {
82 : /* Allow for empty lines and spaces while not in an item. */
83 0 : for (p=line; spacep (p); p++)
84 : ;
85 0 : if (!*p || *p == '#')
86 0 : continue;
87 0 : if (*line != '.' || spacep(line+1))
88 : {
89 0 : log_info (_("file '%s', line %d: %s\n"),
90 : fname, lnr, _("ignoring garbage line"));
91 0 : continue;
92 : }
93 0 : trim_trailing_spaces (line);
94 0 : in_item = 1;
95 0 : if (!strcmp (line+1, key))
96 : {
97 : /* Found. Start collecting. */
98 0 : init_membuf (&mb, 1024);
99 : }
100 0 : continue;
101 : }
102 :
103 : /* If in an item only allow for comments in the first column
104 : and provide ". " as an escape sequence to allow for
105 : leading dots and hash marks in the actual text. */
106 0 : if (*line == '#')
107 0 : continue;
108 0 : if (*line == '.')
109 : {
110 0 : if (spacep(line+1))
111 0 : p = line + 2;
112 : else
113 : {
114 0 : trim_trailing_spaces (line);
115 0 : in_item = 0;
116 0 : if (is_membuf_ready (&mb))
117 : break; /* Yep, found and collected the item. */
118 0 : if (!line[1])
119 0 : continue; /* Just an end of text dot. */
120 0 : goto again; /* A new key line. */
121 : }
122 : }
123 : else
124 0 : p = line;
125 :
126 0 : if (is_membuf_ready (&mb))
127 : {
128 0 : put_membuf_str (&mb, p);
129 0 : put_membuf (&mb, "\n", 1);
130 : }
131 :
132 : }
133 0 : if ( !err && ferror (fp) )
134 : {
135 0 : err = gpg_error_from_syserror ();
136 0 : log_error (_("error reading '%s', line %d: %s\n"),
137 : fname, lnr, gpg_strerror (err));
138 : }
139 :
140 0 : fclose (fp);
141 0 : if (is_membuf_ready (&mb))
142 : {
143 : /* We have collected something. */
144 0 : if (err)
145 : {
146 0 : xfree (get_membuf (&mb, NULL));
147 0 : return NULL;
148 : }
149 : else
150 : {
151 0 : put_membuf (&mb, "", 1); /* Terminate string. */
152 0 : return get_membuf (&mb, NULL);
153 : }
154 : }
155 : else
156 0 : return NULL;
157 : }
158 :
159 :
160 : /* Try the help files depending on the locale. */
161 : static char *
162 0 : findkey_locale (const char *key, const char *locname,
163 : int only_current_locale, const char *dirname)
164 : {
165 : const char *s;
166 : char *fname, *ext, *p;
167 : char *result;
168 :
169 0 : fname = xtrymalloc (strlen (dirname) + 6 + strlen (locname) + 4 + 1);
170 0 : if (!fname)
171 0 : return NULL;
172 0 : ext = stpcpy (stpcpy (fname, dirname), "/help.");
173 : /* Search with locale name and territory. ("help.LL_TT.txt") */
174 0 : if (strchr (locname, '_'))
175 : {
176 0 : strcpy (stpcpy (ext, locname), ".txt");
177 0 : result = findkey_fname (key, fname);
178 : }
179 : else
180 0 : result = NULL; /* No territory. */
181 :
182 0 : if (!result)
183 : {
184 : /* Search with just the locale name - if any. ("help.LL.txt") */
185 0 : if (*locname)
186 : {
187 0 : for (p=ext, s=locname; *s && *s != '_';)
188 0 : *p++ = *s++;
189 0 : strcpy (p, ".txt");
190 0 : result = findkey_fname (key, fname);
191 : }
192 : else
193 0 : result = NULL;
194 : }
195 :
196 0 : if (!result && (!only_current_locale || !*locname) )
197 : {
198 : /* Last try: Search in file without any locale info. ("help.txt") */
199 0 : strcpy (ext, "txt");
200 0 : result = findkey_fname (key, fname);
201 : }
202 :
203 0 : xfree (fname);
204 0 : return result;
205 : }
206 :
207 :
208 : /* Return a malloced help text as identified by KEY. The system takes
209 : the string from an UTF-8 encoded file to be created by an
210 : administrator or as distributed with GnuPG. On a GNU or Unix
211 : system the entry is searched in these files:
212 :
213 : /etc/gnupg/help.LL.txt
214 : /etc/gnupg/help.txt
215 : /usr/share/gnupg/help.LL.txt
216 : /usr/share/gnupg/help.txt
217 :
218 : Here LL denotes the two digit language code of the current locale.
219 : If ONLY_CURRENT_LOCALE is set, the fucntion won;t fallback to the
220 : english valiant ("help.txt") unless that locale has been requested.
221 :
222 : The help file needs to be encoded in UTF-8, lines with a '#' in the
223 : first column are comment lines and entirely ignored. Help keys are
224 : identified by a key consisting of a single word with a single dot
225 : as the first character. All key lines listed without any
226 : intervening lines (except for comment lines) lead to the same help
227 : text. Lines following the key lines make up the actual hep texts.
228 :
229 : */
230 :
231 : char *
232 0 : gnupg_get_help_string (const char *key, int only_current_locale)
233 : {
234 : static const char *locname;
235 : char *result;
236 :
237 0 : if (!locname)
238 : {
239 : char *buffer, *p;
240 0 : int count = 0;
241 0 : const char *s = gnupg_messages_locale_name ();
242 0 : buffer = xtrystrdup (s);
243 0 : if (!buffer)
244 0 : locname = "";
245 : else
246 : {
247 0 : for (p = buffer; *p; p++)
248 0 : if (*p == '.' || *p == '@' || *p == '/' /*(safeguard)*/)
249 0 : *p = 0;
250 0 : else if (*p == '_')
251 : {
252 0 : if (count++)
253 0 : *p = 0; /* Also cut at a underscore in the territory. */
254 : }
255 0 : locname = buffer;
256 : }
257 : }
258 :
259 0 : if (!key || !*key)
260 0 : return NULL;
261 :
262 0 : result = findkey_locale (key, locname, only_current_locale,
263 : gnupg_sysconfdir ());
264 0 : if (!result)
265 0 : result = findkey_locale (key, locname, only_current_locale,
266 : gnupg_datadir ());
267 :
268 0 : if (result)
269 0 : trim_trailing_spaces (result);
270 :
271 0 : return result;
272 : }
|