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