- Improved handling of errors on connection attempts.
[tinc] / src / conf.c
1 /*
2     conf.c -- configuration code
3     Copyright (C) 1998 Emphyrio,
4     Copyright (C) 1998,1999,2000 Ivo Timmermans <itimmermans@bigfoot.com>
5                             2000 Guus Sliepen <guus@sliepen.warande.net>
6                             2000 Cris van Pelt <tribbel@arise.dhs.org>
7
8     This program is free software; you can redistribute it and/or modify
9     it under the terms of the GNU General Public License as published by
10     the Free Software Foundation; either version 2 of the License, or
11     (at your option) any later version.
12
13     This program is distributed in the hope that it will be useful,
14     but WITHOUT ANY WARRANTY; without even the implied warranty of
15     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
16     GNU General Public License for more details.
17
18     You should have received a copy of the GNU General Public License
19     along with this program; if not, write to the Free Software
20     Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
21
22     $Id: conf.c,v 1.9.4.3 2000/06/27 20:10:47 guus Exp $
23 */
24
25
26 #include "config.h"
27
28 #include <ctype.h>
29 #include <errno.h>
30 #include <netdb.h>
31 #include <stdio.h>
32 #include <stdlib.h>
33 #include <string.h>
34
35 #include <xalloc.h>
36
37 #include "conf.h"
38 #include "netutl.h" /* for strtoip */
39
40 #include "system.h"
41
42 config_t *config;
43 int debug_lvl = 0;
44 int timeout = 0; /* seconds before timeout */
45
46 typedef struct internal_config_t {
47   char *name;
48   enum which_t which;
49   int argtype;
50 } internal_config_t;
51
52 /*
53   These are all the possible configurable values
54 */
55 static internal_config_t hazahaza[] = {
56   { "AllowConnect", allowconnect,   TYPE_BOOL },   /* Is not used anywhere. Remove? */
57   { "ConnectTo",    upstreamip,     TYPE_IP },
58   { "ConnectPort",  upstreamport,   TYPE_INT },
59   { "ListenPort",   listenport,     TYPE_INT },
60   { "MyOwnVPNIP",   myvpnip,        TYPE_IP },
61   { "MyVirtualIP",  myvpnip,        TYPE_IP },   /* an alias */
62   { "Passphrases",  passphrasesdir, TYPE_NAME },
63   { "PingTimeout",  pingtimeout,    TYPE_INT },
64   { "TapDevice",    tapdevice,      TYPE_NAME },
65   { "KeyExpire",    keyexpire,      TYPE_INT },
66   { "VpnMask",      vpnmask,        TYPE_IP },
67   { "Hostnames",    resolve_dns,    TYPE_BOOL },
68   { "IndirectData", indirectdata,   TYPE_BOOL },
69   { NULL, 0, 0 }
70 };
71
72 /*
73   Add given value to the list of configs cfg
74 */
75 config_t *
76 add_config_val(config_t **cfg, int argtype, char *val)
77 {
78   config_t *p, *r;
79   char *q;
80
81   p = (config_t*)xmalloc(sizeof(*p));
82   p->data.val = 0;
83   
84   switch(argtype)
85     {
86     case TYPE_INT:
87       p->data.val = strtol(val, &q, 0);
88       if(q && *q)
89         p->data.val = 0;
90       break;
91     case TYPE_NAME:
92       p->data.ptr = xmalloc(strlen(val) + 1);
93       strcpy(p->data.ptr, val);
94       break;
95     case TYPE_IP:
96       p->data.ip = strtoip(val);
97       break;
98     case TYPE_BOOL:
99       if(!strcasecmp("yes", val))
100         p->data.val = stupid_true;
101       else if(!strcasecmp("no", val))
102         p->data.val = stupid_false;
103       else
104         p->data.val = 0;
105     }
106
107   if(p->data.val)
108     {
109       if(*cfg)
110         {
111           r = *cfg;
112           while(r->next)
113             r = r->next;
114           r->next = p;
115         }
116       else
117         *cfg = p;
118       p->next = NULL;
119       return p;
120     }
121
122   free(p);
123   return NULL;
124 }
125
126 /*
127   Get variable from a section in a configfile. returns -1 on failure.
128 */
129 int
130 readconfig(const char *fname, FILE *fp)
131 {
132   char *line, *temp_buf;
133   char *p, *q;
134   int i, lineno = 0;
135   config_t *cfg;
136
137   line = (char *)xmalloc(80 * sizeof(char));
138   temp_buf = (char *)xmalloc(80 * sizeof(char));
139         
140   for(;;)
141     {
142       if(fgets(line, 80, fp) == NULL)
143         return 0;
144
145       while(!index(line, '\n'))
146         {
147           fgets(temp_buf, (strlen(line)+1) * 80, fp);
148           if(!temp_buf)
149             break;
150           strcat(line, temp_buf);
151           line = (char *)xrealloc(line, (strlen(line)+1) * sizeof(char));
152         }        
153       lineno++;
154
155       if((p = strtok(line, "\t\n\r =")) == NULL)
156         continue; /* no tokens on this line */
157
158       if(p[0] == '#')
159         continue; /* comment: ignore */
160
161       for(i = 0; hazahaza[i].name != NULL; i++)
162         if(!strcasecmp(hazahaza[i].name, p))
163           break;
164
165       if(!hazahaza[i].name)
166         {
167           fprintf(stderr, _("%s: %d: Invalid variable name `%s'.\n"),
168                   fname, lineno, p);
169           return -1;
170         }
171
172       if(((q = strtok(NULL, "\t\n\r =")) == NULL) || q[0] == '#')
173         {
174           fprintf(stderr, _("%s: %d: No value given for `%s'.\n"),
175                   fname, lineno, hazahaza[i].name);
176           return -1;
177         }
178
179       cfg = add_config_val(&config, hazahaza[i].argtype, q);
180       if(cfg == NULL)
181         {
182           fprintf(stderr, _("%s: %d: Invalid value `%s' for variable `%s'.\n"),
183                   fname, lineno, q, hazahaza[i].name);
184           return -1;
185         }
186
187       cfg->which = hazahaza[i].which;
188       if(!config)
189         config = cfg;
190     }
191 }
192
193 /*
194   wrapper function for readconfig
195 */
196 int
197 read_config_file(const char *fname)
198 {
199   FILE *fp;
200
201   if((fp = fopen (fname, "r")) == NULL)
202     {
203       fprintf(stderr, _("Could not open %s: %s\n"), fname, sys_errlist[errno]);
204       return 1;
205     }
206
207   if(readconfig(fname, fp))
208     return -1;
209
210   fclose (fp);
211
212   return 0;
213 }
214
215 /*
216   Look up the value of the config option type
217 */
218 const config_t *
219 get_config_val(which_t type)
220 {
221   config_t *p;
222
223   for(p = config; p != NULL; p = p->next)
224     if(p->which == type)
225       return p;
226
227   /* Not found */
228   return NULL;
229 }
230
231 /*
232   Support for multiple config lines.
233   Index is used to get a specific value, 0 being the first, 1 the second etc.
234 */
235 const config_t *
236 get_next_config_val(which_t type, int index)
237 {
238   config_t *p;
239   
240   for(p = config; p != NULL; p = p->next)
241     if(p->which == type)
242       if(--index < 0)
243         return p;
244   
245   /* Not found */
246   return NULL;
247 }