3 /*****************************************************************
5 ** @(#) ncparse.c -- A very simple named.conf parser
7 ** Copyright (c) Apr 2005 - Nov 2007, Holger Zuleger HZnet. All rights reserved.
9 ** This software is open source.
11 ** Redistribution and use in source and binary forms, with or without
12 ** modification, are permitted provided that the following conditions
15 ** Redistributions of source code must retain the above copyright notice,
16 ** this list of conditions and the following disclaimer.
18 ** Redistributions in binary form must reproduce the above copyright notice,
19 ** this list of conditions and the following disclaimer in the documentation
20 ** and/or other materials provided with the distribution.
22 ** Neither the name of Holger Zuleger HZnet nor the names of its contributors may
23 ** be used to endorse or promote products derived from this software without
24 ** specific prior written permission.
26 ** THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
27 ** "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
28 ** TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
29 ** PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE
30 ** LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
31 ** CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
32 ** SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
33 ** INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
34 ** CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
35 ** ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
36 ** POSSIBILITY OF SUCH DAMAGE.
38 *****************************************************************/
50 # define TOK_STRING 257
52 # define TOK_INCLUDE 259
56 # define TOK_MASTER 262
57 # define TOK_SLAVE 263
60 # define TOK_FORWARD 266
61 # define TOK_DELEGATION 267
66 # define TOK_UNKNOWN 511
68 /* list of "named.conf" keywords we are interested in */
69 static struct KeyWords
{
73 { "STRING", TOK_STRING
},
74 { "include", TOK_INCLUDE
},
75 { "directory", TOK_DIR
},
78 #if 0 /* we don't need the type keyword; master, slave etc. is sufficient */
81 { "master", TOK_MASTER
},
82 { "slave", TOK_SLAVE
},
85 { "forward", TOK_FORWARD
},
86 { "delegation-only", TOK_DELEGATION
},
88 { NULL
, TOK_UNKNOWN
},
92 static const char *tok2str (int tok
)
97 while ( kw
[i
].name
&& kw
[i
].tok
!= tok
)
104 static int searchkw (const char *keyword
)
108 dbg_val ("ncparse: searchkw (%s)\n", keyword
);
110 while ( kw
[i
].name
&& strcmp (kw
[i
].name
, keyword
) != 0 )
116 static int gettok (FILE *fp
, char *val
, size_t valsize
)
126 while ( (c
= getc (fp
)) != EOF
&& isspace (c
) )
129 if ( c
== '#' ) /* single line comment ? */
131 while ( (c
= getc (fp
)) != EOF
&& c
!= '\n' )
139 if ( c
== '{' || c
== '}' || c
== ';' )
142 if ( c
== '/' ) /* begin of C comment ? */
144 if ( (c
= getc (fp
)) == '*' ) /* yes! */
146 lastc
= EOF
; /* read until end of c comment */
147 while ( (c
= getc (fp
)) != EOF
&& !(lastc
== '*' && c
== '/') )
150 else if ( c
== '/' ) /* is it a C single line comment ? */
152 while ( (c
= getc (fp
)) != EOF
&& c
!= '\n' )
163 bufend
= val
+ valsize
- 1;
164 while ( (c
= getc (fp
)) != EOF
&& p
< bufend
&& c
!= '\"' )
167 /* if string buffer is too small, eat up rest of string */
168 while ( c
!= EOF
&& c
!= '\"' )
175 bufend
= buf
+ sizeof (buf
) - 1;
178 while ( (c
= getc (fp
)) != EOF
&& p
< bufend
&& isalpha (c
) );
182 if ( (c
= searchkw (buf
)) != TOK_UNKNOWN
)
184 } while ( c
!= EOF
);
189 /*****************************************************************
191 ** parse_namedconf (const char *filename, chroot_dir, dir, dirsize, int (*func) ())
193 ** Very dumb named.conf parser.
194 ** - In a zone declaration the _first_ keyword MUST be "type"
195 ** - For every master zone "func (directory, zone, filename)" will be called
197 *****************************************************************/
198 int parse_namedconf (const char *filename
, const char *chroot_dir
, char *dir
, size_t dirsize
, int (*func
) ())
203 #if 1 /* this is potentialy too small for key data, but we don't need the keys... */
210 char zonefile
[255+1];
212 dbg_val ("parse_namedconf: parsing file \"%s\" \n", filename
);
214 assert (filename
!= NULL
);
215 assert (dir
!= NULL
&& dirsize
!= 0);
216 assert (func
!= NULL
);
219 if ( (fp
= fopen (filename
, "r")) == NULL
)
222 while ( (tok
= gettok (fp
, strval
, sizeof strval
)) != EOF
)
224 if ( tok
> 0 && tok
< 256 )
226 error ("parse_namedconf: token found with value %-10d: %c\n", tok
, tok
);
227 lg_mesg (LG_ERROR
, "parse_namedconf: token found with value %-10d: %c", tok
, tok
);
229 else if ( tok
== TOK_DIR
)
231 if ( gettok (fp
, strval
, sizeof (strval
)) == TOK_STRING
)
233 dbg_val2 ("parse_namedconf: directory found \"%s\" (dir is %s)\n",
235 if ( *strval
!= '/' && *dir
)
236 snprintf (path
, sizeof (path
), "%s/%s", dir
, strval
);
238 snprintf (path
, sizeof (path
), "%s", strval
);
240 /* prepend chroot directory (do it only once) */
241 if ( chroot_dir
&& *chroot_dir
)
243 snprintf (dir
, dirsize
, "%s%s%s", chroot_dir
, *path
== '/' ? "": "/", path
);
247 snprintf (dir
, dirsize
, "%s", path
);
248 dbg_val ("parse_namedconf: new dir \"%s\" \n", dir
);
251 else if ( tok
== TOK_INCLUDE
)
253 if ( gettok (fp
, strval
, sizeof (strval
)) == TOK_STRING
)
255 if ( *strval
!= '/' && *dir
)
256 snprintf (path
, sizeof (path
), "%s/%s", dir
, strval
);
258 snprintf (path
, sizeof (path
), "%s", strval
);
259 if ( !parse_namedconf (path
, chroot_dir
, dir
, dirsize
, func
) )
264 error ("parse_namedconf: need a filename after \"include\"!\n");
265 lg_mesg (LG_ERROR
, "parse_namedconf: need a filename after \"include\"!");
268 else if ( tok
== TOK_VIEW
)
270 if ( gettok (fp
, strval
, sizeof (strval
)) != TOK_STRING
)
272 snprintf (view
, sizeof view
, "%s", strval
); /* store the name of the view */
274 else if ( tok
== TOK_ZONE
)
276 if ( gettok (fp
, strval
, sizeof (strval
)) != TOK_STRING
)
278 snprintf (zone
, sizeof zone
, "%s", strval
); /* store the name of the zone */
280 if ( gettok (fp
, strval
, sizeof (strval
)) != TOK_MASTER
)
282 if ( gettok (fp
, strval
, sizeof (strval
)) != TOK_FILE
)
284 if ( gettok (fp
, strval
, sizeof (strval
)) != TOK_STRING
)
286 snprintf (zonefile
, sizeof zonefile
, "%s", strval
); /* this is the filename */
288 dbg_val4 ("dir %s view %s zone %s file %s\n", dir
, view
, zone
, zonefile
);
289 (*func
) (dir
, view
, zone
, zonefile
);
292 dbg_val3 ("%-10s(%d): %s\n", tok2str(tok
), tok
, strval
);
300 int printzone (const char *dir
, const char *view
, const char *zone
, const char *file
)
302 printf ("printzone ");
303 printf ("view \"%s\" " , view
);
304 printf ("zone \"%s\" " , zone
);
307 printf ("%s/", dir
, file
);
315 main (int argc
, char *argv
[])
317 char directory
[255+1];
323 parse_namedconf ("/var/named/named.conf", NULL
, directory
, sizeof (directory
), printzone
);
325 parse_namedconf (argv
[1], NULL
, directory
, sizeof (directory
), printzone
);