config_parser.cc 26.6 KB
Newer Older
Tomek Mrugalski's avatar
Tomek Mrugalski committed
1
// Copyright (C) 2012-2014 Internet Systems Consortium, Inc. ("ISC")
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
//
// Permission to use, copy, modify, and/or distribute this software for any
// purpose with or without fee is hereby granted, provided that the above
// copyright notice and this permission notice appear in all copies.
//
// THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES WITH
// REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
// AND FITNESS.  IN NO EVENT SHALL ISC BE LIABLE FOR ANY SPECIAL, DIRECT,
// INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
// LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE
// OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
// PERFORMANCE OF THIS SOFTWARE.

#include <config/ccsession.h>
#include <dhcp4/dhcp4_log.h>
17
18
19
#include <dhcp/libdhcp++.h>
#include <dhcp/option_definition.h>
#include <dhcpsrv/cfgmgr.h>
20
#include <dhcp4/config_parser.h>
21
#include <dhcpsrv/dbaccess_parser.h>
22
#include <dhcpsrv/dhcp_parsers.h>
23
#include <dhcpsrv/option_space_container.h>
24
#include <util/encode/hex.h>
25
#include <util/strutil.h>
26

Tomek Mrugalski's avatar
Tomek Mrugalski committed
27
28
29
#include <boost/foreach.hpp>
#include <boost/lexical_cast.hpp>
#include <boost/algorithm/string.hpp>
30

Tomek Mrugalski's avatar
Tomek Mrugalski committed
31
#include <limits>
Tomek Mrugalski's avatar
Tomek Mrugalski committed
32
33
34
35
#include <iostream>
#include <vector>
#include <map>

36
using namespace std;
37
38
using namespace isc;
using namespace isc::dhcp;
39
40
41
using namespace isc::data;
using namespace isc::asiolink;

