// file : libbuild2/types-parsers.cxx -*- C++ -*- // license : MIT; see accompanying LICENSE file #include <libbuild2/types-parsers.hxx> #include <sstream> #include <libbuild2/lexer.hxx> #include <libbuild2/parser.hxx> namespace build2 { namespace build { namespace cli { template <typename T> static void parse_path (T& x, scanner& s) { const char* o (s.next ()); if (!s.more ()) throw missing_value (o); const char* v (s.next ()); try { x = T (v); if (x.empty ()) throw invalid_value (o, v); } catch (const invalid_path&) { throw invalid_value (o, v); } } void parser<path>:: parse (path& x, bool& xs, scanner& s) { xs = true; parse_path (x, s); } void parser<dir_path>:: parse (dir_path& x, bool& xs, scanner& s) { xs = true; parse_path (x, s); } static names parse_names (const char* o, const char* v) { using build2::parser; using std::istringstream; istringstream is (v); is.exceptions (istringstream::failbit | istringstream::badbit); // @@ TODO: currently this issues diagnostics to diag_stream. // Perhaps we should redirect it? Also below. // path_name in (o); lexer l (is, in, 1 /* line */, "\'\"\\$("); // Effective. parser p (nullptr); return p.parse_names (l, nullptr, parser::pattern_mode::preserve); } void parser<name>:: parse (name& x, bool& xs, scanner& s) { const char* o (s.next ()); if (!s.more ()) throw missing_value (o); const char* v (s.next ()); try { names r (parse_names (o, v)); if (r.size () != 1) throw invalid_value (o, v); x = move (r.front ()); xs = true; } catch (const failed&) { throw invalid_value (o, v); } } void parser<pair<name, optional<name>>>:: parse (pair<name, optional<name>>& x, bool& xs, scanner& s) { const char* o (s.next ()); if (!s.more ()) throw missing_value (o); const char* v (s.next ()); try { names r (parse_names (o, v)); if (r.size () == 1) { x.first = move (r.front ()); x.second = nullopt; } else if (r.size () == 2 && r.front ().pair == '@') { x.first = move (r.front ()); x.second = move (r.back ()); } else throw invalid_value (o, v); xs = true; } catch (const failed&) { throw invalid_value (o, v); } } void parser<structured_result_format>:: parse (structured_result_format& x, bool& xs, scanner& s) { xs = true; const char* o (s.next ()); if (!s.more ()) throw missing_value (o); const string v (s.next ()); if (v == "lines") x = structured_result_format::lines; else if (v == "json") x = structured_result_format::json; else throw invalid_value (o, v); } } } }