Add sstream to simplify the generation of exception messages.

Signed-off-by: Leonardo de Moura <leonardo@microsoft.com>
This commit is contained in:
Leonardo de Moura 2013-08-24 11:55:17 -07:00
parent 48ba655bd5
commit 0b112b6637
4 changed files with 32 additions and 6 deletions

View file

@ -16,6 +16,7 @@ Author: Leonardo de Moura
#include "state.h" #include "state.h"
#include "option_declarations.h" #include "option_declarations.h"
#include "expr_maps.h" #include "expr_maps.h"
#include "sstream.h"
#include "lean_frontend.h" #include "lean_frontend.h"
#include "lean_parser.h" #include "lean_parser.h"
#include "lean_scanner.h" #include "lean_scanner.h"
@ -111,6 +112,7 @@ class parser_fn {
struct parser_error : public exception { struct parser_error : public exception {
pos_info m_pos; pos_info m_pos;
parser_error(char const * msg, pos_info const & p):exception(msg), m_pos(p) {} parser_error(char const * msg, pos_info const & p):exception(msg), m_pos(p) {}
parser_error(sstream const & msg, pos_info const & p):exception(msg), m_pos(p) {}
}; };
/** /**
@ -477,14 +479,14 @@ class parser_fn {
if (k == object_kind::Definition || k == object_kind::Postulate) if (k == object_kind::Definition || k == object_kind::Postulate)
return mk_constant(obj.get_name()); return mk_constant(obj.get_name());
else else
throw parser_error("invalid object reference, object is not an expression.", p); throw parser_error(sstream() << "invalid object reference, object '" << id << "' is not an expression.", p);
} }
else { else {
auto it = m_builtins.find(id); auto it = m_builtins.find(id);
if (it != m_builtins.end()) { if (it != m_builtins.end()) {
return it->second; return it->second;
} else { } else {
throw parser_error("unknown identifier", p); throw parser_error(sstream() << "unknown identifier '" << id << "'", p);
} }
} }
} }
@ -1046,12 +1048,12 @@ class parser_fn {
name id = check_identifier_next("invalid set options, identifier (i.e., option name) expected"); name id = check_identifier_next("invalid set options, identifier (i.e., option name) expected");
auto decl_it = get_option_declarations().find(id); auto decl_it = get_option_declarations().find(id);
if (decl_it == get_option_declarations().end()) if (decl_it == get_option_declarations().end())
throw parser_error("unknown option, type 'Help Options.' for list of available options", id_pos); throw parser_error(sstream() << "unknown option '" << id << "', type 'Help Options.' for list of available options", id_pos);
option_kind k = decl_it->second.kind(); option_kind k = decl_it->second.kind();
switch (curr()) { switch (curr()) {
case scanner::token::Id: case scanner::token::Id:
if (k != BoolOption) if (k != BoolOption)
throw parser_error("invalid option value, given option is not Boolean", pos()); throw parser_error(sstream() << "invalid option value, given option is not Boolean", pos());
if (curr_name() == "true") if (curr_name() == "true")
m_frontend.set_option(id, true); m_frontend.set_option(id, true);
else if (curr_name() == "false") else if (curr_name() == "false")
@ -1086,7 +1088,7 @@ class parser_fn {
std::string fname = check_string_next("invalid import command, string (i.e., file name) expected"); std::string fname = check_string_next("invalid import command, string (i.e., file name) expected");
std::ifstream in(fname); std::ifstream in(fname);
if (!in.is_open()) if (!in.is_open())
throw parser_error("invalid import command, failed to open file", m_last_cmd_pos); throw parser_error(sstream() << "invalid import command, failed to open file '" << fname << "'", m_last_cmd_pos);
::lean::parse_commands(m_frontend, in, m_use_exceptions, false); ::lean::parse_commands(m_frontend, in, m_use_exceptions, false);
} }
@ -1151,7 +1153,7 @@ class parser_fn {
else if (cmd_id == g_set_kwd) parse_set(); else if (cmd_id == g_set_kwd) parse_set();
else if (cmd_id == g_import_kwd) parse_import(); else if (cmd_id == g_import_kwd) parse_import();
else if (cmd_id == g_help_kwd) parse_help(); else if (cmd_id == g_help_kwd) parse_help();
else { next(); throw parser_error("invalid command", m_last_cmd_pos); } else { next(); throw parser_error(sstream() << "invalid command '" << cmd_id << "'", m_last_cmd_pos); }
} }
/*@}*/ /*@}*/

View file

@ -6,17 +6,20 @@ Author: Leonardo de Moura
*/ */
#include <sstream> #include <sstream>
#include "exception.h" #include "exception.h"
#include "sstream.h"
namespace lean { namespace lean {
exception::exception(char const * msg):m_msg(msg) {} exception::exception(char const * msg):m_msg(msg) {}
exception::exception(std::string const & msg):m_msg(msg) {} exception::exception(std::string const & msg):m_msg(msg) {}
exception::exception(sstream const & strm):m_msg(strm.str()) {}
exception::exception(exception const & e):m_msg(e.m_msg) {} exception::exception(exception const & e):m_msg(e.m_msg) {}
exception::~exception() noexcept {} exception::~exception() noexcept {}
char const * exception::what() const noexcept { return m_msg.c_str(); } char const * exception::what() const noexcept { return m_msg.c_str(); }
parser_exception::parser_exception(char const * msg, unsigned l, unsigned p):exception(msg), m_line(l), m_pos(p) {} parser_exception::parser_exception(char const * msg, unsigned l, unsigned p):exception(msg), m_line(l), m_pos(p) {}
parser_exception::parser_exception(std::string const & msg, unsigned l, unsigned p):exception(msg), m_line(l), m_pos(p) {} parser_exception::parser_exception(std::string const & msg, unsigned l, unsigned p):exception(msg), m_line(l), m_pos(p) {}
parser_exception::parser_exception(sstream const & msg, unsigned l, unsigned p):exception(msg), m_line(l), m_pos(p) {}
parser_exception::parser_exception(parser_exception const & e):exception(e), m_line(e.m_line), m_pos(e.m_pos) {} parser_exception::parser_exception(parser_exception const & e):exception(e), m_line(e.m_line), m_pos(e.m_pos) {}
parser_exception::~parser_exception() noexcept {} parser_exception::~parser_exception() noexcept {}
char const * parser_exception::what() const noexcept { char const * parser_exception::what() const noexcept {

View file

@ -9,6 +9,7 @@ Author: Leonardo de Moura
#include <string> #include <string>
namespace lean { namespace lean {
class sstream;
/** \brief Base class for all Lean exceptions */ /** \brief Base class for all Lean exceptions */
class exception : public std::exception { class exception : public std::exception {
protected: protected:
@ -17,6 +18,7 @@ protected:
public: public:
exception(char const * msg); exception(char const * msg);
exception(std::string const & msg); exception(std::string const & msg);
exception(sstream const & strm);
exception(exception const & ex); exception(exception const & ex);
virtual ~exception() noexcept; virtual ~exception() noexcept;
virtual char const * what() const noexcept; virtual char const * what() const noexcept;
@ -29,6 +31,7 @@ protected:
public: public:
parser_exception(char const * msg, unsigned l, unsigned p); parser_exception(char const * msg, unsigned l, unsigned p);
parser_exception(std::string const & msg, unsigned l, unsigned p); parser_exception(std::string const & msg, unsigned l, unsigned p);
parser_exception(sstream const & strm, unsigned l, unsigned p);
parser_exception(parser_exception const & ex); parser_exception(parser_exception const & ex);
virtual ~parser_exception() noexcept; virtual ~parser_exception() noexcept;
virtual char const * what() const noexcept; virtual char const * what() const noexcept;

18
src/util/sstream.h Normal file
View file

@ -0,0 +1,18 @@
/*
Copyright (c) 2013 Microsoft Corporation. All rights reserved.
Released under Apache 2.0 license as described in the file LICENSE.
Author: Leonardo de Moura
*/
#pragma once
#include <sstream>
namespace lean {
/** \brief Wrapper for std::ostringstream */
class sstream {
std::ostringstream m_strm;
public:
std::string str() const { return m_strm.str(); }
template<typename T> sstream & operator<<(T const & t) { m_strm << t; return *this; }
};
}