42
43
namespace {

44
/// @brief Parser for DHCP4 option data value.
45
///
46
47
48
49
/// This parser parses configuration entries that specify value of
/// a single option specific to DHCP4.  It provides the DHCP4-specific
/// implementation of the abstract class OptionDataParser.
class Dhcp4OptionDataParser : public OptionDataParser {
50
public:
51
    /// @brief Constructor.
52
    ///
53
54
55
    /// @param dummy first param, option names are always "Dhcp4/option-data[n]"
    /// @param options is the option storage in which to store the parsed option
    /// upon "commit".
56
    /// @param global_context is a pointer to the global context which
57
    /// stores global scope parameters, options, option defintions.
58
59
    Dhcp4OptionDataParser(const std::string&,
        OptionStoragePtr options, ParserContextPtr global_context)
60
        :OptionDataParser("", options, global_context) {
61
62
    }

63
    /// @brief static factory method for instantiating Dhcp4OptionDataParsers
64
    ///
65
    /// @param param_name name of the parameter to be parsed.
66
    /// @param options storage where the parameter value is to be stored.
67
    /// @param global_context is a pointer to the global context which
68
69
70
71
72
73
    /// stores global scope parameters, options, option defintions.
    /// @return returns a pointer to a new OptionDataParser. Caller is
    /// is responsible for deleting it when it is no longer needed.
    static OptionDataParser* factory(const std::string& param_name,
        OptionStoragePtr options, ParserContextPtr global_context) {
        return (new Dhcp4OptionDataParser(param_name, options, global_context));
74
75
    }

76
77
protected:
    /// @brief Finds an option definition within the server's option space
78
79
    ///
    /// Given an option space and an option code, find the correpsonding
80
81
    /// option defintion within the server's option defintion storage.
    ///
82
83
84
    /// @param option_space name of the parameter option space
    /// @param option_code numeric value of the parameter to find
    /// @return OptionDefintionPtr of the option defintion or an
85
    /// empty OptionDefinitionPtr if not found.
86
87
    /// @throw DhcpConfigError if the option space requested is not valid
    /// for this server.
88
89
90
91
92
93
94
95
96
    virtual OptionDefinitionPtr findServerSpaceOptionDefinition (
                std::string& option_space, uint32_t option_code) {
        OptionDefinitionPtr def;
        if (option_space == "dhcp4" &&
            LibDHCP::isStandardOption(Option::V4, option_code)) {
            def = LibDHCP::getOptionDef(Option::V4, option_code);
        } else if (option_space == "dhcp6") {
            isc_throw(DhcpConfigError, "'dhcp6' option space name is reserved"
                     << " for DHCPv6 server");
Tomek Mrugalski's avatar
Tomek Mrugalski committed
97
98
99
100
101
102
103
        } else {
            // Check if this is a vendor-option. If it is, get vendor-specific
            // definition.
            uint32_t vendor_id = SubnetConfigParser::optionSpaceToVendorId(option_space);
            if (vendor_id) {
                def = LibDHCP::getVendorOptionDef(Option::V4, vendor_id, option_code);
            }
104
105
        }

106
        return (def);
107
108
109
    }
};

110
/// @brief Parser for IPv4 pool definitions.
111
///
112
113
114
/// This is the IPv4 derivation of the PoolParser class and handles pool
/// definitions, i.e. a list of entries of one of two syntaxes: min-max and
/// prefix/len for IPv4 pools. Pool4 objects are created and stored in chosen
115
/// PoolStorage container.
116
///
117
118
/// It is useful for parsing Dhcp4/subnet4[X]/pool parameters.
class Pool4Parser : public PoolParser {
119
120
public:

121
122
123
124
125
126
127
128
    /// @brief Constructor.
    ///
    /// @param param_name name of the parameter. Note, it is passed through
    /// but unused, parameter is currently always "Dhcp4/subnet4[X]/pool"
    /// @param pools storage container in which to store the parsed pool
    /// upon "commit"
    Pool4Parser(const std::string& param_name,  PoolStoragePtr pools)
        :PoolParser(param_name, pools) {
129
130
    }

131
132
protected:
    /// @brief Creates a Pool4 object given a IPv4 prefix and the prefix length.
133
    ///
134
135
    /// @param addr is the IPv4 prefix of the pool.
    /// @param len is the prefix length.
136
    /// @param ignored dummy parameter to provide symmetry between the
137
    /// PoolParser derivations. The V6 derivation requires a third value.
138
    /// @return returns a PoolPtr to the new Pool4 object.
139
    PoolPtr poolMaker (IOAddress &addr, uint32_t len, int32_t) {
140
141
        return (PoolPtr(new Pool4(addr, len)));
    }
142

143
144
145
146
    /// @brief Creates a Pool4 object given starting and ending IPv4 addresses.
    ///
    /// @param min is the first IPv4 address in the pool.
    /// @param max is the last IPv4 address in the pool.
147
    /// @param ignored dummy parameter to provide symmetry between the
148
    /// PoolParser derivations. The V6 derivation requires a third value.
149
    /// @return returns a PoolPtr to the new Pool4 object.
150
    PoolPtr poolMaker (IOAddress &min, IOAddress &max, int32_t) {
151
        return (PoolPtr(new Pool4(min, max)));
152
    }
153
};
154

155
156
/// @brief This class parses a single IPv4 subnet.
///
157
158
/// This is the IPv4 derivation of the SubnetConfigParser class and it parses
/// the whole subnet definition. It creates parsersfor received configuration
159
160
161
162
/// parameters as needed.
class Subnet4ConfigParser : public SubnetConfigParser {
public:
    /// @brief Constructor
163
    ///
164
165
    /// @param ignored first parameter
    /// stores global scope parameters, options, option defintions.
166
    Subnet4ConfigParser(const std::string&)
167
        :SubnetConfigParser("", globalContext(), IOAddress("0.0.0.0")) {
168
    }
169

170
171
172
173
174
175
176
    /// @brief Parses a single IPv4 subnet configuration and adds to the
    /// Configuration Manager.
    ///
    /// @param subnet A new subnet being configured.
    void build(ConstElementPtr subnet) {
        SubnetConfigParser::build(subnet);

177
        if (subnet_) {
178
179
180
            Subnet4Ptr sub4ptr = boost::dynamic_pointer_cast<Subnet4>(subnet_);
            if (!sub4ptr) {
                // If we hit this, it is a programming error.
181
                isc_throw(Unexpected,
182
183
184
                          "Invalid cast in Subnet4ConfigParser::commit");
            }

185
186
            // Set relay information if it was parsed
            if (relay_info_) {
187
                sub4ptr->setRelayInfo(*relay_info_);
188
189
            }

190
191
192
193
194
195
196
197
198
            // Adding a subnet to the Configuration Manager may fail if the
            // subnet id is invalid (duplicate). Thus, we catch exceptions
            // here to append a position in the configuration string.
            try {
                isc::dhcp::CfgMgr::instance().addSubnet4(sub4ptr);
            } catch (const std::exception& ex) {
                isc_throw(DhcpConfigError, ex.what() << " ("
                          << subnet->getPosition() << ")");
            }
199
200
201
        }
    }

202
203
204
205
206
207
    /// @brief Commits subnet configuration.
    ///
    /// This function is currently no-op because subnet should already
    /// be added into the Config Manager in the build().
    void commit() { }

208
protected:
209

210
    /// @brief Creates parsers for entries in subnet definition.
211
    ///
212
213
214
215
216
217
218
219
220
221
    /// @param config_id name of the entry
    ///
    /// @return parser object for specified entry name. Note the caller is
    /// responsible for deleting the parser created.
    /// @throw isc::dhcp::DhcpConfigError if trying to create a parser
    /// for unknown config element
    DhcpConfigParser* createSubnetConfigParser(const std::string& config_id) {
        DhcpConfigParser* parser = NULL;
        if ((config_id.compare("valid-lifetime") == 0)  ||
            (config_id.compare("renew-timer") == 0)  ||
222
223
            (config_id.compare("rebind-timer") == 0) ||
            (config_id.compare("id") == 0)) {
224
            parser = new Uint32Parser(config_id, uint32_values_);
225
        } else if ((config_id.compare("subnet") == 0) ||
226
                   (config_id.compare("interface") == 0) ||
227
                   (config_id.compare("client-class") == 0) ||
228
                   (config_id.compare("next-server") == 0)) {
229
230
231
            parser = new StringParser(config_id, string_values_);
        } else if (config_id.compare("pool") == 0) {
            parser = new Pool4Parser(config_id, pools_);
232
        } else if (config_id.compare("relay") == 0) {
Tomek Mrugalski's avatar
Tomek Mrugalski committed
233
            parser = new RelayInfoParser(config_id, relay_info_, Option::V4);
234
        } else if (config_id.compare("option-data") == 0) {
235
           parser = new OptionDataListParser(config_id, options_,
236
237
                                             global_context_,
                                             Dhcp4OptionDataParser::factory);
238
        } else {
239
            isc_throw(NotImplemented, "unsupported parameter: " << config_id);
240
241
        }

242
        return (parser);
243
244
    }

245
    /// @brief Determines if the given option space name and code describe
246
    /// a standard option for the DCHP4 server.
247
    ///
248
249
250
251
252
253
254
255
    /// @param option_space is the name of the option space to consider
    /// @param code is the numeric option code to consider
    /// @return returns true if the space and code are part of the server's
    /// standard options.
    bool isServerStdOption(std::string option_space, uint32_t code) {
        return ((option_space.compare("dhcp4") == 0)
                && LibDHCP::isStandardOption(Option::V4, code));
    }
256

257
258
259
260
261
262
263
    /// @brief Returns the option definition for a given option code from
    /// the DHCP4 server's standard set of options.
    /// @param code is the numeric option code of the desired option definition.
    /// @return returns a pointer the option definition
    OptionDefinitionPtr getServerStdOptionDefinition (uint32_t code) {
        return (LibDHCP::getOptionDef(Option::V4, code));
    }
264

265
    /// @brief Issues a DHCP4 server specific warning regarding duplicate subnet
266
267
    /// options.
    ///
268
    /// @param code is the numeric option code of the duplicate option
269
    /// @param addr is the subnet address
270
    /// @todo a means to know the correct logger and perhaps a common
271
    /// message would allow this method to be emitted by the base class.
272
273
274
275
276
    virtual void duplicate_option_warning(uint32_t code,
                                         isc::asiolink::IOAddress& addr) {
        LOG_WARN(dhcp4_logger, DHCP4_CONFIG_OPTION_DUPLICATE)
            .arg(code).arg(addr.toText());
    }
277

278
    /// @brief Instantiates the IPv4 Subnet based on a given IPv4 address
279
280
    /// and prefix length.
    ///
281
    /// @param addr is IPv4 address of the subnet.
282
    /// @param len is the prefix length
283
    void initSubnet(isc::asiolink::IOAddress addr, uint8_t len) {
284
285
286
287
288
        // Get all 'time' parameters using inheritance.
        // If the subnet-specific value is defined then use it, else
        // use the global value. The global value must always be
        // present. If it is not, it is an internal error and exception
        // is thrown.
289
290
291
        Triplet<uint32_t> t1 = getParam("renew-timer");
        Triplet<uint32_t> t2 = getParam("rebind-timer");
        Triplet<uint32_t> valid = getParam("valid-lifetime");
292
293
294
295
        // Subnet ID is optional. If it is not supplied the value of 0 is used,
        // which means autogenerate.
        SubnetID subnet_id =
            static_cast<SubnetID>(uint32_values_->getOptionalParam("id", 0));
296
297

        stringstream tmp;
298
        tmp << addr << "/" << (int)len
299
300
301
302
            << " with params t1=" << t1 << ", t2=" << t2 << ", valid=" << valid;

        LOG_INFO(dhcp4_logger, DHCP4_CONFIG_NEW_SUBNET).arg(tmp.str());

303
        Subnet4Ptr subnet4(new Subnet4(addr, len, t1, t2, valid, subnet_id));
Tomek Mrugalski's avatar
Tomek Mrugalski committed
304
        subnet_ = subnet4;
305
306
307
308

        // Try global value first
        try {
            string next_server = globalContext()->string_values_->getParam("next-server");
Tomek Mrugalski's avatar
Tomek Mrugalski committed
309
310
311
            if (!next_server.empty()) {
                subnet4->setSiaddr(IOAddress(next_server));
            }
312
        } catch (const DhcpConfigError&) {
Tomek Mrugalski's avatar
Tomek Mrugalski committed
313
            // Don't care. next_server is optional. We can live without it
314
315
316
317
        } catch (...) {
            isc_throw(DhcpConfigError, "invalid parameter next-server ("
                      << globalContext()->string_values_->getPosition("next-server")
                      << ")");
318
319
320
321
322
        }

        // Try subnet specific value if it's available
        try {
            string next_server = string_values_->getParam("next-server");
Tomek Mrugalski's avatar
Tomek Mrugalski committed
323
324
325
            if (!next_server.empty()) {
                subnet4->setSiaddr(IOAddress(next_server));
            }
326
        } catch (const DhcpConfigError&) {
Tomek Mrugalski's avatar
Tomek Mrugalski committed
327
            // Don't care. next_server is optional. We can live without it
328
329
330
331
        } catch (...) {
            isc_throw(DhcpConfigError, "invalid parameter next-server ("
                      << string_values_->getPosition("next-server")
                      << ")");
332
        }
333

334

335
336
337
338
339
340
341
        // Try setting up client class (if specified)
        try {
            string client_class = string_values_->getParam("client-class");
            subnet4->allowClientClass(client_class);
        } catch (const DhcpConfigError&) {
            // That's ok if it fails. client-class is optional.
        }
342
343
344
    }
};

345
/// @brief this class parses list of DHCP4 subnets
346
347
348
349
///
/// This is a wrapper parser that handles the whole list of Subnet4
/// definitions. It iterates over all entries and creates Subnet4ConfigParser
/// for each entry.
350
class Subnets4ListConfigParser : public DhcpConfigParser {
351
352
353
354
public:

