Kea 2.7.5
d2/parser_context.cc
Go to the documentation of this file.
1// Copyright (C) 2017-2024 Internet Systems Consortium, Inc. ("ISC")
2//
3// This Source Code Form is subject to the terms of the Mozilla Public
4// License, v. 2.0. If a copy of the MPL was not distributed with this
5// file, You can obtain one at http://mozilla.org/MPL/2.0/.
6
7#include <config.h>
8
9#include <d2/d2_parser.h>
10#include <d2/parser_context.h>
11#include <d2srv/d2_log.h>
13#include <cc/data.h>
14#include <boost/lexical_cast.hpp>
15#include <fstream>
16#include <sstream>
17#include <limits>
18
19namespace isc {
20namespace d2 {
21
23 : sfile_(0), ctx_(NO_KEYWORD), trace_scanning_(false), trace_parsing_(false)
24{
25}
26
30
32D2ParserContext::parseString(const std::string& str, ParserType parser_type)
33{
34 scanStringBegin(str, parser_type);
35 return (parseCommon());
36}
37
39D2ParserContext::parseFile(const std::string& filename, ParserType parser_type) {
40 FILE* f = fopen(filename.c_str(), "r");
41 if (!f) {
42 isc_throw(D2ParseError, "Unable to open file " << filename);
43 }
44 scanFileBegin(f, filename, parser_type);
45 return (parseCommon());
46}
47
49D2ParserContext::parseCommon() {
50 isc::d2::D2Parser parser(*this);
51 // Uncomment this to get detailed parser logs.
52 // trace_parsing_ = true;
53 parser.set_debug_level(trace_parsing_);
54 try {
55 int res = parser.parse();
56 if (res != 0) {
57 isc_throw(D2ParseError, "Parser abort");
58 }
59 scanEnd();
60 } catch (...) {
61 scanEnd();
62 throw;
63 }
64 if (stack_.size() == 1) {
65 return (stack_[0]);
66 } else {
67 isc_throw(D2ParseError, "Expected exactly one terminal Element, found "
68 << stack_.size());
69 }
70}
71
72void
73D2ParserContext::error(const isc::d2::location& loc,
74 const std::string& what,
75 size_t pos)
76{
77 if (pos == 0) {
78 isc_throw(D2ParseError, loc << ": " << what);
79 } else {
80 isc_throw(D2ParseError, loc << " (near " << pos << "): " << what);
81 }
82}
83
84void
85D2ParserContext::error (const std::string& what)
86{
88}
89
90void
91D2ParserContext::fatal (const std::string& what)
92{
94}
95
97D2ParserContext::loc2pos(isc::d2::location& loc)
98{
99 const std::string& file = *loc.begin.filename;
100 const uint32_t line = loc.begin.line;
101 const uint32_t pos = loc.begin.column;
102 return (isc::data::Element::Position(file, line, pos));
103}
104
105void
106D2ParserContext::require(const std::string& name,
109{
110 ConstElementPtr value = stack_.back()->get(name);
111 if (!value) {
113 "missing parameter '" << name << "' ("
114 << stack_.back()->getPosition() << ") ["
115 << contextName() << " map between "
116 << open_loc << " and " << close_loc << "]");
117 }
118}
119
120void
121D2ParserContext::unique(const std::string& name,
123{
124 ConstElementPtr value = stack_.back()->get(name);
125 if (value) {
126 if (ctx_ != NO_KEYWORD) {
127 isc_throw(D2ParseError, loc << ": duplicate " << name
128 << " entries in " << contextName()
129 << " map (previous at " << value->getPosition() << ")");
130 } else {
131 isc_throw(D2ParseError, loc << ": duplicate " << name
132 << " entries in JSON"
133 << " map (previous at " << value->getPosition() << ")");
134 }
135 }
136}
137
138void
140{
141 cstack_.push_back(ctx_);
142 ctx_ = ctx;
143}
144
145void
147{
148 if (cstack_.empty()) {
149 fatal("unbalanced syntactic context");
150 }
151
152 ctx_ = cstack_.back();
153 cstack_.pop_back();
154}
155
156const std::string
158{
159 switch (ctx_) {
160 case NO_KEYWORD:
161 return ("__no keyword__");
162 case CONFIG:
163 return ("toplevel");
164 case DHCPDDNS:
165 return ("DhcpDdns");
166 case TSIG_KEY:
167 return ("tsig-key");
168 case TSIG_KEYS:
169 return ("tsig-keys");
170 case ALGORITHM:
171 return ("algorithm");
172 case DIGEST_BITS:
173 return ("digest-bits");
174 case SECRET:
175 return ("secret");
176 case FORWARD_DDNS:
177 return ("forward-ddns");
178 case REVERSE_DDNS:
179 return ("reverse-ddns");
180 case DDNS_DOMAIN:
181 return ("ddns-domain");
182 case DDNS_DOMAINS:
183 return ("ddns-domains");
184 case DNS_SERVER:
185 return ("dns-server");
186 case DNS_SERVERS:
187 return ("dns-servers");
188 case CONTROL_SOCKET:
189 return ("control-socket");
191 return ("control-socket-type");
192 case AUTHENTICATION:
193 return ("authentication");
194 case AUTH_TYPE:
195 return ("auth-type");
196 case CLIENTS:
197 return ("clients");
198 case LOGGERS:
199 return ("loggers");
200 case OUTPUT_OPTIONS:
201 return ("output-options");
202 case NCR_PROTOCOL:
203 return ("ncr-protocol");
204 case NCR_FORMAT:
205 return ("ncr-format");
206 case HOOKS_LIBRARIES:
207 return ("hooks-libraries");
208 default:
209 return ("__unknown__");
210 }
211}
212
213void
214D2ParserContext::warning(const isc::d2::location& loc,
215 const std::string& what) {
216 std::ostringstream msg;
217 msg << loc << ": " << what;
219 .arg(msg.str());
220}
221
222void
223D2ParserContext::warnAboutExtraCommas(const isc::d2::location& loc) {
224 warning(loc, "Extraneous comma. A piece of configuration may have been omitted.");
225}
226
227} // namespace d2
228} // namespace isc
Evaluation error exception raised when trying to parse.
ParserType
Defines currently supported scopes.
isc::data::ElementPtr parseString(const std::string &str, ParserType parser_type)
Run the parser on the string specified.
D2ParserContext()
Default constructor.
ParserContext
Defines syntactic contexts for lexical tie-ins.
@ AUTHENTICATION
Used while parsing content of an authentication type.
@ DDNS_DOMAIN
Used while parsing a list of ddns-domains.
@ DNS_SERVERS
Used while parsing content of a control-socket.
@ DDNS_DOMAINS
Used while parsing content of a dns-server.
@ AUTH_TYPE
Used while parsing content of a client.
@ NCR_FORMAT
Used while parsing DhcpDdns/ncr-format.
@ NCR_PROTOCOL
Used while parsing DhcpDdns/ncr-protocol.
@ NO_KEYWORD
This one is used in pure JSON mode.
@ DHCPDDNS
Used while parsing content of a tsig-key.
@ SECRET
Used while parsing content of DhcpDdns/forward-ddns.
@ ALGORITHM
Used while parsing content of DhcpDdns/tsig-keys/digest-bits.
@ OUTPUT_OPTIONS
Used while parsing DhcpDdns/loggers/output-options structures.
@ FORWARD_DDNS
Used while parsing content of DhcpDdns/reverse-ddns.
@ CONTROL_SOCKET_TYPE
Used while parsing content of an authentication.
@ TSIG_KEY
Used while parsing a list of tsig-keys.
@ REVERSE_DDNS
Used while parsing content of a ddns-domain.
@ LOGGERS
Used while parsing DhcpDdns/loggers structures.
@ HOOKS_LIBRARIES
Used while parsing DhcpDdns/hooks-libraries.
@ TSIG_KEYS
Used while parsing content of DhcpDdns/tsig-keys/algorithm.
@ CONTROL_SOCKET
Used while parsing content of a socket-type.
@ DIGEST_BITS
Used while parsing content of DhcpDdns/tsig-keys/secret.
@ CONFIG
Used while parsing content of DhcpDdns.
@ DNS_SERVER
Used while parsing content of list of dns-servers.
void warning(const isc::d2::location &loc, const std::string &what)
Warning handler.
const std::string contextName()
Get the syntax context name.
void error(const isc::d2::location &loc, const std::string &what, size_t pos=0)
Error handler.
void leave()
Leave a syntactic context.
void scanStringBegin(const std::string &str, ParserType type)
Method called before scanning starts on a string.
Definition d2_lexer.cc:3924
isc::data::ElementPtr parseFile(const std::string &filename, ParserType parser_type)
Run the parser on the file specified.
void warnAboutExtraCommas(const isc::d2::location &loc)
Warning for extra commas.
void scanFileBegin(FILE *f, const std::string &filename, ParserType type)
Method called before scanning starts on a file.
Definition d2_lexer.cc:3942
void unique(const std::string &name, isc::data::Element::Position loc)
Check if a parameter is already present.
virtual ~D2ParserContext()
destructor.
ParserContext ctx_
Current syntactic context.
void require(const std::string &name, isc::data::Element::Position open_loc, isc::data::Element::Position close_loc)
Check if a required parameter is present.
std::vector< isc::data::ElementPtr > stack_
JSON elements being parsed.
void enter(const ParserContext &ctx)
Enter a new syntactic context.
static void fatal(const std::string &what)
Fatal error handler.
void scanEnd()
Method called after the last tokens are scanned.
Definition d2_lexer.cc:3964
isc::data::Element::Position loc2pos(isc::d2::location &loc)
Converts bison's position to one understood by isc::data::Element.
A Bison parser.
Definition d2_parser.h:216
Define the isc::d2::parser class.
#define isc_throw(type, stream)
A shortcut macro to insert known values into exception arguments.
#define LOG_WARN(LOGGER, MESSAGE)
Macro to conveniently test warn output and log it.
Definition macros.h:26
isc::log::Logger d2_to_dns_logger("d2-to-dns")
Definition d2_log.h:20
const isc::log::MessageID DHCP_DDNS_CONFIG_SYNTAX_WARNING
Definition d2_messages.h:19
boost::shared_ptr< const Element > ConstElementPtr
Definition data.h:29
boost::shared_ptr< Element > ElementPtr
Definition data.h:28
Defines the logger used by the top-level component of kea-lfc.
Represents the position of the data element within a configuration string.
Definition data.h:94