2014-01-14 00:54:21 +00:00
|
|
|
/*
|
|
|
|
Copyright (c) 2013 Microsoft Corporation. All rights reserved.
|
|
|
|
Released under Apache 2.0 license as described in the file LICENSE.
|
|
|
|
|
|
|
|
Author: Leonardo de Moura
|
|
|
|
*/
|
|
|
|
#include <utility>
|
|
|
|
#include <string>
|
|
|
|
#include "util/script_exception.h"
|
|
|
|
#include "util/name_set.h"
|
2015-12-04 21:00:57 +00:00
|
|
|
#include "kernel/ext_exception.h"
|
2014-01-14 00:54:21 +00:00
|
|
|
#include "kernel/for_each_fn.h"
|
|
|
|
#include "library/io_state_stream.h"
|
2014-06-25 16:41:25 +00:00
|
|
|
#include "library/unifier.h"
|
2014-01-14 00:54:21 +00:00
|
|
|
#include "library/parser_nested_exception.h"
|
2014-10-15 01:26:21 +00:00
|
|
|
#include "library/flycheck.h"
|
2015-06-12 21:46:51 +00:00
|
|
|
#include "library/pp_options.h"
|
2015-12-29 23:31:40 +00:00
|
|
|
#include "library/error_handling.h"
|
2014-01-14 00:54:21 +00:00
|
|
|
|
|
|
|
namespace lean {
|
2015-03-31 18:53:55 +00:00
|
|
|
void display_pos(std::ostream & out, options const & o, char const * strm_name, unsigned line, unsigned pos) {
|
|
|
|
out << strm_name << ":";
|
|
|
|
if (o.get_bool("flycheck", false)) {
|
2014-09-15 03:15:22 +00:00
|
|
|
// generate valid line and column for flycheck mode
|
|
|
|
if (line == static_cast<unsigned>(-1))
|
|
|
|
line = 1;
|
|
|
|
if (pos == static_cast<unsigned>(-1))
|
|
|
|
pos = 0;
|
|
|
|
}
|
2014-08-14 22:47:46 +00:00
|
|
|
if (line != static_cast<unsigned>(-1))
|
2015-03-31 18:53:55 +00:00
|
|
|
out << line << ":";
|
2014-01-14 00:54:21 +00:00
|
|
|
if (pos != static_cast<unsigned>(-1))
|
2015-03-31 18:53:55 +00:00
|
|
|
out << pos << ":";
|
|
|
|
}
|
|
|
|
|
|
|
|
void display_pos(std::ostream & out, char const * strm_name, unsigned line, unsigned pos) {
|
|
|
|
display_pos(out, options(), strm_name, line, pos);
|
|
|
|
}
|
|
|
|
|
|
|
|
void display_pos(io_state_stream const & ios, char const * strm_name, unsigned line, unsigned pos) {
|
|
|
|
display_pos(ios.get_stream(), ios.get_options(), strm_name, line, pos);
|
2014-08-02 06:21:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void display_error_pos(io_state_stream const & ios, char const * strm_name, unsigned line, unsigned pos) {
|
|
|
|
display_pos(ios, strm_name, line, pos);
|
2014-04-30 17:28:07 +00:00
|
|
|
ios << " error:";
|
2014-01-14 00:54:21 +00:00
|
|
|
}
|
|
|
|
|
2014-08-02 06:21:55 +00:00
|
|
|
void display_warning_pos(io_state_stream const & ios, char const * strm_name, unsigned line, unsigned pos) {
|
|
|
|
display_pos(ios, strm_name, line, pos);
|
|
|
|
ios << " warning:";
|
|
|
|
}
|
|
|
|
|
2014-11-24 16:35:49 +00:00
|
|
|
void display_information_pos(io_state_stream const & ios, char const * strm_name, unsigned line, unsigned pos) {
|
|
|
|
display_pos(ios, strm_name, line, pos);
|
|
|
|
ios << " information:";
|
|
|
|
}
|
|
|
|
|
2014-04-30 17:28:07 +00:00
|
|
|
void display_error_pos(io_state_stream const & ios, pos_info_provider const * p, expr const & e) {
|
2014-01-14 00:54:21 +00:00
|
|
|
if (p) {
|
2014-08-02 03:17:26 +00:00
|
|
|
auto pos = p->get_pos_info_or_some(e);
|
2014-01-14 00:54:21 +00:00
|
|
|
display_error_pos(ios, p->get_file_name(), pos.first, pos.second);
|
|
|
|
} else {
|
2014-04-30 17:28:07 +00:00
|
|
|
ios << "error:";
|
2014-01-14 00:54:21 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-04-30 17:28:07 +00:00
|
|
|
void display_error_pos(io_state_stream const & ios, pos_info_provider const * p, optional<expr> const & e) {
|
2014-01-14 00:54:21 +00:00
|
|
|
if (e) {
|
|
|
|
display_error_pos(ios, p, *e);
|
|
|
|
} else if (p) {
|
|
|
|
auto pos = p->get_some_pos();
|
|
|
|
display_error_pos(ios, p->get_file_name(), pos.first, pos.second);
|
|
|
|
} else {
|
2014-04-30 17:28:07 +00:00
|
|
|
ios << "error:";
|
2014-01-14 00:54:21 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-15 23:05:47 +00:00
|
|
|
void display_error(io_state_stream const & ios, pos_info_provider const * p, throwable const & ex);
|
2014-01-14 00:54:21 +00:00
|
|
|
|
2015-12-04 21:00:57 +00:00
|
|
|
static void display_error(io_state_stream const & ios, pos_info_provider const * p, ext_exception const & ex) {
|
2014-12-15 19:04:55 +00:00
|
|
|
display_error_pos(ios, p, ex.get_main_expr());
|
|
|
|
ios << " " << ex << endl;
|
|
|
|
}
|
|
|
|
|
2014-06-25 16:41:25 +00:00
|
|
|
static void display_error(io_state_stream const & ios, pos_info_provider const * p, unifier_exception const & ex) {
|
|
|
|
formatter fmt = ios.get_formatter();
|
|
|
|
options opts = ios.get_options();
|
|
|
|
auto j = ex.get_justification();
|
2014-06-26 15:49:50 +00:00
|
|
|
display_error_pos(ios, p, j.get_main_expr());
|
2014-07-10 17:32:00 +00:00
|
|
|
ios << " " << mk_pair(j.pp(fmt, p, ex.get_substitution()), opts) << endl;
|
2014-06-25 16:41:25 +00:00
|
|
|
}
|
2014-04-29 18:52:09 +00:00
|
|
|
|
2014-04-30 17:28:07 +00:00
|
|
|
static void display_error(io_state_stream const & ios, pos_info_provider const * p, script_exception const & ex) {
|
2014-01-14 00:54:21 +00:00
|
|
|
if (p) {
|
|
|
|
char const * msg = ex.get_msg();
|
|
|
|
char const * space = msg && *msg == ' ' ? "" : " ";
|
|
|
|
switch (ex.get_source()) {
|
|
|
|
case script_exception::source::String:
|
|
|
|
display_error_pos(ios, p->get_file_name(), ex.get_line() + p->get_some_pos().first - 1, static_cast<unsigned>(-1));
|
2014-04-30 17:28:07 +00:00
|
|
|
ios << " executing script," << space << msg << endl;
|
2014-01-14 00:54:21 +00:00
|
|
|
break;
|
|
|
|
case script_exception::source::File:
|
|
|
|
display_error_pos(ios, p->get_file_name(), p->get_some_pos().first, p->get_some_pos().second);
|
2014-04-30 17:28:07 +00:00
|
|
|
ios << " executing external script (" << ex.get_file_name() << ":" << ex.get_line() << ")," << space << msg << endl;
|
2014-01-14 00:54:21 +00:00
|
|
|
break;
|
|
|
|
case script_exception::source::Unknown:
|
|
|
|
display_error_pos(ios, p->get_file_name(), p->get_some_pos().first, p->get_some_pos().second);
|
2014-04-30 17:28:07 +00:00
|
|
|
ios << " executing script, exact error position is not available, " << ex.what() << endl;
|
2014-01-14 00:54:21 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
} else {
|
2014-04-30 17:28:07 +00:00
|
|
|
ios << ex.what() << endl;
|
2014-01-14 00:54:21 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-04-30 17:28:07 +00:00
|
|
|
static void display_error(io_state_stream const & ios, pos_info_provider const * p, script_nested_exception const & ex) {
|
2014-01-14 00:54:21 +00:00
|
|
|
switch (ex.get_source()) {
|
|
|
|
case script_exception::source::String:
|
|
|
|
if (p) {
|
|
|
|
display_error_pos(ios, p->get_file_name(), ex.get_line() + p->get_some_pos().first - 1, static_cast<unsigned>(-1));
|
2014-04-30 17:28:07 +00:00
|
|
|
ios << " executing script" << endl;
|
2014-01-14 00:54:21 +00:00
|
|
|
}
|
|
|
|
display_error(ios, nullptr, ex.get_exception());
|
|
|
|
break;
|
|
|
|
case script_exception::source::File:
|
|
|
|
if (p) {
|
|
|
|
display_error_pos(ios, p->get_file_name(), p->get_some_pos().first, p->get_some_pos().second);
|
2014-04-30 17:28:07 +00:00
|
|
|
ios << " executing external script (" << ex.get_file_name() << ":" << ex.get_line() << ")" << endl;
|
2014-01-14 00:54:21 +00:00
|
|
|
} else {
|
|
|
|
display_error_pos(ios, ex.get_file_name(), ex.get_line(), -1);
|
2014-04-30 17:28:07 +00:00
|
|
|
ios << " executing script" << endl;
|
2014-01-14 00:54:21 +00:00
|
|
|
}
|
|
|
|
display_error(ios, nullptr, ex.get_exception());
|
|
|
|
break;
|
|
|
|
case script_exception::source::Unknown:
|
|
|
|
display_error(ios, nullptr, ex.get_exception());
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-06-17 18:59:49 +00:00
|
|
|
void display_error(io_state_stream const & ios, pos_info_provider const * p, throwable const & ex) {
|
2014-08-01 00:41:39 +00:00
|
|
|
flycheck_error err(ios);
|
2015-12-04 21:00:57 +00:00
|
|
|
if (auto k_ex = dynamic_cast<ext_exception const *>(&ex)) {
|
2014-01-14 00:54:21 +00:00
|
|
|
display_error(ios, p, *k_ex);
|
2014-06-25 16:41:25 +00:00
|
|
|
} else if (auto e_ex = dynamic_cast<unifier_exception const *>(&ex)) {
|
|
|
|
display_error(ios, p, *e_ex);
|
2014-01-14 00:54:21 +00:00
|
|
|
} else if (auto ls_ex = dynamic_cast<script_nested_exception const *>(&ex)) {
|
|
|
|
display_error(ios, p, *ls_ex);
|
|
|
|
} else if (auto s_ex = dynamic_cast<script_exception const *>(&ex)) {
|
|
|
|
display_error(ios, p, *s_ex);
|
|
|
|
} else if (p) {
|
|
|
|
display_error_pos(ios, p->get_file_name(), p->get_some_pos().first, p->get_some_pos().second);
|
2014-04-30 17:28:07 +00:00
|
|
|
ios << " " << ex.what() << endl;
|
2014-01-14 00:54:21 +00:00
|
|
|
} else {
|
2014-04-30 17:28:07 +00:00
|
|
|
ios << "error: " << ex.what() << endl;
|
2014-01-14 00:54:21 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|