    /// @brief constructor
    ///
355
    /// @param dummy first argument, always ignored. All parsers accept a
356
    /// string parameter "name" as their first argument.
357
358
359
360
361
362
363
364
365
366
    Subnets4ListConfigParser(const std::string&) {
    }

    /// @brief parses contents of the list
    ///
    /// Iterates over all entries on the list and creates Subnet4ConfigParser
    /// for each entry.
    ///
    /// @param subnets_list pointer to a list of IPv4 subnets
    void build(ConstElementPtr subnets_list) {
367
368
369
370
371
372
        // @todo: Implement more subtle reconfiguration than toss
        // the old one and replace with the new one.

        // remove old subnets
        CfgMgr::instance().deleteSubnets4();

373
        BOOST_FOREACH(ConstElementPtr subnet, subnets_list->listValue()) {
374
            ParserPtr parser(new Subnet4ConfigParser("subnet"));
375
376
377
378
379
380
381
            parser->build(subnet);
            subnets_.push_back(parser);
        }
    }

    /// @brief commits subnets definitions.
    ///
382
383
    /// Iterates over all Subnet4 parsers. Each parser contains definitions of
    /// a single subnet and its parameters and commits each subnet separately.
384
385
386
387
388
389
390
391
392
393
    void commit() {
        BOOST_FOREACH(ParserPtr subnet, subnets_) {
            subnet->commit();
        }

    }

