Imported Upstream version 2.2.15
[platform/upstream/gpg2.git] / common / server-help.c
1 /* server-help.h - Helper functions for writing Assuan servers.
2  *      Copyright (C) 2003, 2009, 2010 g10 Code GmbH
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 <https://www.gnu.org/licenses/>.
28  */
29
30 #include <config.h>
31 #include <string.h>
32
33 #include "util.h"
34 #include "server-help.h"
35
36
37 static GPGRT_INLINE gpg_error_t
38 my_error (int e)
39 {
40   return gpg_err_make (default_errsource, (e));
41 }
42
43 static GPGRT_INLINE gpg_error_t
44 my_error_from_syserror (void)
45 {
46   return gpg_err_make (default_errsource, gpg_err_code_from_syserror ());
47 }
48
49
50 /* Skip over options in LINE.
51
52    Blanks after the options are also removed.  Options are indicated
53    by two leading dashes followed by a string consisting of non-space
54    characters.  The special option "--" indicates an explicit end of
55    options; all what follows will not be considered an option.  The
56    first no-option string also indicates the end of option parsing. */
57 char *
58 skip_options (const char *line)
59 {
60   while (spacep (line))
61     line++;
62   while (*line == '-' && line[1] == '-')
63     {
64       while (*line && !spacep (line))
65         line++;
66       while (spacep (line))
67         line++;
68     }
69   return (char*) line;
70 }
71
72
73 /* Check whether the option NAME appears in LINE.  */
74 int
75 has_option (const char *line, const char *name)
76 {
77   const char *s;
78   int n = strlen (name);
79
80   s = strstr (line, name);
81   if (s && s >= skip_options (line))
82     return 0;
83   return (s && (s == line || spacep (s-1)) && (!s[n] || spacep (s+n)));
84 }
85
86
87 /* Same as has_option but only considers options at the begin of the
88    line.  This is useful for commands which allow arbitrary strings on
89    the line.  */
90 int
91 has_leading_option (const char *line, const char *name)
92 {
93   const char *s;
94   int n;
95
96   if (name[0] != '-' || name[1] != '-' || !name[2] || spacep (name+2))
97     return 0;
98   n = strlen (name);
99   while ( *line == '-' && line[1] == '-' )
100     {
101       s = line;
102       while (*line && !spacep (line))
103         line++;
104       if (n == (line - s) && !strncmp (s, name, n))
105         return 1;
106       while (spacep (line))
107         line++;
108     }
109   return 0;
110 }
111
112
113 /* Same as has_option but does only test for the name of the option
114    and ignores an argument, i.e. with NAME being "--hash" it would
115    return a pointer for "--hash" as well as for "--hash=foo".  If
116    there is no such option NULL is returned.  The pointer returned
117    points right behind the option name, this may be an equal sign, Nul
118    or a space.  */
119 const char *
120 has_option_name (const char *line, const char *name)
121 {
122   const char *s;
123   int n = strlen (name);
124
125   s = strstr (line, name);
126   return (s && (s == line || spacep (s-1))
127           && (!s[n] || spacep (s+n) || s[n] == '=')) ? (s+n) : NULL;
128 }
129
130
131 /* Parse an option with the format "--NAME=VALUE" which must occur in
132  * LINE before a double-dash.  LINE is written to but not modified by
133  * this function.  If the option is found and has a value the value is
134  * stored as a malloced string at R_VALUE.  If the option was not
135  * found or an error occurred NULL is stored there.  Note that
136  * currently the value must be a string without any space; we may
137  * eventually update this function to allow for a quoted value.  */
138 gpg_error_t
139 get_option_value (char *line, const char *name, char **r_value)
140 {
141   char *p, *pend;
142   int c;
143
144   *r_value = NULL;
145
146   p = (char*)has_option_name (line, name);
147   if (!p || p >= skip_options (line))
148     return 0;
149
150   if (*p != '=' || !p[1] || spacep (p+1))
151     return my_error (GPG_ERR_INV_ARG);
152   p++;
153   for (pend = p; *pend && !spacep (pend); pend++)
154     ;
155   c = *pend;
156   *pend = 0;
157   *r_value = xtrystrdup (p);
158   *pend = c;
159   if (!p)
160     return my_error_from_syserror ();
161   return 0;
162 }
163
164
165 /* Return a pointer to the argument of the option with NAME.  If such
166    an option is not given, NULL is returned. */
167 char *
168 option_value (const char *line, const char *name)
169 {
170   char *s;
171   int n = strlen (name);
172
173   s = strstr (line, name);
174   if (s && s >= skip_options (line))
175     return NULL;
176   if (s && (s == line || spacep (s-1))
177       && s[n] && (spacep (s+n) || s[n] == '='))
178     {
179       s += n + 1;
180       s += strspn (s, " ");
181       if (*s && !spacep(s))
182         return s;
183     }
184   return NULL;
185 }