    /// @brief Returns Subnet4ListConfigParser object
    /// @param param_name name of the parameter
    /// @return Subnets4ListConfigParser object
394
    static DhcpConfigParser* factory(const std::string& param_name) {
395
396
397
398
399
        return (new Subnets4ListConfigParser(param_name));
    }

    /// @brief collection of subnet parsers.
    ParserCollection subnets_;
400

401
402
};

403
404
405
406
407
} // anonymous namespace

namespace isc {
namespace dhcp {

408
409
410
411
412
413
414
/// @brief creates global parsers
///
/// This method creates global parsers that parse global parameters, i.e.
/// those that take format of Dhcp4/param1, Dhcp4/param2 and so forth.
///
/// @param config_id pointer to received global configuration entry
/// @return parser for specified global DHCPv4 parameter
415
/// @throw NotImplemented if trying to create a parser for unknown
416
/// config element
417
418
    DhcpConfigParser* createGlobalDhcp4ConfigParser(const std::string& config_id,
                                                    ConstElementPtr element) {
419
    DhcpConfigParser* parser = NULL;
420
421
422
    if ((config_id.compare("valid-lifetime") == 0)  ||
        (config_id.compare("renew-timer") == 0)  ||
        (config_id.compare("rebind-timer") == 0))  {
423
        parser = new Uint32Parser(config_id,
424
                                 globalContext()->uint32_values_);
425
    } else if (config_id.compare("interfaces") == 0) {
426
        parser = new InterfaceListConfigParser(config_id);
427
    } else if (config_id.compare("subnet4") == 0) {
428
        parser = new Subnets4ListConfigParser(config_id);
429
    } else if (config_id.compare("option-data") == 0) {
430
431
        parser = new OptionDataListParser(config_id,
                                          globalContext()->options_,
432
                                          globalContext(),
433
                                          Dhcp4OptionDataParser::factory);
434
    } else if (config_id.compare("option-def") == 0) {
435
        parser  = new OptionDefListParser(config_id, globalContext());
436
437
    } else if ((config_id.compare("version") == 0) ||
               (config_id.compare("next-server") == 0)) {
438
        parser  = new StringParser(config_id,
439
                                    globalContext()->string_values_);
440
    } else if (config_id.compare("lease-database") == 0) {
441
        parser = new DbAccessParser(config_id, *globalContext());
442
443
    } else if (config_id.compare("hooks-libraries") == 0) {
        parser = new HooksLibrariesParser(config_id);
444
445
    } else if (config_id.compare("echo-client-id") == 0) {
        parser = new BooleanParser(config_id, globalContext()->boolean_values_);
446
447
    } else if (config_id.compare("dhcp-ddns") == 0) {
        parser = new D2ClientConfigParser(config_id);
448
    } else {
449
450
451
        isc_throw(DhcpConfigError,
                "unsupported global configuration parameter: "
                  << config_id << " (" << element->getPosition() << ")");
452
    }
453
454

    return (parser);
455
456
}

457
void commitGlobalOptions() {
Tomek Mrugalski's avatar
Tomek Mrugalski committed
458
459
    // Although the function is modest for now, it is certain that the number
    // of global switches will increase over time, hence the name.
460
461
462
463
464
465
466
467
468
469
470

    // Set whether v4 server is supposed to echo back client-id (yes = RFC6842
    // compatible, no = backward compatibility)
    try {
        bool echo_client_id = globalContext()->boolean_values_->getParam("echo-client-id");
        CfgMgr::instance().echoClientId(echo_client_id);
    } catch (...) {
        // Ignore errors. This flag is optional
    }
}

471
isc::data::ConstElementPtr
472
configureDhcp4Server(Dhcpv4Srv&, isc::data::ConstElementPtr config_set) {
473
    if (!config_set) {
Tomek Mrugalski's avatar
Tomek Mrugalski committed
474
475
476
        ConstElementPtr answer = isc::config::createAnswer(1,
                                 string("Can't parse NULL config"));
        return (answer);
477
478
    }

479
    /// @todo: Append most essential info here (like "2 new subnets configured")
480
481
    string config_details;

482
    LOG_DEBUG(dhcp4_logger, DBG_DHCP4_COMMAND,
483
              DHCP4_CONFIG_START).arg(config_set->str());
484

485
486
487
488
    // Before starting any subnet operations, let's reset the subnet-id counter,
    // so newly recreated configuration starts with first subnet-id equal 1.
    Subnet::resetSubnetID();

489
    // Some of the values specified in the configuration depend on
490
491
492
    // other values. Typically, the values in the subnet4 structure
    // depend on the global values. Also, option values configuration
    // must be performed after the option definitions configurations.
493
494
495
    // Thus we group parsers and will fire them in the right order:
    // all parsers other than subnet4 and option-data parser,
    // option-data parser, subnet4 parser.
496
    ParserCollection independent_parsers;
497
498
    ParserPtr subnet_parser;
    ParserPtr option_parser;
499
    ParserPtr iface_parser;
500

501
502
    // Some of the parsers alter the state of the system in a way that can't
    // easily be undone. (Or alter it in a way such that undoing the change has
503
    // the same risk of failure as doing the change.)
504
    ParserPtr hooks_parser;
505

506
    // The subnet parsers implement data inheritance by directly
507
    // accessing global storage. For this reason the global data
508
    // parsers must store the parsed data into global storages
509
510
511
512
    // immediately. This may cause data inconsistency if the
    // parsing operation fails after the global storage has been
    // modified. We need to preserve the original global data here
    // so as we can rollback changes when an error occurs.
513
    ParserContext original_context(*globalContext());
514

515
    // Answer will hold the result.
516
    ConstElementPtr answer;
517
    // Rollback informs whether error occured and original data
518
519
    // have to be restored to global storages.
    bool rollback = false;
520
    // config_pair holds the details of the current parser when iterating over
521
522
    // the parsers.  It is declared outside the loops so in case of an error,
    // the name of the failing parser can be retrieved in the "catch" clause.
523
    ConfigPair config_pair;
524
    try {
525
        // Make parsers grouping.
526
        const std::map<std::string, ConstElementPtr>& values_map =
527
                                                        config_set->mapValue();
528
        BOOST_FOREACH(config_pair, values_map) {
529
530
            ParserPtr parser(createGlobalDhcp4ConfigParser(config_pair.first,
                                                           config_pair.second));
531
532
            LOG_DEBUG(dhcp4_logger, DBG_DHCP4_DETAIL, DHCP4_PARSER_CREATED)
                      .arg(config_pair.first);
533
534
535
536
            if (config_pair.first == "subnet4") {
                subnet_parser = parser;
            } else if (config_pair.first == "option-data") {
                option_parser = parser;
537
538
539
540
541
            } else if (config_pair.first == "interfaces") {
                // The interface parser is independent from any other
                // parser and can be run here before any other parsers.
                iface_parser = parser;
                parser->build(config_pair.second);
542
543
544
545
            } else if (config_pair.first == "hooks-libraries") {
                // Executing commit will alter currently-loaded hooks
                // libraries.  Check if the supplied libraries are valid,
                // but defer the commit until everything else has committed.
546
                hooks_parser = parser;
547
                parser->build(config_pair.second);
548
            } else {
549
550
                // Those parsers should be started before other
                // parsers so we can call build straight away.
551
552
553
554
555
556
557
558
                independent_parsers.push_back(parser);
                parser->build(config_pair.second);
                // The commit operation here may modify the global storage
                // but we need it so as the subnet6 parser can access the
                // parsed data.
                parser->commit();
            }
        }
559

560
        // The option values parser is the next one to be run.
561
562
563
        std::map<std::string, ConstElementPtr>::const_iterator option_config =
            values_map.find("option-data");
        if (option_config != values_map.end()) {
564
            config_pair.first = "option-data";
565
566
567
568
            option_parser->build(option_config->second);
            option_parser->commit();
        }

569
        // The subnet parser is the last one to be run.
570
571
572
        std::map<std::string, ConstElementPtr>::const_iterator subnet_config =
            values_map.find("subnet4");
        if (subnet_config != values_map.end()) {
573
            config_pair.first = "subnet4";
574
            subnet_parser->build(subnet_config->second);
575
        }
576

577
    } catch (const isc::Exception& ex) {
578
        LOG_ERROR(dhcp4_logger, DHCP4_PARSER_FAIL)
579
580
581
                  .arg(config_pair.first).arg(ex.what());
        answer = isc::config::createAnswer(1,
                     string("Configuration parsing failed: ") + ex.what());
582
583
584
585

        // An error occured, so make sure that we restore original data.
        rollback = true;

586
    } catch (...) {
587
        // For things like bad_cast in boost::lexical_cast
588
        LOG_ERROR(dhcp4_logger, DHCP4_PARSER_EXCEPTION).arg(config_pair.first);
589
590
        answer = isc::config::createAnswer(1,
                     string("Configuration parsing failed"));
591
592
593

        // An error occured, so make sure that we restore original data.
        rollback = true;
594
595
    }

596
597
598
599
600
601
    // So far so good, there was no parsing error so let's commit the
    // configuration. This will add created subnets and option values into
    // the server's configuration.
    // This operation should be exception safe but let's make sure.
    if (!rollback) {
        try {
602
603
            if (subnet_parser) {
                subnet_parser->commit();
604
            }
605
606
607
608

            if (iface_parser) {
                iface_parser->commit();
            }
609

610
611
612
            // Apply global options
            commitGlobalOptions();

613
614
615
            // This occurs last as if it succeeds, there is no easy way
            // revert it.  As a result, the failure to commit a subsequent
            // change causes problems when trying to roll back.
616
617
            if (hooks_parser) {
                hooks_parser->commit();
618
            }
619
620
        }
        catch (const isc::Exception& ex) {
621
            LOG_ERROR(dhcp4_logger, DHCP4_PARSER_COMMIT_FAIL).arg(ex.what());
622
623
            answer = isc::config::createAnswer(2,
                         string("Configuration commit failed: ") + ex.what());
624
625
            rollback = true;
        } catch (...) {
626
            // For things like bad_cast in boost::lexical_cast
627
            LOG_ERROR(dhcp4_logger, DHCP4_PARSER_COMMIT_EXCEPTION);
628
629
            answer = isc::config::createAnswer(2,
                         string("Configuration commit failed"));
630
            rollback = true;
631
632
        }
    }
633
634
635

    // Rollback changes as the configuration parsing failed.
    if (rollback) {
636
        globalContext().reset(new ParserContext(original_context));
637
638
639
640
641
        return (answer);
    }

    LOG_INFO(dhcp4_logger, DHCP4_CONFIG_COMPLETE).arg(config_details);

642
    // Everything was fine. Configuration is successful.
Shane Kerr's avatar
Shane Kerr committed
643
    answer = isc::config::createAnswer(0, "Configuration committed.");
644
645
646
    return (answer);
}

647
ParserContextPtr& globalContext() {
648
649
    static ParserContextPtr global_context_ptr(new ParserContext(Option::V4));
    return (global_context_ptr);
650
651
}

652
653


654
655
}; // end of isc::dhcp namespace
}; // end of isc namespace